util.go 4.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149
  1. package server
  2. import (
  3. "encoding/json"
  4. "heckel.io/ntfy/util"
  5. "net/http"
  6. "strings"
  7. )
  8. const (
  9. actionIDLength = 10
  10. actionsMax = 3
  11. )
  12. func readBoolParam(r *http.Request, defaultValue bool, names ...string) bool {
  13. value := strings.ToLower(readParam(r, names...))
  14. if value == "" {
  15. return defaultValue
  16. }
  17. return value == "1" || value == "yes" || value == "true"
  18. }
  19. func readParam(r *http.Request, names ...string) string {
  20. value := readHeaderParam(r, names...)
  21. if value != "" {
  22. return value
  23. }
  24. return readQueryParam(r, names...)
  25. }
  26. func readHeaderParam(r *http.Request, names ...string) string {
  27. for _, name := range names {
  28. value := r.Header.Get(name)
  29. if value != "" {
  30. return strings.TrimSpace(value)
  31. }
  32. }
  33. return ""
  34. }
  35. func readQueryParam(r *http.Request, names ...string) string {
  36. for _, name := range names {
  37. value := r.URL.Query().Get(strings.ToLower(name))
  38. if value != "" {
  39. return strings.TrimSpace(value)
  40. }
  41. }
  42. return ""
  43. }
  44. func parseActions(s string) (actions []*action, err error) {
  45. // Parse JSON or simple format
  46. s = strings.TrimSpace(s)
  47. if strings.HasPrefix(s, "[") {
  48. actions, err = parseActionsFromJSON(s)
  49. } else {
  50. actions, err = parseActionsFromSimple(s)
  51. }
  52. if err != nil {
  53. return nil, err
  54. }
  55. // Add ID field, ensure correct uppercase/lowercase
  56. for i := range actions {
  57. actions[i].ID = util.RandomString(actionIDLength)
  58. actions[i].Action = strings.ToLower(actions[i].Action)
  59. actions[i].Method = strings.ToUpper(actions[i].Method)
  60. }
  61. // Validate
  62. if len(actions) > actionsMax {
  63. return nil, wrapErrHTTP(errHTTPBadRequestActionsInvalid, "only %d actions allowed", actionsMax)
  64. }
  65. for _, action := range actions {
  66. if !util.InStringList([]string{"view", "broadcast", "http"}, action.Action) {
  67. return nil, wrapErrHTTP(errHTTPBadRequestActionsInvalid, "action '%s' unknown", action.Action)
  68. } else if action.Label == "" {
  69. return nil, wrapErrHTTP(errHTTPBadRequestActionsInvalid, "parameter 'label' is required")
  70. } else if util.InStringList([]string{"view", "http"}, action.Action) && action.URL == "" {
  71. return nil, wrapErrHTTP(errHTTPBadRequestActionsInvalid, "parameter 'url' is required for action '%s'", action.Action)
  72. } else if action.Action == "http" && util.InStringList([]string{"GET", "HEAD"}, action.Method) && action.Body != "" {
  73. return nil, wrapErrHTTP(errHTTPBadRequestActionsInvalid, "parameter 'body' cannot be set if method is %s", action.Method)
  74. }
  75. }
  76. return actions, nil
  77. }
  78. func parseActionsFromJSON(s string) ([]*action, error) {
  79. actions := make([]*action, 0)
  80. if err := json.Unmarshal([]byte(s), &actions); err != nil {
  81. return nil, err
  82. }
  83. return actions, nil
  84. }
  85. func parseActionsFromSimple(s string) ([]*action, error) {
  86. actions := make([]*action, 0)
  87. rawActions := util.SplitNoEmpty(s, ";")
  88. for _, rawAction := range rawActions {
  89. newAction := &action{
  90. Headers: make(map[string]string),
  91. Extras: make(map[string]string),
  92. }
  93. parts := util.SplitNoEmpty(rawAction, ",")
  94. if len(parts) < 3 {
  95. return nil, wrapErrHTTP(errHTTPBadRequestActionsInvalid, "action requires at least keys 'action', 'label' and one parameter: %s", rawAction)
  96. }
  97. for i, part := range parts {
  98. key, value := util.SplitKV(part, "=")
  99. if key == "" && i == 0 {
  100. newAction.Action = value
  101. } else if key == "" && i == 1 {
  102. newAction.Label = value
  103. } else if key == "" && util.InStringList([]string{"view", "http"}, newAction.Action) && i == 2 {
  104. newAction.URL = value
  105. } else if strings.HasPrefix(key, "headers.") {
  106. newAction.Headers[strings.TrimPrefix(key, "headers.")] = value
  107. } else if strings.HasPrefix(key, "extras.") {
  108. newAction.Extras[strings.TrimPrefix(key, "extras.")] = value
  109. } else if key != "" {
  110. switch strings.ToLower(key) {
  111. case "action":
  112. newAction.Action = value
  113. case "label":
  114. newAction.Label = value
  115. case "clear":
  116. lvalue := strings.ToLower(value)
  117. if !util.InStringList([]string{"true", "yes", "1", "false", "no", "0"}, lvalue) {
  118. return nil, wrapErrHTTP(errHTTPBadRequestActionsInvalid, "'clear=%s' not allowed", value)
  119. }
  120. newAction.Clear = lvalue == "true" || lvalue == "yes" || lvalue == "1"
  121. case "url":
  122. newAction.URL = value
  123. case "method":
  124. newAction.Method = value
  125. case "body":
  126. newAction.Body = value
  127. default:
  128. return nil, wrapErrHTTP(errHTTPBadRequestActionsInvalid, "key '%s' unknown", key)
  129. }
  130. } else {
  131. return nil, wrapErrHTTP(errHTTPBadRequestActionsInvalid, "unknown term '%s'", part)
  132. }
  133. }
  134. actions = append(actions, newAction)
  135. }
  136. return actions, nil
  137. }