/util.go

http://github.com/robfig/revel · Go · 172 lines · 133 code · 20 blank · 19 comment · 38 complexity · 96a50ec6e5312ccb87e6b266d6a33b45 MD5 · raw file

  1. package revel
  2. import (
  3. "bytes"
  4. "io"
  5. "io/ioutil"
  6. "net/url"
  7. "os"
  8. "reflect"
  9. "regexp"
  10. "strings"
  11. )
  12. // Add some more methods to the default Template.
  13. type ExecutableTemplate interface {
  14. Execute(io.Writer, interface{}) error
  15. }
  16. // Execute a template and returns the result as a string.
  17. func ExecuteTemplate(tmpl ExecutableTemplate, data interface{}) string {
  18. var b bytes.Buffer
  19. tmpl.Execute(&b, data)
  20. return b.String()
  21. }
  22. // Reads the lines of the given file. Panics in the case of error.
  23. func MustReadLines(filename string) []string {
  24. r, err := ReadLines(filename)
  25. if err != nil {
  26. panic(err)
  27. }
  28. return r
  29. }
  30. // Reads the lines of the given file. Panics in the case of error.
  31. func ReadLines(filename string) ([]string, error) {
  32. bytes, err := ioutil.ReadFile(filename)
  33. if err != nil {
  34. return nil, err
  35. }
  36. return strings.Split(string(bytes), "\n"), nil
  37. }
  38. func ContainsString(list []string, target string) bool {
  39. for _, el := range list {
  40. if el == target {
  41. return true
  42. }
  43. }
  44. return false
  45. }
  46. // Return the reflect.Method, given a Receiver type and Func value.
  47. func FindMethod(recvType reflect.Type, funcVal reflect.Value) *reflect.Method {
  48. // It is not possible to get the name of the method from the Func.
  49. // Instead, compare it to each method of the Controller.
  50. for i := 0; i < recvType.NumMethod(); i++ {
  51. method := recvType.Method(i)
  52. if method.Func.Pointer() == funcVal.Pointer() {
  53. return &method
  54. }
  55. }
  56. return nil
  57. }
  58. var (
  59. cookieKeyValueParser = regexp.MustCompile("\x00([^:]*):([^\x00]*)\x00")
  60. )
  61. // Takes the raw (escaped) cookie value and parses out key values.
  62. func ParseKeyValueCookie(val string, cb func(key, val string)) {
  63. val, _ = url.QueryUnescape(val)
  64. if matches := cookieKeyValueParser.FindAllStringSubmatch(val, -1); matches != nil {
  65. for _, match := range matches {
  66. cb(match[1], match[2])
  67. }
  68. }
  69. }
  70. const DefaultFileContentType = "application/octet-stream"
  71. var mimeConfig *MergedConfig
  72. // Load mime-types.conf on init.
  73. func LoadMimeConfig() {
  74. var err error
  75. mimeConfig, err = LoadConfig("mime-types.conf")
  76. if err != nil {
  77. ERROR.Fatalln("Failed to load mime type config:", err)
  78. }
  79. }
  80. func init() {
  81. OnAppStart(LoadMimeConfig)
  82. }
  83. // Returns a MIME content type based on the filename's extension.
  84. // If no appropriate one is found, returns "application/octet-stream" by default.
  85. // Additionally, specifies the charset as UTF-8 for text/* types.
  86. func ContentTypeByFilename(filename string) string {
  87. dot := strings.LastIndex(filename, ".")
  88. if dot == -1 || dot+1 >= len(filename) {
  89. return DefaultFileContentType
  90. }
  91. extension := filename[dot+1:]
  92. contentType := mimeConfig.StringDefault(extension, "")
  93. if contentType == "" {
  94. return DefaultFileContentType
  95. }
  96. if strings.HasPrefix(contentType, "text/") {
  97. return contentType + "; charset=utf-8"
  98. }
  99. return contentType
  100. }
  101. // DirExists returns true if the given path exists and is a directory.
  102. func DirExists(filename string) bool {
  103. fileInfo, err := os.Stat(filename)
  104. return err == nil && fileInfo.IsDir()
  105. }
  106. func FirstNonEmpty(strs ...string) string {
  107. for _, str := range strs {
  108. if len(str) > 0 {
  109. return str
  110. }
  111. }
  112. return ""
  113. }
  114. // Equal is a helper for comparing value equality, following these rules:
  115. // - Values with equivalent types are compared with reflect.DeepEqual
  116. // - int, uint, and float values are compared without regard to the type width.
  117. // for example, Equal(int32(5), int64(5)) == true
  118. // - strings and byte slices are converted to strings before comparison.
  119. // - else, return false.
  120. func Equal(a, b interface{}) bool {
  121. if reflect.TypeOf(a) == reflect.TypeOf(b) {
  122. return reflect.DeepEqual(a, b)
  123. }
  124. switch a.(type) {
  125. case int, int8, int16, int32, int64:
  126. switch b.(type) {
  127. case int, int8, int16, int32, int64:
  128. return reflect.ValueOf(a).Int() == reflect.ValueOf(b).Int()
  129. }
  130. case uint, uint8, uint16, uint32, uint64:
  131. switch b.(type) {
  132. case uint, uint8, uint16, uint32, uint64:
  133. return reflect.ValueOf(a).Uint() == reflect.ValueOf(b).Uint()
  134. }
  135. case float32, float64:
  136. switch b.(type) {
  137. case float32, float64:
  138. return reflect.ValueOf(a).Float() == reflect.ValueOf(b).Float()
  139. }
  140. case string:
  141. switch b.(type) {
  142. case []byte:
  143. return a.(string) == string(b.([]byte))
  144. }
  145. case []byte:
  146. switch b.(type) {
  147. case string:
  148. return b.(string) == string(a.([]byte))
  149. }
  150. }
  151. return false
  152. }