utils.go 2.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131
  1. package main
  2. import (
  3. "bytes"
  4. "crypto/rand"
  5. "fmt"
  6. "path/filepath"
  7. "regexp"
  8. "strconv"
  9. "strings"
  10. "github.com/lib/pq"
  11. )
  12. var (
  13. tagRegexpSpaces = regexp.MustCompile(`[\s]+`)
  14. )
  15. // inArray checks if a string is present in a list of strings.
  16. func inArray(val string, vals []string) (ok bool) {
  17. for _, v := range vals {
  18. if v == val {
  19. return true
  20. }
  21. }
  22. return false
  23. }
  24. // makeFilename sanitizes a filename (user supplied upload filenames).
  25. func makeFilename(fName string) string {
  26. name := strings.TrimSpace(fName)
  27. if name == "" {
  28. name, _ = generateRandomString(10)
  29. }
  30. return filepath.Base(name)
  31. }
  32. // Given an error, pqErrMsg will try to return pq error details
  33. // if it's a pq error.
  34. func pqErrMsg(err error) string {
  35. if err, ok := err.(*pq.Error); ok {
  36. if err.Detail != "" {
  37. return fmt.Sprintf("%s. %s", err, err.Detail)
  38. }
  39. }
  40. return err.Error()
  41. }
  42. // normalizeTags takes a list of string tags and normalizes them by
  43. // lowercasing and removing all special characters except for dashes.
  44. func normalizeTags(tags []string) []string {
  45. var (
  46. out []string
  47. dash = []byte("-")
  48. )
  49. for _, t := range tags {
  50. rep := tagRegexpSpaces.ReplaceAll(bytes.TrimSpace([]byte(t)), dash)
  51. if len(rep) > 0 {
  52. out = append(out, string(rep))
  53. }
  54. }
  55. return out
  56. }
  57. // makeMsgTpl takes a page title, heading, and message and returns
  58. // a msgTpl that can be rendered as a HTML view. This is used for
  59. // rendering arbitrary HTML views with error and success messages.
  60. func makeMsgTpl(pageTitle, heading, msg string) msgTpl {
  61. if heading == "" {
  62. heading = pageTitle
  63. }
  64. err := msgTpl{}
  65. err.Title = pageTitle
  66. err.MessageTitle = heading
  67. err.Message = msg
  68. return err
  69. }
  70. // parseStringIDs takes a slice of numeric string IDs and
  71. // parses each number into an int64 and returns a slice of the
  72. // resultant values.
  73. func parseStringIDs(s []string) ([]int64, error) {
  74. vals := make([]int64, 0, len(s))
  75. for _, v := range s {
  76. i, err := strconv.ParseInt(v, 10, 64)
  77. if err != nil {
  78. return nil, err
  79. }
  80. if i < 1 {
  81. return nil, fmt.Errorf("%d is not a valid ID", i)
  82. }
  83. vals = append(vals, i)
  84. }
  85. return vals, nil
  86. }
  87. // generateRandomString generates a cryptographically random, alphanumeric string of length n.
  88. func generateRandomString(n int) (string, error) {
  89. const dictionary = "0123456789ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz"
  90. var bytes = make([]byte, n)
  91. if _, err := rand.Read(bytes); err != nil {
  92. return "", err
  93. }
  94. for k, v := range bytes {
  95. bytes[k] = dictionary[v%byte(len(dictionary))]
  96. }
  97. return string(bytes), nil
  98. }
  99. // strHasLen checks if the given string has a length within min-max.
  100. func strHasLen(str string, min, max int) bool {
  101. return len(str) >= min && len(str) <= max
  102. }
  103. // strSliceContains checks if a string is present in the string slice.
  104. func strSliceContains(str string, sl []string) bool {
  105. for _, s := range sl {
  106. if s == str {
  107. return true
  108. }
  109. }
  110. return false
  111. }