Treetop中的CSS / HSS解析器和嵌套样式表规则

时间:2009-04-24 12:56:15

标签: ruby treetop

我是Treetop的新手并尝试编写CSS / HSS解析器。 HSS使用嵌套样式,变量和一种mixin功能来增强CSS的基本功能。

我非常接近 - 解析器可以处理CSS - 但是当我在样式中实现样式时,我会失败。 e.g:

#rule #one {
  #two {
    color: red;
  }
  color: blue;
}

我已经拍了两张照片,一张处理空白,一张不处理空白。我不能上班。树梢文件有点稀疏,我真的觉得我缺少一些基本的东西。希望有人可以让我直截了当。

A:

 grammar Stylesheet

      rule stylesheet
        space* style*
      end

      rule style
        selectors space* '{' space* properties? space* '}' space*
      end

      rule properties
        property space* (';' space* property)* ';'?
      end

      rule property
        property_name space* [:] space* property_value
      end

      rule property_name
        [^:;}]+
      end

      rule property_value
        [^:;}]+
      end

      rule space
        [\t ]
      end

      rule selectors
        selector space* ([,] space* selector)*
      end

      rule selector
        element (space+ ![{] element)*
      end

      rule element
        class / id
      end

      rule id
        [#] [a-zA-Z-]+
      end

      rule class
       [.] [a-zA-Z-]+
      end
end

B:

grammar Stylesheet

  rule stylesheet
   style*
  end

  rule style
    selectors closure
  end

  rule closure
    '{' ( style / property )* '}'
  end

  rule property
    property_name ':' property_value ';'
  end

  rule property_name
    [^:}]+
    <PropertyNode>
  end

  rule property_value
    [^;]+
    <PropertyNode>
  end

  rule selectors
    selector ( !closure ',' selector )*
    <SelectorNode>
  end

  rule selector
    element ( space+ !closure element )*
    <SelectorNode>
  end

  rule element
    class / id
  end

  rule id
    ('#' [a-zA-Z]+)
  end

  rule class
    ('.' [a-zA-Z]+)
  end

  rule space
    [\t ]
  end

end

线束代码:

require 'rubygems'
require 'treetop'

class PropertyNode < Treetop::Runtime::SyntaxNode
  def value
    "property:(#{text_value})"
  end
end

class SelectorNode < Treetop::Runtime::SyntaxNode
  def value
    "--> #{text_value}"
  end
end

Treetop.load('css')

parser = StylesheetParser.new
parser.consume_all_input = false

string = <<EOS
#hello-there .my-friend {
  font-family:Verdana;
  font-size:12px;
}
.my-friend, #is-cool {
  font: 12px Verdana;
  #he .likes-jam, #very-much {asaads:there;}
  hello: there;
}
EOS

root_node = parser.parse(string)

def print_node(node, output = [])
  output << node.value if node.respond_to?(:value)
  node.elements.each {|element| print_node(element, output)} if node.elements
  output
end

puts print_node(root_node).join("\n") if root_node

#puts parser.methods.sort.join(',')
puts parser.input
puts string[0...parser.failure_index] + '<--'
puts parser.failure_reason
puts parser.terminal_failures

2 个答案:

答案 0 :(得分:3)

我认为你遇到了left recursion个问题?如果是这样,请记住TreeTop会生成recursive descent parsers,因此,您无法在语法中使用左递归。 (我仍然喜欢使用ocamlyacc / ocamllex而不是TreeTop的主要原因之一,尽管它非常性感。)这意味着你需要从左递归形式转换为右递归。既然您无疑拥有Dragon Book(对吗?),我将引导您阅读涉及该问题的4.3.3,4.3.4和4.4.1节。通常情况下,它很难理解,但解析器并没有得到任何声誉。还有一个很好的left recursion elimination tutorial,ANTLR的人提出了这个问题。它有点特定于ANTLR / ANTLRworks,但它比Dragon Book中的内容更容易理解。对于那些至少几次没有做过这件事的人来说,这是其中之一。但/ / p>

另外,小评论,如果您要使用TreeTop,我建议改为:

def ws
  [\t ]*
end

你不太可能需要匹配单个空格字符,而且几乎每个语法规则都需要它,因此将其命名为非常短的事情是有意义的。顺便提一下,单独的lexing步骤具有 优势。这是其中之一。

答案 1 :(得分:1)

看起来有人打败了我:

http://lesscss.org/

虽然我注意到他们使用正则表达式和eval()来解析输入文件而不是解析器。

编辑:现在他们使用TreeTop!这就像有人为我做了所有艰苦的工作。