-
Notifications
You must be signed in to change notification settings - Fork 2
/
Copy pathtools_edn.pl
140 lines (110 loc) · 4.12 KB
/
tools_edn.pl
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
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
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
:- module(tools_edn,[read_edn/2]).
/* Prolog module to read in EDN files */
/* (c) 2020 Michael Leuschel */
/* under construction */
read_edn(Filename,EdnTerm) :-
open(Filename,read,S),
read_file(S,Content),
close(S),
parse_edn(Content,EdnTerm).
read_file(Stream,Out) :-
get_code(Stream,Code),
read_file(Code,Stream,Out).
read_file(-1,_Stream,[]) :- !.
read_file(Code,Stream,[Code|L]) :-
read_file(Stream,L).
parse_edn(SourceCodes, EdnTerm) :-
reset_parser,
top_level_val(EdnTerm,SourceCodes,[]).
top_level_val(V) --> edn_val(V),ws.
edn_val(Nr) --> edn_number(Nr),!.
edn_val(true) --> "true",!.
edn_val(false) --> "false",!.
edn_val(nil) --> "nil",!.
edn_val(string(S)) --> [34],any_chars_but_quotation(S),[34],!.
edn_val(vector(V)) --> "[", !, edn_val_list(0'],V).
edn_val(list(V)) --> "(", !, edn_val_list(0'),V).
edn_val(set(V)) --> "#{", !, edn_val_list(0'},V).
edn_val(map(V)) --> "{", !, edn_pair_list(V).
edn_val(keyword(KeyW)) --> ":", id(L), {atom_codes(KeyW,L)},!.
edn_val(_) --> print_error('Not EDN value').
% list of pairs inside map, also parsing ending brace
edn_pair_list([]) --> "}",!.
edn_pair_list(L)--> " ", !, edn_pair_list(L).
edn_pair_list(L) --> newline, !, edn_pair_list(L).
edn_pair_list([H|T]) --> edn_pair(H), !, edn_pair_list(T).
edn_pair_list(_) --> print_error('Not EDN map').
edn_pair(Key/Val) -->
edn_val(Key), !,
%{print_msg(key(Key))},
real_ws, edn_val(Val),
{print_msg(pair(Key,Val))},
ws.
% list of values inside a vector, also parsing ending square bracket
edn_val_list(EndSymbol,[]) --> [EndSymbol],!.
edn_val_list(EndSymbol,L) --> " ", !, edn_val_list(EndSymbol,L).
edn_val_list(EndSymbol,L) --> newline, !, edn_val_list(EndSymbol,L).
edn_val_list(EndSymbol,[H|T]) --> edn_val(H),!, edn_val_list(EndSymbol,T).
edn_val_list(_EndSymbol,_) --> print_error('Not EDN vector/list/set').
% identifier
id(ID) --> "'",!,id2(ID),"'".
id([H|T]) --> alpha(H), id2(T).
id2([H|T]) --> alphadigit(H),!,id2(T).
id2([]) --> [].
% EDN number; TO DO: floats
edn_number(MN) --> "-",digit(D),!, edn_nr2(D,N), {MN is -N}.
edn_number(N) --> digit(D),!, edn_nr2(D,N).
edn_nr2(D,N) --> digit(D2), {Acc is D*10+D2}, edn_nr2(Acc,N).
edn_nr2(A,A) --> "".
% a few utilities:
alpha(X) --> [X],{X>=97, X=<122}.
alphadigit(X) --> [X], ({X>=97,X=<122} ; {X>=65, X=<90} ; {X=45} ; {X=95} ; {X>=48, X=<57}).
digit(D) --> [X],{X>=48, X=<57, D is X-48}.
%:- assert_must_succeed(is_octal_digit(0'7,7)).
is_octal_digit(X,Nr) :- X>=48, X=<55, Nr is X-48.
%:- assert_must_succeed(is_hex_digit(0'f,15)).
%:- assert_must_succeed(is_hex_digit(0'F,15)).
%:- assert_must_succeed(is_hex_digit(0'9,9)).
is_hex_digit(X,Nr) :- X>=48, X=<57, Nr is X-48.
is_hex_digit(X,Nr) :- X>=65, X=<70, Nr is X-55. % upper-case A-F
is_hex_digit(X,Nr) :- X>=97, X=<102, Nr is X-87. % lower-case a-f
is_lowcase(X) :- X>=97, X=<122.
% \X --> new char
escape_conversion(39,39). % single quote '
escape_conversion(34,34). % double quote "
escape_conversion(92,92).
escape_conversion(0'n,10).
escape_conversion(0't,9).
escape_conversion(0'b,8).
escape_conversion(0'r,13).
% possible chars in strings; TO DO: deal with escaping
any_chars_but_quotation([C|T]) --> [C], {C \= 34}, any_chars_but_quotation(T).
any_chars_but_quotation([]) --> "".
% whitespaces
ws --> " ", !, ws.
ws --> newline, !, ws.
ws --> ",", ws. % EDN allows commas everywhere as whitespace
ws --> "".
% non-optional whitespace:
real_ws --> " ", !, ws.
real_ws --> ",", !, ws.
real_ws --> newline, !, ws.
real_ws --> print_error('Expecting whitespace').
% processing line numbers and newlines
newline --> "\n", {inc_line_counter}.
reset_parser :- retractall(line_counter(_)), assert(line_counter(1)).
:- dynamic line_counter/1.
line_counter(1).
inc_line_counter :-
retract(line_counter(N)),
N1 is N+1, assert(line_counter(N1)).
% printing message with position info
print_msg(Msg) :-
line_counter(LineNr),
format(user_output,'Line: ~w, ~w~n',[LineNr,Msg]).
% printing error with position info and failing
print_error(Error,L,_) :-
(L = [LH|_] -> Next = [LH] ; Next= "EOF"),
line_counter(LineNr),
format(user_error,'~n! Line: ~w, char: ~s~n! ~w~n',[LineNr,Next,Error]),
fail.