Class: Parser

Inherits:
Object
  • Object
show all
Defined in:
lib/linmeric/Parser.rb

Overview

This is a stack-based parser that returns a tree in array format. Sub arrays are operations with higher priority. Operators are converted to symbol. It must be associated to Scp. Eg: 3 + 4 * 9 => ‘[3, :+, [4, :*, 9]]`

Author

Massimiliano Dal Mas ([email protected])

License

Distributed under MIT license

Instance Method Summary collapse

Constructor Details

#initializeParser

Initialize two new variables: a counter and a boolean one to report errors and stop the parser



18
19
20
21
# File 'lib/linmeric/Parser.rb', line 18

def initialize()
  @d = 0
  @error = false
end

Instance Method Details

#extract(expr) ⇒ Object

Extracts the content between quotes (it must not be parsed)

  • argument: string to extract the sub-string from

  • returns: sub-string



130
131
132
133
134
135
136
137
138
# File 'lib/linmeric/Parser.rb', line 130

def extract(expr)
  ext = '"'
  while expr[@d] != '"' do
    ext += expr[@d]
    @d += 1
  end
  @d += 1
  return ext + '"'
end

#parse(expr) ⇒ Object

Main function that parses the string

  • argument: string to be parsed; the string must be the

output of scopify (see: Scp )

  • returns: abstract tree array



34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
# File 'lib/linmeric/Parser.rb', line 34

def parse(expr)
  array = Array.new
  until @d == expr.length
    c = expr[@d]
    case c
      when "("
        @d += 1
        calc = parse(expr)
        array.push calc if calc != nil 
      when ")"
        @d += 1
        return array
      when /[\*\/]/
        @d +=1
        array.push c.to_sym
      when /[\+\-\^]/
        @d+=1
        array.push c.to_sym
      when /\=/
        @d += 1
        array.push c.to_sym
      when /\>/
        @d += 1
        array.push c.to_sym
      when /\"/
        @d += 1
        array.push extract(expr)
      when /\~/
        @d += 1
        if expr[@d] == "("
          @d +=1
          calc = parse(expr)
          array.push calc unless calc == nil
        else
          array.push expr[@d]
          @d += 1
        end
      when /\./
        if expr[@d-1] =~ /[0-9]+/
          x = array.pop.to_s + c + expr[@d+1]
          array.push x.to_n
        else
          unless @error
            @error = true
            puts "Problem evaluating expression at index:#{@d}"
            puts "Invalid char '#{expr[@d]}' in string variable"
          end
          return
        end
        @d+=2
      when /\:/
        if expr[@d - 1] =~ /[a-zA-Z]+/ then
          x = array.pop.to_s + c
          array.push x.downcase.to_sym
        else
          unless @error
            @error = true
            puts "Problem evaluating expression at index:#{@d}"
            puts "Invalid char '#{expr[@d]}' in numeric variable"
          end
          return
        end
        @d += 1
      when /\_/
        @d += 1
        x = array.pop.to_s + c
        array.push x
      when /\p{Alnum}/ 
        if expr[@d-1] =~ /[0-9]/ && array.count > 0
          x = array.pop.to_s + c
          array.push x.to_n
        elsif (expr[@d-1] =~ /[a-zA-Z]/ or expr[@d-1] == '_') && array.count > 0
          x = array.pop.to_s + c 
          array.push x 
        else 
          array.push c.to_n if c =~ /[0-9]/
          array.push c if c =~ /[a-zA-Z]/
        end          
        @d += 1
      else
        unless @error
          @error = true
          puts "Problem evaluating expression at index:#{@d}"
          puts "Char '#{expr[@d]}' not recognized"
        end
        return
    end
  end

  return array
end

#resetObject

Re initializes the counter and the boolean variable



24
25
26
27
# File 'lib/linmeric/Parser.rb', line 24

def reset()
  @d = 0
  @error = false
end