node-map-generation.scm
1 | ;;;; Copyright (C) 2020 Julien Lepiller <julien@lepiller.eu> |
2 | ;;;; |
3 | ;;;; This library is free software; you can redistribute it and/or |
4 | ;;;; modify it under the terms of the GNU Lesser General Public |
5 | ;;;; License as published by the Free Software Foundation; either |
6 | ;;;; version 3 of the License, or (at your option) any later version. |
7 | ;;;; |
8 | ;;;; This library is distributed in the hope that it will be useful, |
9 | ;;;; but WITHOUT ANY WARRANTY; without even the implied warranty of |
10 | ;;;; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU |
11 | ;;;; Lesser General Public License for more details. |
12 | ;;;; |
13 | ;;;; You should have received a copy of the GNU Lesser General Public |
14 | ;;;; License along with this library; if not, write to the Free Software |
15 | ;;;; Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA |
16 | ;;;; |
17 | |
18 | (define-module (jsonld node-map-generation) |
19 | #:use-module (jsonld json) |
20 | #:export (get-node-map-generation)) |
21 | |
22 | (define (contains-json? array json) |
23 | (when (equal? array #f) |
24 | (set! array #())) |
25 | (unless (json-array? array) |
26 | (set! array `#(,array))) |
27 | (not (null? (filter (lambda (o) (same-json? o json)) (array->list array))))) |
28 | |
29 | (define (add-to-list array element) |
30 | (when (equal? array #f) |
31 | (set! array #())) |
32 | (unless (json-array? array) |
33 | (set! array `#(,array))) |
34 | (list->array 1 (append |
35 | (array->list array) |
36 | (list element)))) |
37 | |
38 | (define (append-to-list a1 a2) |
39 | (when (equal? a1 #f) |
40 | (set! a1 #())) |
41 | (unless (json-array? a1) |
42 | (set! a1 `#(,a1))) |
43 | (when (equal? a2 #f) |
44 | (set! a2 #())) |
45 | (unless (json-array? a2) |
46 | (set! a2 `#(,a2))) |
47 | (list->array 1 (append |
48 | (array->list a1) |
49 | (array->list a2)))) |
50 | |
51 | (define (add-if-not-in object property element) |
52 | (let ((array (assoc-ref object property))) |
53 | (if (contains-json? array element) |
54 | object |
55 | (alist-set |
56 | object property |
57 | (add-to-list array element))))) |
58 | |
59 | (define (append-if-not-in object property elements) |
60 | (if (json-array? elements) |
61 | (set! elements (array->list elements)) |
62 | (set! elements (list elements))) |
63 | (let loop ((elements elements) (object object)) |
64 | (if (null? elements) |
65 | object |
66 | (loop (cdr elements) (add-if-not-in object property (car elements)))))) |
67 | |
68 | (define (get-node-map-generation generate-blank-node) |
69 | (define* (node-map-generation element node-map |
70 | #:key (active-graph "@default") |
71 | (active-subject #nil) |
72 | (active-property #nil) |
73 | (lst #nil)) |
74 | ;; 1 |
75 | (if (json-array? element) |
76 | (list->array 1 |
77 | (map |
78 | (lambda (item) |
79 | (let ((res (node-map-generation item node-map |
80 | #:active-graph active-graph |
81 | #:active-subject active-subject |
82 | #:active-property active-property |
83 | #:lst lst))) |
84 | (set! node-map (assoc-ref res "node-map")) |
85 | (set! lst (assoc-ref res "list")))) |
86 | (array->list element))) |
87 | ;; 2: otherwise |
88 | (let* ((graph (or (assoc-ref node-map active-graph) '())) |
89 | (subject-node (if (equal? active-subject #nil) |
90 | '() |
91 | (or (assoc-ref graph active-subject) '()))) |
92 | (types (assoc-ref element "@type")) |
93 | (types (if (json-array? types) |
94 | (array->list types) |
95 | (if types (list types) '()))) |
96 | ;; 3 |
97 | (types |
98 | (map |
99 | (lambda (item) |
100 | (if (blank-node? item) |
101 | (generate-blank-node item) |
102 | item)) |
103 | types))) |
104 | (unless (null? types) |
105 | (if (json-array? (assoc-ref element "@type")) |
106 | (set! element (alist-set element "@type" (list->array 1 types))) |
107 | (set! element (alist-set element "@type" (car types))))) |
108 | ;; 4 |
109 | (when (json-has-key? element "@value") |
110 | (if (equal? lst #nil) |
111 | (begin |
112 | ;; 4.1.1 |
113 | (unless (json-has-key? subject-node active-property) |
114 | (set! subject-node |
115 | (alist-set subject-node active-property `#(,element)))) |
116 | ;; 4.1.2 |
117 | (set! subject-node |
118 | (add-if-not-in subject-node active-property element))) |
119 | ;; 4.2 |
120 | (set! lst |
121 | (alist-set lst "@list" |
122 | (add-to-list (assoc-ref lst "@list") element))))) |
123 | ;; 5 |
124 | (when (json-has-key? element "@list") |
125 | (let ((result `(("@list" . #())))) |
126 | ;; 5.2 |
127 | (let ((res (node-map-generation |
128 | (assoc-ref element "@list") node-map |
129 | #:active-graph active-graph |
130 | #:active-subject active-subject |
131 | #:active-property active-property |
132 | #:lst result))) |
133 | (set! result (assoc-ref res "list")) |
134 | (set! node-map (assoc-ref res "node-map")) |
135 | (set! graph (assoc-ref node-map active-graph)) |
136 | (set! subject-node (or (assoc-ref graph active-subject) '())) |
137 | (if (equal? lst #nil) |
138 | ;; 5.3 |
139 | (begin |
140 | (set! subject-node |
141 | (alist-set |
142 | subject-node |
143 | active-property |
144 | (add-to-list (assoc-ref subject-node active-property) |
145 | result))) |
146 | (set! graph (alist-set graph active-subject subject-node)) |
147 | (set! node-map (alist-set node-map active-graph graph))) |
148 | (set! lst |
149 | (alist-set lst "@list" |
150 | (add-to-list (assoc-ref lst "@list") |
151 | result))))))) |
152 | ;; 6 |
153 | (when (node-object? element) |
154 | ;; 6.1 |
155 | (let ((id (if (json-has-key? element "@id") |
156 | (if (blank-node? (assoc-ref element "@id")) |
157 | (generate-blank-node (assoc-ref element "@id")) |
158 | (assoc-ref element "@id")) |
159 | (generate-blank-node #nil)))) |
160 | ;; 6.3 |
161 | (unless (or (json-has-key? graph id) (not id)) |
162 | (set! graph |
163 | (alist-set graph id `(("@id" . ,id)))) |
164 | (set! node-map (alist-set node-map active-graph graph))) |
165 | ;; 6.4 |
166 | (let ((node (or (assoc-ref graph id) '()))) |
167 | (cond |
168 | ;; 6.5 |
169 | ((json-object? active-subject) |
170 | (if (json-has-key? node active-property) |
171 | ;; 6.5.2 |
172 | (set! node |
173 | (add-if-not-in node active-property active-subject)) |
174 | ;; 6.5.1 |
175 | (set! node |
176 | (alist-set node active-property `#(,active-subject)))) |
177 | (when id |
178 | (set! graph (alist-set graph id node))) |
179 | (set! node-map (alist-set node-map active-graph graph))) |
180 | ;; 6.6 |
181 | ((not (equal? active-property #nil)) |
182 | ;; 6.6.1 |
183 | (let ((reference `(("@id" . ,id)))) |
184 | (if (equal? lst #nil) |
185 | ;; 6.6.2 |
186 | (begin |
187 | (if (json-has-key? subject-node active-property) |
188 | ;; 6.6.2.2 |
189 | (set! subject-node |
190 | (add-if-not-in subject-node active-property reference)) |
191 | ;; 6.6.2.1 |
192 | (set! subject-node |
193 | (alist-set subject-node active-property `#(,reference)))) |
194 | (set! graph (alist-set graph active-subject subject-node)) |
195 | (set! node-map (alist-set node-map active-graph graph))) |
196 | ;; 6.6.3 |
197 | (set! lst |
198 | (alist-set lst "@list" |
199 | (add-to-list (assoc-ref lst "@list") |
200 | reference))))))) |
201 | ;; 6.7 |
202 | (when (json-has-key? element "@type") |
203 | (set! node |
204 | (append-if-not-in node "@type" (assoc-ref element "@type"))) |
205 | (when id |
206 | (set! graph (alist-set graph id node))) |
207 | (set! node-map (alist-set node-map active-graph graph))) |
208 | ;; 6.8 |
209 | (when (json-has-key? element "@index") |
210 | (when (json-has-key? node "@index") |
211 | (throw 'conflicting-indexes)) |
212 | (set! node |
213 | (alist-set node "@index" (assoc-ref element "@index"))) |
214 | (unless (string? id) |
215 | (throw 'eee)) |
216 | (set! graph (alist-set graph id node)) |
217 | (set! node-map (alist-set node-map active-graph graph)) |
218 | (set! element (alist-remove element "@index"))) |
219 | ;; 6.9 |
220 | (when (json-has-key? element "@reverse") |
221 | ;; 6.9.1 |
222 | (let ((referenced-node `(("@id" . ,id))) |
223 | ;; 6.9.2 |
224 | (reverse-map (assoc-ref element "@reverse"))) |
225 | ;; 6.9.3 |
226 | (for-each-pair |
227 | (lambda (property values) |
228 | (for-each |
229 | (lambda (value) |
230 | (let ((res (node-map-generation |
231 | value node-map |
232 | #:active-graph active-graph |
233 | #:active-subject referenced-node |
234 | #:active-property property))) |
235 | (set! node-map (assoc-ref res "node-map")) |
236 | (set! graph (assoc-ref node-map active-graph)) |
237 | (set! subject-node (or (assoc-ref graph active-subject) |
238 | '())) |
239 | (when id |
240 | (set! node (assoc-ref graph id))))) |
241 | (array->list values))) |
242 | reverse-map) |
243 | ;; 6.9.4 |
244 | (set! element (alist-remove element "@reverse")))) |
245 | ;; 6.10 |
246 | (when (json-has-key? element "@graph") |
247 | (let ((res (node-map-generation |
248 | (assoc-ref element "@graph") |
249 | node-map |
250 | #:active-graph id))) |
251 | (set! node-map (assoc-ref res "node-map")) |
252 | (set! graph (assoc-ref node-map active-graph)) |
253 | (set! subject-node (or (assoc-ref graph active-subject) '())) |
254 | (when id |
255 | (set! node (assoc-ref graph id)))) |
256 | (set! element (alist-remove element "@graph"))) |
257 | ;; 6.11 |
258 | (when (json-has-key? element "@included") |
259 | (let ((res (node-map-generation |
260 | (assoc-ref element "@included") |
261 | node-map |
262 | #:active-graph active-graph))) |
263 | (set! node-map (assoc-ref res "node-map")) |
264 | (set! graph (assoc-ref node-map active-graph)) |
265 | (set! subject-node (or (assoc-ref graph active-subject) '())) |
266 | (when id |
267 | (set! node (assoc-ref graph id)))) |
268 | (set! element (alist-remove element "@included"))) |
269 | ;; 6.12 |
270 | (for-each-pair |
271 | (lambda (property value) |
272 | ;; 6.12.1 |
273 | (when (blank-node? property) |
274 | (set! property (generate-blank-node property))) |
275 | ;; 6.12.2 |
276 | (unless (json-has-key? node property) |
277 | (set! node (alist-set node property #())) |
278 | (when id |
279 | (set! graph (alist-set graph id node))) |
280 | (set! node-map (alist-set node-map active-graph graph))) |
281 | ;; 6.12.3 |
282 | (let ((res (node-map-generation |
283 | value node-map #:active-graph active-graph |
284 | #:active-subject id |
285 | #:active-property property))) |
286 | (set! node-map (assoc-ref res "node-map")) |
287 | (set! graph (assoc-ref node-map active-graph)) |
288 | (set! subject-node (or (assoc-ref graph active-subject) '())) |
289 | (when id |
290 | (set! node (assoc-ref graph id))))) |
291 | (alist-sort-by-key element)) |
292 | (when id |
293 | (set! graph (alist-set graph id node)))))) |
294 | (when (string? active-subject) |
295 | (set! graph (alist-set graph active-subject subject-node))) |
296 | (set! node-map (alist-set node-map active-graph graph)))) |
297 | ;; The algorithm returns nothing, but may have modified these two references |
298 | `(("node-map" . ,node-map) ("list" . ,lst))) |
299 | node-map-generation) |
300 |