interscript 0.1.4 → 2.0.5
Sign up to get free protection for your applications and to get access to all the features.
- checksums.yaml +4 -4
- data/.gitignore +11 -0
- data/.rspec +3 -0
- data/Gemfile +29 -0
- data/LICENSE.adoc +31 -0
- data/README.md +3 -0
- data/Rakefile +53 -0
- data/bin/console +14 -0
- data/bin/interscript +3 -39
- data/bin/maps_analyze_staging +168 -0
- data/bin/maps_debug_compilers +58 -0
- data/bin/maps_debug_ordering +88 -0
- data/bin/maps_debug_ruby_compile +24 -0
- data/bin/maps_debug_step_by_step +44 -0
- data/bin/maps_optimize_order +112 -0
- data/bin/maps_v1_analyze_regexps +45 -0
- data/bin/maps_v1_to_v2 +426 -0
- data/exe/interscript +6 -0
- data/interscript.gemspec +31 -0
- data/lib/interscript.rb +76 -128
- data/lib/interscript/command.rb +6 -5
- data/lib/interscript/compiler.rb +22 -0
- data/lib/interscript/compiler/javascript.rb +292 -0
- data/lib/interscript/compiler/ruby.rb +262 -0
- data/lib/interscript/dsl.rb +67 -0
- data/lib/interscript/dsl/aliases.rb +23 -0
- data/lib/interscript/dsl/document.rb +46 -0
- data/lib/interscript/dsl/group.rb +45 -0
- data/lib/interscript/dsl/group/parallel.rb +6 -0
- data/lib/interscript/dsl/items.rb +89 -0
- data/lib/interscript/dsl/metadata.rb +26 -0
- data/lib/interscript/dsl/stage.rb +6 -0
- data/lib/interscript/dsl/symbol_mm.rb +11 -0
- data/lib/interscript/dsl/tests.rb +12 -0
- data/lib/interscript/interpreter.rb +251 -0
- data/lib/interscript/node.rb +25 -0
- data/lib/interscript/node/alias_def.rb +15 -0
- data/lib/interscript/node/dependency.rb +13 -0
- data/lib/interscript/node/document.rb +45 -0
- data/lib/interscript/node/group.rb +34 -0
- data/lib/interscript/node/group/parallel.rb +9 -0
- data/lib/interscript/node/group/sequential.rb +2 -0
- data/lib/interscript/node/item.rb +52 -0
- data/lib/interscript/node/item/alias.rb +42 -0
- data/lib/interscript/node/item/any.rb +61 -0
- data/lib/interscript/node/item/capture.rb +50 -0
- data/lib/interscript/node/item/group.rb +51 -0
- data/lib/interscript/node/item/repeat.rb +40 -0
- data/lib/interscript/node/item/stage.rb +23 -0
- data/lib/interscript/node/item/string.rb +51 -0
- data/lib/interscript/node/metadata.rb +18 -0
- data/lib/interscript/node/rule.rb +6 -0
- data/lib/interscript/node/rule/funcall.rb +18 -0
- data/lib/interscript/node/rule/run.rb +15 -0
- data/lib/interscript/node/rule/sub.rb +65 -0
- data/lib/interscript/node/stage.rb +19 -0
- data/lib/interscript/node/tests.rb +15 -0
- data/lib/interscript/stdlib.rb +211 -0
- data/lib/interscript/utils/regexp_converter.rb +283 -0
- data/lib/interscript/version.rb +1 -1
- data/requirements.txt +1 -0
- metadata +73 -223
- data/README.adoc +0 -297
- data/bin/rspec +0 -29
- data/lib/g2pwrapper.py +0 -34
- data/lib/interscript/mapping.rb +0 -125
- data/lib/model-7 +0 -0
- data/lib/tha-pt-b-7 +0 -0
- data/maps/acadsin-zho-Hani-Latn-2002.yaml +0 -38912
- data/maps/alalc-aze-Cyrl-Latn-1997.yaml +0 -141
- data/maps/alalc-bel-cyrl-latn-1997.yaml +0 -125
- data/maps/alalc-ben-Beng-Latn-2017.yaml +0 -130
- data/maps/alalc-bul-Cyrl-Latn-1997.yaml +0 -94
- data/maps/alalc-ell-Grek-Latn-1997.yaml +0 -625
- data/maps/alalc-ell-Grek-Latn-2010.yaml +0 -628
- data/maps/alalc-kat-Geok-Latn-1997.yaml +0 -112
- data/maps/alalc-kat-Geor-Latn-1997.yaml +0 -146
- data/maps/alalc-kor-Hang-Latn-1997.yaml +0 -94
- data/maps/alalc-mkd-Cyrl-Latn-2013.yaml +0 -103
- data/maps/alalc-mkd-cyrl-latn-1997.yaml +0 -114
- data/maps/alalc-rus-Cyrl-Latn-1997.yaml +0 -222
- data/maps/alalc-rus-Cyrl-Latn-2012.yaml +0 -162
- data/maps/alalc-srp-Cyrl-Latn-1997.yaml +0 -114
- data/maps/alalc-srp-cyrl-latn-2013.yaml +0 -135
- data/maps/alalc-ukr-Cyrl-Latn-1997.yaml +0 -141
- data/maps/alalc-ukr-Cyrl-Latn-2011.yaml +0 -16
- data/maps/apcbg-bul-Cyrl-Latn-1995.yaml +0 -283
- data/maps/bas-rus-Cyrl-Latn-2017-bss.yaml +0 -175
- data/maps/bas-rus-Cyrl-Latn-2017-oss.yaml +0 -169
- data/maps/bgn-jpn-Hrkt-Latn-1962.yaml +0 -294
- data/maps/bgn-kor-Hang-Latn-1943.yaml +0 -31
- data/maps/bgn-kor-Kore-Latn-1943.yaml +0 -31
- data/maps/bgna-bul-Cyrl-Latn-2006.yaml +0 -208
- data/maps/bgna-bul-Cyrl-Latn-2009.yaml +0 -208
- data/maps/bgnpcgn-arm-Armn-Latn-1981.yaml +0 -108
- data/maps/bgnpcgn-aze-Cyrl-Latn-1993.yaml +0 -104
- data/maps/bgnpcgn-bak-Cyrl-Latn-2007.yaml +0 -184
- data/maps/bgnpcgn-bel-cyrl-latn-1979.yaml +0 -285
- data/maps/bgnpcgn-bul-Cyrl-Latn-1952.yaml +0 -115
- data/maps/bgnpcgn-bul-Cyrl-Latn-2013.yaml +0 -38
- data/maps/bgnpcgn-chn-Hans-Latn-1979.yaml +0 -7456
- data/maps/bgnpcgn-ell-Grek-Latn-1962.yaml +0 -702
- data/maps/bgnpcgn-ell-Grek-Latn-1996.yaml +0 -20
- data/maps/bgnpcgn-jpn-Hrkt-Latn-1976.yaml +0 -257
- data/maps/bgnpcgn-kat-Geor-Latn-1981.yaml +0 -127
- data/maps/bgnpcgn-kat-Geor-Latn-2009.yaml +0 -43
- data/maps/bgnpcgn-kor-Hang-Latn-kn-1945.yaml +0 -253
- data/maps/bgnpcgn-kor-Hang-Latn-rok-2011.yaml +0 -48
- data/maps/bgnpcgn-kor-Kore-Latn-rok-2011.yaml +0 -48
- data/maps/bgnpcgn-mkd-Cyrl-Latn-1981.yaml +0 -159
- data/maps/bgnpcgn-mkd-Cyrl-Latn-2013.yaml +0 -190
- data/maps/bgnpcgn-per-Arab-Latn-1956.yaml +0 -93
- data/maps/bgnpcgn-rus-Cyrl-Latn-1947.yaml +0 -314
- data/maps/bgnpcgn-srp-Cyrl-Latn-2005.yaml +0 -166
- data/maps/bgnpcgn-ukr-Cyrl-Latn-1965.yaml +0 -163
- data/maps/bgnpcgn-ukr-Cyrl-Latn-2019.yaml +0 -208
- data/maps/by-bel-Cyrl-Latn-1998.yaml +0 -168
- data/maps/by-bel-Cyrl-Latn-2007.yaml +0 -115
- data/maps/elot-ell-Grek-Latn-743-1982-tl.yaml +0 -685
- data/maps/elot-ell-Grek-Latn-743-1982-ts.yaml +0 -681
- data/maps/elot-ell-Grek-Latn-743-2001-tl.yaml +0 -20
- data/maps/elot-ell-Grek-Latn-743-2001-ts.yaml +0 -32
- data/maps/ggg-kat-Geor-Latn-2002.yaml +0 -89
- data/maps/gki-bel-cyrl-latn-1992.yaml +0 -33
- data/maps/gki-bel-cyrl-latn-2000.yaml +0 -201
- data/maps/gost-rus-cyrl-latn-16876-71-1983.yaml +0 -186
- data/maps/hk-yue-Hani-Latn-1888.yaml +0 -38497
- data/maps/icao-bel-Cyrl-Latn-9303.yaml +0 -141
- data/maps/icao-bul-Cyrl-Latn-9303.yaml +0 -122
- data/maps/icao-heb-Hebr-Latn-9303.yaml +0 -151
- data/maps/icao-mkd-Cyrl-Latn-9303.yaml +0 -117
- data/maps/icao-per-Arab-Latn-9303.yaml +0 -104
- data/maps/icao-rus-Cyrl-Latn-9303.yaml +0 -118
- data/maps/icao-srp-Cyrl-Latn-9303.yaml +0 -117
- data/maps/icao-ukr-Cyrl-Latn-9303.yaml +0 -120
- data/maps/iso-ell-Grek-Latn-843-1997-t1.yaml +0 -610
- data/maps/iso-ell-Grek-Latn-843-1997-t2.yaml +0 -41
- data/maps/iso-jpn-Hrkt-Latn-3602-1989.yaml +0 -62
- data/maps/iso-rus-Cyrl-Latn-9-1995.yaml +0 -272
- data/maps/iso-tha-Thai-Latn-11940-1998.yaml +0 -109
- data/maps/kp-kor-Hang-Latn-2002.yaml +0 -901
- data/maps/lshk-yue-Hani-Latn-jyutping-1993.yaml +0 -44820
- data/maps/mext-jpn-Hrkt-Latn-1954.yaml +0 -411
- data/maps/moct-kor-Hang-Latn-2000.yaml +0 -803
- data/maps/mofa-jpn-Hrkt-Latn-1989.yaml +0 -541
- data/maps/mvd-bel-Cyrl-Latn-2008.yaml +0 -225
- data/maps/mvd-bel-Cyrl-Latn-2010.yaml +0 -63
- data/maps/mvd-rus-Cyrl-Latn-2008.yaml +0 -110
- data/maps/mvd-rus-Cyrl-Latn-2010.yaml +0 -37
- data/maps/nil-kor-Hang-Hang-jamo.yaml +0 -11193
- data/maps/odni-bel-Cyrl-Latn-2015.yaml +0 -148
- data/maps/odni-bul-Cyrl-Latn-2015.yaml +0 -96
- data/maps/odni-kat-Geor-Latn-2015.yaml +0 -88
- data/maps/odni-rus-Cyrl-Latn-2015.yaml +0 -77
- data/maps/odni-srp-Cyrl-Latn-2015.yaml +0 -129
- data/maps/odni-ukr-Cyrl-Latn-2015.yaml +0 -157
- data/maps/odni-uzb-Cyrl-Latn-2015.yaml +0 -167
- data/maps/royin-tha-Thai-Latn-1939-generic.yaml +0 -90
- data/maps/royin-tha-Thai-Latn-1968.yaml +0 -179
- data/maps/royin-tha-Thai-Latn-1999-chained.yaml +0 -180
- data/maps/royin-tha-Thai-Latn-1999.yaml +0 -76
- data/maps/sac-zho-Hans-Latn-1979.yaml +0 -24759
- data/maps/stategeocadastre-ukr-Cyrl-Latn-1993.yaml +0 -222
- data/maps/ua-ukr-Cyrl-Latn-1996.yaml +0 -193
- data/maps/un-bel-Cyrl-Latn-2007.yaml +0 -114
- data/maps/un-ben-Beng-Latn-2016.yaml +0 -534
- data/maps/un-ell-Grek-Latn-1987-tl.yaml +0 -32
- data/maps/un-ell-Grek-Latn-1987-ts.yaml +0 -20
- data/maps/un-ell-Grek-Latn-phonetic-1987.yaml +0 -780
- data/maps/un-mon-Mong-Latn-2013.yaml +0 -93
- data/maps/un-rus-Cyrl-Latn-1987.yaml +0 -166
- data/maps/un-ukr-cyrl-latn-1998.yaml +0 -30
- data/maps/var-jpn-Hrkt-Latn-hepburn-1886.yaml +0 -406
- data/maps/var-jpn-Hrkt-Latn-hepburn-1954.yaml +0 -386
- data/maps/var-kor-Hang-Latn-mr-1939.yaml +0 -1054
- data/maps/var-kor-Kore-Hang-2013.yaml +0 -59754
- data/maps/var-kor-Kore-Latn-mr-1939.yaml +0 -37
- data/maps/var-tha-Thai-Thai-phonemic.yaml +0 -59
- data/maps/var-tha-Thai-Zsym-ipa.yaml +0 -301
- data/maps/var-zho-Hani-Latn-1979.yaml +0 -38908
- data/spec/interscript/mapping_spec.rb +0 -42
- data/spec/interscript_spec.rb +0 -26
- data/spec/spec_helper.rb +0 -3
@@ -0,0 +1,262 @@
|
|
1
|
+
$main_binding = binding
|
2
|
+
|
3
|
+
class Interscript::Compiler::Ruby < Interscript::Compiler
|
4
|
+
def compile(map, debug: false)
|
5
|
+
@map = map
|
6
|
+
@debug = debug
|
7
|
+
@parallel_trees = {}
|
8
|
+
@parallel_regexps = {}
|
9
|
+
c = "require 'interscript/stdlib'\n"
|
10
|
+
c << "if !defined?(Interscript::Maps); module Interscript; module Maps\n"
|
11
|
+
c << "module Cache; end\n"
|
12
|
+
c << "class Map < Struct.new(:stages, :aliases, :aliases_re); end\n"
|
13
|
+
c << "@maps = Hash.new { |h,id| h[id] = Map.new({},{},{}) }\n"
|
14
|
+
c << "def self.has_map?(map); @maps.include?(map); end\n"
|
15
|
+
c << "def self.add_map_alias(map,name,value) @maps[map].aliases[name] = value; end\n"
|
16
|
+
c << "def self.add_map_alias_re(map,name,value) @maps[map].aliases_re[name] = value; end\n"
|
17
|
+
c << "def self.add_map_stage(map,stage,&block); @maps[map].stages[stage] = block; end\n"
|
18
|
+
c << "def self.get_alias(map,name); @maps[map].aliases[name]; end\n"
|
19
|
+
c << "def self.get_alias_re(map,name); @maps[map].aliases_re[name]; end\n"
|
20
|
+
c << "def self.transliterate(map,string,stage=:main); @maps[map].stages[stage].(string); end\n"
|
21
|
+
c << "end; end; end\n"
|
22
|
+
c
|
23
|
+
|
24
|
+
map.aliases.each do |name, value|
|
25
|
+
val = compile_item(value.data, map, :str)
|
26
|
+
c << "Interscript::Maps.add_map_alias(#{map.name.inspect}, #{name.inspect}, #{val})\n"
|
27
|
+
val = '/'+compile_item(value.data, map, :re).gsub('/', '\\\\/')+'/'
|
28
|
+
c << "Interscript::Maps.add_map_alias_re(#{map.name.inspect}, #{name.inspect}, #{val})\n"
|
29
|
+
end
|
30
|
+
|
31
|
+
map.stages.each do |_, stage|
|
32
|
+
c << compile_rule(stage, @map, true)
|
33
|
+
end
|
34
|
+
@parallel_trees.each do |k,v|
|
35
|
+
c << "Interscript::Maps::Cache::PTREE_#{k} ||= #{v.inspect}\n"
|
36
|
+
end
|
37
|
+
@parallel_regexps.each do |k,v|
|
38
|
+
c << "Interscript::Maps::Cache::PRE_#{k} ||= #{v.inspect}\n"
|
39
|
+
end
|
40
|
+
@code = c
|
41
|
+
end
|
42
|
+
|
43
|
+
def compile_rule(r, map = @map, wrapper = false)
|
44
|
+
c = ""
|
45
|
+
case r
|
46
|
+
when Interscript::Node::Stage
|
47
|
+
c += "Interscript::Maps.add_map_stage \"#{@map.name}\", #{r.name.inspect} do |s|\n"
|
48
|
+
c += "$map_debug ||= []\n" if @debug
|
49
|
+
c += "s = s.dup\n"
|
50
|
+
r.children.each do |t|
|
51
|
+
comp = compile_rule(t, map)
|
52
|
+
c += comp
|
53
|
+
c += %{$map_debug << [s.dup, #{@map.name.to_s.inspect}, #{r.name.to_s.inspect}, #{t.inspect.inspect}, #{comp.inspect}]\n} if @debug
|
54
|
+
end
|
55
|
+
c += "s\n"
|
56
|
+
c += "end\n"
|
57
|
+
when Interscript::Node::Group::Parallel
|
58
|
+
begin
|
59
|
+
# Try to build a tree
|
60
|
+
a = []
|
61
|
+
r.children.each do |i|
|
62
|
+
raise ArgumentError, "Can't parallelize #{i.class}" unless Interscript::Node::Rule::Sub === i
|
63
|
+
raise ArgumentError, "Can't parallelize rules with :before" if i.before
|
64
|
+
raise ArgumentError, "Can't parallelize rules with :after" if i.after
|
65
|
+
raise ArgumentError, "Can't parallelize rules with :not_before" if i.not_before
|
66
|
+
raise ArgumentError, "Can't parallelize rules with :not_after" if i.not_after
|
67
|
+
|
68
|
+
a << [compile_item(i.from, map, :par), compile_item(i.to, map, :parstr)]
|
69
|
+
end
|
70
|
+
ah = a.hash.abs
|
71
|
+
unless @parallel_trees.include? ah
|
72
|
+
tree = Interscript::Stdlib.parallel_replace_compile_tree(a)
|
73
|
+
@parallel_trees[ah] = tree
|
74
|
+
end
|
75
|
+
c += "s = Interscript::Stdlib.parallel_replace_tree(s, Interscript::Maps::Cache::PTREE_#{ah})\n"
|
76
|
+
rescue
|
77
|
+
# Otherwise let's build a megaregexp
|
78
|
+
a = []
|
79
|
+
Interscript::Stdlib.deterministic_sort_by_max_length(r.children).each do |i|
|
80
|
+
raise ArgumentError, "Can't parallelize #{i.class}" unless Interscript::Node::Rule::Sub === i
|
81
|
+
|
82
|
+
a << [build_regexp(i, map), compile_item(i.to, map, :parstr)]
|
83
|
+
end
|
84
|
+
ah = a.hash.abs
|
85
|
+
unless @parallel_regexps.include? ah
|
86
|
+
re = Interscript::Stdlib.parallel_regexp_compile(a)
|
87
|
+
@parallel_regexps[ah] = [re, a.map(&:last)]
|
88
|
+
end
|
89
|
+
c += "s = Interscript::Stdlib.parallel_regexp_gsub(s, *Interscript::Maps::Cache::PRE_#{ah})\n"
|
90
|
+
end
|
91
|
+
when Interscript::Node::Rule::Sub
|
92
|
+
from = "/#{build_regexp(r, map).gsub("/", "\\\\/")}/"
|
93
|
+
if r.to == :upcase
|
94
|
+
to = '&:upcase'
|
95
|
+
else
|
96
|
+
to = compile_item(r.to, map, :str)
|
97
|
+
end
|
98
|
+
c += "s.gsub!(#{from}, #{to})\n"
|
99
|
+
when Interscript::Node::Rule::Funcall
|
100
|
+
c += "s = Interscript::Stdlib::Functions.#{r.name}(s, #{r.kwargs.inspect[1..-2]})\n"
|
101
|
+
when Interscript::Node::Rule::Run
|
102
|
+
if r.stage.map
|
103
|
+
doc = map.dep_aliases[r.stage.map].document
|
104
|
+
stage = doc.imported_stages[r.stage.name]
|
105
|
+
else
|
106
|
+
stage = map.imported_stages[r.stage.name]
|
107
|
+
end
|
108
|
+
c += "s = Interscript::Maps.transliterate(#{stage.doc_name.inspect}, s, #{stage.name.inspect})\n"
|
109
|
+
else
|
110
|
+
raise ArgumentError, "Can't compile unhandled #{r.class}"
|
111
|
+
end
|
112
|
+
c
|
113
|
+
end
|
114
|
+
|
115
|
+
def build_regexp(r, map=@map)
|
116
|
+
from = compile_item(r.from, map, :re)
|
117
|
+
before = compile_item(r.before, map, :re) if r.before
|
118
|
+
after = compile_item(r.after, map, :re) if r.after
|
119
|
+
not_before = compile_item(r.not_before, map, :re) if r.not_before
|
120
|
+
not_after = compile_item(r.not_after, map, :re) if r.not_after
|
121
|
+
|
122
|
+
re = ""
|
123
|
+
re += "(?<=#{before})" if before
|
124
|
+
re += "(?<!#{not_before})" if not_before
|
125
|
+
re += from
|
126
|
+
re += "(?!#{not_after})" if not_after
|
127
|
+
re += "(?=#{after})" if after
|
128
|
+
re
|
129
|
+
end
|
130
|
+
|
131
|
+
def compile_item i, doc=@map, target=nil
|
132
|
+
i = i.first_string if %i[str parstr].include? target
|
133
|
+
i = Interscript::Node::Item.try_convert(i)
|
134
|
+
if target == :parstr
|
135
|
+
parstr = true
|
136
|
+
target = :par
|
137
|
+
end
|
138
|
+
|
139
|
+
out = case i
|
140
|
+
when Interscript::Node::Item::Alias
|
141
|
+
astr = if i.map
|
142
|
+
d = doc.dep_aliases[i.map].document
|
143
|
+
a = d.imported_aliases[i.name]
|
144
|
+
raise ArgumentError, "Alias #{i.name} of #{i.stage.map} not found" unless a
|
145
|
+
"Interscript::Maps.get_alias_ALIASTYPE(#{a.doc_name.inspect}, #{a.name.inspect})"
|
146
|
+
elsif Interscript::Stdlib::ALIASES.include?(i.name)
|
147
|
+
if target != :re && Interscript::Stdlib.re_only_alias?(i.name)
|
148
|
+
raise ArgumentError, "Can't use #{i.name} in a #{target} context"
|
149
|
+
end
|
150
|
+
stdlib_alias = true
|
151
|
+
"Interscript::Stdlib::ALIASES[#{i.name.inspect}]"
|
152
|
+
else
|
153
|
+
a = doc.imported_aliases[i.name]
|
154
|
+
raise ArgumentError, "Alias #{i.name} not found" unless a
|
155
|
+
|
156
|
+
"Interscript::Maps.get_alias_ALIASTYPE(#{a.doc_name.inspect}, #{a.name.inspect})"
|
157
|
+
end
|
158
|
+
|
159
|
+
if target == :str
|
160
|
+
astr = astr.sub("_ALIASTYPE(", "(")
|
161
|
+
elsif target == :re
|
162
|
+
astr = "\#{#{astr.sub("_ALIASTYPE(", "_re(")}}"
|
163
|
+
elsif parstr && stdlib_alias
|
164
|
+
astr = Interscript::Stdlib::ALIASES[i.name]
|
165
|
+
elsif target == :par
|
166
|
+
# raise NotImplementedError, "Can't use aliases in parallel mode yet"
|
167
|
+
astr = Interscript::Stdlib::ALIASES[i.name]
|
168
|
+
end
|
169
|
+
when Interscript::Node::Item::String
|
170
|
+
if target == :str
|
171
|
+
# Replace \1 with \\1, this is weird, but it works!
|
172
|
+
i.data.gsub("\\", "\\\\\\\\").inspect
|
173
|
+
elsif target == :par
|
174
|
+
i.data
|
175
|
+
elsif target == :re
|
176
|
+
Regexp.escape(i.data)
|
177
|
+
end
|
178
|
+
when Interscript::Node::Item::Group
|
179
|
+
if target == :par
|
180
|
+
i.children.map do |j|
|
181
|
+
compile_item(j, doc, target)
|
182
|
+
end.reduce([""]) do |j,k|
|
183
|
+
Array(j).product(Array(k)).map(&:join)
|
184
|
+
end
|
185
|
+
elsif target == :str
|
186
|
+
i.children.map { |j| compile_item(j, doc, target) }.join("+")
|
187
|
+
elsif target == :re
|
188
|
+
i.children.map { |j| compile_item(j, doc, target) }.join
|
189
|
+
end
|
190
|
+
when Interscript::Node::Item::CaptureGroup
|
191
|
+
if target != :re
|
192
|
+
raise ArgumentError, "Can't use a CaptureGroup in a #{target} context"
|
193
|
+
end
|
194
|
+
"(" + compile_item(i.data, doc, target) + ")"
|
195
|
+
when Interscript::Node::Item::Maybe,
|
196
|
+
Interscript::Node::Item::MaybeSome,
|
197
|
+
Interscript::Node::Item::Some
|
198
|
+
|
199
|
+
resuffix = { Interscript::Node::Item::Maybe => "?" ,
|
200
|
+
Interscript::Node::Item::Some => "+" ,
|
201
|
+
Interscript::Node::Item::MaybeSome => "*" }[i.class]
|
202
|
+
|
203
|
+
if target == :par
|
204
|
+
raise ArgumentError, "Can't use a Maybe in a #{target} context"
|
205
|
+
end
|
206
|
+
if Interscript::Node::Item::String === i.data && i.data.data.length != 1
|
207
|
+
"(?:" + compile_item(i.data, doc, target) + ")" + resuffix
|
208
|
+
else
|
209
|
+
compile_item(i.data, doc, target) + resuffix
|
210
|
+
end
|
211
|
+
when Interscript::Node::Item::CaptureRef
|
212
|
+
if target == :par
|
213
|
+
raise ArgumentError, "Can't use CaptureRef in parallel mode"
|
214
|
+
elsif target == :re
|
215
|
+
"\\#{i.id}"
|
216
|
+
elsif target == :str
|
217
|
+
"\"\\\\#{i.id}\""
|
218
|
+
end
|
219
|
+
when Interscript::Node::Item::Any
|
220
|
+
if target == :str
|
221
|
+
raise ArgumentError, "Can't use Any in a string context" # A linter could find this!
|
222
|
+
elsif target == :par
|
223
|
+
i.data.map(&:data)
|
224
|
+
elsif target == :re
|
225
|
+
case i.value
|
226
|
+
when Array
|
227
|
+
data = i.data.map { |j| compile_item(j, doc, target) }
|
228
|
+
"(?:"+data.join("|")+")"
|
229
|
+
when String
|
230
|
+
"[#{Regexp.escape(i.value)}]"
|
231
|
+
when Range
|
232
|
+
"[#{Regexp.escape(i.value.first)}-#{Regexp.escape(i.value.last)}]"
|
233
|
+
end
|
234
|
+
end
|
235
|
+
end
|
236
|
+
end
|
237
|
+
|
238
|
+
def load
|
239
|
+
if !defined?(Interscript::Maps) || !Interscript::Maps.has_map?(@map.name)
|
240
|
+
@map.dependencies.each do |dep|
|
241
|
+
dep = dep.full_name
|
242
|
+
if !defined?(Interscript::Maps) || !Interscript::Maps.has_map?(dep)
|
243
|
+
Interscript.load(dep, compiler: self.class).load
|
244
|
+
end
|
245
|
+
end
|
246
|
+
eval(@code, $main_binding)
|
247
|
+
end
|
248
|
+
end
|
249
|
+
|
250
|
+
def call(str, stage=:main)
|
251
|
+
load
|
252
|
+
Interscript::Maps.transliterate(@map.name, str, stage)
|
253
|
+
end
|
254
|
+
|
255
|
+
def self.read_debug_data
|
256
|
+
$map_debug || []
|
257
|
+
end
|
258
|
+
|
259
|
+
def self.reset_debug_data
|
260
|
+
$map_debug = []
|
261
|
+
end
|
262
|
+
end
|
@@ -0,0 +1,67 @@
|
|
1
|
+
require "yaml"
|
2
|
+
|
3
|
+
module Interscript::DSL
|
4
|
+
@cache = {}
|
5
|
+
def self.parse(map_name)
|
6
|
+
# map name aliases? here may be a place to wrap it
|
7
|
+
|
8
|
+
return @cache[map_name] if @cache[map_name]
|
9
|
+
path = Interscript.locate(map_name)
|
10
|
+
map_name = File.basename(path, ".imp")
|
11
|
+
map_name = File.basename(map_name, ".iml")
|
12
|
+
|
13
|
+
ruby = []
|
14
|
+
yaml = []
|
15
|
+
|
16
|
+
file = File.read(path).split("\n")
|
17
|
+
exc_fname = File.expand_path(path, Dir.pwd)
|
18
|
+
|
19
|
+
md_reading = false
|
20
|
+
md_indent = nil
|
21
|
+
md_inner_indent = nil
|
22
|
+
file.each do |l|
|
23
|
+
if md_reading && l =~ /\A#{md_indent}\}\s*\z/
|
24
|
+
md_reading = false
|
25
|
+
elsif md_reading
|
26
|
+
ruby << ""
|
27
|
+
yaml << l
|
28
|
+
elsif l =~ /\A(\s*)metadata\s*\{\z/
|
29
|
+
md_indent = $1
|
30
|
+
md_reading = true
|
31
|
+
else
|
32
|
+
yaml << ""
|
33
|
+
ruby << l
|
34
|
+
end
|
35
|
+
end
|
36
|
+
raise ArgumentError, "metadata stage isn't terminated" if md_reading
|
37
|
+
ruby, yaml = ruby.join("\n"), yaml.join("\n")
|
38
|
+
|
39
|
+
obj = Interscript::DSL::Document.new(map_name)
|
40
|
+
obj.instance_eval ruby, exc_fname, 1
|
41
|
+
|
42
|
+
yaml = if yaml =~ /\A\s*\z/
|
43
|
+
{}
|
44
|
+
else
|
45
|
+
YAML.load(yaml, exc_fname)
|
46
|
+
end
|
47
|
+
|
48
|
+
md = Interscript::DSL::Metadata.new(yaml: true) do
|
49
|
+
yaml.each do |k,v|
|
50
|
+
public_send(k.to_sym, v)
|
51
|
+
end
|
52
|
+
end
|
53
|
+
obj.node.metadata = md.node
|
54
|
+
|
55
|
+
@cache[map_name] = obj.node
|
56
|
+
end
|
57
|
+
end
|
58
|
+
|
59
|
+
require 'interscript/dsl/symbol_mm'
|
60
|
+
require 'interscript/dsl/items'
|
61
|
+
|
62
|
+
require 'interscript/dsl/document'
|
63
|
+
require 'interscript/dsl/group'
|
64
|
+
require 'interscript/dsl/stage'
|
65
|
+
require 'interscript/dsl/metadata'
|
66
|
+
require 'interscript/dsl/tests'
|
67
|
+
require 'interscript/dsl/aliases'
|
@@ -0,0 +1,23 @@
|
|
1
|
+
class Interscript::DSL::Aliases
|
2
|
+
include Interscript::DSL::Items
|
3
|
+
|
4
|
+
attr_accessor :node
|
5
|
+
|
6
|
+
def initialize(&block)
|
7
|
+
@node = {}
|
8
|
+
self.instance_exec(&block)
|
9
|
+
end
|
10
|
+
|
11
|
+
def def_alias(name, value)
|
12
|
+
if Interscript::Node::Item::Alias === name
|
13
|
+
name = name.name
|
14
|
+
end
|
15
|
+
|
16
|
+
unless Symbol === name
|
17
|
+
raise TypeError, "Alias name must be a Symbol, given #{name.class}"
|
18
|
+
end
|
19
|
+
|
20
|
+
puts "def_alias(#{name.inspect}, #{thing.inspect})" if $DEBUG
|
21
|
+
@node[name] = Interscript::Node::AliasDef.new(name, value)
|
22
|
+
end
|
23
|
+
end
|
@@ -0,0 +1,46 @@
|
|
1
|
+
class Interscript::DSL::Document
|
2
|
+
include Interscript::DSL::SymbolMM
|
3
|
+
|
4
|
+
attr_accessor :node
|
5
|
+
|
6
|
+
def initialize(name = nil, &block)
|
7
|
+
@node = Interscript::Node::Document.new
|
8
|
+
@node.name = name if name
|
9
|
+
self.instance_exec &block if block_given?
|
10
|
+
end
|
11
|
+
|
12
|
+
def metadata(&block)
|
13
|
+
metadata = Interscript::DSL::Metadata.new(&block)
|
14
|
+
@node.metadata = metadata.node
|
15
|
+
end
|
16
|
+
|
17
|
+
def tests(&block)
|
18
|
+
tests = Interscript::DSL::Tests.new(&block)
|
19
|
+
@node.tests = tests.node
|
20
|
+
end
|
21
|
+
|
22
|
+
def aliases(&block)
|
23
|
+
aliases = Interscript::DSL::Aliases.new(&block)
|
24
|
+
@node.aliases = aliases.node
|
25
|
+
@node.aliases.transform_values { |v| v.doc_name = @node.name; v }
|
26
|
+
end
|
27
|
+
|
28
|
+
def dependency(full_name, **kargs)
|
29
|
+
puts "dependency(#{name.inspect}, #{kargs.inspect}" if $DEBUG
|
30
|
+
dep = Interscript::Node::Dependency.new
|
31
|
+
dep.name = kargs[:as]
|
32
|
+
dep.full_name = full_name
|
33
|
+
dep.import = kargs[:import] || false
|
34
|
+
|
35
|
+
dep.document = Interscript::DSL.parse(full_name)
|
36
|
+
@node.dependencies << dep
|
37
|
+
@node.dep_aliases[dep.name] = dep if dep.name
|
38
|
+
end
|
39
|
+
|
40
|
+
def stage(name = :main, &block)
|
41
|
+
puts "stage(#{name}) from #{self.inspect}" if $DEBUG
|
42
|
+
stage = Interscript::DSL::Stage.new(name, &block)
|
43
|
+
stage.node.doc_name = @node.name
|
44
|
+
@node.stages[name] = stage.node
|
45
|
+
end
|
46
|
+
end
|
@@ -0,0 +1,45 @@
|
|
1
|
+
class Interscript::DSL::Group
|
2
|
+
include Interscript::DSL::Items
|
3
|
+
|
4
|
+
attr_accessor :node
|
5
|
+
|
6
|
+
def initialize(&block)
|
7
|
+
@node = Interscript::Node::Group.new
|
8
|
+
self.instance_exec(&block)
|
9
|
+
end
|
10
|
+
|
11
|
+
def run(stage)
|
12
|
+
if stage.class != Interscript::Node::Item::Stage
|
13
|
+
raise TypeError, "I::Node::Item::Stage expected, got #{stage.class}"
|
14
|
+
end
|
15
|
+
@node.children << Interscript::Node::Rule::Run.new(stage)
|
16
|
+
end
|
17
|
+
|
18
|
+
def sub(from, to, **kwargs, &block)
|
19
|
+
puts "sub(#{from.inspect},#{to}, kargs = #{
|
20
|
+
kargs.inspect
|
21
|
+
}) from #{self.inspect}" if $DEBUG
|
22
|
+
|
23
|
+
rule = Interscript::Node::Rule::Sub.new(from, to, **kwargs)
|
24
|
+
@node.children << rule
|
25
|
+
end
|
26
|
+
|
27
|
+
def upcase; :upcase; end
|
28
|
+
|
29
|
+
Interscript::Stdlib.available_functions.each do |fun|
|
30
|
+
define_method fun do |**kwargs|
|
31
|
+
puts "funcall(#{fun}, #{kwargs.inspect}) from #{self.inspect}" if $DEBUG
|
32
|
+
|
33
|
+
rule = Interscript::Node::Rule::Funcall.new(fun, **kwargs)
|
34
|
+
@node.children << rule
|
35
|
+
end
|
36
|
+
end
|
37
|
+
|
38
|
+
def parallel(&block)
|
39
|
+
puts "parallel(#{chars.inspect}) from #{self.inspect}" if $DEBUG
|
40
|
+
group = Interscript::DSL::Group::Parallel.new(&block)
|
41
|
+
@node.children << group.node
|
42
|
+
end
|
43
|
+
end
|
44
|
+
|
45
|
+
require 'interscript/dsl/group/parallel'
|