1 <?xml version="1.0" encoding="UTF-8"?>
2 <xsl:stylesheet version="3.0" xmlns:xsl="http://www.w3.org/1999/XSL/Transform" xmlns:xs="http://www.w3.org/2001/XMLSchema" xmlns:fn="http://www.w3.org/2005/02/xpath-functions" exclude-result-prefixes="xsl xs fn">
3 <!-- Expects a "tokens" in current scope -->
4 <xsl:template name="malreader-peek">
5 <!-- <xsl:message>PEEK <xsl:sequence select=".">
10 <xsl:variable name="context">
12 <xsl:sequence select="tokens/*" />
15 <xsl:for-each select="tokens/token[1]">
16 <xsl:sequence select="."></xsl:sequence>
19 <xsl:sequence select="lvalue"></xsl:sequence>
20 <xsl:sequence select="error"></xsl:sequence>
22 <xsl:for-each select="$context"><xsl:sequence select="." /></xsl:for-each>
26 <xsl:template name="malreader-next">
27 <!-- <xsl:message>NEXT <xsl:sequence select=".">
33 <xsl:variable name="context">
35 <xsl:for-each select="tokens/token[position() != 1]">
36 <xsl:sequence select="." />
40 <xsl:for-each select="tokens/token[1]">
41 <xsl:sequence select="."></xsl:sequence>
44 <xsl:sequence select="lvalue"></xsl:sequence>
45 <xsl:sequence select="error"></xsl:sequence>
47 <xsl:for-each select="$context"><xsl:sequence select="./*" /></xsl:for-each>
50 <xsl:template name="malreader-read_str">
51 <xsl:variable name="context">
52 <input value="{str}"/>
54 <xsl:call-template name="malreader-tokenize"></xsl:call-template>
57 <xsl:for-each select="$context">
58 <xsl:call-template name="malreader-read_form"></xsl:call-template>
60 <!-- <xsl:sequence select="$context" /> -->
63 <xsl:template name="malreader-tokenize">
64 <xsl:analyze-string select="str" regex="[\s,]*(~@|[\[\]{{}}()'`~^@]|"(?:\\.|[^\\"])*"?|;.*|[^\s\[\]{{}}('"`,;)]+)" flags=";j">
65 <xsl:matching-substring>
66 <xsl:variable name="match">
67 <xsl:sequence select="regex-group(1)" />
69 <xsl:if test="string-length($match) > 0">
71 <xsl:when test="starts-with($match, '"')">
73 <xsl:when test="fn:check_string($match)">
74 <token type="error" text="EOF while reading string or invalid escape in string"></token>
76 <xsl:when test="ends-with($match, '"')">
77 <token type="string" text="{fn:process-string(replace($match, '"(.*)"', '$1'))}"> </token>
80 <token type="error" text="EOF while reading string"></token>
84 <xsl:when test="starts-with($match, ':')">
85 <token type="keyword" text="{replace($match, ':(.*)', '$1')}"/>
87 <xsl:when test="starts-with($match, ';')">
88 <!-- ignore comments -->
90 <xsl:when test="starts-with($match, '~@')">
91 <token type="special" text="~@" />
93 <xsl:when test="matches($match, '[\[\]\{\}()''`~^@]')">
94 <token type="special" text="{$match}"></token>
98 <xsl:when test="$match = 'false'">
99 <token type="false"></token>
101 <xsl:when test="$match = 'true'">
102 <token type="true"></token>
104 <xsl:when test="$match = 'nil'">
105 <token type="nil"></token>
107 <xsl:when test="matches($match, '^-?\d+$')">
108 <token type="number" text="{$match}"></token>
111 <token type="symbol" text="{$match}"></token>
117 </xsl:matching-substring>
118 </xsl:analyze-string>
121 <xsl:template name="malreader-read_form">
122 <xsl:variable name="peek">
123 <xsl:call-template name="malreader-peek"></xsl:call-template>
125 <xsl:variable name="next">
126 <xsl:call-template name="malreader-next"></xsl:call-template>
128 <xsl:for-each select="$peek">
130 <xsl:when test="value/token/@type = 'error'">
131 <error><malval kind="error"><xsl:value-of select="value/token/@text"/></malval></error>
133 <xsl:when test="contains('([{', value/token/@text) and value/token/@type = 'special'">
134 <xsl:variable name="next">
135 <xsl:call-template name="malreader-next"></xsl:call-template>
137 <xsl:for-each select="$next">
138 <xsl:variable name="listkind">
139 <xsl:value-of select="value/token/@text" /> <!-- listkind [/(/{ -->
141 <xsl:call-template name="malreader-read_list"><xsl:with-param name="listkind" select="$listkind"/></xsl:call-template>
144 <xsl:when test="value/token/@text = "'" and value/token/@type = 'special'">
145 <xsl:for-each select="$next">
146 <xsl:variable name="inner">
147 <xsl:variable name="ctx">
148 <xsl:sequence select="tokens"/>
150 <xsl:for-each select="$ctx">
151 <xsl:call-template name="malreader-read_form"></xsl:call-template>
154 <xsl:for-each select="$inner">
158 <malval kind="symbol" value="quote"></malval>
159 <xsl:sequence select="/value/malval" />
163 <xsl:sequence select="tokens"/>
164 <xsl:sequence select="error"/>
166 <xsl:sequence select="lvalue"/> <!-- preserve previous list (if any) -->
169 <xsl:when test="value/token/@text = '`' and value/token/@type = 'special'">
170 <xsl:for-each select="$next">
171 <xsl:variable name="inner">
172 <xsl:variable name="ctx">
173 <xsl:sequence select="tokens"/>
175 <xsl:for-each select="$ctx">
176 <xsl:call-template name="malreader-read_form"></xsl:call-template>
179 <xsl:for-each select="$inner">
183 <malval kind="symbol" value="quasiquote"></malval>
184 <xsl:sequence select="/value/malval" />
188 <xsl:sequence select="tokens"/>
189 <xsl:sequence select="error"/>
191 <xsl:sequence select="lvalue"/> <!-- preserve previous list (if any) -->
194 <xsl:when test="value/token/@text = '~' and value/token/@type = 'special'">
195 <xsl:for-each select="$next">
196 <xsl:variable name="inner">
197 <xsl:variable name="ctx">
198 <xsl:sequence select="tokens"/>
200 <xsl:for-each select="$ctx">
201 <xsl:call-template name="malreader-read_form"></xsl:call-template>
204 <xsl:for-each select="$inner">
208 <malval kind="symbol" value="unquote"></malval>
209 <xsl:sequence select="/value/malval" />
213 <xsl:sequence select="tokens"/>
214 <xsl:sequence select="error"/>
216 <xsl:sequence select="lvalue"/> <!-- preserve previous list (if any) -->
219 <xsl:when test="value/token/@text = '~@' and value/token/@type = 'special'">
220 <xsl:for-each select="$next">
221 <xsl:variable name="inner">
222 <xsl:variable name="ctx">
223 <xsl:sequence select="tokens"/>
225 <xsl:for-each select="$ctx">
226 <xsl:call-template name="malreader-read_form"></xsl:call-template>
229 <xsl:for-each select="$inner">
233 <malval kind="symbol" value="splice-unquote"></malval>
234 <xsl:sequence select="/value/malval" />
238 <xsl:sequence select="tokens"/>
239 <xsl:sequence select="error"/>
241 <xsl:sequence select="lvalue"/> <!-- preserve previous list (if any) -->
244 <xsl:when test="value/token/@text = '@' and value/token/@type = 'special'">
245 <xsl:for-each select="$next">
246 <xsl:variable name="inner">
247 <xsl:variable name="ctx">
248 <xsl:sequence select="tokens"/>
250 <xsl:for-each select="$ctx">
251 <xsl:call-template name="malreader-read_form"></xsl:call-template>
254 <xsl:for-each select="$inner">
258 <malval kind="symbol" value="deref"></malval>
259 <xsl:sequence select="/value/malval" />
263 <xsl:sequence select="tokens"/>
264 <xsl:sequence select="error"/>
266 <xsl:sequence select="lvalue"/> <!-- preserve previous list (if any) -->
269 <xsl:when test="value/token/@text = '^' and value/token/@type = 'special'">
270 <xsl:for-each select="$next">
271 <xsl:variable name="meta">
272 <xsl:variable name="ctx">
273 <xsl:sequence select="tokens"/>
275 <xsl:for-each select="$ctx">
276 <xsl:call-template name="malreader-read_form"></xsl:call-template>
279 <xsl:variable name="form">
280 <xsl:for-each select="$meta">
281 <xsl:variable name="ctx">
282 <xsl:sequence select="tokens"/>
284 <xsl:for-each select="$ctx">
285 <xsl:call-template name="malreader-read_form"></xsl:call-template>
289 <xsl:for-each select="$form">
293 <malval kind="symbol" value="with-meta"></malval>
294 <xsl:sequence select="/value/malval" />
295 <xsl:for-each select="$meta">
296 <xsl:sequence select="/value/malval"/>
301 <xsl:sequence select="tokens"/>
302 <xsl:sequence select="error"/>
304 <xsl:sequence select="lvalue"/> <!-- preserve previous list (if any) -->
308 <xsl:call-template name="malreader-read_atom"></xsl:call-template>
314 <xsl:template name="malreader-read_list">
315 <xsl:param name="listkind" as="xs:string" />
316 <xsl:variable name="prev_lvalue">
317 <xsl:copy-of select="lvalue"/>
319 <xsl:variable name="value">
320 <xsl:variable name="ctx">
321 <xsl:sequence select="tokens"/>
323 <xsl:for-each select="$ctx">
324 <xsl:call-template name="malreader-read_list_helper"><xsl:with-param name="listkind" select="$listkind"/></xsl:call-template>
327 <xsl:for-each select="$value">
328 <xsl:sequence select="tokens" />
329 <xsl:variable name="value">
331 <malval kind="{fn:list-kind($listkind)}">
332 <xsl:sequence select="lvalue[1]"/>
337 <xsl:when test="$listkind = '{'">
339 <xsl:when test="count($value/value/malval/lvalue/malval) mod 2 = 1">
340 <error><malval kind="error">Odd number of values to hash</malval></error>
343 <xsl:sequence select="$value"/>
348 <xsl:sequence select="$value"/>
349 <xsl:sequence select="error" />
352 <xsl:sequence select="$prev_lvalue"/>
356 <xsl:template name="malreader-read_list_helper">
357 <xsl:param name="listkind" as="xs:string" />
359 <xsl:when test="count(tokens/*) > 0">
360 <xsl:variable name="peek">
361 <xsl:call-template name="malreader-peek"></xsl:call-template>
363 <xsl:for-each select="$peek">
365 <xsl:when test="value/token/@text = fn:list-ender($listkind) and value/token/@type = 'special'">
367 <xsl:variable name="next">
368 <xsl:call-template name="malreader-next"></xsl:call-template>
370 <xsl:sequence select="lvalue" />
371 <xsl:sequence select="$next/tokens" />
372 <xsl:sequence select="error" />
375 <xsl:variable name="form">
376 <xsl:call-template name="malreader-read_form"></xsl:call-template>
378 <xsl:variable name="context">
379 <xsl:for-each select="$form">
380 <!-- <xsl:message>READ_FORM <xsl:sequence select=".">
385 <xsl:sequence select="tokens" />
387 <xsl:for-each select="lvalue/malval"><xsl:sequence select="." /></xsl:for-each>
388 <xsl:for-each select="value/malval"><xsl:sequence select="."/></xsl:for-each>
392 <xsl:for-each select="$context">
393 <xsl:call-template name="malreader-read_list_helper"><xsl:with-param name="listkind" select="$listkind"/></xsl:call-template>
397 <xsl:sequence select="lvalue"/>
401 <error><malval kind="error">EOF while reading list</malval></error>
406 <xsl:template name="malreader-read_atom">
407 <xsl:variable name="next">
408 <xsl:call-template name="malreader-next"></xsl:call-template>
410 <xsl:for-each select="$next">
411 <xsl:sequence select="tokens"/>
412 <xsl:sequence select="lvalue"/>
413 <xsl:sequence select="error"/>
415 <xsl:when test="value/token/@type = 'number'">
417 <malval kind="number" value="{value/token/@text}" />
420 <xsl:when test="value/token/@type = 'symbol'">
422 <malval kind="symbol" value="{value/token/@text}" />
425 <xsl:when test="value/token/@type = 'string'">
427 <malval kind="string" value="{value/token/@text}" />
430 <xsl:when test="value/token/@type = 'keyword'">
432 <malval kind="keyword" value="{value/token/@text}" />
435 <xsl:when test="value/token/@type = 'true'">
437 <malval kind="true"/>
440 <xsl:when test="value/token/@type = 'false'">
442 <malval kind="false"/>
445 <xsl:when test="value/token/@type = 'nil'">
452 <malval kind="error"><xsl:sequence select="value"/></malval>
459 <xsl:function name="fn:check_string" as="xs:boolean">
460 <xsl:param name="str" as="xs:string" />
461 <xsl:sequence select="$str = '"' or matches($str, '[^\\](\\[^n\\"]|\\(\\\\)*"$)')" />
464 <xsl:function name="fn:process-string" as="xs:string">
465 <xsl:param name="str" as="xs:string" />
466 <xsl:sequence select="replace(replace($str, '([^\\]|^)\\n', '$1 '), '\\([\\"])', '$1')" />
469 <xsl:function name="fn:list-ender" as="xs:string">
470 <xsl:param name="str" as="xs:string" />
472 <xsl:when test="$str = '('"><xsl:sequence select="')'" /></xsl:when>
473 <xsl:when test="$str = '['"><xsl:sequence select="']'" /></xsl:when>
474 <xsl:otherwise><xsl:sequence select="'}'" /></xsl:otherwise>
478 <xsl:function name="fn:list-kind" as="xs:string">
479 <xsl:param name="str" as="xs:string" />
481 <xsl:when test="$str = '('"><xsl:sequence select="'list'" /></xsl:when>
482 <xsl:when test="$str = '['"><xsl:sequence select="'vector'" /></xsl:when>
483 <xsl:otherwise><xsl:sequence select="'hash'" /></xsl:otherwise>