rley 0.8.14 → 0.9.01
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.
- checksums.yaml +4 -4
- data/.rubocop.yml +20 -2
- data/CHANGELOG.md +14 -0
- data/Gemfile +8 -0
- data/examples/general/calc_iter1/spec/calculator_spec.rb +9 -9
- data/examples/general/calc_iter2/spec/calculator_spec.rb +39 -39
- data/examples/general/recursive_right.rb +2 -2
- data/lib/rley/constants.rb +2 -2
- data/lib/rley/engine.rb +3 -1
- data/lib/rley/formatter/asciitree.rb +4 -1
- data/lib/rley/gfg/epsilon_edge.rb +0 -2
- data/lib/rley/gfg/grm_flow_graph.rb +7 -9
- data/lib/rley/gfg/item_vertex.rb +1 -1
- data/lib/rley/gfg/vertex.rb +11 -0
- data/lib/rley/lexical/token.rb +10 -3
- data/lib/rley/parse_forest_visitor.rb +3 -3
- data/lib/rley/parse_rep/ast_base_builder.rb +6 -5
- data/lib/rley/parse_rep/parse_forest_builder.rb +4 -2
- data/lib/rley/parse_rep/parse_tree_builder.rb +14 -2
- data/lib/rley/parse_rep/parse_tree_factory.rb +1 -1
- data/lib/rley/parser/error_reason.rb +2 -2
- data/lib/rley/parser/gfg_chart.rb +2 -2
- data/lib/rley/parser/gfg_earley_parser.rb +2 -2
- data/lib/rley/parser/gfg_parsing.rb +1 -1
- data/lib/rley/parser/parse_entry.rb +4 -4
- data/lib/rley/parser/parse_entry_set.rb +4 -2
- data/lib/rley/parser/parse_entry_tracker.rb +10 -7
- data/lib/rley/parser/parse_walker_factory.rb +9 -8
- data/lib/rley/ptree/parse_tree_node.rb +6 -0
- data/lib/rley/ptree/terminal_node.rb +1 -1
- data/lib/rley/rgn/ast_builder.rb +2 -2
- data/lib/rley/rgn/ast_node.rb +11 -1
- data/lib/rley/rgn/ast_visitor.rb +2 -2
- data/lib/rley/rgn/composite_node.rb +1 -1
- data/lib/rley/rgn/grammar_builder.rb +12 -14
- data/lib/rley/rgn/parser.rb +2 -2
- data/lib/rley/rgn/tokenizer.rb +1 -1
- data/lib/rley/rley_error.rb +0 -4
- data/lib/rley/sppf/composite_node.rb +6 -0
- data/lib/rley/sppf/parse_forest.rb +7 -7
- data/lib/rley/sppf/sppf_node.rb +15 -1
- data/lib/rley/syntax/base_grammar_builder.rb +3 -12
- data/lib/rley/syntax/grammar.rb +9 -4
- data/lib/rley/syntax/production.rb +1 -1
- data/spec/rley/base/dotted_item_spec.rb +46 -46
- data/spec/rley/base/grm_items_builder_spec.rb +1 -1
- data/spec/rley/engine_spec.rb +50 -50
- data/spec/rley/formatter/asciitree_spec.rb +8 -8
- data/spec/rley/formatter/bracket_notation_spec.rb +10 -10
- data/spec/rley/formatter/debug_spec.rb +10 -10
- data/spec/rley/formatter/json_spec.rb +6 -7
- data/spec/rley/gfg/call_edge_spec.rb +6 -6
- data/spec/rley/gfg/edge_spec.rb +8 -7
- data/spec/rley/gfg/end_vertex_spec.rb +8 -7
- data/spec/rley/gfg/epsilon_edge_spec.rb +5 -4
- data/spec/rley/gfg/grm_flow_graph_spec.rb +33 -34
- data/spec/rley/gfg/item_vertex_spec.rb +34 -36
- data/spec/rley/gfg/non_terminal_vertex_spec.rb +12 -12
- data/spec/rley/gfg/return_edge_spec.rb +6 -6
- data/spec/rley/gfg/scan_edge_spec.rb +7 -6
- data/spec/rley/gfg/shortcut_edge_spec.rb +15 -15
- data/spec/rley/gfg/start_vertex_spec.rb +8 -8
- data/spec/rley/gfg/vertex_spec.rb +18 -18
- data/spec/rley/lexical/literal_spec.rb +5 -5
- data/spec/rley/lexical/token_range_spec.rb +55 -55
- data/spec/rley/lexical/token_spec.rb +17 -16
- data/spec/rley/parse_forest_visitor_spec.rb +30 -32
- data/spec/rley/parse_rep/ambiguous_parse_spec.rb +2 -2
- data/spec/rley/parse_rep/ast_builder_spec.rb +30 -30
- data/spec/rley/parse_rep/cst_builder_spec.rb +85 -85
- data/spec/rley/parse_rep/groucho_spec.rb +23 -23
- data/spec/rley/parse_rep/parse_forest_builder_spec.rb +42 -42
- data/spec/rley/parse_rep/parse_forest_factory_spec.rb +10 -12
- data/spec/rley/parse_rep/parse_tree_factory_spec.rb +10 -15
- data/spec/rley/parse_tree_visitor_spec.rb +43 -46
- data/spec/rley/parser/dangling_else_spec.rb +12 -12
- data/spec/rley/parser/error_reason_spec.rb +37 -37
- data/spec/rley/parser/gfg_chart_spec.rb +27 -29
- data/spec/rley/parser/gfg_earley_parser_spec.rb +55 -56
- data/spec/rley/parser/gfg_parsing_spec.rb +106 -103
- data/spec/rley/parser/parse_entry_set_spec.rb +63 -61
- data/spec/rley/parser/parse_entry_spec.rb +73 -71
- data/spec/rley/parser/parse_walker_factory_spec.rb +14 -15
- data/spec/rley/ptree/non_terminal_node_spec.rb +16 -16
- data/spec/rley/ptree/parse_tree_node_spec.rb +11 -11
- data/spec/rley/ptree/parse_tree_spec.rb +6 -8
- data/spec/rley/ptree/terminal_node_spec.rb +6 -6
- data/spec/rley/rgn/grammar_builder_spec.rb +69 -67
- data/spec/rley/rgn/parser_spec.rb +63 -63
- data/spec/rley/rgn/repetition_node_spec.rb +15 -15
- data/spec/rley/rgn/sequence_node_spec.rb +10 -10
- data/spec/rley/rgn/symbol_node_spec.rb +5 -6
- data/spec/rley/rgn/tokenizer_spec.rb +68 -67
- data/spec/rley/sppf/alternative_node_spec.rb +16 -16
- data/spec/rley/sppf/non_terminal_node_spec.rb +20 -20
- data/spec/rley/sppf/token_node_spec.rb +13 -13
- data/spec/rley/syntax/base_grammar_builder_spec.rb +76 -86
- data/spec/rley/syntax/grammar_spec.rb +40 -78
- data/spec/rley/syntax/grm_symbol_spec.rb +7 -7
- data/spec/rley/syntax/match_closest_spec.rb +8 -8
- data/spec/rley/syntax/non_terminal_spec.rb +25 -25
- data/spec/rley/syntax/production_spec.rb +33 -33
- data/spec/rley/syntax/symbol_seq_spec.rb +27 -27
- data/spec/rley/syntax/terminal_spec.rb +12 -11
- data/spec/support/base_tokenizer_spec.rb +9 -8
- metadata +8 -25
checksums.yaml
CHANGED
@@ -1,7 +1,7 @@
|
|
1
1
|
---
|
2
2
|
SHA256:
|
3
|
-
metadata.gz:
|
4
|
-
data.tar.gz:
|
3
|
+
metadata.gz: 7fc61215b7d8262c0ffa08c6149558e79915dcc3fbe58ba1bb4b8c57d8097423
|
4
|
+
data.tar.gz: a5e363a07f5daf5ef99799e6abe8b88dd322b6c68fbe752d3be77270329058c5
|
5
5
|
SHA512:
|
6
|
-
metadata.gz:
|
7
|
-
data.tar.gz:
|
6
|
+
metadata.gz: '0605876019c02c17ba22d3e916bbdbfae96182da06384276939805569aaac83f11fb62ad3c22c7e837a59f3718e018ca3577cbf22566fcc38bfe044832c8a54b'
|
7
|
+
data.tar.gz: 6d56cf19e500a36e1abb5298bd7d46418d548bee380cd2ed1abab5b72a2e74c05a2a6e9dcf877dd6bf0f8af133192bf86d227a57ba912884f499fe31a9e0e37b
|
data/.rubocop.yml
CHANGED
@@ -438,6 +438,9 @@ Lint/TopLevelReturnWithArgument:
|
|
438
438
|
Lint/UnreachableLoop:
|
439
439
|
Enabled: true
|
440
440
|
|
441
|
+
Naming/BlockForwarding:
|
442
|
+
EnforcedStyle: explicit
|
443
|
+
|
441
444
|
Style/AccessorGrouping:
|
442
445
|
Enabled: true
|
443
446
|
|
@@ -505,7 +508,22 @@ Style/StringConcatenation:
|
|
505
508
|
# Enabled: false
|
506
509
|
#
|
507
510
|
#RSpec/ExampleLength:
|
508
|
-
# Max:
|
511
|
+
# Max: 350
|
512
|
+
#
|
513
|
+
#RSpec/IndexedLet:
|
514
|
+
# Enabled: false
|
515
|
+
#
|
516
|
+
#RSpec/MultipleDescribes:
|
517
|
+
# Enabled: false
|
509
518
|
#
|
510
519
|
#RSpec/MultipleExpectations:
|
511
|
-
# Max:
|
520
|
+
# Max: 55
|
521
|
+
#
|
522
|
+
#RSpec/MultipleMemoizedHelpers:
|
523
|
+
# Max: 30
|
524
|
+
#
|
525
|
+
#RSpec/NoExpectationExample:
|
526
|
+
# Enabled: false
|
527
|
+
#
|
528
|
+
#RSpec/VerifiedDoubles:
|
529
|
+
# Enabled: false
|
data/CHANGELOG.md
CHANGED
@@ -1,3 +1,17 @@
|
|
1
|
+
### 0.9.01 / 2025-03-16
|
2
|
+
Version bumped.
|
3
|
+
* [CHANGED] Updated `.rbs` signature files.
|
4
|
+
|
5
|
+
### 0.9.00 / 2025-03-08
|
6
|
+
Version bumped.
|
7
|
+
* [NEW] Added `sig` folder with `.rbs` signature files.
|
8
|
+
- [FIX] Broken Cucumber test. Replaced reference to `GrammarBuilder` by `BaseGrammarBuilder`.
|
9
|
+
- [FIX] File `rley.gemspec` now returns a GemSpecification as expected by Bundler.
|
10
|
+
- [CHANGE] File `Gemfile` refers to gemspec
|
11
|
+
|
12
|
+
### 0.8.15 / 2025-02-28
|
13
|
+
- [FIX] Re-styled spec files
|
14
|
+
|
1
15
|
### 0.8.14 / 2025-02-26
|
2
16
|
- [FIX] Re-styled spec files of tutorial/iter4 folder
|
3
17
|
|
data/Gemfile
ADDED
@@ -33,39 +33,39 @@ describe 'Calculator' do
|
|
33
33
|
return result
|
34
34
|
end
|
35
35
|
|
36
|
-
it '
|
36
|
+
it 'evaluates simple number literals' do
|
37
37
|
expect_expr('2').to eq(2)
|
38
38
|
end
|
39
39
|
|
40
|
-
it '
|
40
|
+
it 'evaluates addition' do
|
41
41
|
expect_expr('2 + 2').to eq(4)
|
42
42
|
end
|
43
43
|
|
44
|
-
it '
|
44
|
+
it 'evaluates subtraction' do
|
45
45
|
expect_expr('2.1 - 2').to be_within(0.000000000000001).of(0.1)
|
46
46
|
end
|
47
47
|
|
48
|
-
it '
|
48
|
+
it 'evaluates division' do
|
49
49
|
expect_expr('10.5 / 5').to eq(2.1)
|
50
50
|
end
|
51
51
|
|
52
|
-
it '
|
52
|
+
it 'evaluates multiplication' do
|
53
53
|
expect_expr('2 * 3.1').to eq(6.2)
|
54
54
|
end
|
55
55
|
|
56
|
-
it '
|
56
|
+
it 'evaluates parentheses' do
|
57
57
|
expect_expr('2 * (2.1 + 1)').to eq(6.2)
|
58
58
|
end
|
59
59
|
|
60
|
-
it '
|
60
|
+
it 'evaluates regardless of whitespace' do
|
61
61
|
expect_expr("2*(1+\t1)").to eq(4)
|
62
62
|
end
|
63
63
|
|
64
|
-
it '
|
64
|
+
it 'evaluates order of operations' do
|
65
65
|
expect_expr('2 * 2.1 + 1 / 2').to eq 4.7
|
66
66
|
end
|
67
67
|
|
68
|
-
it '
|
68
|
+
it 'evaluates multiple levels of parentheses' do
|
69
69
|
expect_expr('2*(1/(1+3))').to eq(0.5)
|
70
70
|
end
|
71
71
|
end # describe
|
@@ -41,152 +41,152 @@ describe 'Calculator' do
|
|
41
41
|
context 'Parsing valid expressions' do
|
42
42
|
let(:epsilon) { 0.0000000001 }
|
43
43
|
|
44
|
-
it '
|
44
|
+
it 'evaluates simple integer literals' do
|
45
45
|
expect_expr('2').to eq(2)
|
46
46
|
end
|
47
47
|
|
48
|
-
it '
|
48
|
+
it 'evaluates simple floating-point literals' do
|
49
49
|
expect_expr('3.1').to eq(3.1)
|
50
50
|
end
|
51
51
|
|
52
|
-
it '
|
52
|
+
it 'evaluates positive integer literals' do
|
53
53
|
expect_expr('+2').to eq(2)
|
54
54
|
expect_expr('+ 2').to eq(2)
|
55
55
|
end
|
56
56
|
|
57
|
-
it '
|
57
|
+
it 'evaluates positive floating-point literals' do
|
58
58
|
expect_expr('+3.1').to eq(3.1)
|
59
59
|
expect_expr('+ 3.1').to eq(3.1)
|
60
60
|
end
|
61
61
|
|
62
|
-
it '
|
62
|
+
it 'evaluates negative integer literals' do
|
63
63
|
expect_expr('-2').to eq(-2)
|
64
64
|
expect_expr('- 2').to eq(-2)
|
65
65
|
end
|
66
66
|
|
67
|
-
it '
|
67
|
+
it 'evaluates negative floating-point literals' do
|
68
68
|
expect_expr('-3.1').to eq(-3.1)
|
69
69
|
expect_expr('- 3.1').to eq(-3.1)
|
70
70
|
end
|
71
71
|
|
72
|
-
it '
|
72
|
+
it 'evaluates the pi constant' do
|
73
73
|
expect_expr('PI').to eq(3.141592653589793)
|
74
74
|
end
|
75
75
|
|
76
|
-
it '
|
76
|
+
it 'evaluates the negated pi constant' do
|
77
77
|
expect_expr('-PI').to eq(-3.141592653589793)
|
78
78
|
end
|
79
79
|
|
80
|
-
it "
|
80
|
+
it "evaluates Neper's e constant" do
|
81
81
|
expect_expr('E').to eq(2.718281828459045)
|
82
82
|
end
|
83
83
|
|
84
|
-
it "
|
84
|
+
it "evaluates negated Neper's e constant" do
|
85
85
|
expect_expr('-E').to eq(-2.718281828459045)
|
86
86
|
end
|
87
87
|
|
88
|
-
it '
|
88
|
+
it 'evaluates integer addition' do
|
89
89
|
expect_expr('2 + 2').to eq(4)
|
90
90
|
end
|
91
91
|
|
92
|
-
it '
|
92
|
+
it 'evaluates floating-point addition' do
|
93
93
|
expect_expr('2.0 + 3.1').to eq(5.1)
|
94
94
|
end
|
95
95
|
|
96
|
-
it '
|
96
|
+
it 'evaluates integer subtraction' do
|
97
97
|
expect_expr('2 - 3').to eq(-1)
|
98
98
|
end
|
99
99
|
|
100
|
-
it '
|
100
|
+
it 'evaluates floating-point subtraction' do
|
101
101
|
expect_expr('3.1 - 2').to eq(1.1)
|
102
102
|
end
|
103
103
|
|
104
|
-
it '
|
104
|
+
it 'handles negation of negative numbers' do
|
105
105
|
expect_expr('3--2').to eq(5)
|
106
106
|
end
|
107
107
|
|
108
|
-
it '
|
108
|
+
it 'evaluates integer multiplication' do
|
109
109
|
expect_expr('2 * 3').to eq(6)
|
110
110
|
end
|
111
111
|
|
112
|
-
it '
|
112
|
+
it 'evaluates floating-point multiplication' do
|
113
113
|
expect_expr('2 * 3.1').to eq(6.2)
|
114
114
|
end
|
115
115
|
|
116
|
-
it '
|
116
|
+
it 'evaluates division of integers' do
|
117
117
|
expect_expr('5 / 2').to eq(2.5)
|
118
118
|
end
|
119
119
|
|
120
|
-
it '
|
120
|
+
it 'evaluates floating-point division' do
|
121
121
|
expect_expr('10.5 / 5').to eq(2.1)
|
122
122
|
end
|
123
123
|
|
124
|
-
it '
|
124
|
+
it 'evaluates integer exponent' do
|
125
125
|
expect_expr('5 ** (3 - 1)').to eq(25)
|
126
126
|
end
|
127
127
|
|
128
|
-
it '
|
128
|
+
it 'evaluates floating-point exponent' do
|
129
129
|
expect_expr('25 ** 0.5').to eq(5)
|
130
130
|
end
|
131
131
|
|
132
|
-
it '
|
132
|
+
it 'evaluates negative exponent' do
|
133
133
|
expect_expr('5 ** -2').to eq(0.04)
|
134
134
|
end
|
135
135
|
|
136
|
-
it '
|
136
|
+
it 'handles nested exponentiations' do
|
137
137
|
expect_expr('2 ** 2**2').to eq(16)
|
138
138
|
end
|
139
139
|
|
140
|
-
it '
|
140
|
+
it 'changes sign of expression in parentheses' do
|
141
141
|
expect_expr('- (2 * 5)').to eq(-10)
|
142
142
|
end
|
143
143
|
|
144
|
-
it '
|
144
|
+
it 'evaluates parentheses' do
|
145
145
|
expect_expr('2 * (2.1 + 1)').to eq(6.2)
|
146
146
|
end
|
147
147
|
|
148
|
-
it '
|
148
|
+
it 'evaluates regardless of whitespace' do
|
149
149
|
expect_expr("2*(1+\t1)").to eq(4)
|
150
150
|
end
|
151
151
|
|
152
|
-
it '
|
152
|
+
it 'evaluates order of operations' do
|
153
153
|
expect_expr('2 * 2.1 + 1 / 2').to eq 4.7
|
154
154
|
end
|
155
155
|
|
156
|
-
it '
|
156
|
+
it 'evaluates multiple levels of parentheses' do
|
157
157
|
expect_expr('2*(1/(1+3))').to eq(0.5)
|
158
158
|
end
|
159
159
|
|
160
160
|
# Some special functions
|
161
|
-
it '
|
161
|
+
it 'evaluates square root of expressions' do
|
162
162
|
expect_expr('sqrt(0)').to eq(0)
|
163
163
|
expect_expr('sqrt(1)').to eq(1)
|
164
164
|
expect_expr('sqrt(1 + 1)').to eq(Math.sqrt(2))
|
165
165
|
expect_expr('sqrt(5 * 5)').to eq(5)
|
166
166
|
end
|
167
167
|
|
168
|
-
it '
|
168
|
+
it 'evaluates cubic root of expressions' do
|
169
169
|
expect_expr('cbrt(0)').to eq(0)
|
170
170
|
expect_expr('cbrt(1)').to eq(1)
|
171
171
|
expect_expr('cbrt(1 + 1)').to eq(Math.cbrt(2))
|
172
172
|
expect_expr('cbrt(5 * 5 * 5)').to eq(5)
|
173
173
|
end
|
174
174
|
|
175
|
-
it '
|
175
|
+
it 'evaluates exponential of expressions' do
|
176
176
|
expect_expr('exp(-1)').to eq(1 / Math::E)
|
177
177
|
expect_expr('exp(0)').to eq(1)
|
178
178
|
expect_expr('exp(1)').to eq(Math::E)
|
179
179
|
expect_expr('exp(2)').to be_within(epsilon).of(Math::E * Math::E)
|
180
180
|
end
|
181
181
|
|
182
|
-
it '
|
182
|
+
it 'evaluates natural logarithm of expressions' do
|
183
183
|
expect_expr('ln(1/E)').to eq(-1)
|
184
184
|
expect_expr('ln(1)').to eq(0)
|
185
185
|
expect_expr('ln(E)').to eq(1)
|
186
186
|
expect_expr('ln(E * E)').to eq(2)
|
187
187
|
end
|
188
188
|
|
189
|
-
it '
|
189
|
+
it 'evaluates the logarithm base 10 of expressions' do
|
190
190
|
expect_expr('log(1/10)').to eq(-1)
|
191
191
|
expect_expr('log(1)').to eq(0)
|
192
192
|
expect_expr('log(10)').to eq(1)
|
@@ -195,19 +195,19 @@ describe 'Calculator' do
|
|
195
195
|
|
196
196
|
# Trigonometric functions
|
197
197
|
|
198
|
-
it '
|
198
|
+
it 'computes the sinus of an expression' do
|
199
199
|
expect_expr('sin(0)').to eq(0)
|
200
200
|
expect_expr('sin(PI/6)').to be_within(epsilon).of(0.5)
|
201
201
|
expect_expr('sin(PI/2)').to eq(1)
|
202
202
|
end
|
203
203
|
|
204
|
-
it '
|
204
|
+
it 'computes the cosinus of an expression' do
|
205
205
|
expect_expr('cos(0)').to eq(1)
|
206
206
|
expect_expr('cos(PI/3)').to be_within(epsilon).of(0.5)
|
207
207
|
expect_expr('cos(PI/2)').to be_within(epsilon).of(0)
|
208
208
|
end
|
209
209
|
|
210
|
-
it '
|
210
|
+
it 'computes the tangent of an expression' do
|
211
211
|
expect_expr('tan(0)').to eq(0)
|
212
212
|
expect_expr('tan(PI/4)').to be_within(epsilon).of(1)
|
213
213
|
expect_expr('tan(5*PI/12)').to be_within(epsilon).of(2 + Math.sqrt(3))
|
@@ -215,19 +215,19 @@ describe 'Calculator' do
|
|
215
215
|
|
216
216
|
# Inverse trigonometric functions
|
217
217
|
|
218
|
-
it '
|
218
|
+
it 'computes the arcsinus of an expression' do
|
219
219
|
expect_expr('asin(0)').to eq(0)
|
220
220
|
expect_expr('asin(0.5)').to be_within(epsilon).of(Math::PI / 6)
|
221
221
|
expect_expr('asin(1)').to eq(Math::PI / 2)
|
222
222
|
end
|
223
223
|
|
224
|
-
it '
|
224
|
+
it 'computes the arccosinus of an expression' do
|
225
225
|
expect_expr('acos(1)').to eq(0)
|
226
226
|
expect_expr('acos(0.5)').to be_within(epsilon).of(Math::PI / 3)
|
227
227
|
expect_expr('acos(0)').to be_within(epsilon).of(Math::PI / 2)
|
228
228
|
end
|
229
229
|
|
230
|
-
it '
|
230
|
+
it 'computes the arctangent of an expression' do
|
231
231
|
expect_expr('atan(0)').to eq(0)
|
232
232
|
pi = Math::PI
|
233
233
|
expect_expr('atan(1)').to be_within(epsilon).of(pi / 4)
|
@@ -143,7 +143,7 @@ find every scan edge
|
|
143
143
|
def build_gfg(aGrammar)
|
144
144
|
items_builder = Object.new.extend(Rley::Base::GrmItemsBuilder)
|
145
145
|
items = items_builder.build_dotted_items(aGrammar)
|
146
|
-
|
146
|
+
Rley::GFG::GrmFlowGraph.new(items)
|
147
147
|
end
|
148
148
|
|
149
149
|
graph = build_gfg(right_recursive_grammar)
|
@@ -326,7 +326,7 @@ module GraphMixin
|
|
326
326
|
|
327
327
|
# visit_action takes two arguments: a vertex and an edge
|
328
328
|
# return true/false if false stop traversal
|
329
|
-
def df_traversal(aVertex, &
|
329
|
+
def df_traversal(aVertex, &_visit_action)
|
330
330
|
visit = [aVertex]
|
331
331
|
vtx2count = { aVertex => 0 }
|
332
332
|
|
data/lib/rley/constants.rb
CHANGED
@@ -5,10 +5,10 @@
|
|
5
5
|
|
6
6
|
module Rley # Module used as a namespace
|
7
7
|
# The version number of the gem.
|
8
|
-
Version = '0.
|
8
|
+
Version = '0.9.01'
|
9
9
|
|
10
10
|
# Brief description of the gem.
|
11
|
-
Description = "
|
11
|
+
Description = "A parsing library based on the Earley's parsing algorithm."
|
12
12
|
|
13
13
|
# Constant Rley::RootDir contains the absolute path of Rley's
|
14
14
|
# start directory. Note: it also ends with a slash character.
|
data/lib/rley/engine.rb
CHANGED
@@ -19,6 +19,7 @@ module Rley # This module is used as a namespace
|
|
19
19
|
# Constructor with default initialization.
|
20
20
|
def initialize
|
21
21
|
super()
|
22
|
+
# @type self: EngineConfig
|
22
23
|
self.parse_repr = :parse_tree
|
23
24
|
self.repr_builder = :default
|
24
25
|
self.diagnose = false
|
@@ -80,7 +81,7 @@ module Rley # This module is used as a namespace
|
|
80
81
|
aTokenizer.each do |a_token|
|
81
82
|
next unless a_token
|
82
83
|
|
83
|
-
if a_token.terminal.
|
84
|
+
if a_token.terminal.is_a?(String)
|
84
85
|
term_name = a_token.terminal
|
85
86
|
term_symb = grammar.name2symbol[term_name]
|
86
87
|
a_token.instance_variable_set(:@terminal, term_symb)
|
@@ -133,6 +134,7 @@ module Rley # This module is used as a namespace
|
|
133
134
|
result = factory.create(configuration.repr_builder)
|
134
135
|
end
|
135
136
|
|
137
|
+
# @type var result : Rley::SPPF::ParseForest
|
136
138
|
result
|
137
139
|
end
|
138
140
|
|
@@ -82,9 +82,10 @@ module Rley # This module is used as a namespace
|
|
82
82
|
def rank_of(aChild)
|
83
83
|
if curr_path.empty?
|
84
84
|
rank = 'root'
|
85
|
-
elsif curr_path[-1].subnodes.size == 1
|
85
|
+
elsif curr_path[-1].subnodes.size == 1 # steep:ignore
|
86
86
|
rank = 'first_and_last'
|
87
87
|
else
|
88
|
+
# @type var parent : Rley::PTree::NonTerminalNode
|
88
89
|
parent = curr_path[-1]
|
89
90
|
siblings = parent.subnodes
|
90
91
|
siblings_last_index = siblings.size - 1
|
@@ -96,6 +97,8 @@ module Rley # This module is used as a namespace
|
|
96
97
|
end
|
97
98
|
end
|
98
99
|
ranks << rank
|
100
|
+
|
101
|
+
rank
|
99
102
|
end
|
100
103
|
|
101
104
|
# 'root', 'first', 'first_and_last', 'last', 'other'
|
@@ -1,6 +1,5 @@
|
|
1
1
|
# frozen_string_literal: true
|
2
2
|
|
3
|
-
require 'set'
|
4
3
|
require_relative 'start_vertex'
|
5
4
|
require_relative 'end_vertex'
|
6
5
|
require_relative 'item_vertex'
|
@@ -41,7 +40,7 @@ module Rley # This module is used as a namespace
|
|
41
40
|
attr_reader :vertices
|
42
41
|
|
43
42
|
# The vertex marked as start node of the graph
|
44
|
-
# @return [StartVertex
|
43
|
+
# @return [StartVertex]
|
45
44
|
attr_reader :start_vertex
|
46
45
|
|
47
46
|
# A Hash with pairs of the form: non-terminal symbol => start node
|
@@ -189,7 +188,7 @@ module Rley # This module is used as a namespace
|
|
189
188
|
# For debugging purposes
|
190
189
|
def print_vertex(aText, aVertex)
|
191
190
|
print "#{aText} "
|
192
|
-
if aVertex.
|
191
|
+
if aVertex.is_a?(NonTerminalVertex)
|
193
192
|
puts "#{aVertex.class} #{aVertex.non_terminal.name}"
|
194
193
|
else
|
195
194
|
p(aVertex.label)
|
@@ -223,7 +222,7 @@ module Rley # This module is used as a namespace
|
|
223
222
|
productions.each do |prod|
|
224
223
|
all_nterms << prod.lhs
|
225
224
|
nterms_of_rhs = prod.rhs.members.select do |symb|
|
226
|
-
symb.
|
225
|
+
symb.is_a?(Syntax::NonTerminal)
|
227
226
|
end
|
228
227
|
all_nterms.merge(nterms_of_rhs)
|
229
228
|
end
|
@@ -286,7 +285,7 @@ module Rley # This module is used as a namespace
|
|
286
285
|
# At least one symbol before the dot
|
287
286
|
# Retrieve the symbol before the dot...
|
288
287
|
prev_symbol = current_item.prev_symbol
|
289
|
-
if prev_symbol.
|
288
|
+
if prev_symbol.is_a?(Syntax::Terminal)
|
290
289
|
build_scan_edge(vertices[-2], new_vertex)
|
291
290
|
else
|
292
291
|
# ...non-terminal
|
@@ -295,7 +294,7 @@ module Rley # This module is used as a namespace
|
|
295
294
|
end
|
296
295
|
|
297
296
|
prev_symbol = current_item.prev_symbol
|
298
|
-
if prev_symbol&.
|
297
|
+
if prev_symbol&.is_a?(Syntax::NonTerminal)
|
299
298
|
build_shortcut_edge(prev_vertex, new_vertex)
|
300
299
|
end
|
301
300
|
prev_vertex = new_vertex
|
@@ -356,8 +355,7 @@ module Rley # This module is used as a namespace
|
|
356
355
|
end_vertex = end_vertex_for[aCallEdge.successor.non_terminal]
|
357
356
|
|
358
357
|
# Retrieve the return edge with specified key
|
359
|
-
|
360
|
-
return return_edge
|
358
|
+
end_vertex.edges.find { |edge| edge.key == ret_key }
|
361
359
|
end
|
362
360
|
|
363
361
|
# Mark non-terminal symbols that cannot be derived from the start symbol.
|
@@ -373,7 +371,7 @@ module Rley # This module is used as a namespace
|
|
373
371
|
# and mark all visited non-terminals as reachable
|
374
372
|
traverse_df(start_vertex) do |a_vertex|
|
375
373
|
# print_vertex(' Visiting', a_vertex)
|
376
|
-
if a_vertex.
|
374
|
+
if a_vertex.is_a?(StartVertex)
|
377
375
|
a_vertex.non_terminal.unreachable = false
|
378
376
|
end
|
379
377
|
end
|
data/lib/rley/gfg/item_vertex.rb
CHANGED
@@ -30,7 +30,7 @@ module Rley # This module is used as a namespace
|
|
30
30
|
# Set the "shortcut" edge.
|
31
31
|
# @param aShortcut [ShortcutEdge] the "shortcut" edge.
|
32
32
|
def shortcut=(aShortcut)
|
33
|
-
unless aShortcut.
|
33
|
+
unless aShortcut.is_a?(ShortcutEdge)
|
34
34
|
raise StandardError, 'Invalid shortcut argument'
|
35
35
|
end
|
36
36
|
|
data/lib/rley/gfg/vertex.rb
CHANGED
@@ -69,6 +69,17 @@ module Rley # This module is used as a namespace
|
|
69
69
|
edges << arrow
|
70
70
|
end
|
71
71
|
|
72
|
+
# The label of this vertex.
|
73
|
+
# It is the same as the label of the corresponding dotted item.
|
74
|
+
# @return [String] Label for this vertex
|
75
|
+
def label
|
76
|
+
raise NotImplementedError
|
77
|
+
end
|
78
|
+
|
79
|
+
def dotted_item
|
80
|
+
raise NotImplementedError
|
81
|
+
end
|
82
|
+
|
72
83
|
protected
|
73
84
|
|
74
85
|
# Validation method for adding an outgoing edge to the vertex.
|
data/lib/rley/lexical/token.rb
CHANGED
@@ -5,13 +5,20 @@ module Rley # This module is used as a namespace
|
|
5
5
|
# as return values from a tokenizer / lexer.
|
6
6
|
module Lexical
|
7
7
|
# A Position is the location of a lexeme within a source file.
|
8
|
-
Position
|
8
|
+
class Position
|
9
|
+
attr_reader :line
|
10
|
+
attr_reader :column
|
11
|
+
|
12
|
+
def initialize(aLine, aCol)
|
13
|
+
@line = aLine
|
14
|
+
@column = aCol
|
15
|
+
end
|
16
|
+
|
9
17
|
def to_s
|
10
18
|
"line #{line}, column #{column}"
|
11
19
|
end
|
12
20
|
end
|
13
21
|
|
14
|
-
|
15
22
|
# In Rley, a (lexical) token is an object created by a lexer (tokenizer)
|
16
23
|
# and passed to the parser. Such token an object is created when a lexer
|
17
24
|
# detects that a sequence of characters(a lexeme) from the input stream
|
@@ -27,7 +34,7 @@ module Rley # This module is used as a namespace
|
|
27
34
|
# @return [String] Input substring that is an instance of the terminal.
|
28
35
|
attr_reader(:lexeme)
|
29
36
|
|
30
|
-
# @return [
|
37
|
+
# @return [Syntax::Terminal] The name of terminal symbol matching the lexeme.
|
31
38
|
attr_reader(:terminal)
|
32
39
|
|
33
40
|
# @return [Position] The position -in "editor" coordinates- of the lexeme in the source file.
|
@@ -1,6 +1,5 @@
|
|
1
1
|
# frozen_string_literal: true
|
2
2
|
|
3
|
-
# require 'pry'
|
4
3
|
require 'prime'
|
5
4
|
|
6
5
|
module Rley # This module is used as a namespace
|
@@ -145,9 +144,10 @@ module Rley # This module is used as a namespace
|
|
145
144
|
# Let's proceed with the visit of children
|
146
145
|
children.each_with_index do |a_node, i|
|
147
146
|
edge_sign = aParentNode.signatures[i]
|
148
|
-
if a_node.
|
147
|
+
if a_node.is_a?(SPPF::CompositeNode)
|
149
148
|
push_node(a_node, edge_sign)
|
150
149
|
access_paths = node_accesses[a_node]
|
150
|
+
# @type var last_path : Integer
|
151
151
|
last_path = legs.last[-1]
|
152
152
|
path_reused = access_paths.include?(last_path)
|
153
153
|
unless path_reused
|
@@ -178,8 +178,8 @@ module Rley # This module is used as a namespace
|
|
178
178
|
if legs.empty?
|
179
179
|
legs << [aCompositeNode, anEdgeSignature]
|
180
180
|
else
|
181
|
+
# @type var path_signature : Integer
|
181
182
|
path_signature = legs.last[-1]
|
182
|
-
# binding.pry if anEdgeSignature == 37 && path_signature != 230
|
183
183
|
if (path_signature % anEdgeSignature).zero?
|
184
184
|
legs << [aCompositeNode, path_signature]
|
185
185
|
else
|
@@ -94,9 +94,10 @@ module Rley # This module is used as a namespace
|
|
94
94
|
klass = terminal2node.fetch(aTerminal.name, terminalnode_class)
|
95
95
|
if klass.is_a?(Hash)
|
96
96
|
# Lexical ambiguity...
|
97
|
-
klass = klass.fetch(aProduction.name)
|
97
|
+
klass = klass.fetch(aProduction.name) # steep:ignore ArgumentTypeMismatch
|
98
98
|
end
|
99
|
-
|
99
|
+
|
100
|
+
klass.new(aToken, aTokenPosition) # steep:ignore
|
100
101
|
end
|
101
102
|
|
102
103
|
# Method to override.
|
@@ -106,7 +107,7 @@ module Rley # This module is used as a namespace
|
|
106
107
|
# @param theTokens [Array] The input tokens
|
107
108
|
# @param theChildren [Array] Children nodes (one per rhs symbol)
|
108
109
|
def new_parent_node(aProduction, aRange, theTokens, theChildren)
|
109
|
-
mth_name = method_name(aProduction.name)
|
110
|
+
mth_name = method_name(aProduction.name) # steep:ignore ArgumentTypeMismatch
|
110
111
|
if respond_to?(mth_name, true)
|
111
112
|
node = send(mth_name, aProduction, aRange, theTokens, theChildren)
|
112
113
|
else
|
@@ -139,7 +140,7 @@ module Rley # This module is used as a namespace
|
|
139
140
|
# Implicit rule generated for * modifier
|
140
141
|
# rule('X') => 'X item'.as '_star_more'
|
141
142
|
def reduce__star_more(_production, _range, _tokens, theChildren)
|
142
|
-
theChildren[0] << theChildren[1]
|
143
|
+
theChildren[0] << theChildren[1] # steep:ignore NoMethod
|
143
144
|
theChildren[0]
|
144
145
|
end
|
145
146
|
|
@@ -152,7 +153,7 @@ module Rley # This module is used as a namespace
|
|
152
153
|
# Implicit rule generated for + modifier
|
153
154
|
# rule('X') => 'X item'.as '_plus_more'
|
154
155
|
def reduce__plus_more(_production, _range, _tokens, theChildren)
|
155
|
-
theChildren[0] << theChildren[1]
|
156
|
+
theChildren[0] << theChildren[1] # steep:ignore NoMethod
|
156
157
|
theChildren[0]
|
157
158
|
end
|
158
159
|
|