Skip to content

Commit

Permalink
Improve parse_declarations! performance
Browse files Browse the repository at this point in the history
  • Loading branch information
leonid-shevtsov committed Aug 16, 2024
1 parent 4fc5b07 commit 9b3ac77
Show file tree
Hide file tree
Showing 3 changed files with 38 additions and 8 deletions.
1 change: 1 addition & 0 deletions CHANGELOG.md
Original file line number Diff line number Diff line change
Expand Up @@ -7,6 +7,7 @@
* RuleSet initialize now takes keyword argument, positional arguments are still supported but deprecated
* Removed OffsetAwareRuleSet, it's a RuleSet with optional attributes filename and offset
* Improved performance of block parsing by using StringScanner
* Improve `RuleSet#parse_declarations!` performance by using substring search istead of regexps

### Version v1.18.0

Expand Down
32 changes: 24 additions & 8 deletions lib/css_parser/rule_set.rb
Original file line number Diff line number Diff line change
Expand Up @@ -26,6 +26,11 @@ class RuleSet

WHITESPACE_REPLACEMENT = '___SPACE___'

# Tokens for parse_declarations!
COLON = ':'.freeze
SEMICOLON = ';'.freeze
LPAREN = '('.freeze
RPAREN = ')'.freeze
class Declarations
class Value
attr_reader :value
Expand Down Expand Up @@ -647,20 +652,31 @@ def parse_declarations!(block) # :nodoc:
return unless block

continuation = nil
block.split(/[;$]+/m).each do |decs|
decs = (continuation ? continuation + decs : decs)
if decs =~ /\([^)]*\Z/ # if it has an unmatched parenthesis
continuation = "#{decs};"
elsif (matches = decs.match(/\s*(.[^:]*)\s*:\s*(?m:(.+))(?:;?\s*\Z)/i))
# skip end_of_declaration
property = matches[1]
value = matches[2]
block.split(SEMICOLON) do |decs|
decs = (continuation ? "#{continuation};#{decs}" : decs)
if unmatched_parenthesis?(decs)
continuation = decs
else
colon = decs.index(COLON)
next if colon.nil?

property = decs[0, colon]
value = decs[colon + 1, decs.length - colon - 1]
property.strip!
value.strip!
next if property.empty? || value.empty?

add_declaration!(property, value)
continuation = nil
end
end
end

def unmatched_parenthesis?(declarations)
lparen_index = declarations.index(LPAREN)
!lparen_index.nil? && declarations.index(RPAREN, lparen_index).nil?
end

#--
# TODO: way too simplistic
#++
Expand Down
13 changes: 13 additions & 0 deletions test/test_rule_set.rb
Original file line number Diff line number Diff line change
Expand Up @@ -78,6 +78,19 @@ def test_each_declaration_containing_semicolons
assert_equal('no-repeat;', rs['background-repeat'])
end

def test_each_declaration_with_newlines
expected = Set[
{property: 'background-image', value: 'url(foo;bar)', is_important: false},
{property: 'font-weight', value: 'bold', is_important: true},
]
rs = RuleSet.new(block: "background-image\n:\nurl(foo;bar);\n\n\n\n\n;;font-weight\n\n\n:bold\n\n\n!important")
actual = Set.new
rs.each_declaration do |prop, val, imp|
actual << {property: prop, value: val, is_important: imp}
end
assert_equal(expected, actual)
end

def test_selector_sanitization
selectors = "h1, h2,\nh3 "
rs = RuleSet.new(selectors: selectors, block: "color: #fff;")
Expand Down

0 comments on commit 9b3ac77

Please sign in to comment.