map.go 5.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222
  1. package objx
  2. import (
  3. "encoding/base64"
  4. "encoding/json"
  5. "errors"
  6. "io/ioutil"
  7. "net/url"
  8. "strings"
  9. )
  10. // MSIConvertable is an interface that defines methods for converting your
  11. // custom types to a map[string]interface{} representation.
  12. type MSIConvertable interface {
  13. // MSI gets a map[string]interface{} (msi) representing the
  14. // object.
  15. MSI() map[string]interface{}
  16. }
  17. // Map provides extended functionality for working with
  18. // untyped data, in particular map[string]interface (msi).
  19. type Map map[string]interface{}
  20. // Value returns the internal value instance
  21. func (m Map) Value() *Value {
  22. return &Value{data: m}
  23. }
  24. // Nil represents a nil Map.
  25. var Nil Map = New(nil)
  26. // New creates a new Map containing the map[string]interface{} in the data argument.
  27. // If the data argument is not a map[string]interface, New attempts to call the
  28. // MSI() method on the MSIConvertable interface to create one.
  29. func New(data interface{}) Map {
  30. if _, ok := data.(map[string]interface{}); !ok {
  31. if converter, ok := data.(MSIConvertable); ok {
  32. data = converter.MSI()
  33. } else {
  34. return nil
  35. }
  36. }
  37. return Map(data.(map[string]interface{}))
  38. }
  39. // MSI creates a map[string]interface{} and puts it inside a new Map.
  40. //
  41. // The arguments follow a key, value pattern.
  42. //
  43. // Panics
  44. //
  45. // Panics if any key arugment is non-string or if there are an odd number of arguments.
  46. //
  47. // Example
  48. //
  49. // To easily create Maps:
  50. //
  51. // m := objx.MSI("name", "Mat", "age", 29, "subobj", objx.MSI("active", true))
  52. //
  53. // // creates an Map equivalent to
  54. // m := objx.New(map[string]interface{}{"name": "Mat", "age": 29, "subobj": map[string]interface{}{"active": true}})
  55. func MSI(keyAndValuePairs ...interface{}) Map {
  56. newMap := make(map[string]interface{})
  57. keyAndValuePairsLen := len(keyAndValuePairs)
  58. if keyAndValuePairsLen%2 != 0 {
  59. panic("objx: MSI must have an even number of arguments following the 'key, value' pattern.")
  60. }
  61. for i := 0; i < keyAndValuePairsLen; i = i + 2 {
  62. key := keyAndValuePairs[i]
  63. value := keyAndValuePairs[i+1]
  64. // make sure the key is a string
  65. keyString, keyStringOK := key.(string)
  66. if !keyStringOK {
  67. panic("objx: MSI must follow 'string, interface{}' pattern. " + keyString + " is not a valid key.")
  68. }
  69. newMap[keyString] = value
  70. }
  71. return New(newMap)
  72. }
  73. // ****** Conversion Constructors
  74. // MustFromJSON creates a new Map containing the data specified in the
  75. // jsonString.
  76. //
  77. // Panics if the JSON is invalid.
  78. func MustFromJSON(jsonString string) Map {
  79. o, err := FromJSON(jsonString)
  80. if err != nil {
  81. panic("objx: MustFromJSON failed with error: " + err.Error())
  82. }
  83. return o
  84. }
  85. // FromJSON creates a new Map containing the data specified in the
  86. // jsonString.
  87. //
  88. // Returns an error if the JSON is invalid.
  89. func FromJSON(jsonString string) (Map, error) {
  90. var data interface{}
  91. err := json.Unmarshal([]byte(jsonString), &data)
  92. if err != nil {
  93. return Nil, err
  94. }
  95. return New(data), nil
  96. }
  97. // FromBase64 creates a new Obj containing the data specified
  98. // in the Base64 string.
  99. //
  100. // The string is an encoded JSON string returned by Base64
  101. func FromBase64(base64String string) (Map, error) {
  102. decoder := base64.NewDecoder(base64.StdEncoding, strings.NewReader(base64String))
  103. decoded, err := ioutil.ReadAll(decoder)
  104. if err != nil {
  105. return nil, err
  106. }
  107. return FromJSON(string(decoded))
  108. }
  109. // MustFromBase64 creates a new Obj containing the data specified
  110. // in the Base64 string and panics if there is an error.
  111. //
  112. // The string is an encoded JSON string returned by Base64
  113. func MustFromBase64(base64String string) Map {
  114. result, err := FromBase64(base64String)
  115. if err != nil {
  116. panic("objx: MustFromBase64 failed with error: " + err.Error())
  117. }
  118. return result
  119. }
  120. // FromSignedBase64 creates a new Obj containing the data specified
  121. // in the Base64 string.
  122. //
  123. // The string is an encoded JSON string returned by SignedBase64
  124. func FromSignedBase64(base64String, key string) (Map, error) {
  125. parts := strings.Split(base64String, SignatureSeparator)
  126. if len(parts) != 2 {
  127. return nil, errors.New("objx: Signed base64 string is malformed.")
  128. }
  129. sig := HashWithKey(parts[0], key)
  130. if parts[1] != sig {
  131. return nil, errors.New("objx: Signature for base64 data does not match.")
  132. }
  133. return FromBase64(parts[0])
  134. }
  135. // MustFromSignedBase64 creates a new Obj containing the data specified
  136. // in the Base64 string and panics if there is an error.
  137. //
  138. // The string is an encoded JSON string returned by Base64
  139. func MustFromSignedBase64(base64String, key string) Map {
  140. result, err := FromSignedBase64(base64String, key)
  141. if err != nil {
  142. panic("objx: MustFromSignedBase64 failed with error: " + err.Error())
  143. }
  144. return result
  145. }
  146. // FromURLQuery generates a new Obj by parsing the specified
  147. // query.
  148. //
  149. // For queries with multiple values, the first value is selected.
  150. func FromURLQuery(query string) (Map, error) {
  151. vals, err := url.ParseQuery(query)
  152. if err != nil {
  153. return nil, err
  154. }
  155. m := make(map[string]interface{})
  156. for k, vals := range vals {
  157. m[k] = vals[0]
  158. }
  159. return New(m), nil
  160. }
  161. // MustFromURLQuery generates a new Obj by parsing the specified
  162. // query.
  163. //
  164. // For queries with multiple values, the first value is selected.
  165. //
  166. // Panics if it encounters an error
  167. func MustFromURLQuery(query string) Map {
  168. o, err := FromURLQuery(query)
  169. if err != nil {
  170. panic("objx: MustFromURLQuery failed with error: " + err.Error())
  171. }
  172. return o
  173. }