util_test.go 8.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264
  1. package util
  2. import (
  3. "errors"
  4. "io"
  5. "net/netip"
  6. "os"
  7. "path/filepath"
  8. "strings"
  9. "testing"
  10. "time"
  11. "github.com/stretchr/testify/require"
  12. )
  13. func TestRandomString(t *testing.T) {
  14. s1 := RandomString(10)
  15. s2 := RandomString(10)
  16. s3 := RandomString(12)
  17. require.Equal(t, 10, len(s1))
  18. require.Equal(t, 10, len(s2))
  19. require.Equal(t, 12, len(s3))
  20. require.NotEqual(t, s1, s2)
  21. }
  22. func TestFileExists(t *testing.T) {
  23. filename := filepath.Join(t.TempDir(), "somefile.txt")
  24. require.Nil(t, os.WriteFile(filename, []byte{0x25, 0x86}, 0600))
  25. require.True(t, FileExists(filename))
  26. require.False(t, FileExists(filename+".doesnotexist"))
  27. }
  28. func TestInStringList(t *testing.T) {
  29. s := []string{"one", "two"}
  30. require.True(t, Contains(s, "two"))
  31. require.False(t, Contains(s, "three"))
  32. }
  33. func TestInStringListAll(t *testing.T) {
  34. s := []string{"one", "two", "three", "four"}
  35. require.True(t, ContainsAll(s, []string{"two", "four"}))
  36. require.False(t, ContainsAll(s, []string{"three", "five"}))
  37. }
  38. func TestContains(t *testing.T) {
  39. s := []int{1, 2}
  40. require.True(t, Contains(s, 2))
  41. require.False(t, Contains(s, 3))
  42. }
  43. func TestContainsIP(t *testing.T) {
  44. require.True(t, ContainsIP([]netip.Prefix{netip.MustParsePrefix("fd00::/8"), netip.MustParsePrefix("1.1.0.0/16")}, netip.MustParseAddr("1.1.1.1")))
  45. require.True(t, ContainsIP([]netip.Prefix{netip.MustParsePrefix("fd00::/8"), netip.MustParsePrefix("1.1.0.0/16")}, netip.MustParseAddr("fd12:1234:5678::9876")))
  46. require.False(t, ContainsIP([]netip.Prefix{netip.MustParsePrefix("fd00::/8"), netip.MustParsePrefix("1.1.0.0/16")}, netip.MustParseAddr("1.2.0.1")))
  47. require.False(t, ContainsIP([]netip.Prefix{netip.MustParsePrefix("fd00::/8"), netip.MustParsePrefix("1.1.0.0/16")}, netip.MustParseAddr("fc00::1")))
  48. }
  49. func TestSplitNoEmpty(t *testing.T) {
  50. require.Equal(t, []string{}, SplitNoEmpty("", ","))
  51. require.Equal(t, []string{}, SplitNoEmpty(",,,", ","))
  52. require.Equal(t, []string{"tag1", "tag2"}, SplitNoEmpty("tag1,tag2", ","))
  53. require.Equal(t, []string{"tag1", "tag2"}, SplitNoEmpty("tag1,tag2,", ","))
  54. }
  55. func TestParsePriority(t *testing.T) {
  56. priorities := []string{"", "1", "2", "3", "4", "5", "min", "LOW", " default ", "HIgh", "max", "urgent"}
  57. expected := []int{0, 1, 2, 3, 4, 5, 1, 2, 3, 4, 5, 5}
  58. for i, priority := range priorities {
  59. actual, err := ParsePriority(priority)
  60. require.Nil(t, err)
  61. require.Equal(t, expected[i], actual)
  62. }
  63. }
  64. func TestParsePriority_Invalid(t *testing.T) {
  65. priorities := []string{"-1", "6", "aa", "-", "o=1"}
  66. for _, priority := range priorities {
  67. _, err := ParsePriority(priority)
  68. require.Equal(t, errInvalidPriority, err)
  69. }
  70. }
  71. func TestParsePriority_HTTPSpecPriority(t *testing.T) {
  72. priorities := []string{"u=1", "u=3", "u=7, i"} // see https://datatracker.ietf.org/doc/html/draft-ietf-httpbis-priority
  73. for _, priority := range priorities {
  74. actual, err := ParsePriority(priority)
  75. require.Nil(t, err)
  76. require.Equal(t, 3, actual) // Always expect 3!
  77. }
  78. }
  79. func TestPriorityString(t *testing.T) {
  80. priorities := []int{0, 1, 2, 3, 4, 5}
  81. expected := []string{"default", "min", "low", "default", "high", "max"}
  82. for i, priority := range priorities {
  83. actual, err := PriorityString(priority)
  84. require.Nil(t, err)
  85. require.Equal(t, expected[i], actual)
  86. }
  87. }
  88. func TestPriorityString_Invalid(t *testing.T) {
  89. _, err := PriorityString(99)
  90. require.Equal(t, err, errInvalidPriority)
  91. }
  92. func TestShortTopicURL(t *testing.T) {
  93. require.Equal(t, "ntfy.sh/mytopic", ShortTopicURL("https://ntfy.sh/mytopic"))
  94. require.Equal(t, "ntfy.sh/mytopic", ShortTopicURL("http://ntfy.sh/mytopic"))
  95. require.Equal(t, "lalala", ShortTopicURL("lalala"))
  96. }
  97. func TestParseSize_10GSuccess(t *testing.T) {
  98. s, err := ParseSize("10G")
  99. if err != nil {
  100. t.Fatal(err)
  101. }
  102. require.Equal(t, int64(10*1024*1024*1024), s)
  103. }
  104. func TestParseSize_10MUpperCaseSuccess(t *testing.T) {
  105. s, err := ParseSize("10M")
  106. if err != nil {
  107. t.Fatal(err)
  108. }
  109. require.Equal(t, int64(10*1024*1024), s)
  110. }
  111. func TestParseSize_10kLowerCaseSuccess(t *testing.T) {
  112. s, err := ParseSize("10k")
  113. if err != nil {
  114. t.Fatal(err)
  115. }
  116. require.Equal(t, int64(10*1024), s)
  117. }
  118. func TestParseSize_FailureInvalid(t *testing.T) {
  119. _, err := ParseSize("not a size")
  120. if err == nil {
  121. t.Fatalf("expected error, but got none")
  122. }
  123. }
  124. func TestSplitKV(t *testing.T) {
  125. key, value := SplitKV(" key = value ", "=")
  126. require.Equal(t, "key", key)
  127. require.Equal(t, "value", value)
  128. key, value = SplitKV(" value ", "=")
  129. require.Equal(t, "", key)
  130. require.Equal(t, "value", value)
  131. key, value = SplitKV("mykey=value=with=separator ", "=")
  132. require.Equal(t, "mykey", key)
  133. require.Equal(t, "value=with=separator", value)
  134. }
  135. func TestLastString(t *testing.T) {
  136. require.Equal(t, "last", LastString([]string{"first", "second", "last"}, "default"))
  137. require.Equal(t, "default", LastString([]string{}, "default"))
  138. }
  139. func TestQuoteCommand(t *testing.T) {
  140. require.Equal(t, `ls -al "Document Folder"`, QuoteCommand([]string{"ls", "-al", "Document Folder"}))
  141. require.Equal(t, `rsync -av /home/phil/ root@example.com:/home/phil/`, QuoteCommand([]string{"rsync", "-av", "/home/phil/", "root@example.com:/home/phil/"}))
  142. require.Equal(t, `/home/sweet/home "Äöü this is a test" "\a\b"`, QuoteCommand([]string{"/home/sweet/home", "Äöü this is a test", "\\a\\b"}))
  143. }
  144. func TestBasicAuth(t *testing.T) {
  145. require.Equal(t, "Basic cGhpbDpwaGls", BasicAuth("phil", "phil"))
  146. }
  147. func TestBearerAuth(t *testing.T) {
  148. require.Equal(t, "Bearer sometoken", BearerAuth("sometoken"))
  149. }
  150. type testJSON struct {
  151. Name string `json:"name"`
  152. Something int `json:"something"`
  153. }
  154. func TestReadJSON_Success(t *testing.T) {
  155. v, err := UnmarshalJSON[testJSON](io.NopCloser(strings.NewReader(`{"name":"some name","something":99}`)))
  156. require.Nil(t, err)
  157. require.Equal(t, "some name", v.Name)
  158. require.Equal(t, 99, v.Something)
  159. }
  160. func TestReadJSON_Failure(t *testing.T) {
  161. _, err := UnmarshalJSON[testJSON](io.NopCloser(strings.NewReader(`{"na`)))
  162. require.Equal(t, ErrUnmarshalJSON, err)
  163. }
  164. func TestReadJSONWithLimit_Success(t *testing.T) {
  165. v, err := UnmarshalJSONWithLimit[testJSON](io.NopCloser(strings.NewReader(`{"name":"some name","something":99}`)), 100, false)
  166. require.Nil(t, err)
  167. require.Equal(t, "some name", v.Name)
  168. require.Equal(t, 99, v.Something)
  169. }
  170. func TestReadJSONWithLimit_FailureTooLong(t *testing.T) {
  171. _, err := UnmarshalJSONWithLimit[testJSON](io.NopCloser(strings.NewReader(`{"name":"some name","something":99}`)), 10, false)
  172. require.Equal(t, ErrTooLargeJSON, err)
  173. }
  174. func TestReadJSONWithLimit_AllowEmpty(t *testing.T) {
  175. v, err := UnmarshalJSONWithLimit[testJSON](io.NopCloser(strings.NewReader(` `)), 10, true)
  176. require.Nil(t, err)
  177. require.Equal(t, "", v.Name)
  178. require.Equal(t, 0, v.Something)
  179. }
  180. func TestReadJSONWithLimit_NoAllowEmpty(t *testing.T) {
  181. _, err := UnmarshalJSONWithLimit[testJSON](io.NopCloser(strings.NewReader(` `)), 10, false)
  182. require.Equal(t, ErrUnmarshalJSON, err)
  183. }
  184. func TestRetry_Succeeds(t *testing.T) {
  185. start := time.Now()
  186. delays, i := []time.Duration{10 * time.Millisecond, 50 * time.Millisecond, 100 * time.Millisecond, time.Second}, 0
  187. fn := func() (*int, error) {
  188. i++
  189. if i < len(delays) {
  190. return nil, errors.New("error")
  191. }
  192. return Int(99), nil
  193. }
  194. result, err := Retry[int](fn, delays...)
  195. require.Nil(t, err)
  196. require.Equal(t, 99, *result)
  197. require.True(t, time.Since(start).Milliseconds() > 150)
  198. }
  199. func TestRetry_Fails(t *testing.T) {
  200. fn := func() (*int, error) {
  201. return nil, errors.New("fails")
  202. }
  203. _, err := Retry[int](fn, 10*time.Millisecond)
  204. require.Error(t, err)
  205. }
  206. func TestMinMax(t *testing.T) {
  207. require.Equal(t, 10, MinMax(9, 10, 99))
  208. require.Equal(t, 99, MinMax(100, 10, 99))
  209. require.Equal(t, 50, MinMax(50, 10, 99))
  210. }
  211. func TestPointerFunctions(t *testing.T) {
  212. i, s, ti := Int(99), String("abc"), Time(time.Unix(99, 0))
  213. require.Equal(t, 99, *i)
  214. require.Equal(t, "abc", *s)
  215. require.Equal(t, time.Unix(99, 0), *ti)
  216. }
  217. func TestMaybeMarshalJSON(t *testing.T) {
  218. require.Equal(t, `"aa"`, MaybeMarshalJSON("aa"))
  219. require.Equal(t, `[
  220. "aa",
  221. "bb"
  222. ]`, MaybeMarshalJSON([]string{"aa", "bb"}))
  223. require.Equal(t, "<cannot serialize>", MaybeMarshalJSON(func() {}))
  224. require.Equal(t, `"`+strings.Repeat("x", 4999), MaybeMarshalJSON(strings.Repeat("x", 6000)))
  225. }