/stylesheet/component.go

https://bitbucket.org/romanoff/ahc · Go · 214 lines · 197 code · 17 blank · 0 comment · 57 complexity · f8437afb7e530dc4003f0bb4ad4a3f79 MD5 · raw file

  1. package stylesheet
  2. import (
  3. "bitbucket.org/romanoff/ahc/ti"
  4. "bytes"
  5. "encoding/base64"
  6. "errors"
  7. "fmt"
  8. "io/ioutil"
  9. "os"
  10. "os/exec"
  11. "regexp"
  12. "strings"
  13. "time"
  14. )
  15. type Component struct {
  16. File os.FileInfo
  17. Path string
  18. Content string
  19. CompiledContent string
  20. Provides []string
  21. Requires []string
  22. Preprocessor string
  23. Templates []string
  24. Font string
  25. }
  26. func (self *Component) ModTime() (time time.Time, err error) {
  27. cssFile := self.File
  28. if cssFile == nil {
  29. err = errors.New("No file in component")
  30. return
  31. }
  32. return
  33. }
  34. func (self *Component) Compile(components []*Component, deps map[string][]string) error {
  35. if len(self.Provides) == 0 {
  36. return errors.New("Component doesn't provide anything")
  37. }
  38. componentRequirements := deps[self.Provides[0]]
  39. usedComponents := make([]*Component, 0)
  40. for _, requirement := range componentRequirements {
  41. for _, component := range components {
  42. usedComponents = addComponentIfMatches(requirement, component, usedComponents)
  43. }
  44. }
  45. content := ""
  46. for _, component := range usedComponents {
  47. content += "\n"
  48. content = component.Content + content
  49. if component.Preprocessor != self.Preprocessor {
  50. return errors.New(fmt.Sprintf("%v file and its dependency - %v have different preprocessors", self.Path, component.Path))
  51. }
  52. }
  53. content += self.Content
  54. self.CompiledContent = content
  55. if self.Preprocessor != "" {
  56. return self.PreprocessContent()
  57. }
  58. return nil
  59. }
  60. func (self *Component) PreprocessContent() error {
  61. tempfile, err := ioutil.TempFile("", "ahc")
  62. defer os.Remove(tempfile.Name())
  63. defer tempfile.Close()
  64. if err != nil {
  65. return err
  66. }
  67. _, err = tempfile.WriteString(self.CompiledContent)
  68. if err != nil {
  69. return err
  70. }
  71. name, arg, err := GetPreprocessorInfo(self.Preprocessor)
  72. if err != nil {
  73. return err
  74. }
  75. arg = append(arg, tempfile.Name())
  76. cmd := exec.Command(name, arg...)
  77. output, err := cmd.Output()
  78. if err != nil {
  79. return err
  80. }
  81. self.CompiledContent = string(output)
  82. return nil
  83. }
  84. func (self *Component) Init() (err error) {
  85. bytesContent, err := ioutil.ReadFile(self.Path)
  86. if err != nil {
  87. return err
  88. }
  89. content := string(bytesContent)
  90. self.Content = content
  91. self.readAnnotations()
  92. err = self.replaceImagesWithDataUri()
  93. if err != nil {
  94. return err
  95. }
  96. return
  97. }
  98. func addComponentIfMatches(requirement string, component *Component, usedComponents []*Component) []*Component {
  99. for _, componentProvide := range component.Provides {
  100. if componentProvide == requirement {
  101. for _, usedComponent := range usedComponents {
  102. if usedComponent.Path == component.Path {
  103. return usedComponents
  104. }
  105. }
  106. usedComponents = append(usedComponents, component)
  107. return usedComponents
  108. }
  109. }
  110. return usedComponents
  111. }
  112. var provideRe *regexp.Regexp = regexp.MustCompile("@provide\\s+['\"](.+)['\"]")
  113. var RequireRe *regexp.Regexp = regexp.MustCompile("@require\\s+['\"](.+)['\"]")
  114. var FontRe *regexp.Regexp = regexp.MustCompile("@font\\s+['\"](.+)['\"]")
  115. var preprocessorRe *regexp.Regexp = regexp.MustCompile("@preprocessor\\s+['\"](.+)['\"]")
  116. func (self *Component) readAnnotations() {
  117. provides := make([]string, 0)
  118. requires := make([]string, 0)
  119. content := self.Content
  120. lines := strings.Split(content, "\n")
  121. for _, line := range lines {
  122. matches := provideRe.FindStringSubmatch(line)
  123. if len(matches) == 2 {
  124. provides = append(provides, matches[1])
  125. }
  126. matches = RequireRe.FindStringSubmatch(line)
  127. if len(matches) == 2 {
  128. requires = append(requires, matches[1])
  129. }
  130. matches = preprocessorRe.FindStringSubmatch(line)
  131. if len(matches) == 2 {
  132. self.Preprocessor = matches[1]
  133. }
  134. matches = FontRe.FindStringSubmatch(line)
  135. if len(matches) == 2 {
  136. self.Font = matches[1]
  137. }
  138. }
  139. self.Provides = provides
  140. self.Requires = requires
  141. return
  142. }
  143. var backgroundImageRe *regexp.Regexp = regexp.MustCompile("background-image:\\s+url\\((.+)\\);")
  144. func (self *Component) replaceImagesWithDataUri() error {
  145. lines := strings.Split(self.Content, "\n")
  146. for _, line := range lines {
  147. matches := backgroundImageRe.FindStringSubmatch(line)
  148. if len(matches) == 2 {
  149. path := strings.Replace(self.Path, self.File.Name(), "", 1) + matches[1]
  150. contentBytes, err := ioutil.ReadFile(path)
  151. if err != nil {
  152. return err
  153. }
  154. dataUri := &bytes.Buffer{}
  155. encoder := base64.NewEncoder(base64.StdEncoding, dataUri)
  156. encoder.Write(contentBytes)
  157. encoder.Close()
  158. newLine := "background-image: url('data:image/png;base64," + dataUri.String() + "');"
  159. self.Content = strings.Replace(self.Content, line, newLine, 1)
  160. }
  161. }
  162. return nil
  163. }
  164. func (self *Component) GetHtml() string {
  165. htmlPath := strings.Replace(self.Path, ".css", ".html", 1)
  166. contentBytes, err := ioutil.ReadFile(htmlPath)
  167. if err != nil {
  168. return ""
  169. }
  170. return string(contentBytes)
  171. }
  172. func (self *Component) CheckTemplate(extension string) {
  173. templatePath := strings.Replace(self.Path, ".css", "."+extension, 1)
  174. if _, err := os.Stat(templatePath); err == nil {
  175. self.Templates = append(self.Templates, extension)
  176. }
  177. }
  178. func (self *Component) GetTemplateExec(templatesInfo []*ti.TemplateInfo) (string, string, error) {
  179. for _, templateInfo := range templatesInfo {
  180. for _, templateExtension := range self.Templates {
  181. if templateInfo.Extension == templateExtension {
  182. return templateInfo.Command, strings.Replace(self.Path, ".css", "."+templateExtension, 1), nil
  183. }
  184. }
  185. }
  186. return "", "", errors.New(fmt.Sprintf("No template found for %v", self.Provides[0]))
  187. }
  188. func FindComponent(components []*Component, templateName string) (*Component, error) {
  189. for _, component := range components {
  190. for _, provide := range component.Provides {
  191. if provide == templateName {
  192. return component, nil
  193. }
  194. }
  195. }
  196. return nil, errors.New(fmt.Sprintf("%v component not found", templateName))
  197. }