sort.go 3.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131
  1. package qs
  2. import (
  3. "errors"
  4. "net/http"
  5. "net/url"
  6. "regexp"
  7. )
  8. // Query error.
  9. var (
  10. ErrInvalidSort = errors.New("invalid sort")
  11. ErrTooManySorts = errors.New("too many sorts")
  12. )
  13. var sortRegexp = regexp.MustCompile("^([A-z0-9]+) (asc|desc)$")
  14. // ReadSortsOptions configures the behaviour of ReadSorts.
  15. type ReadSortsOptions struct {
  16. Key string // Query string key. The default value is "sort"
  17. MaxSorts int // If this is > 0, a maximum number of sorts is imposed
  18. }
  19. // Sort represents a sort order for, most likely, a database query.
  20. type Sort struct {
  21. Field string `json:"field"` // Field by which to sort.
  22. Direction string `json:"direction"` // Direction in which to sort, namely asc or desc.
  23. }
  24. // Sorts is a slice of Sort structs.
  25. type Sorts []Sort
  26. // Field returns a new Sorts slice containing only sorts for the specified field.
  27. // The original order of sorts is preserved.
  28. func (sorts Sorts) Field(field string) Sorts {
  29. ff := Sorts{}
  30. for _, sort := range sorts {
  31. if sort.Field == field {
  32. ff = append(ff, sort)
  33. }
  34. }
  35. return ff
  36. }
  37. // Fields returns a new Sorts slice containing sorts for any of the specified fields.
  38. // The original order of sorts is preserved.
  39. func (sorts Sorts) Fields(fields ...string) Sorts {
  40. ff := Sorts{}
  41. for _, sort := range sorts {
  42. for _, field := range fields {
  43. if sort.Field == field {
  44. ff = append(ff, sort)
  45. }
  46. }
  47. }
  48. return ff
  49. }
  50. // HasField returns true if the Sorts slice includes any sorts for the specified field.
  51. func (sorts Sorts) HasField(field string) bool {
  52. for _, sort := range sorts {
  53. if sort.Field == field {
  54. return true
  55. }
  56. }
  57. return false
  58. }
  59. // ReadRequestSorts parses a request's query string into a slice of sorts.
  60. // This function returns nil if no sorts are found.
  61. func ReadRequestSorts(req *http.Request, opt *ReadSortsOptions) (Sorts, error) {
  62. return ReadSorts(req.URL.Query(), opt)
  63. }
  64. // ReadSorts parses URL values into a slice of sorts.
  65. // This function returns nil if no sorts are found.
  66. func ReadSorts(values url.Values, opt *ReadSortsOptions) (Sorts, error) {
  67. opt = initSortsOptions(opt)
  68. if !values.Has(opt.Key) {
  69. return nil, nil
  70. }
  71. if opt.MaxSorts > 0 && len(values[opt.Key]) > opt.MaxSorts {
  72. return nil, ErrTooManySorts
  73. }
  74. sorts := []Sort{}
  75. for _, sortStr := range values[opt.Key] {
  76. match := sortRegexp.FindStringSubmatch(sortStr)
  77. if match == nil {
  78. return nil, ErrInvalidSort
  79. }
  80. sort := Sort{
  81. Field: match[1],
  82. Direction: match[2],
  83. }
  84. sorts = append(sorts, sort)
  85. }
  86. return sorts, nil
  87. }
  88. // ReadStringSorts parses a query string literal into a slice of sorts.
  89. // This function returns nil if no sorts are found.
  90. func ReadStringSorts(qs string, opt *ReadSortsOptions) (Sorts, error) {
  91. values, err := url.ParseQuery(qs)
  92. if err != nil {
  93. return nil, err
  94. }
  95. return ReadSorts(values, opt)
  96. }
  97. func initSortsOptions(opt *ReadSortsOptions) *ReadSortsOptions {
  98. def := &ReadSortsOptions{
  99. Key: "sort",
  100. }
  101. if opt != nil {
  102. if len(opt.Key) > 0 {
  103. def.Key = opt.Key
  104. }
  105. if opt.MaxSorts > def.MaxSorts {
  106. def.MaxSorts = opt.MaxSorts
  107. }
  108. }
  109. return def
  110. }