hamlit 1.7.2 → 2.0.1

Sign up to get free protection for your applications and to get access to all the features.
Files changed (283) hide show
  1. checksums.yaml +4 -4
  2. data/.gitignore +4 -3
  3. data/.gitmodules +3 -0
  4. data/.travis.yml +25 -37
  5. data/CHANGELOG.md +18 -0
  6. data/Gemfile +16 -0
  7. data/LICENSE.txt +23 -2
  8. data/README.md +106 -48
  9. data/REFERENCE.md +222 -0
  10. data/Rakefile +77 -19
  11. data/benchmark/boolean_attribute.haml +6 -0
  12. data/benchmark/class_attribute.haml +5 -0
  13. data/benchmark/common_attribute.haml +3 -0
  14. data/benchmark/data_attribute.haml +4 -0
  15. data/benchmark/dynamic_attributes/boolean_attribute.haml +4 -0
  16. data/benchmark/dynamic_attributes/class_attribute.haml +4 -0
  17. data/benchmark/dynamic_attributes/common_attribute.haml +2 -0
  18. data/benchmark/dynamic_attributes/data_attribute.haml +2 -0
  19. data/benchmark/dynamic_attributes/id_attribute.haml +2 -0
  20. data/benchmark/etc/attribute_builder.haml +5 -0
  21. data/benchmark/etc/real_sample.haml +888 -0
  22. data/benchmark/etc/real_sample.rb +11 -0
  23. data/benchmark/etc/static_analyzer.haml +1 -0
  24. data/benchmark/etc/tags.haml +3 -0
  25. data/benchmark/ext/build_data.rb +15 -0
  26. data/benchmark/ext/build_id.rb +13 -0
  27. data/benchmark/id_attribute.haml +3 -0
  28. data/benchmark/plain.haml +4 -0
  29. data/benchmark/script.haml +4 -0
  30. data/benchmark/slim/LICENSE +21 -0
  31. data/{benchmarks → benchmark/slim}/context.rb +2 -4
  32. data/benchmark/slim/run-benchmarks.rb +94 -0
  33. data/{benchmarks → benchmark/slim}/view.erb +3 -3
  34. data/{benchmarks → benchmark/slim}/view.haml +0 -0
  35. data/{benchmarks/view.escaped.slim → benchmark/slim/view.slim} +1 -1
  36. data/benchmark/string_interpolation.haml +2 -0
  37. data/benchmark/utils/benchmark_ips_extension.rb +43 -0
  38. data/bin/bench +85 -0
  39. data/bin/clone +14 -0
  40. data/bin/console +11 -0
  41. data/bin/lineprof +48 -0
  42. data/bin/ruby +3 -0
  43. data/bin/setup +7 -0
  44. data/bin/stackprof +27 -0
  45. data/{test → bin/test} +6 -10
  46. data/{bin → exe}/hamlit +0 -0
  47. data/ext/hamlit/extconf.rb +14 -0
  48. data/ext/hamlit/hamlit.c +512 -0
  49. data/ext/hamlit/houdini/.gitignore +3 -0
  50. data/ext/hamlit/houdini/COPYING +7 -0
  51. data/ext/hamlit/houdini/Makefile +79 -0
  52. data/ext/hamlit/houdini/README.md +59 -0
  53. data/ext/hamlit/houdini/buffer.c +249 -0
  54. data/ext/hamlit/houdini/buffer.h +113 -0
  55. data/ext/hamlit/houdini/houdini.h +46 -0
  56. data/ext/hamlit/houdini/houdini_href_e.c +115 -0
  57. data/ext/hamlit/houdini/houdini_html_e.c +90 -0
  58. data/ext/hamlit/houdini/houdini_html_u.c +122 -0
  59. data/ext/hamlit/houdini/houdini_js_e.c +90 -0
  60. data/ext/hamlit/houdini/houdini_js_u.c +60 -0
  61. data/ext/hamlit/houdini/houdini_uri_e.c +107 -0
  62. data/ext/hamlit/houdini/houdini_uri_u.c +68 -0
  63. data/ext/hamlit/houdini/houdini_xml_e.c +136 -0
  64. data/ext/hamlit/houdini/html_unescape.gperf +258 -0
  65. data/ext/hamlit/houdini/html_unescape.h +754 -0
  66. data/ext/hamlit/houdini/tools/build_table.py +13 -0
  67. data/ext/hamlit/houdini/tools/build_tables.c +51 -0
  68. data/ext/hamlit/houdini/tools/wikipedia_table.txt +2025 -0
  69. data/hamlit.gemspec +30 -31
  70. data/lib/hamlit.rb +3 -1
  71. data/lib/hamlit/attribute_builder.rb +12 -0
  72. data/lib/hamlit/cli.rb +44 -43
  73. data/lib/hamlit/compiler.rb +92 -16
  74. data/lib/hamlit/compiler/attribute_compiler.rb +148 -0
  75. data/lib/hamlit/compiler/children_compiler.rb +111 -0
  76. data/lib/hamlit/compiler/comment_compiler.rb +36 -0
  77. data/lib/hamlit/compiler/doctype_compiler.rb +45 -0
  78. data/lib/hamlit/compiler/script_compiler.rb +97 -0
  79. data/lib/hamlit/compiler/silent_script_compiler.rb +24 -0
  80. data/lib/hamlit/compiler/tag_compiler.rb +69 -0
  81. data/lib/hamlit/engine.rb +12 -7
  82. data/lib/hamlit/error.rb +14 -0
  83. data/lib/hamlit/escapable.rb +12 -0
  84. data/lib/hamlit/filters.rb +65 -0
  85. data/lib/hamlit/filters/base.rb +4 -62
  86. data/lib/hamlit/filters/coffee.rb +9 -7
  87. data/lib/hamlit/filters/css.rb +25 -8
  88. data/lib/hamlit/filters/erb.rb +4 -6
  89. data/lib/hamlit/filters/escaped.rb +11 -9
  90. data/lib/hamlit/filters/javascript.rb +25 -8
  91. data/lib/hamlit/filters/less.rb +9 -7
  92. data/lib/hamlit/filters/markdown.rb +5 -6
  93. data/lib/hamlit/filters/plain.rb +11 -15
  94. data/lib/hamlit/filters/preserve.rb +15 -5
  95. data/lib/hamlit/filters/ruby.rb +3 -5
  96. data/lib/hamlit/filters/sass.rb +9 -7
  97. data/lib/hamlit/filters/scss.rb +9 -7
  98. data/lib/hamlit/filters/text_base.rb +24 -0
  99. data/lib/hamlit/filters/tilt_base.rb +47 -0
  100. data/lib/hamlit/hash_parser.rb +107 -0
  101. data/lib/hamlit/html.rb +9 -6
  102. data/lib/hamlit/identity.rb +12 -0
  103. data/lib/hamlit/object_ref.rb +29 -0
  104. data/lib/hamlit/parser.rb +25 -142
  105. data/lib/hamlit/parser/MIT-LICENSE +20 -0
  106. data/lib/hamlit/parser/README.md +28 -0
  107. data/lib/hamlit/parser/haml_buffer.rb +348 -0
  108. data/lib/hamlit/parser/haml_compiler.rb +553 -0
  109. data/lib/hamlit/parser/haml_error.rb +61 -0
  110. data/lib/hamlit/parser/haml_helpers.rb +727 -0
  111. data/lib/hamlit/parser/haml_options.rb +286 -0
  112. data/lib/hamlit/parser/haml_parser.rb +801 -0
  113. data/lib/hamlit/parser/haml_util.rb +283 -0
  114. data/lib/hamlit/parser/haml_xss_mods.rb +109 -0
  115. data/lib/hamlit/{helpers.rb → rails_helpers.rb} +2 -7
  116. data/lib/hamlit/rails_template.rb +30 -0
  117. data/lib/hamlit/railtie.rb +1 -12
  118. data/lib/hamlit/ruby_expression.rb +31 -0
  119. data/lib/hamlit/static_analyzer.rb +49 -0
  120. data/lib/hamlit/string_interpolation.rb +69 -0
  121. data/lib/hamlit/template.rb +8 -0
  122. data/lib/hamlit/utils.rb +9 -0
  123. data/lib/hamlit/version.rb +1 -1
  124. metadata +116 -324
  125. data/.rspec +0 -2
  126. data/benchmarks/benchmark.rb +0 -110
  127. data/benchmarks/view.slim +0 -17
  128. data/doc/README.md +0 -19
  129. data/doc/engine/indent.md +0 -48
  130. data/doc/engine/new_attribute.md +0 -77
  131. data/doc/engine/old_attributes.md +0 -198
  132. data/doc/engine/silent_script.md +0 -97
  133. data/doc/engine/tag.md +0 -48
  134. data/doc/engine/text.md +0 -64
  135. data/doc/faml/README.md +0 -16
  136. data/doc/faml/engine/indent.md +0 -48
  137. data/doc/faml/engine/old_attributes.md +0 -111
  138. data/doc/faml/engine/silent_script.md +0 -97
  139. data/doc/faml/engine/text.md +0 -59
  140. data/doc/faml/filters/erb.md +0 -24
  141. data/doc/faml/filters/javascript.md +0 -27
  142. data/doc/faml/filters/less.md +0 -57
  143. data/doc/faml/filters/plain.md +0 -25
  144. data/doc/filters/erb.md +0 -31
  145. data/doc/filters/javascript.md +0 -83
  146. data/doc/filters/less.md +0 -57
  147. data/doc/filters/markdown.md +0 -31
  148. data/doc/filters/plain.md +0 -25
  149. data/doc/haml/README.md +0 -15
  150. data/doc/haml/engine/new_attribute.md +0 -77
  151. data/doc/haml/engine/old_attributes.md +0 -142
  152. data/doc/haml/engine/tag.md +0 -48
  153. data/doc/haml/engine/text.md +0 -29
  154. data/doc/haml/filters/erb.md +0 -26
  155. data/doc/haml/filters/javascript.md +0 -76
  156. data/doc/haml/filters/markdown.md +0 -31
  157. data/lib/hamlit/attribute.rb +0 -78
  158. data/lib/hamlit/compilers/attributes.rb +0 -108
  159. data/lib/hamlit/compilers/comment.rb +0 -13
  160. data/lib/hamlit/compilers/doctype.rb +0 -39
  161. data/lib/hamlit/compilers/filter.rb +0 -53
  162. data/lib/hamlit/compilers/new_attribute.rb +0 -115
  163. data/lib/hamlit/compilers/old_attribute.rb +0 -241
  164. data/lib/hamlit/compilers/runtime_attribute.rb +0 -58
  165. data/lib/hamlit/compilers/script.rb +0 -31
  166. data/lib/hamlit/compilers/strip.rb +0 -19
  167. data/lib/hamlit/compilers/text.rb +0 -111
  168. data/lib/hamlit/concerns/attribute_builder.rb +0 -22
  169. data/lib/hamlit/concerns/balanceable.rb +0 -68
  170. data/lib/hamlit/concerns/deprecation.rb +0 -20
  171. data/lib/hamlit/concerns/error.rb +0 -31
  172. data/lib/hamlit/concerns/escapable.rb +0 -17
  173. data/lib/hamlit/concerns/included.rb +0 -28
  174. data/lib/hamlit/concerns/indentable.rb +0 -117
  175. data/lib/hamlit/concerns/lexable.rb +0 -32
  176. data/lib/hamlit/concerns/line_reader.rb +0 -62
  177. data/lib/hamlit/concerns/registerable.rb +0 -24
  178. data/lib/hamlit/concerns/string_interpolation.rb +0 -48
  179. data/lib/hamlit/concerns/whitespace.rb +0 -91
  180. data/lib/hamlit/filters/tilt.rb +0 -41
  181. data/lib/hamlit/parsers/attribute.rb +0 -71
  182. data/lib/hamlit/parsers/comment.rb +0 -30
  183. data/lib/hamlit/parsers/doctype.rb +0 -18
  184. data/lib/hamlit/parsers/filter.rb +0 -18
  185. data/lib/hamlit/parsers/multiline.rb +0 -58
  186. data/lib/hamlit/parsers/script.rb +0 -126
  187. data/lib/hamlit/parsers/tag.rb +0 -83
  188. data/lib/hamlit/parsers/text.rb +0 -28
  189. data/lib/hamlit/temple.rb +0 -9
  190. data/release +0 -6
  191. data/spec/Rakefile +0 -72
  192. data/spec/hamlit/engine/comment_spec.rb +0 -56
  193. data/spec/hamlit/engine/doctype_spec.rb +0 -19
  194. data/spec/hamlit/engine/error_spec.rb +0 -135
  195. data/spec/hamlit/engine/indent_spec.rb +0 -42
  196. data/spec/hamlit/engine/multiline_spec.rb +0 -44
  197. data/spec/hamlit/engine/new_attribute_spec.rb +0 -110
  198. data/spec/hamlit/engine/old_attributes_spec.rb +0 -404
  199. data/spec/hamlit/engine/script_spec.rb +0 -116
  200. data/spec/hamlit/engine/silent_script_spec.rb +0 -213
  201. data/spec/hamlit/engine/tag_spec.rb +0 -295
  202. data/spec/hamlit/engine/text_spec.rb +0 -239
  203. data/spec/hamlit/engine_spec.rb +0 -58
  204. data/spec/hamlit/filters/coffee_spec.rb +0 -60
  205. data/spec/hamlit/filters/css_spec.rb +0 -33
  206. data/spec/hamlit/filters/erb_spec.rb +0 -16
  207. data/spec/hamlit/filters/javascript_spec.rb +0 -82
  208. data/spec/hamlit/filters/less_spec.rb +0 -37
  209. data/spec/hamlit/filters/markdown_spec.rb +0 -30
  210. data/spec/hamlit/filters/plain_spec.rb +0 -15
  211. data/spec/hamlit/filters/ruby_spec.rb +0 -24
  212. data/spec/hamlit/filters/sass_spec.rb +0 -33
  213. data/spec/hamlit/filters/scss_spec.rb +0 -37
  214. data/spec/hamlit/haml_spec.rb +0 -910
  215. data/spec/rails/.gitignore +0 -18
  216. data/spec/rails/.rspec +0 -2
  217. data/spec/rails/Gemfile +0 -19
  218. data/spec/rails/README.rdoc +0 -28
  219. data/spec/rails/Rakefile +0 -6
  220. data/spec/rails/app/assets/images/.keep +0 -0
  221. data/spec/rails/app/assets/javascripts/application.js +0 -15
  222. data/spec/rails/app/assets/stylesheets/application.css +0 -15
  223. data/spec/rails/app/controllers/application_controller.rb +0 -8
  224. data/spec/rails/app/controllers/concerns/.keep +0 -0
  225. data/spec/rails/app/controllers/users_controller.rb +0 -23
  226. data/spec/rails/app/helpers/application_helper.rb +0 -2
  227. data/spec/rails/app/mailers/.keep +0 -0
  228. data/spec/rails/app/models/.keep +0 -0
  229. data/spec/rails/app/models/concerns/.keep +0 -0
  230. data/spec/rails/app/views/application/index.html.haml +0 -18
  231. data/spec/rails/app/views/layouts/application.html.haml +0 -12
  232. data/spec/rails/app/views/users/capture.html.haml +0 -5
  233. data/spec/rails/app/views/users/capture_haml.html.haml +0 -5
  234. data/spec/rails/app/views/users/form.html.haml +0 -2
  235. data/spec/rails/app/views/users/helpers.html.haml +0 -10
  236. data/spec/rails/app/views/users/index.html.haml +0 -9
  237. data/spec/rails/app/views/users/inline.html.haml +0 -6
  238. data/spec/rails/app/views/users/old_attributes.html.haml +0 -5
  239. data/spec/rails/app/views/users/safe_buffer.html.haml +0 -4
  240. data/spec/rails/app/views/users/whitespace.html.haml +0 -4
  241. data/spec/rails/bin/bundle +0 -3
  242. data/spec/rails/bin/rails +0 -8
  243. data/spec/rails/bin/rake +0 -8
  244. data/spec/rails/bin/setup +0 -29
  245. data/spec/rails/bin/spring +0 -15
  246. data/spec/rails/config.ru +0 -4
  247. data/spec/rails/config/application.rb +0 -34
  248. data/spec/rails/config/boot.rb +0 -3
  249. data/spec/rails/config/database.yml +0 -25
  250. data/spec/rails/config/environment.rb +0 -5
  251. data/spec/rails/config/environments/development.rb +0 -41
  252. data/spec/rails/config/environments/production.rb +0 -79
  253. data/spec/rails/config/environments/test.rb +0 -42
  254. data/spec/rails/config/initializers/assets.rb +0 -11
  255. data/spec/rails/config/initializers/backtrace_silencers.rb +0 -7
  256. data/spec/rails/config/initializers/cookies_serializer.rb +0 -3
  257. data/spec/rails/config/initializers/filter_parameter_logging.rb +0 -4
  258. data/spec/rails/config/initializers/inflections.rb +0 -16
  259. data/spec/rails/config/initializers/mime_types.rb +0 -4
  260. data/spec/rails/config/initializers/session_store.rb +0 -3
  261. data/spec/rails/config/initializers/wrap_parameters.rb +0 -14
  262. data/spec/rails/config/locales/en.yml +0 -24
  263. data/spec/rails/config/routes.rb +0 -16
  264. data/spec/rails/config/secrets.yml +0 -22
  265. data/spec/rails/db/schema.rb +0 -16
  266. data/spec/rails/db/seeds.rb +0 -7
  267. data/spec/rails/lib/assets/.keep +0 -0
  268. data/spec/rails/lib/tasks/.keep +0 -0
  269. data/spec/rails/log/.keep +0 -0
  270. data/spec/rails/public/404.html +0 -67
  271. data/spec/rails/public/422.html +0 -67
  272. data/spec/rails/public/500.html +0 -66
  273. data/spec/rails/public/favicon.ico +0 -0
  274. data/spec/rails/public/robots.txt +0 -5
  275. data/spec/rails/spec/hamlit_spec.rb +0 -123
  276. data/spec/rails/spec/rails_helper.rb +0 -56
  277. data/spec/rails/spec/spec_helper.rb +0 -91
  278. data/spec/rails/vendor/assets/javascripts/.keep +0 -0
  279. data/spec/rails/vendor/assets/stylesheets/.keep +0 -0
  280. data/spec/spec_helper.rb +0 -36
  281. data/spec/spec_helper/document_generator.rb +0 -93
  282. data/spec/spec_helper/render_helper.rb +0 -120
  283. data/spec/spec_helper/test_case.rb +0 -55
@@ -0,0 +1,283 @@
1
+ # encoding: utf-8
2
+
3
+ begin
4
+ require 'erubis/tiny'
5
+ rescue LoadError
6
+ require 'erb'
7
+ end
8
+ require 'set'
9
+ require 'stringio'
10
+ require 'strscan'
11
+
12
+ module Hamlit
13
+ # A module containing various useful functions.
14
+ module HamlUtil
15
+ extend self
16
+
17
+ # Silence all output to STDERR within a block.
18
+ #
19
+ # @yield A block in which no output will be printed to STDERR
20
+ def silence_warnings
21
+ the_real_stderr, $stderr = $stderr, StringIO.new
22
+ yield
23
+ ensure
24
+ $stderr = the_real_stderr
25
+ end
26
+
27
+ ## Rails XSS Safety
28
+
29
+ # Whether or not ActionView's XSS protection is available and enabled,
30
+ # as is the default for Rails 3.0+, and optional for version 2.3.5+.
31
+ # Overridden in haml/template.rb if this is the case.
32
+ #
33
+ # @return [Boolean]
34
+ def rails_xss_safe?
35
+ false
36
+ end
37
+
38
+ # Returns the given text, marked as being HTML-safe.
39
+ # With older versions of the Rails XSS-safety mechanism,
40
+ # this destructively modifies the HTML-safety of `text`.
41
+ #
42
+ # It only works if you are using ActiveSupport or the parameter `text`
43
+ # implements the #html_safe method.
44
+ #
45
+ # @param text [String, nil]
46
+ # @return [String, nil] `text`, marked as HTML-safe
47
+ def html_safe(text)
48
+ return unless text
49
+ text.html_safe
50
+ end
51
+
52
+ # Checks that the encoding of a string is valid
53
+ # and cleans up potential encoding gotchas like the UTF-8 BOM.
54
+ # If it's not, yields an error string describing the invalid character
55
+ # and the line on which it occurs.
56
+ #
57
+ # @param str [String] The string of which to check the encoding
58
+ # @yield [msg] A block in which an encoding error can be raised.
59
+ # Only yields if there is an encoding error
60
+ # @yieldparam msg [String] The error message to be raised
61
+ # @return [String] `str`, potentially with encoding gotchas like BOMs removed
62
+ def check_encoding(str)
63
+ if str.valid_encoding?
64
+ # Get rid of the Unicode BOM if possible
65
+ # Shortcut for UTF-8 which might be the majority case
66
+ if str.encoding == Encoding::UTF_8
67
+ return str.gsub(/\A\uFEFF/, '')
68
+ elsif str.encoding.name =~ /^UTF-(16|32)(BE|LE)?$/
69
+ return str.gsub(Regexp.new("\\A\uFEFF".encode(str.encoding)), '')
70
+ else
71
+ return str
72
+ end
73
+ end
74
+
75
+ encoding = str.encoding
76
+ newlines = Regexp.new("\r\n|\r|\n".encode(encoding).force_encoding(Encoding::ASCII_8BIT))
77
+ str.force_encoding(Encoding::ASCII_8BIT).split(newlines).each_with_index do |line, i|
78
+ begin
79
+ line.encode(encoding)
80
+ rescue Encoding::UndefinedConversionError => e
81
+ yield <<MSG.rstrip, i + 1
82
+ Invalid #{encoding.name} character #{e.error_char.dump}
83
+ MSG
84
+ end
85
+ end
86
+ return str
87
+ end
88
+
89
+ # Like {\#check\_encoding}, but also checks for a Ruby-style `-# coding:` comment
90
+ # at the beginning of the template and uses that encoding if it exists.
91
+ #
92
+ # The Haml encoding rules are simple.
93
+ # If a `-# coding:` comment exists,
94
+ # we assume that that's the original encoding of the document.
95
+ # Otherwise, we use whatever encoding Ruby has.
96
+ #
97
+ # Haml uses the same rules for parsing coding comments as Ruby.
98
+ # This means that it can understand Emacs-style comments
99
+ # (e.g. `-*- encoding: "utf-8" -*-`),
100
+ # and also that it cannot understand non-ASCII-compatible encodings
101
+ # such as `UTF-16` and `UTF-32`.
102
+ #
103
+ # @param str [String] The Haml template of which to check the encoding
104
+ # @yield [msg] A block in which an encoding error can be raised.
105
+ # Only yields if there is an encoding error
106
+ # @yieldparam msg [String] The error message to be raised
107
+ # @return [String] The original string encoded properly
108
+ # @raise [ArgumentError] if the document declares an unknown encoding
109
+ def check_haml_encoding(str, &block)
110
+ str = str.dup if str.frozen?
111
+
112
+ bom, encoding = parse_haml_magic_comment(str)
113
+ if encoding; str.force_encoding(encoding)
114
+ elsif bom; str.force_encoding(Encoding::UTF_8)
115
+ end
116
+
117
+ return check_encoding(str, &block)
118
+ end
119
+
120
+ # Like `Object#inspect`, but preserves non-ASCII characters rather than escaping them.
121
+ # This is necessary so that the precompiled Haml template can be `#encode`d into `@options[:encoding]`
122
+ # before being evaluated.
123
+ #
124
+ # @param obj {Object}
125
+ # @return {String}
126
+ def inspect_obj(obj)
127
+ case obj
128
+ when String
129
+ %Q!"#{obj.gsub(/[\x00-\x7F]+/) {|s| s.inspect[1...-1]}}"!
130
+ when Symbol
131
+ ":#{inspect_obj(obj.to_s)}"
132
+ else
133
+ obj.inspect
134
+ end
135
+ end
136
+
137
+ # Scans through a string looking for the interoplation-opening `#{`
138
+ # and, when it's found, yields the scanner to the calling code
139
+ # so it can handle it properly.
140
+ #
141
+ # The scanner will have any backslashes immediately in front of the `#{`
142
+ # as the second capture group (`scan[2]`),
143
+ # and the text prior to that as the first (`scan[1]`).
144
+ #
145
+ # @yieldparam scan [StringScanner] The scanner scanning through the string
146
+ # @return [String] The text remaining in the scanner after all `#{`s have been processed
147
+ def handle_interpolation(str)
148
+ scan = StringScanner.new(str)
149
+ yield scan while scan.scan(/(.*?)(\\*)#([\{@$])/)
150
+ scan.rest
151
+ end
152
+
153
+ # Moves a scanner through a balanced pair of characters.
154
+ # For example:
155
+ #
156
+ # Foo (Bar (Baz bang) bop) (Bang (bop bip))
157
+ # ^ ^
158
+ # from to
159
+ #
160
+ # @param scanner [StringScanner] The string scanner to move
161
+ # @param start [String] The character opening the balanced pair.
162
+ # @param finish [String] The character closing the balanced pair.
163
+ # @param count [Fixnum] The number of opening characters matched
164
+ # before calling this method
165
+ # @return [(String, String)] The string matched within the balanced pair
166
+ # and the rest of the string.
167
+ # `["Foo (Bar (Baz bang) bop)", " (Bang (bop bip))"]` in the example above.
168
+ def balance(scanner, start, finish, count = 0)
169
+ str = ''
170
+ scanner = StringScanner.new(scanner) unless scanner.is_a? StringScanner
171
+ regexp = Regexp.new("(.*?)[\\#{start.chr}\\#{finish.chr}]", Regexp::MULTILINE)
172
+ while scanner.scan(regexp)
173
+ str << scanner.matched
174
+ count += 1 if scanner.matched[-1] == start
175
+ count -= 1 if scanner.matched[-1] == finish
176
+ return [str.strip, scanner.rest] if count == 0
177
+ end
178
+ end
179
+
180
+ # Formats a string for use in error messages about indentation.
181
+ #
182
+ # @param indentation [String] The string used for indentation
183
+ # @return [String] The name of the indentation (e.g. `"12 spaces"`, `"1 tab"`)
184
+ def human_indentation(indentation)
185
+ if !indentation.include?(?\t)
186
+ noun = 'space'
187
+ elsif !indentation.include?(?\s)
188
+ noun = 'tab'
189
+ else
190
+ return indentation.inspect
191
+ end
192
+
193
+ singular = indentation.length == 1
194
+ "#{indentation.length} #{noun}#{'s' unless singular}"
195
+ end
196
+
197
+ def contains_interpolation?(str)
198
+ /#[\{$@]/ === str
199
+ end
200
+
201
+ # Original Haml::Util.unescape_interpolation
202
+ def slow_unescape_interpolation(str, escape_html = nil)
203
+ res = ''
204
+ rest = ::Hamlit::HamlUtil.handle_interpolation str.dump do |scan|
205
+ escapes = (scan[2].size - 1) / 2
206
+ char = scan[3] # '{', '@' or '$'
207
+ res << scan.matched[0...-3 - escapes]
208
+ if escapes % 2 == 1
209
+ res << "\##{char}"
210
+ else
211
+ interpolated = if char == '{'
212
+ balance(scan, ?{, ?}, 1)[0][0...-1]
213
+ else
214
+ scan.scan(/\w+/)
215
+ end
216
+ content = eval('"' + interpolated + '"')
217
+ content.prepend(char) if char == '@' || char == '$'
218
+ content = "::Hamlit::HamlHelpers.html_escape((#{content}))" if escape_html
219
+
220
+ res << "\#{#{content}}"
221
+ end
222
+ end
223
+ res + rest
224
+ end
225
+
226
+ # Customized Haml::Util.unescape_interpolation to handle escape by Hamlit
227
+ def unescape_interpolation(str)
228
+ res = ''
229
+ rest = ::Hamlit::HamlUtil.handle_interpolation str.dump do |scan|
230
+ escapes = (scan[2].size - 1) / 2
231
+ char = scan[3] # '{', '@' or '$'
232
+ res << scan.matched[0...-3 - escapes]
233
+ if escapes % 2 == 1
234
+ res << "\##{char}"
235
+ else
236
+ interpolated = if char == '{'
237
+ balance(scan, ?{, ?}, 1)[0][0...-1]
238
+ else
239
+ scan.scan(/\w+/)
240
+ end
241
+ content = eval('"' + interpolated + '"')
242
+ content.prepend(char) if char == '@' || char == '$'
243
+
244
+ res << "\#{#{content}}"
245
+ end
246
+ end
247
+ res + rest
248
+ end
249
+
250
+ private
251
+
252
+ # Parses a magic comment at the beginning of a Haml file.
253
+ # The parsing rules are basically the same as Ruby's.
254
+ #
255
+ # @return [(Boolean, String or nil)]
256
+ # Whether the document begins with a UTF-8 BOM,
257
+ # and the declared encoding of the document (or nil if none is declared)
258
+ def parse_haml_magic_comment(str)
259
+ scanner = StringScanner.new(str.dup.force_encoding(Encoding::ASCII_8BIT))
260
+ bom = scanner.scan(/\xEF\xBB\xBF/n)
261
+ return bom unless scanner.scan(/-\s*#\s*/n)
262
+ if coding = try_parse_haml_emacs_magic_comment(scanner)
263
+ return bom, coding
264
+ end
265
+
266
+ return bom unless scanner.scan(/.*?coding[=:]\s*([\w-]+)/in)
267
+ return bom, scanner[1]
268
+ end
269
+
270
+ def try_parse_haml_emacs_magic_comment(scanner)
271
+ pos = scanner.pos
272
+ return unless scanner.scan(/.*?-\*-\s*/n)
273
+ # From Ruby's parse.y
274
+ return unless scanner.scan(/([^\s'":;]+)\s*:\s*("(?:\\.|[^"])*"|[^"\s;]+?)[\s;]*-\*-/n)
275
+ name, val = scanner[1], scanner[2]
276
+ return unless name =~ /(en)?coding/in
277
+ val = $1 if val =~ /^"(.*)"$/n
278
+ return val
279
+ ensure
280
+ scanner.pos = pos
281
+ end
282
+ end
283
+ end
@@ -0,0 +1,109 @@
1
+ module Hamlit
2
+ module HamlHelpers
3
+ # This module overrides Haml helpers to work properly
4
+ # in the context of ActionView.
5
+ # Currently it's only used for modifying the helpers
6
+ # to work with Rails' XSS protection methods.
7
+ module XssMods
8
+ def self.included(base)
9
+ %w[html_escape find_and_preserve preserve list_of surround
10
+ precede succeed capture_haml haml_concat haml_internal_concat haml_indent
11
+ escape_once].each do |name|
12
+ base.send(:alias_method, "#{name}_without_haml_xss", name)
13
+ base.send(:alias_method, name, "#{name}_with_haml_xss")
14
+ end
15
+ end
16
+
17
+ # Don't escape text that's already safe,
18
+ # output is always HTML safe
19
+ def html_escape_with_haml_xss(text)
20
+ str = text.to_s
21
+ return text if str.html_safe?
22
+ ::Hamlit::HamlUtil.html_safe(html_escape_without_haml_xss(str))
23
+ end
24
+
25
+ # Output is always HTML safe
26
+ def find_and_preserve_with_haml_xss(*args, &block)
27
+ ::Hamlit::HamlUtil.html_safe(find_and_preserve_without_haml_xss(*args, &block))
28
+ end
29
+
30
+ # Output is always HTML safe
31
+ def preserve_with_haml_xss(*args, &block)
32
+ ::Hamlit::HamlUtil.html_safe(preserve_without_haml_xss(*args, &block))
33
+ end
34
+
35
+ # Output is always HTML safe
36
+ def list_of_with_haml_xss(*args, &block)
37
+ ::Hamlit::HamlUtil.html_safe(list_of_without_haml_xss(*args, &block))
38
+ end
39
+
40
+ # Input is escaped, output is always HTML safe
41
+ def surround_with_haml_xss(front, back = front, &block)
42
+ ::Hamlit::HamlUtil.html_safe(
43
+ surround_without_haml_xss(
44
+ haml_xss_html_escape(front),
45
+ haml_xss_html_escape(back),
46
+ &block))
47
+ end
48
+
49
+ # Input is escaped, output is always HTML safe
50
+ def precede_with_haml_xss(str, &block)
51
+ ::Hamlit::HamlUtil.html_safe(precede_without_haml_xss(haml_xss_html_escape(str), &block))
52
+ end
53
+
54
+ # Input is escaped, output is always HTML safe
55
+ def succeed_with_haml_xss(str, &block)
56
+ ::Hamlit::HamlUtil.html_safe(succeed_without_haml_xss(haml_xss_html_escape(str), &block))
57
+ end
58
+
59
+ # Output is always HTML safe
60
+ def capture_haml_with_haml_xss(*args, &block)
61
+ ::Hamlit::HamlUtil.html_safe(capture_haml_without_haml_xss(*args, &block))
62
+ end
63
+
64
+ # Input will be escaped unless this is in a `with_raw_haml_concat`
65
+ # block. See #Haml::Helpers::ActionViewExtensions#with_raw_haml_concat.
66
+ def haml_concat_with_haml_xss(text = "")
67
+ raw = instance_variable_defined?(:@_haml_concat_raw) ? @_haml_concat_raw : false
68
+ if raw
69
+ haml_internal_concat_raw text
70
+ else
71
+ haml_internal_concat text
72
+ end
73
+ ErrorReturn.new("haml_concat")
74
+ end
75
+
76
+ # Input is escaped
77
+ def haml_internal_concat_with_haml_xss(text="", newline=true, indent=true)
78
+ haml_internal_concat_without_haml_xss(haml_xss_html_escape(text), newline, indent)
79
+ end
80
+ private :haml_internal_concat_with_haml_xss
81
+
82
+ # Output is always HTML safe
83
+ def haml_indent_with_haml_xss
84
+ ::Hamlit::HamlUtil.html_safe(haml_indent_without_haml_xss)
85
+ end
86
+
87
+ # Output is always HTML safe
88
+ def escape_once_with_haml_xss(*args)
89
+ ::Hamlit::HamlUtil.html_safe(escape_once_without_haml_xss(*args))
90
+ end
91
+
92
+ private
93
+
94
+ # Escapes the HTML in the text if and only if
95
+ # Rails XSS protection is enabled *and* the `:escape_html` option is set.
96
+ def haml_xss_html_escape(text)
97
+ return text unless ::Hamlit::HamlUtil.rails_xss_safe? && haml_buffer.options[:escape_html]
98
+ html_escape(text)
99
+ end
100
+ end
101
+
102
+ class ErrorReturn
103
+ # Any attempt to treat ErrorReturn as a string should cause it to blow up.
104
+ alias_method :html_safe, :to_s
105
+ alias_method :html_safe?, :to_s
106
+ alias_method :html_safe!, :to_s
107
+ end
108
+ end
109
+ end
@@ -1,12 +1,7 @@
1
- # This is a module compatible with Haml::Helpers.
2
- # It is included by ActionView in initializer.
3
- #
4
- # NOTE: currently Hamlit::Helpers is enabled by default
5
- # on only Rails environment.
6
- # And currently this Hamlit::Helpers depends on
1
+ # Currently this Hamlit::Helpers depends on
7
2
  # ActionView internal implementation. (not desired)
8
3
  module Hamlit
9
- module Helpers
4
+ module RailsHelpers
10
5
  extend self
11
6
 
12
7
  DEFAULT_PRESERVE_TAGS = %w[textarea pre code].freeze
@@ -0,0 +1,30 @@
1
+ require 'temple'
2
+ require 'hamlit/engine'
3
+ require 'hamlit/rails_helpers'
4
+ require 'hamlit/parser/haml_helpers'
5
+ require 'hamlit/parser/haml_util'
6
+
7
+ module Hamlit
8
+ RailsTemplate = Temple::Templates::Rails.create(
9
+ Hamlit::Engine,
10
+ generator: Temple::Generators::RailsOutputBuffer,
11
+ register_as: :haml,
12
+ use_html_safe: true,
13
+ streaming: true,
14
+ )
15
+
16
+ # https://github.com/haml/haml/blob/4.0.7/lib/haml/template.rb
17
+ module HamlHelpers
18
+ require 'hamlit/parser/haml_xss_mods'
19
+ include Hamlit::HamlHelpers::XssMods
20
+ end
21
+
22
+ module HamlUtil
23
+ undef :rails_xss_safe? if defined? rails_xss_safe?
24
+ def rails_xss_safe?; true; end
25
+ end
26
+ end
27
+
28
+ # Haml extends Haml::Helpers in ActionView each time.
29
+ # It costs much, so Hamlit includes a compatible module at first.
30
+ ActionView::Base.send :include, Hamlit::RailsHelpers
@@ -1,20 +1,9 @@
1
1
  require 'rails'
2
- require 'temple'
3
2
 
4
3
  module Hamlit
5
4
  class Railtie < ::Rails::Railtie
6
5
  initializer :hamlit do |app|
7
- Hamlit::RailsTemplate = Temple::Templates::Rails.create(
8
- Hamlit::Engine,
9
- generator: Temple::Generators::RailsOutputBuffer,
10
- register_as: :haml,
11
- escape_html: true,
12
- streaming: true,
13
- )
14
-
15
- # Haml extends Haml::Helpers in ActionView each time.
16
- # It costs much, so Hamlit includes a compatible module at first.
17
- ActionView::Base.send :include, Hamlit::Helpers
6
+ require 'hamlit/rails_template'
18
7
  end
19
8
  end
20
9
  end