oreorenasass 3.4.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (268) hide show
  1. checksums.yaml +7 -0
  2. data/.yardopts +11 -0
  3. data/CONTRIBUTING +3 -0
  4. data/MIT-LICENSE +20 -0
  5. data/README.md +221 -0
  6. data/Rakefile +370 -0
  7. data/VERSION +1 -0
  8. data/VERSION_NAME +1 -0
  9. data/bin/sass +13 -0
  10. data/bin/sass-convert +12 -0
  11. data/bin/scss +13 -0
  12. data/extra/update_watch.rb +13 -0
  13. data/init.rb +18 -0
  14. data/lib/sass/cache_stores/base.rb +88 -0
  15. data/lib/sass/cache_stores/chain.rb +34 -0
  16. data/lib/sass/cache_stores/filesystem.rb +60 -0
  17. data/lib/sass/cache_stores/memory.rb +47 -0
  18. data/lib/sass/cache_stores/null.rb +25 -0
  19. data/lib/sass/cache_stores.rb +15 -0
  20. data/lib/sass/callbacks.rb +67 -0
  21. data/lib/sass/css.rb +407 -0
  22. data/lib/sass/engine.rb +1181 -0
  23. data/lib/sass/environment.rb +191 -0
  24. data/lib/sass/error.rb +198 -0
  25. data/lib/sass/exec/base.rb +187 -0
  26. data/lib/sass/exec/sass_convert.rb +264 -0
  27. data/lib/sass/exec/sass_scss.rb +424 -0
  28. data/lib/sass/exec.rb +9 -0
  29. data/lib/sass/features.rb +47 -0
  30. data/lib/sass/importers/base.rb +182 -0
  31. data/lib/sass/importers/filesystem.rb +211 -0
  32. data/lib/sass/importers.rb +22 -0
  33. data/lib/sass/logger/base.rb +30 -0
  34. data/lib/sass/logger/log_level.rb +45 -0
  35. data/lib/sass/logger.rb +12 -0
  36. data/lib/sass/media.rb +210 -0
  37. data/lib/sass/plugin/compiler.rb +565 -0
  38. data/lib/sass/plugin/configuration.rb +118 -0
  39. data/lib/sass/plugin/generic.rb +15 -0
  40. data/lib/sass/plugin/merb.rb +48 -0
  41. data/lib/sass/plugin/rack.rb +60 -0
  42. data/lib/sass/plugin/rails.rb +47 -0
  43. data/lib/sass/plugin/staleness_checker.rb +199 -0
  44. data/lib/sass/plugin.rb +133 -0
  45. data/lib/sass/railtie.rb +10 -0
  46. data/lib/sass/repl.rb +57 -0
  47. data/lib/sass/root.rb +7 -0
  48. data/lib/sass/script/css_lexer.rb +33 -0
  49. data/lib/sass/script/css_parser.rb +34 -0
  50. data/lib/sass/script/functions.rb +2626 -0
  51. data/lib/sass/script/lexer.rb +449 -0
  52. data/lib/sass/script/parser.rb +637 -0
  53. data/lib/sass/script/tree/funcall.rb +306 -0
  54. data/lib/sass/script/tree/interpolation.rb +118 -0
  55. data/lib/sass/script/tree/list_literal.rb +77 -0
  56. data/lib/sass/script/tree/literal.rb +45 -0
  57. data/lib/sass/script/tree/map_literal.rb +64 -0
  58. data/lib/sass/script/tree/node.rb +109 -0
  59. data/lib/sass/script/tree/operation.rb +103 -0
  60. data/lib/sass/script/tree/selector.rb +26 -0
  61. data/lib/sass/script/tree/string_interpolation.rb +104 -0
  62. data/lib/sass/script/tree/unary_operation.rb +69 -0
  63. data/lib/sass/script/tree/variable.rb +57 -0
  64. data/lib/sass/script/tree.rb +16 -0
  65. data/lib/sass/script/value/arg_list.rb +36 -0
  66. data/lib/sass/script/value/base.rb +240 -0
  67. data/lib/sass/script/value/bool.rb +35 -0
  68. data/lib/sass/script/value/color.rb +680 -0
  69. data/lib/sass/script/value/helpers.rb +262 -0
  70. data/lib/sass/script/value/list.rb +113 -0
  71. data/lib/sass/script/value/map.rb +70 -0
  72. data/lib/sass/script/value/null.rb +44 -0
  73. data/lib/sass/script/value/number.rb +530 -0
  74. data/lib/sass/script/value/string.rb +97 -0
  75. data/lib/sass/script/value.rb +11 -0
  76. data/lib/sass/script.rb +66 -0
  77. data/lib/sass/scss/css_parser.rb +42 -0
  78. data/lib/sass/scss/parser.rb +1209 -0
  79. data/lib/sass/scss/rx.rb +141 -0
  80. data/lib/sass/scss/script_lexer.rb +15 -0
  81. data/lib/sass/scss/script_parser.rb +25 -0
  82. data/lib/sass/scss/static_parser.rb +368 -0
  83. data/lib/sass/scss.rb +16 -0
  84. data/lib/sass/selector/abstract_sequence.rb +109 -0
  85. data/lib/sass/selector/comma_sequence.rb +175 -0
  86. data/lib/sass/selector/pseudo.rb +256 -0
  87. data/lib/sass/selector/sequence.rb +600 -0
  88. data/lib/sass/selector/simple.rb +117 -0
  89. data/lib/sass/selector/simple_sequence.rb +325 -0
  90. data/lib/sass/selector.rb +326 -0
  91. data/lib/sass/shared.rb +76 -0
  92. data/lib/sass/source/map.rb +210 -0
  93. data/lib/sass/source/position.rb +39 -0
  94. data/lib/sass/source/range.rb +41 -0
  95. data/lib/sass/stack.rb +120 -0
  96. data/lib/sass/supports.rb +227 -0
  97. data/lib/sass/tree/at_root_node.rb +83 -0
  98. data/lib/sass/tree/charset_node.rb +22 -0
  99. data/lib/sass/tree/comment_node.rb +82 -0
  100. data/lib/sass/tree/content_node.rb +9 -0
  101. data/lib/sass/tree/css_import_node.rb +60 -0
  102. data/lib/sass/tree/debug_node.rb +18 -0
  103. data/lib/sass/tree/directive_node.rb +59 -0
  104. data/lib/sass/tree/each_node.rb +24 -0
  105. data/lib/sass/tree/error_node.rb +18 -0
  106. data/lib/sass/tree/extend_node.rb +43 -0
  107. data/lib/sass/tree/for_node.rb +36 -0
  108. data/lib/sass/tree/function_node.rb +39 -0
  109. data/lib/sass/tree/if_node.rb +52 -0
  110. data/lib/sass/tree/import_node.rb +74 -0
  111. data/lib/sass/tree/keyframe_rule_node.rb +15 -0
  112. data/lib/sass/tree/media_node.rb +48 -0
  113. data/lib/sass/tree/mixin_def_node.rb +38 -0
  114. data/lib/sass/tree/mixin_node.rb +52 -0
  115. data/lib/sass/tree/node.rb +238 -0
  116. data/lib/sass/tree/prop_node.rb +171 -0
  117. data/lib/sass/tree/return_node.rb +19 -0
  118. data/lib/sass/tree/root_node.rb +44 -0
  119. data/lib/sass/tree/rule_node.rb +145 -0
  120. data/lib/sass/tree/supports_node.rb +38 -0
  121. data/lib/sass/tree/trace_node.rb +33 -0
  122. data/lib/sass/tree/variable_node.rb +36 -0
  123. data/lib/sass/tree/visitors/base.rb +72 -0
  124. data/lib/sass/tree/visitors/check_nesting.rb +177 -0
  125. data/lib/sass/tree/visitors/convert.rb +334 -0
  126. data/lib/sass/tree/visitors/cssize.rb +369 -0
  127. data/lib/sass/tree/visitors/deep_copy.rb +107 -0
  128. data/lib/sass/tree/visitors/extend.rb +68 -0
  129. data/lib/sass/tree/visitors/perform.rb +539 -0
  130. data/lib/sass/tree/visitors/set_options.rb +139 -0
  131. data/lib/sass/tree/visitors/to_css.rb +381 -0
  132. data/lib/sass/tree/warn_node.rb +18 -0
  133. data/lib/sass/tree/while_node.rb +18 -0
  134. data/lib/sass/util/cross_platform_random.rb +19 -0
  135. data/lib/sass/util/multibyte_string_scanner.rb +157 -0
  136. data/lib/sass/util/normalized_map.rb +130 -0
  137. data/lib/sass/util/ordered_hash.rb +192 -0
  138. data/lib/sass/util/subset_map.rb +110 -0
  139. data/lib/sass/util/test.rb +9 -0
  140. data/lib/sass/util.rb +1318 -0
  141. data/lib/sass/version.rb +124 -0
  142. data/lib/sass.rb +102 -0
  143. data/rails/init.rb +1 -0
  144. data/test/sass/cache_test.rb +131 -0
  145. data/test/sass/callbacks_test.rb +61 -0
  146. data/test/sass/compiler_test.rb +232 -0
  147. data/test/sass/conversion_test.rb +2054 -0
  148. data/test/sass/css2sass_test.rb +477 -0
  149. data/test/sass/data/hsl-rgb.txt +319 -0
  150. data/test/sass/encoding_test.rb +219 -0
  151. data/test/sass/engine_test.rb +3301 -0
  152. data/test/sass/exec_test.rb +86 -0
  153. data/test/sass/extend_test.rb +1661 -0
  154. data/test/sass/fixtures/test_staleness_check_across_importers.css +1 -0
  155. data/test/sass/fixtures/test_staleness_check_across_importers.scss +1 -0
  156. data/test/sass/functions_test.rb +1926 -0
  157. data/test/sass/importer_test.rb +412 -0
  158. data/test/sass/logger_test.rb +58 -0
  159. data/test/sass/mock_importer.rb +49 -0
  160. data/test/sass/more_results/more1.css +9 -0
  161. data/test/sass/more_results/more1_with_line_comments.css +26 -0
  162. data/test/sass/more_results/more_import.css +29 -0
  163. data/test/sass/more_templates/_more_partial.sass +2 -0
  164. data/test/sass/more_templates/more1.sass +23 -0
  165. data/test/sass/more_templates/more_import.sass +11 -0
  166. data/test/sass/plugin_test.rb +554 -0
  167. data/test/sass/results/alt.css +4 -0
  168. data/test/sass/results/basic.css +9 -0
  169. data/test/sass/results/cached_import_option.css +3 -0
  170. data/test/sass/results/compact.css +5 -0
  171. data/test/sass/results/complex.css +86 -0
  172. data/test/sass/results/compressed.css +1 -0
  173. data/test/sass/results/expanded.css +19 -0
  174. data/test/sass/results/filename_fn.css +3 -0
  175. data/test/sass/results/if.css +3 -0
  176. data/test/sass/results/import.css +31 -0
  177. data/test/sass/results/import_charset.css +5 -0
  178. data/test/sass/results/import_charset_1_8.css +5 -0
  179. data/test/sass/results/import_charset_ibm866.css +5 -0
  180. data/test/sass/results/import_content.css +1 -0
  181. data/test/sass/results/line_numbers.css +49 -0
  182. data/test/sass/results/mixins.css +95 -0
  183. data/test/sass/results/multiline.css +24 -0
  184. data/test/sass/results/nested.css +22 -0
  185. data/test/sass/results/options.css +1 -0
  186. data/test/sass/results/parent_ref.css +13 -0
  187. data/test/sass/results/script.css +16 -0
  188. data/test/sass/results/scss_import.css +31 -0
  189. data/test/sass/results/scss_importee.css +2 -0
  190. data/test/sass/results/subdir/nested_subdir/nested_subdir.css +1 -0
  191. data/test/sass/results/subdir/subdir.css +3 -0
  192. data/test/sass/results/units.css +11 -0
  193. data/test/sass/results/warn.css +0 -0
  194. data/test/sass/results/warn_imported.css +0 -0
  195. data/test/sass/script_conversion_test.rb +328 -0
  196. data/test/sass/script_test.rb +1054 -0
  197. data/test/sass/scss/css_test.rb +1215 -0
  198. data/test/sass/scss/rx_test.rb +156 -0
  199. data/test/sass/scss/scss_test.rb +3900 -0
  200. data/test/sass/scss/test_helper.rb +37 -0
  201. data/test/sass/source_map_test.rb +977 -0
  202. data/test/sass/superselector_test.rb +191 -0
  203. data/test/sass/templates/_cached_import_option_partial.scss +1 -0
  204. data/test/sass/templates/_double_import_loop2.sass +1 -0
  205. data/test/sass/templates/_filename_fn_import.scss +11 -0
  206. data/test/sass/templates/_imported_charset_ibm866.sass +4 -0
  207. data/test/sass/templates/_imported_charset_utf8.sass +4 -0
  208. data/test/sass/templates/_imported_content.sass +3 -0
  209. data/test/sass/templates/_partial.sass +2 -0
  210. data/test/sass/templates/_same_name_different_partiality.scss +1 -0
  211. data/test/sass/templates/alt.sass +16 -0
  212. data/test/sass/templates/basic.sass +23 -0
  213. data/test/sass/templates/bork1.sass +2 -0
  214. data/test/sass/templates/bork2.sass +2 -0
  215. data/test/sass/templates/bork3.sass +2 -0
  216. data/test/sass/templates/bork4.sass +2 -0
  217. data/test/sass/templates/bork5.sass +3 -0
  218. data/test/sass/templates/cached_import_option.scss +3 -0
  219. data/test/sass/templates/compact.sass +17 -0
  220. data/test/sass/templates/complex.sass +305 -0
  221. data/test/sass/templates/compressed.sass +15 -0
  222. data/test/sass/templates/double_import_loop1.sass +1 -0
  223. data/test/sass/templates/expanded.sass +17 -0
  224. data/test/sass/templates/filename_fn.scss +18 -0
  225. data/test/sass/templates/if.sass +11 -0
  226. data/test/sass/templates/import.sass +12 -0
  227. data/test/sass/templates/import_charset.sass +9 -0
  228. data/test/sass/templates/import_charset_1_8.sass +6 -0
  229. data/test/sass/templates/import_charset_ibm866.sass +11 -0
  230. data/test/sass/templates/import_content.sass +4 -0
  231. data/test/sass/templates/importee.less +2 -0
  232. data/test/sass/templates/importee.sass +19 -0
  233. data/test/sass/templates/line_numbers.sass +13 -0
  234. data/test/sass/templates/mixin_bork.sass +5 -0
  235. data/test/sass/templates/mixins.sass +76 -0
  236. data/test/sass/templates/multiline.sass +20 -0
  237. data/test/sass/templates/nested.sass +25 -0
  238. data/test/sass/templates/nested_bork1.sass +2 -0
  239. data/test/sass/templates/nested_bork2.sass +2 -0
  240. data/test/sass/templates/nested_bork3.sass +2 -0
  241. data/test/sass/templates/nested_bork4.sass +2 -0
  242. data/test/sass/templates/nested_import.sass +2 -0
  243. data/test/sass/templates/nested_mixin_bork.sass +6 -0
  244. data/test/sass/templates/options.sass +2 -0
  245. data/test/sass/templates/parent_ref.sass +25 -0
  246. data/test/sass/templates/same_name_different_ext.sass +2 -0
  247. data/test/sass/templates/same_name_different_ext.scss +1 -0
  248. data/test/sass/templates/same_name_different_partiality.scss +1 -0
  249. data/test/sass/templates/script.sass +101 -0
  250. data/test/sass/templates/scss_import.scss +12 -0
  251. data/test/sass/templates/scss_importee.scss +1 -0
  252. data/test/sass/templates/single_import_loop.sass +1 -0
  253. data/test/sass/templates/subdir/import_up1.scss +1 -0
  254. data/test/sass/templates/subdir/import_up2.scss +1 -0
  255. data/test/sass/templates/subdir/nested_subdir/_nested_partial.sass +2 -0
  256. data/test/sass/templates/subdir/nested_subdir/nested_subdir.sass +3 -0
  257. data/test/sass/templates/subdir/subdir.sass +6 -0
  258. data/test/sass/templates/units.sass +11 -0
  259. data/test/sass/templates/warn.sass +3 -0
  260. data/test/sass/templates/warn_imported.sass +4 -0
  261. data/test/sass/test_helper.rb +8 -0
  262. data/test/sass/util/multibyte_string_scanner_test.rb +147 -0
  263. data/test/sass/util/normalized_map_test.rb +51 -0
  264. data/test/sass/util/subset_map_test.rb +91 -0
  265. data/test/sass/util_test.rb +467 -0
  266. data/test/sass/value_helpers_test.rb +179 -0
  267. data/test/test_helper.rb +109 -0
  268. metadata +386 -0
@@ -0,0 +1,118 @@
1
+ module Sass
2
+ module Plugin
3
+ # We keep configuration in its own self-contained file
4
+ # so that we can load it independently in Rails 3,
5
+ # where the full plugin stuff is lazy-loaded.
6
+ module Configuration
7
+ # Returns the default options for a {Sass::Plugin::Compiler}.
8
+ #
9
+ # @return [{Symbol => Object}]
10
+ def default_options
11
+ @default_options ||= {
12
+ :css_location => './public/stylesheets',
13
+ :always_update => false,
14
+ :always_check => true,
15
+ :full_exception => true,
16
+ :cache_location => ".sass-cache"
17
+ }.freeze
18
+ end
19
+
20
+ # Resets the options and
21
+ # {Sass::Callbacks::InstanceMethods#clear_callbacks! clears all callbacks}.
22
+ def reset!
23
+ @options = nil
24
+ clear_callbacks!
25
+ end
26
+
27
+ # An options hash.
28
+ # See {file:SASS_REFERENCE.md#sass_options the Sass options documentation}.
29
+ #
30
+ # @return [{Symbol => Object}]
31
+ def options
32
+ @options ||= default_options.dup
33
+ end
34
+
35
+ # Adds a new template-location/css-location mapping.
36
+ # This means that Sass/SCSS files in `template_location`
37
+ # will be compiled to CSS files in `css_location`.
38
+ #
39
+ # This is preferred over manually manipulating the
40
+ # {file:SASS_REFERENCE.md#template_location-option `:template_location` option}
41
+ # since the option can be in multiple formats.
42
+ #
43
+ # Note that this method will change `options[:template_location]`
44
+ # to be in the Array format.
45
+ # This means that even if `options[:template_location]`
46
+ # had previously been a Hash or a String,
47
+ # it will now be an Array.
48
+ #
49
+ # @param template_location [String] The location where Sass/SCSS files will be.
50
+ # @param css_location [String] The location where compiled CSS files will go.
51
+ def add_template_location(template_location, css_location = options[:css_location])
52
+ normalize_template_location!
53
+ template_location_array << [template_location, css_location]
54
+ end
55
+
56
+ # Removes a template-location/css-location mapping.
57
+ # This means that Sass/SCSS files in `template_location`
58
+ # will no longer be compiled to CSS files in `css_location`.
59
+ #
60
+ # This is preferred over manually manipulating the
61
+ # {file:SASS_REFERENCE.md#template_location-option `:template_location` option}
62
+ # since the option can be in multiple formats.
63
+ #
64
+ # Note that this method will change `options[:template_location]`
65
+ # to be in the Array format.
66
+ # This means that even if `options[:template_location]`
67
+ # had previously been a Hash or a String,
68
+ # it will now be an Array.
69
+ #
70
+ # @param template_location [String]
71
+ # The location where Sass/SCSS files were,
72
+ # which is now going to be ignored.
73
+ # @param css_location [String]
74
+ # The location where compiled CSS files went, but will no longer go.
75
+ # @return [Boolean]
76
+ # Non-`nil` if the given mapping already existed and was removed,
77
+ # or `nil` if nothing was changed.
78
+ def remove_template_location(template_location, css_location = options[:css_location])
79
+ normalize_template_location!
80
+ template_location_array.delete([template_location, css_location])
81
+ end
82
+
83
+ # Returns the template locations configured for Sass
84
+ # as an array of `[template_location, css_location]` pairs.
85
+ # See the {file:SASS_REFERENCE.md#template_location-option `:template_location` option}
86
+ # for details.
87
+ #
88
+ # @return [Array<(String, String)>]
89
+ # An array of `[template_location, css_location]` pairs.
90
+ def template_location_array
91
+ old_template_location = options[:template_location]
92
+ normalize_template_location!
93
+ options[:template_location]
94
+ ensure
95
+ options[:template_location] = old_template_location
96
+ end
97
+
98
+ private
99
+
100
+ def normalize_template_location!
101
+ return if options[:template_location].is_a?(Array)
102
+ options[:template_location] =
103
+ case options[:template_location]
104
+ when nil
105
+ if options[:css_location]
106
+ [[File.join(options[:css_location], 'sass'), options[:css_location]]]
107
+ else
108
+ []
109
+ end
110
+ when String
111
+ [[options[:template_location], options[:css_location]]]
112
+ else
113
+ options[:template_location].to_a
114
+ end
115
+ end
116
+ end
117
+ end
118
+ end
@@ -0,0 +1,15 @@
1
+ # The reason some options are declared here rather than in sass/plugin/configuration.rb
2
+ # is that otherwise they'd clobber the Rails-specific options.
3
+ # Since Rails' options are lazy-loaded in Rails 3,
4
+ # they're reverse-merged with the default options
5
+ # so that user configuration is preserved.
6
+ # This means that defaults that differ from Rails'
7
+ # must be declared here.
8
+
9
+ unless defined?(Sass::GENERIC_LOADED)
10
+ Sass::GENERIC_LOADED = true
11
+
12
+ Sass::Plugin.options.merge!(:css_location => './public/stylesheets',
13
+ :always_update => false,
14
+ :always_check => true)
15
+ end
@@ -0,0 +1,48 @@
1
+ unless defined?(Sass::MERB_LOADED)
2
+ Sass::MERB_LOADED = true
3
+
4
+ module Sass::Plugin::Configuration
5
+ # Different default options in a m envirionment.
6
+ def default_options
7
+ @default_options ||= begin
8
+ version = Merb::VERSION.split('.').map {|n| n.to_i}
9
+ if version[0] <= 0 && version[1] < 5
10
+ root = MERB_ROOT
11
+ env = MERB_ENV
12
+ else
13
+ root = Merb.root.to_s
14
+ env = Merb.environment
15
+ end
16
+
17
+ {
18
+ :always_update => false,
19
+ :template_location => root + '/public/stylesheets/sass',
20
+ :css_location => root + '/public/stylesheets',
21
+ :cache_location => root + '/tmp/sass-cache',
22
+ :always_check => env != "production",
23
+ :quiet => env != "production",
24
+ :full_exception => env != "production"
25
+ }.freeze
26
+ end
27
+ end
28
+ end
29
+
30
+ config = Merb::Plugins.config[:sass] || Merb::Plugins.config["sass"] || {}
31
+
32
+ if defined? config.symbolize_keys!
33
+ config.symbolize_keys!
34
+ end
35
+
36
+ Sass::Plugin.options.merge!(config)
37
+
38
+ require 'sass/plugin/rack'
39
+ class Sass::Plugin::MerbBootLoader < Merb::BootLoader
40
+ after Merb::BootLoader::RackUpApplication
41
+
42
+ def self.run
43
+ # Apparently there's no better way than this to add Sass
44
+ # to Merb's Rack stack.
45
+ Merb::Config[:app] = Sass::Plugin::Rack.new(Merb::Config[:app])
46
+ end
47
+ end
48
+ end
@@ -0,0 +1,60 @@
1
+ module Sass
2
+ module Plugin
3
+ # Rack middleware for compiling Sass code.
4
+ #
5
+ # ## Activate
6
+ #
7
+ # require 'sass/plugin/rack'
8
+ # use Sass::Plugin::Rack
9
+ #
10
+ # ## Customize
11
+ #
12
+ # Sass::Plugin.options.merge(
13
+ # :cache_location => './tmp/sass-cache',
14
+ # :never_update => environment != :production,
15
+ # :full_exception => environment != :production)
16
+ #
17
+ # {file:SASS_REFERENCE.md#options See the Reference for more options}.
18
+ #
19
+ # ## Use
20
+ #
21
+ # Put your Sass files in `public/stylesheets/sass`.
22
+ # Your CSS will be generated in `public/stylesheets`,
23
+ # and regenerated every request if necessary.
24
+ # The locations and frequency {file:SASS_REFERENCE.md#options can be customized}.
25
+ # That's all there is to it!
26
+ class Rack
27
+ # The delay, in seconds, between update checks.
28
+ # Useful when many resources are requested for a single page.
29
+ # `nil` means no delay at all.
30
+ #
31
+ # @return [Float]
32
+ attr_accessor :dwell
33
+
34
+ # Initialize the middleware.
35
+ #
36
+ # @param app [#call] The Rack application
37
+ # @param dwell [Float] See \{#dwell}
38
+ def initialize(app, dwell = 1.0)
39
+ @app = app
40
+ @dwell = dwell
41
+ @check_after = Time.now.to_f
42
+ end
43
+
44
+ # Process a request, checking the Sass stylesheets for changes
45
+ # and updating them if necessary.
46
+ #
47
+ # @param env The Rack request environment
48
+ # @return [(#to_i, {String => String}, Object)] The Rack response
49
+ def call(env)
50
+ if @dwell.nil? || Time.now.to_f > @check_after
51
+ Sass::Plugin.check_for_updates
52
+ @check_after = Time.now.to_f + @dwell if @dwell
53
+ end
54
+ @app.call(env)
55
+ end
56
+ end
57
+ end
58
+ end
59
+
60
+ require 'sass/plugin'
@@ -0,0 +1,47 @@
1
+ unless defined?(Sass::RAILS_LOADED)
2
+ Sass::RAILS_LOADED = true
3
+
4
+ module Sass::Plugin::Configuration
5
+ # Different default options in a rails envirionment.
6
+ def default_options
7
+ return @default_options if @default_options
8
+ opts = {
9
+ :quiet => Sass::Util.rails_env != "production",
10
+ :full_exception => Sass::Util.rails_env != "production",
11
+ :cache_location => Sass::Util.rails_root + '/tmp/sass-cache'
12
+ }
13
+
14
+ opts.merge!(
15
+ :always_update => false,
16
+ :template_location => Sass::Util.rails_root + '/public/stylesheets/sass',
17
+ :css_location => Sass::Util.rails_root + '/public/stylesheets',
18
+ :always_check => Sass::Util.rails_env == "development")
19
+
20
+ @default_options = opts.freeze
21
+ end
22
+ end
23
+
24
+ Sass::Plugin.options.reverse_merge!(Sass::Plugin.default_options)
25
+
26
+ # Rails 3.1 loads and handles Sass all on its own
27
+ if defined?(ActionController::Metal)
28
+ # 3.1 > Rails >= 3.0
29
+ require 'sass/plugin/rack'
30
+ Rails.configuration.middleware.use(Sass::Plugin::Rack)
31
+ elsif defined?(ActionController::Dispatcher) &&
32
+ defined?(ActionController::Dispatcher.middleware)
33
+ # Rails >= 2.3
34
+ require 'sass/plugin/rack'
35
+ ActionController::Dispatcher.middleware.use(Sass::Plugin::Rack)
36
+ else
37
+ module ActionController
38
+ class Base
39
+ alias_method :sass_old_process, :process
40
+ def process(*args)
41
+ Sass::Plugin.check_for_updates
42
+ sass_old_process(*args)
43
+ end
44
+ end
45
+ end
46
+ end
47
+ end
@@ -0,0 +1,199 @@
1
+ require 'thread'
2
+
3
+ module Sass
4
+ module Plugin
5
+ # The class handles `.s[ca]ss` file staleness checks via their mtime timestamps.
6
+ #
7
+ # To speed things up two level of caches are employed:
8
+ #
9
+ # * A class-level dependency cache which stores @import paths for each file.
10
+ # This is a long-lived cache that is reused by every StalenessChecker instance.
11
+ # * Three short-lived instance-level caches, one for file mtimes,
12
+ # one for whether a file is stale during this particular run.
13
+ # and one for the parse tree for a file.
14
+ # These are only used by a single StalenessChecker instance.
15
+ #
16
+ # Usage:
17
+ #
18
+ # * For a one-off staleness check of a single `.s[ca]ss` file,
19
+ # the class-level {stylesheet_needs_update?} method
20
+ # should be used.
21
+ # * For a series of staleness checks (e.g. checking all files for staleness)
22
+ # a StalenessChecker instance should be created,
23
+ # and the instance-level \{#stylesheet\_needs\_update?} method should be used.
24
+ # the caches should make the whole process significantly faster.
25
+ # *WARNING*: It is important not to retain the instance for too long,
26
+ # as its instance-level caches are never explicitly expired.
27
+ class StalenessChecker
28
+ @dependencies_cache = {}
29
+ @dependency_cache_mutex = Mutex.new
30
+
31
+ class << self
32
+ # TODO: attach this to a compiler instance.
33
+ # @private
34
+ attr_accessor :dependencies_cache
35
+ attr_reader :dependency_cache_mutex
36
+ end
37
+
38
+ # Creates a new StalenessChecker
39
+ # for checking the staleness of several stylesheets at once.
40
+ #
41
+ # @param options [{Symbol => Object}]
42
+ # See {file:SASS_REFERENCE.md#sass_options the Sass options documentation}.
43
+ def initialize(options)
44
+ # URIs that are being actively checked for staleness. Protects against
45
+ # import loops.
46
+ @actively_checking = Set.new
47
+
48
+ # Entries in the following instance-level caches are never explicitly expired.
49
+ # Instead they are supposed to automatically go out of scope when a series of staleness
50
+ # checks (this instance of StalenessChecker was created for) is finished.
51
+ @mtimes, @dependencies_stale, @parse_trees = {}, {}, {}
52
+ @options = Sass::Engine.normalize_options(options)
53
+ end
54
+
55
+ # Returns whether or not a given CSS file is out of date
56
+ # and needs to be regenerated.
57
+ #
58
+ # @param css_file [String] The location of the CSS file to check.
59
+ # @param template_file [String] The location of the Sass or SCSS template
60
+ # that is compiled to `css_file`.
61
+ # @return [Boolean] Whether the stylesheet needs to be updated.
62
+ def stylesheet_needs_update?(css_file, template_file, importer = nil)
63
+ template_file = File.expand_path(template_file)
64
+ begin
65
+ css_mtime = File.mtime(css_file)
66
+ rescue Errno::ENOENT
67
+ return true
68
+ end
69
+ stylesheet_modified_since?(template_file, css_mtime, importer)
70
+ end
71
+
72
+ # Returns whether a Sass or SCSS stylesheet has been modified since a given time.
73
+ #
74
+ # @param template_file [String] The location of the Sass or SCSS template.
75
+ # @param mtime [Fixnum] The modification time to check against.
76
+ # @param importer [Sass::Importers::Base] The importer used to locate the stylesheet.
77
+ # Defaults to the filesystem importer.
78
+ # @return [Boolean] Whether the stylesheet has been modified.
79
+ def stylesheet_modified_since?(template_file, mtime, importer = nil)
80
+ importer ||= @options[:filesystem_importer].new(".")
81
+ dependency_updated?(mtime).call(template_file, importer)
82
+ end
83
+
84
+ # Returns whether or not a given CSS file is out of date
85
+ # and needs to be regenerated.
86
+ #
87
+ # The distinction between this method and the instance-level \{#stylesheet\_needs\_update?}
88
+ # is that the instance method preserves mtime and stale-dependency caches,
89
+ # so it's better to use when checking multiple stylesheets at once.
90
+ #
91
+ # @param css_file [String] The location of the CSS file to check.
92
+ # @param template_file [String] The location of the Sass or SCSS template
93
+ # that is compiled to `css_file`.
94
+ # @return [Boolean] Whether the stylesheet needs to be updated.
95
+ def self.stylesheet_needs_update?(css_file, template_file, importer = nil)
96
+ new(Plugin.engine_options).stylesheet_needs_update?(css_file, template_file, importer)
97
+ end
98
+
99
+ # Returns whether a Sass or SCSS stylesheet has been modified since a given time.
100
+ #
101
+ # The distinction between this method and the instance-level \{#stylesheet\_modified\_since?}
102
+ # is that the instance method preserves mtime and stale-dependency caches,
103
+ # so it's better to use when checking multiple stylesheets at once.
104
+ #
105
+ # @param template_file [String] The location of the Sass or SCSS template.
106
+ # @param mtime [Fixnum] The modification time to check against.
107
+ # @param importer [Sass::Importers::Base] The importer used to locate the stylesheet.
108
+ # Defaults to the filesystem importer.
109
+ # @return [Boolean] Whether the stylesheet has been modified.
110
+ def self.stylesheet_modified_since?(template_file, mtime, importer = nil)
111
+ new(Plugin.engine_options).stylesheet_modified_since?(template_file, mtime, importer)
112
+ end
113
+
114
+ private
115
+
116
+ def dependencies_stale?(uri, importer, css_mtime)
117
+ timestamps = @dependencies_stale[[uri, importer]] ||= {}
118
+ timestamps.each_pair do |checked_css_mtime, is_stale|
119
+ if checked_css_mtime <= css_mtime && !is_stale
120
+ return false
121
+ elsif checked_css_mtime > css_mtime && is_stale
122
+ return true
123
+ end
124
+ end
125
+ timestamps[css_mtime] = dependencies(uri, importer).any?(&dependency_updated?(css_mtime))
126
+ rescue Sass::SyntaxError
127
+ # If there's an error finding dependencies, default to recompiling.
128
+ true
129
+ end
130
+
131
+ def mtime(uri, importer)
132
+ @mtimes[[uri, importer]] ||=
133
+ begin
134
+ mtime = importer.mtime(uri, @options)
135
+ if mtime.nil?
136
+ with_dependency_cache {|cache| cache.delete([uri, importer])}
137
+ nil
138
+ else
139
+ mtime
140
+ end
141
+ end
142
+ end
143
+
144
+ def dependencies(uri, importer)
145
+ stored_mtime, dependencies =
146
+ with_dependency_cache {|cache| Sass::Util.destructure(cache[[uri, importer]])}
147
+
148
+ if !stored_mtime || stored_mtime < mtime(uri, importer)
149
+ dependencies = compute_dependencies(uri, importer)
150
+ with_dependency_cache do |cache|
151
+ cache[[uri, importer]] = [mtime(uri, importer), dependencies]
152
+ end
153
+ end
154
+
155
+ dependencies
156
+ end
157
+
158
+ def dependency_updated?(css_mtime)
159
+ proc do |uri, importer|
160
+ next true if @actively_checking.include?(uri)
161
+ begin
162
+ @actively_checking << uri
163
+ sass_mtime = mtime(uri, importer)
164
+ !sass_mtime ||
165
+ sass_mtime > css_mtime ||
166
+ dependencies_stale?(uri, importer, css_mtime)
167
+ ensure
168
+ @actively_checking.delete uri
169
+ end
170
+ end
171
+ end
172
+
173
+ def compute_dependencies(uri, importer)
174
+ tree(uri, importer).grep(Tree::ImportNode) do |n|
175
+ next if n.css_import?
176
+ file = n.imported_file
177
+ key = [file.options[:filename], file.options[:importer]]
178
+ @parse_trees[key] = file.to_tree
179
+ key
180
+ end.compact
181
+ end
182
+
183
+ def tree(uri, importer)
184
+ @parse_trees[[uri, importer]] ||= importer.find(uri, @options).to_tree
185
+ end
186
+
187
+ # Get access to the global dependency cache in a threadsafe manner.
188
+ # Inside the block, no other thread can access the dependency cache.
189
+ #
190
+ # @yieldparam cache [Hash] The hash that is the global dependency cache
191
+ # @return The value returned by the block to which this method yields
192
+ def with_dependency_cache
193
+ StalenessChecker.dependency_cache_mutex.synchronize do
194
+ yield StalenessChecker.dependencies_cache
195
+ end
196
+ end
197
+ end
198
+ end
199
+ end