@hydrofoil/talos-core 0.1.0

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.
package/CHANGELOG.md ADDED
@@ -0,0 +1,7 @@
1
+ # @hydrofoil/talos-core
2
+
3
+ ## 0.1.0
4
+
5
+ ### Minor Changes
6
+
7
+ - 085a4a6: Extracted core functionality of loading resource from file system to a new library `@hydrofoil/talos-core`
package/LICENSE ADDED
@@ -0,0 +1,21 @@
1
+ MIT License
2
+
3
+ Copyright (c) 2022 hypermedia.app
4
+
5
+ Permission is hereby granted, free of charge, to any person obtaining a copy
6
+ of this software and associated documentation files (the "Software"), to deal
7
+ in the Software without restriction, including without limitation the rights
8
+ to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
9
+ copies of the Software, and to permit persons to whom the Software is
10
+ furnished to do so, subject to the following conditions:
11
+
12
+ The above copyright notice and this permission notice shall be included in all
13
+ copies or substantial portions of the Software.
14
+
15
+ THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
16
+ IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
17
+ FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
18
+ AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
19
+ LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
20
+ OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
21
+ SOFTWARE.
package/index.d.ts ADDED
@@ -0,0 +1,3 @@
1
+ import { DatasetCore } from 'rdf-js';
2
+ export { talosNs as ns } from './lib/ns.js';
3
+ export declare function fromDirectories(directories: string[], api: string): Promise<DatasetCore>;
package/index.js ADDED
@@ -0,0 +1,90 @@
1
+ import path from 'path';
2
+ import fs from 'fs';
3
+ import walk from '@fcostarodrigo/walk';
4
+ import $rdf from '@zazuko/env';
5
+ import deleteMatch from 'rdf-dataset-ext/deleteMatch.js';
6
+ import addAll from 'rdf-dataset-ext/addAll.js';
7
+ import log from './lib/log.js';
8
+ import { getPatchedStream } from './lib/fileStream.js';
9
+ import { optionsFromPrefixes } from './lib/prefixHandler.js';
10
+ import { talosNs } from './lib/ns.js';
11
+ export { talosNs as ns } from './lib/ns.js';
12
+ export async function fromDirectories(directories, api) {
13
+ const validDirs = directories.filter(isValidDir);
14
+ const dataset = await validDirs.reduce(toGraphs(api), Promise.resolve($rdf.dataset()));
15
+ setDefaultAction(dataset);
16
+ return dataset;
17
+ }
18
+ function setDefaultAction(dataset) {
19
+ $rdf.clownface({ dataset, graph: talosNs.resources })
20
+ .has(talosNs.action, talosNs.default)
21
+ .deleteOut(talosNs.action, talosNs.default)
22
+ .addOut(talosNs.action, talosNs.overwrite);
23
+ }
24
+ function toGraphs(api) {
25
+ return async function (previousPromise, dir) {
26
+ let previous = await previousPromise;
27
+ const dataset = $rdf.dataset();
28
+ log.debug('Processing dir %s', dir);
29
+ for await (const file of walk(dir)) {
30
+ const relative = path.relative(dir, file);
31
+ const resourcePath = path.relative(dir, file)
32
+ .replace(/\.[^.]+$/, '')
33
+ .replace(/\/?index$/, '');
34
+ const url = resourcePath === ''
35
+ ? encodeURI(api)
36
+ : encodeURI(`${api}/${resourcePath}`);
37
+ const parserStream = getPatchedStream(file, dir, api, url);
38
+ if (!parserStream) {
39
+ continue;
40
+ }
41
+ log.debug('Parsing %s', relative);
42
+ const parsedResourceOptions = {};
43
+ parserStream.on('prefix', optionsFromPrefixes(parsedResourceOptions));
44
+ const resources = $rdf.termSet();
45
+ const resourceOptions = $rdf.clownface({ dataset: previous, graph: talosNs.resources });
46
+ try {
47
+ for await (const { subject, predicate, object, ...quad } of parserStream) {
48
+ const graph = quad.graph.equals($rdf.defaultGraph()) ? $rdf.namedNode(url) : quad.graph;
49
+ if (!resources.has(graph)) {
50
+ log.debug('Parsed resource %s', graph.value);
51
+ }
52
+ resources.add(graph);
53
+ dataset.add($rdf.quad(subject, predicate, object, graph));
54
+ }
55
+ }
56
+ catch (e) {
57
+ if (e instanceof Error) {
58
+ log('Failed to parse %s: %s', relative, e.message);
59
+ }
60
+ }
61
+ resources.forEach(id => {
62
+ const action = parsedResourceOptions.existingResource || 'default';
63
+ const environmentRepresentation = parsedResourceOptions.environmentRepresentation || 'default';
64
+ const options = resourceOptions.node(id);
65
+ options
66
+ .deleteOut(talosNs.action, talosNs.default)
67
+ .addOut(talosNs.action, talosNs(action))
68
+ .deleteOut(talosNs.environmentRepresentation, talosNs.default)
69
+ .addOut(talosNs.environmentRepresentation, talosNs(environmentRepresentation));
70
+ if (options.has(talosNs.environmentRepresentation, talosNs.replace).terms.length) {
71
+ previous = deleteMatch(previous, undefined, undefined, undefined, id);
72
+ }
73
+ });
74
+ }
75
+ addAll(previous, dataset);
76
+ return previous;
77
+ };
78
+ }
79
+ function isValidDir(dir) {
80
+ if (!fs.existsSync(dir)) {
81
+ log('Skipping path %s which does not exist', dir);
82
+ return false;
83
+ }
84
+ if (!fs.statSync(dir).isDirectory()) {
85
+ log('Skipping path %s which is not a directory', dir);
86
+ return false;
87
+ }
88
+ return true;
89
+ }
90
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyJpbmRleC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLElBQUksTUFBTSxNQUFNLENBQUE7QUFDdkIsT0FBTyxFQUFFLE1BQU0sSUFBSSxDQUFBO0FBRW5CLE9BQU8sSUFBSSxNQUFNLHFCQUFxQixDQUFBO0FBQ3RDLE9BQU8sSUFBSSxNQUFNLGFBQWEsQ0FBQTtBQUM5QixPQUFPLFdBQVcsTUFBTSxnQ0FBZ0MsQ0FBQTtBQUN4RCxPQUFPLE1BQU0sTUFBTSwyQkFBMkIsQ0FBQTtBQUM5QyxPQUFPLEdBQUcsTUFBTSxjQUFjLENBQUE7QUFDOUIsT0FBTyxFQUFFLGdCQUFnQixFQUFFLE1BQU0scUJBQXFCLENBQUE7QUFDdEQsT0FBTyxFQUFFLG1CQUFtQixFQUFFLE1BQU0sd0JBQXdCLENBQUE7QUFDNUQsT0FBTyxFQUFFLE9BQU8sRUFBRSxNQUFNLGFBQWEsQ0FBQTtBQUVyQyxPQUFPLEVBQUUsT0FBTyxJQUFJLEVBQUUsRUFBRSxNQUFNLGFBQWEsQ0FBQTtBQU8zQyxNQUFNLENBQUMsS0FBSyxVQUFVLGVBQWUsQ0FBQyxXQUFxQixFQUFFLEdBQVc7SUFDdEUsTUFBTSxTQUFTLEdBQUcsV0FBVyxDQUFDLE1BQU0sQ0FBQyxVQUFVLENBQUMsQ0FBQTtJQUNoRCxNQUFNLE9BQU8sR0FBRyxNQUFNLFNBQVMsQ0FBQyxNQUFNLENBQUMsUUFBUSxDQUFDLEdBQUcsQ0FBQyxFQUFFLE9BQU8sQ0FBQyxPQUFPLENBQUMsSUFBSSxDQUFDLE9BQU8sRUFBRSxDQUFDLENBQUMsQ0FBQTtJQUV0RixnQkFBZ0IsQ0FBQyxPQUFPLENBQUMsQ0FBQTtJQUV6QixPQUFPLE9BQU8sQ0FBQTtBQUNoQixDQUFDO0FBRUQsU0FBUyxnQkFBZ0IsQ0FBQyxPQUFvQjtJQUM1QyxJQUFJLENBQUMsU0FBUyxDQUFDLEVBQUUsT0FBTyxFQUFFLEtBQUssRUFBRSxPQUFPLENBQUMsU0FBUyxFQUFFLENBQUM7U0FDbEQsR0FBRyxDQUFDLE9BQU8sQ0FBQyxNQUFNLEVBQUUsT0FBTyxDQUFDLE9BQU8sQ0FBQztTQUNwQyxTQUFTLENBQUMsT0FBTyxDQUFDLE1BQU0sRUFBRSxPQUFPLENBQUMsT0FBTyxDQUFDO1NBQzFDLE1BQU0sQ0FBQyxPQUFPLENBQUMsTUFBTSxFQUFFLE9BQU8sQ0FBQyxTQUFTLENBQUMsQ0FBQTtBQUM5QyxDQUFDO0FBRUQsU0FBUyxRQUFRLENBQUMsR0FBVztJQUMzQixPQUFPLEtBQUssV0FBVyxlQUFxQyxFQUFFLEdBQVc7UUFDdkUsSUFBSSxRQUFRLEdBQUcsTUFBTSxlQUFlLENBQUE7UUFDcEMsTUFBTSxPQUFPLEdBQUcsSUFBSSxDQUFDLE9BQU8sRUFBRSxDQUFBO1FBRTlCLEdBQUcsQ0FBQyxLQUFLLENBQUMsbUJBQW1CLEVBQUUsR0FBRyxDQUFDLENBQUE7UUFFbkMsSUFBSSxLQUFLLEVBQUUsTUFBTSxJQUFJLElBQUksSUFBSSxDQUFDLEdBQUcsQ0FBQyxFQUFFO1lBQ2xDLE1BQU0sUUFBUSxHQUFHLElBQUksQ0FBQyxRQUFRLENBQUMsR0FBRyxFQUFFLElBQUksQ0FBQyxDQUFBO1lBQ3pDLE1BQU0sWUFBWSxHQUFHLElBQUksQ0FBQyxRQUFRLENBQUMsR0FBRyxFQUFFLElBQUksQ0FBQztpQkFDMUMsT0FBTyxDQUFDLFVBQVUsRUFBRSxFQUFFLENBQUM7aUJBQ3ZCLE9BQU8sQ0FBQyxXQUFXLEVBQUUsRUFBRSxDQUFDLENBQUE7WUFFM0IsTUFBTSxHQUFHLEdBQUcsWUFBWSxLQUFLLEVBQUU7Z0JBQzdCLENBQUMsQ0FBQyxTQUFTLENBQUMsR0FBRyxDQUFDO2dCQUNoQixDQUFDLENBQUMsU0FBUyxDQUFDLEdBQUcsR0FBRyxJQUFJLFlBQVksRUFBRSxDQUFDLENBQUE7WUFFdkMsTUFBTSxZQUFZLEdBQUcsZ0JBQWdCLENBQUMsSUFBSSxFQUFFLEdBQUcsRUFBRSxHQUFHLEVBQUUsR0FBRyxDQUFDLENBQUE7WUFDMUQsSUFBSSxDQUFDLFlBQVksRUFBRTtnQkFDakIsU0FBUTthQUNUO1lBRUQsR0FBRyxDQUFDLEtBQUssQ0FBQyxZQUFZLEVBQUUsUUFBUSxDQUFDLENBQUE7WUFDakMsTUFBTSxxQkFBcUIsR0FBNkIsRUFBRyxDQUFBO1lBQzNELFlBQVksQ0FBQyxFQUFFLENBQUMsUUFBUSxFQUFFLG1CQUFtQixDQUFDLHFCQUFxQixDQUFDLENBQUMsQ0FBQTtZQUVyRSxNQUFNLFNBQVMsR0FBRyxJQUFJLENBQUMsT0FBTyxFQUFhLENBQUE7WUFDM0MsTUFBTSxlQUFlLEdBQUcsSUFBSSxDQUFDLFNBQVMsQ0FBQyxFQUFFLE9BQU8sRUFBRSxRQUFRLEVBQUUsS0FBSyxFQUFFLE9BQU8sQ0FBQyxTQUFTLEVBQUUsQ0FBQyxDQUFBO1lBQ3ZGLElBQUk7Z0JBQ0YsSUFBSSxLQUFLLEVBQUUsTUFBTSxFQUFFLE9BQU8sRUFBRSxTQUFTLEVBQUUsTUFBTSxFQUFFLEdBQUcsSUFBSSxFQUFFLElBQUksWUFBWSxFQUFFO29CQUN4RSxNQUFNLEtBQUssR0FBYyxJQUFJLENBQUMsS0FBSyxDQUFDLE1BQU0sQ0FBQyxJQUFJLENBQUMsWUFBWSxFQUFFLENBQUMsQ0FBQyxDQUFDLENBQUMsSUFBSSxDQUFDLFNBQVMsQ0FBQyxHQUFHLENBQUMsQ0FBQyxDQUFDLENBQUMsSUFBSSxDQUFDLEtBQUssQ0FBQTtvQkFFbEcsSUFBSSxDQUFDLFNBQVMsQ0FBQyxHQUFHLENBQUMsS0FBSyxDQUFDLEVBQUU7d0JBQ3pCLEdBQUcsQ0FBQyxLQUFLLENBQUMsb0JBQW9CLEVBQUUsS0FBSyxDQUFDLEtBQUssQ0FBQyxDQUFBO3FCQUM3QztvQkFDRCxTQUFTLENBQUMsR0FBRyxDQUFDLEtBQUssQ0FBQyxDQUFBO29CQUNwQixPQUFPLENBQUMsR0FBRyxDQUFDLElBQUksQ0FBQyxJQUFJLENBQUMsT0FBTyxFQUFFLFNBQVMsRUFBRSxNQUFNLEVBQUUsS0FBSyxDQUFDLENBQUMsQ0FBQTtpQkFDMUQ7YUFDRjtZQUFDLE9BQU8sQ0FBVSxFQUFFO2dCQUNuQixJQUFJLENBQUMsWUFBWSxLQUFLLEVBQUU7b0JBQ3RCLEdBQUcsQ0FBQyx3QkFBd0IsRUFBRSxRQUFRLEVBQUUsQ0FBQyxDQUFDLE9BQU8sQ0FBQyxDQUFBO2lCQUNuRDthQUNGO1lBRUQsU0FBUyxDQUFDLE9BQU8sQ0FBQyxFQUFFLENBQUMsRUFBRTtnQkFDckIsTUFBTSxNQUFNLEdBQUcscUJBQXFCLENBQUMsZ0JBQWdCLElBQUksU0FBUyxDQUFBO2dCQUNsRSxNQUFNLHlCQUF5QixHQUFHLHFCQUFxQixDQUFDLHlCQUF5QixJQUFJLFNBQVMsQ0FBQTtnQkFDOUYsTUFBTSxPQUFPLEdBQUcsZUFBZSxDQUFDLElBQUksQ0FBQyxFQUFFLENBQUMsQ0FBQTtnQkFDeEMsT0FBTztxQkFDSixTQUFTLENBQUMsT0FBTyxDQUFDLE1BQU0sRUFBRSxPQUFPLENBQUMsT0FBTyxDQUFDO3FCQUMxQyxNQUFNLENBQUMsT0FBTyxDQUFDLE1BQU0sRUFBRSxPQUFPLENBQUMsTUFBTSxDQUFDLENBQUM7cUJBQ3ZDLFNBQVMsQ0FBQyxPQUFPLENBQUMseUJBQXlCLEVBQUUsT0FBTyxDQUFDLE9BQU8sQ0FBQztxQkFDN0QsTUFBTSxDQUFDLE9BQU8sQ0FBQyx5QkFBeUIsRUFBRSxPQUFPLENBQUMseUJBQXlCLENBQUMsQ0FBQyxDQUFBO2dCQUVoRixJQUFJLE9BQU8sQ0FBQyxHQUFHLENBQUMsT0FBTyxDQUFDLHlCQUF5QixFQUFFLE9BQU8sQ0FBQyxPQUFPLENBQUMsQ0FBQyxLQUFLLENBQUMsTUFBTSxFQUFFO29CQUNoRixRQUFRLEdBQUcsV0FBVyxDQUFDLFFBQVEsRUFBRSxTQUFTLEVBQUUsU0FBUyxFQUFFLFNBQVMsRUFBRSxFQUFFLENBQUMsQ0FBQTtpQkFDdEU7WUFDSCxDQUFDLENBQUMsQ0FBQTtTQUNIO1FBRUQsTUFBTSxDQUFDLFFBQVEsRUFBRSxPQUFPLENBQUMsQ0FBQTtRQUN6QixPQUFPLFFBQVEsQ0FBQTtJQUNqQixDQUFDLENBQUE7QUFDSCxDQUFDO0FBRUQsU0FBUyxVQUFVLENBQUMsR0FBVztJQUM3QixJQUFJLENBQUMsRUFBRSxDQUFDLFVBQVUsQ0FBQyxHQUFHLENBQUMsRUFBRTtRQUN2QixHQUFHLENBQUMsdUNBQXVDLEVBQUUsR0FBRyxDQUFDLENBQUE7UUFDakQsT0FBTyxLQUFLLENBQUE7S0FDYjtJQUNELElBQUksQ0FBQyxFQUFFLENBQUMsUUFBUSxDQUFDLEdBQUcsQ0FBQyxDQUFDLFdBQVcsRUFBRSxFQUFFO1FBQ25DLEdBQUcsQ0FBQywyQ0FBMkMsRUFBRSxHQUFHLENBQUMsQ0FBQTtRQUNyRCxPQUFPLEtBQUssQ0FBQTtLQUNiO0lBRUQsT0FBTyxJQUFJLENBQUE7QUFDYixDQUFDIn0=
package/index.ts ADDED
@@ -0,0 +1,112 @@
1
+ import path from 'path'
2
+ import fs from 'fs'
3
+ import { NamedNode, DatasetCore } from 'rdf-js'
4
+ import walk from '@fcostarodrigo/walk'
5
+ import $rdf from '@zazuko/env'
6
+ import deleteMatch from 'rdf-dataset-ext/deleteMatch.js'
7
+ import addAll from 'rdf-dataset-ext/addAll.js'
8
+ import log from './lib/log.js'
9
+ import { getPatchedStream } from './lib/fileStream.js'
10
+ import { optionsFromPrefixes } from './lib/prefixHandler.js'
11
+ import { talosNs } from './lib/ns.js'
12
+
13
+ export { talosNs as ns } from './lib/ns.js'
14
+
15
+ interface ResourceOptions {
16
+ existingResource: 'merge' | 'overwrite' | 'skip'
17
+ environmentRepresentation: 'default' | 'replace'
18
+ }
19
+
20
+ export async function fromDirectories(directories: string[], api: string): Promise<DatasetCore> {
21
+ const validDirs = directories.filter(isValidDir)
22
+ const dataset = await validDirs.reduce(toGraphs(api), Promise.resolve($rdf.dataset()))
23
+
24
+ setDefaultAction(dataset)
25
+
26
+ return dataset
27
+ }
28
+
29
+ function setDefaultAction(dataset: DatasetCore) {
30
+ $rdf.clownface({ dataset, graph: talosNs.resources })
31
+ .has(talosNs.action, talosNs.default)
32
+ .deleteOut(talosNs.action, talosNs.default)
33
+ .addOut(talosNs.action, talosNs.overwrite)
34
+ }
35
+
36
+ function toGraphs(api: string) {
37
+ return async function (previousPromise: Promise<DatasetCore>, dir: string): Promise<DatasetCore> {
38
+ let previous = await previousPromise
39
+ const dataset = $rdf.dataset()
40
+
41
+ log.debug('Processing dir %s', dir)
42
+
43
+ for await (const file of walk(dir)) {
44
+ const relative = path.relative(dir, file)
45
+ const resourcePath = path.relative(dir, file)
46
+ .replace(/\.[^.]+$/, '')
47
+ .replace(/\/?index$/, '')
48
+
49
+ const url = resourcePath === ''
50
+ ? encodeURI(api)
51
+ : encodeURI(`${api}/${resourcePath}`)
52
+
53
+ const parserStream = getPatchedStream(file, dir, api, url)
54
+ if (!parserStream) {
55
+ continue
56
+ }
57
+
58
+ log.debug('Parsing %s', relative)
59
+ const parsedResourceOptions: Partial<ResourceOptions> = { }
60
+ parserStream.on('prefix', optionsFromPrefixes(parsedResourceOptions))
61
+
62
+ const resources = $rdf.termSet<NamedNode>()
63
+ const resourceOptions = $rdf.clownface({ dataset: previous, graph: talosNs.resources })
64
+ try {
65
+ for await (const { subject, predicate, object, ...quad } of parserStream) {
66
+ const graph: NamedNode = quad.graph.equals($rdf.defaultGraph()) ? $rdf.namedNode(url) : quad.graph
67
+
68
+ if (!resources.has(graph)) {
69
+ log.debug('Parsed resource %s', graph.value)
70
+ }
71
+ resources.add(graph)
72
+ dataset.add($rdf.quad(subject, predicate, object, graph))
73
+ }
74
+ } catch (e: unknown) {
75
+ if (e instanceof Error) {
76
+ log('Failed to parse %s: %s', relative, e.message)
77
+ }
78
+ }
79
+
80
+ resources.forEach(id => {
81
+ const action = parsedResourceOptions.existingResource || 'default'
82
+ const environmentRepresentation = parsedResourceOptions.environmentRepresentation || 'default'
83
+ const options = resourceOptions.node(id)
84
+ options
85
+ .deleteOut(talosNs.action, talosNs.default)
86
+ .addOut(talosNs.action, talosNs(action))
87
+ .deleteOut(talosNs.environmentRepresentation, talosNs.default)
88
+ .addOut(talosNs.environmentRepresentation, talosNs(environmentRepresentation))
89
+
90
+ if (options.has(talosNs.environmentRepresentation, talosNs.replace).terms.length) {
91
+ previous = deleteMatch(previous, undefined, undefined, undefined, id)
92
+ }
93
+ })
94
+ }
95
+
96
+ addAll(previous, dataset)
97
+ return previous
98
+ }
99
+ }
100
+
101
+ function isValidDir(dir: string) {
102
+ if (!fs.existsSync(dir)) {
103
+ log('Skipping path %s which does not exist', dir)
104
+ return false
105
+ }
106
+ if (!fs.statSync(dir).isDirectory()) {
107
+ log('Skipping path %s which is not a directory', dir)
108
+ return false
109
+ }
110
+
111
+ return true
112
+ }
@@ -0,0 +1,3 @@
1
+ /// <reference types="node" />
2
+ import { Readable } from 'stream';
3
+ export declare function getPatchedStream(file: string, cwd: string, api: string, resourceUrl: string): Readable | null;
@@ -0,0 +1,61 @@
1
+ import fs from 'fs';
2
+ import path from 'path';
3
+ import * as mime from 'mime-types';
4
+ import replaceStream from 'replacestream';
5
+ import { parsers } from '@rdfjs/formats-common';
6
+ import isAbsoluteUrl from 'is-absolute-url';
7
+ import log from './log.js';
8
+ function replacer(basePath, resourceUrl, s, e = s) {
9
+ return (_, match) => {
10
+ let absolute;
11
+ if (isAbsoluteUrl(match)) {
12
+ return `${s}${match}${e}`;
13
+ }
14
+ if (match.startsWith('/') && basePath !== '/') {
15
+ absolute = basePath + match;
16
+ }
17
+ else {
18
+ absolute = new URL(match, resourceUrl).toString();
19
+ if (!match.endsWith('/')) {
20
+ absolute = absolute.replace(/\/?$/, '');
21
+ }
22
+ }
23
+ const absoluteUrl = new URL(absolute, resourceUrl);
24
+ if (absoluteUrl.pathname === '/' && basePath !== '/') {
25
+ absoluteUrl.pathname = basePath;
26
+ absolute = absoluteUrl.toString();
27
+ }
28
+ return `${s}${absolute}${e}`;
29
+ };
30
+ }
31
+ const angleBracketTransform = (basePath, resourceUrl) => replaceStream(/<([^>]+)>(?=([^"\\]*(\\.|"([^"\\]*\\.)*[^"\\]*"))*[^"]*$)/g, replacer(basePath, resourceUrl, '<', '>'));
32
+ const jsonTransform = (basePath, resourceUrl) => replaceStream(/"([./][^"]+)"/g, replacer(basePath, resourceUrl, '"'));
33
+ const filePatchTransforms = new Map([
34
+ ['text/turtle', angleBracketTransform],
35
+ ['application/n-triples', angleBracketTransform],
36
+ ['application/n-quads', angleBracketTransform],
37
+ ['application/trig', angleBracketTransform],
38
+ ['application/ld+json', jsonTransform],
39
+ ]);
40
+ export function getPatchedStream(file, cwd, api, resourceUrl) {
41
+ const relative = path.relative(cwd, file);
42
+ const basePath = new URL(api).pathname;
43
+ const mediaType = mime.lookup(file);
44
+ if (!mediaType) {
45
+ log('Could not determine media type for file %s', relative);
46
+ return null;
47
+ }
48
+ let fileStream = fs.createReadStream(file);
49
+ if (filePatchTransforms.has(mediaType)) {
50
+ fileStream = fileStream.pipe(filePatchTransforms.get(mediaType)(basePath, resourceUrl));
51
+ }
52
+ const parserStream = parsers.import(mediaType, fileStream, {
53
+ baseIRI: resourceUrl,
54
+ blankNodePrefix: '',
55
+ });
56
+ if (!parserStream) {
57
+ log('Unsupported media type %s from %s', mediaType, relative);
58
+ }
59
+ return parserStream;
60
+ }
61
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZmlsZVN0cmVhbS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbImZpbGVTdHJlYW0udHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLE1BQU0sSUFBSSxDQUFBO0FBRW5CLE9BQU8sSUFBSSxNQUFNLE1BQU0sQ0FBQTtBQUN2QixPQUFPLEtBQUssSUFBSSxNQUFNLFlBQVksQ0FBQTtBQUNsQyxPQUFPLGFBQWEsTUFBTSxlQUFlLENBQUE7QUFDekMsT0FBTyxFQUFFLE9BQU8sRUFBRSxNQUFNLHVCQUF1QixDQUFBO0FBQy9DLE9BQU8sYUFBYSxNQUFNLGlCQUFpQixDQUFBO0FBQzNDLE9BQU8sR0FBRyxNQUFNLFVBQVUsQ0FBQTtBQUUxQixTQUFTLFFBQVEsQ0FBQyxRQUFnQixFQUFFLFdBQW1CLEVBQUUsQ0FBUyxFQUFFLENBQUMsR0FBRyxDQUFDO0lBQ3ZFLE9BQU8sQ0FBQyxDQUFVLEVBQUUsS0FBYSxFQUFFLEVBQUU7UUFDbkMsSUFBSSxRQUFnQixDQUFBO1FBQ3BCLElBQUksYUFBYSxDQUFDLEtBQUssQ0FBQyxFQUFFO1lBQ3hCLE9BQU8sR0FBRyxDQUFDLEdBQUcsS0FBSyxHQUFHLENBQUMsRUFBRSxDQUFBO1NBQzFCO1FBRUQsSUFBSSxLQUFLLENBQUMsVUFBVSxDQUFDLEdBQUcsQ0FBQyxJQUFJLFFBQVEsS0FBSyxHQUFHLEVBQUU7WUFDN0MsUUFBUSxHQUFHLFFBQVEsR0FBRyxLQUFLLENBQUE7U0FDNUI7YUFBTTtZQUNMLFFBQVEsR0FBRyxJQUFJLEdBQUcsQ0FBQyxLQUFLLEVBQUUsV0FBVyxDQUFDLENBQUMsUUFBUSxFQUFFLENBQUE7WUFDakQsSUFBSSxDQUFDLEtBQUssQ0FBQyxRQUFRLENBQUMsR0FBRyxDQUFDLEVBQUU7Z0JBQ3hCLFFBQVEsR0FBRyxRQUFRLENBQUMsT0FBTyxDQUFDLE1BQU0sRUFBRSxFQUFFLENBQUMsQ0FBQTthQUN4QztTQUNGO1FBRUQsTUFBTSxXQUFXLEdBQUcsSUFBSSxHQUFHLENBQUMsUUFBUSxFQUFFLFdBQVcsQ0FBQyxDQUFBO1FBQ2xELElBQUksV0FBVyxDQUFDLFFBQVEsS0FBSyxHQUFHLElBQUksUUFBUSxLQUFLLEdBQUcsRUFBRTtZQUNwRCxXQUFXLENBQUMsUUFBUSxHQUFHLFFBQVEsQ0FBQTtZQUMvQixRQUFRLEdBQUcsV0FBVyxDQUFDLFFBQVEsRUFBRSxDQUFBO1NBQ2xDO1FBRUQsT0FBTyxHQUFHLENBQUMsR0FBRyxRQUFRLEdBQUcsQ0FBQyxFQUFFLENBQUE7SUFDOUIsQ0FBQyxDQUFBO0FBQ0gsQ0FBQztBQUVELE1BQU0scUJBQXFCLEdBQUcsQ0FBQyxRQUFnQixFQUFFLFdBQW1CLEVBQUUsRUFBRSxDQUFDLGFBQWEsQ0FBQyw0REFBNEQsRUFBRSxRQUFRLENBQUMsUUFBUSxFQUFFLFdBQVcsRUFBRSxHQUFHLEVBQUUsR0FBRyxDQUFDLENBQUMsQ0FBQTtBQUMvTCxNQUFNLGFBQWEsR0FBRyxDQUFDLFFBQWdCLEVBQUUsV0FBbUIsRUFBRSxFQUFFLENBQUMsYUFBYSxDQUFDLGdCQUFnQixFQUFFLFFBQVEsQ0FBQyxRQUFRLEVBQUUsV0FBVyxFQUFFLEdBQUcsQ0FBQyxDQUFDLENBQUE7QUFFdEksTUFBTSxtQkFBbUIsR0FBRyxJQUFJLEdBQUcsQ0FBQztJQUNsQyxDQUFDLGFBQWEsRUFBRSxxQkFBcUIsQ0FBQztJQUN0QyxDQUFDLHVCQUF1QixFQUFFLHFCQUFxQixDQUFDO0lBQ2hELENBQUMscUJBQXFCLEVBQUUscUJBQXFCLENBQUM7SUFDOUMsQ0FBQyxrQkFBa0IsRUFBRSxxQkFBcUIsQ0FBQztJQUMzQyxDQUFDLHFCQUFxQixFQUFFLGFBQWEsQ0FBQztDQUN2QyxDQUFDLENBQUE7QUFFRixNQUFNLFVBQVUsZ0JBQWdCLENBQUMsSUFBWSxFQUFFLEdBQVcsRUFBRSxHQUFXLEVBQUUsV0FBbUI7SUFDMUYsTUFBTSxRQUFRLEdBQUcsSUFBSSxDQUFDLFFBQVEsQ0FBQyxHQUFHLEVBQUUsSUFBSSxDQUFDLENBQUE7SUFDekMsTUFBTSxRQUFRLEdBQUcsSUFBSSxHQUFHLENBQUMsR0FBRyxDQUFDLENBQUMsUUFBUSxDQUFBO0lBQ3RDLE1BQU0sU0FBUyxHQUFHLElBQUksQ0FBQyxNQUFNLENBQUMsSUFBSSxDQUFDLENBQUE7SUFDbkMsSUFBSSxDQUFDLFNBQVMsRUFBRTtRQUNkLEdBQUcsQ0FBQyw0Q0FBNEMsRUFBRSxRQUFRLENBQUMsQ0FBQTtRQUMzRCxPQUFPLElBQUksQ0FBQTtLQUNaO0lBRUQsSUFBSSxVQUFVLEdBQUcsRUFBRSxDQUFDLGdCQUFnQixDQUFDLElBQUksQ0FBQyxDQUFBO0lBQzFDLElBQUksbUJBQW1CLENBQUMsR0FBRyxDQUFDLFNBQVMsQ0FBQyxFQUFFO1FBQ3RDLFVBQVUsR0FBRyxVQUFVLENBQUMsSUFBSSxDQUFDLG1CQUFtQixDQUFDLEdBQUcsQ0FBQyxTQUFTLENBQUUsQ0FBQyxRQUFRLEVBQUUsV0FBVyxDQUFDLENBQUMsQ0FBQTtLQUN6RjtJQUNELE1BQU0sWUFBWSxHQUFHLE9BQU8sQ0FBQyxNQUFNLENBQUMsU0FBUyxFQUFFLFVBQVUsRUFBRTtRQUN6RCxPQUFPLEVBQUUsV0FBVztRQUNwQixlQUFlLEVBQUUsRUFBRTtLQUNwQixDQUFDLENBQUE7SUFFRixJQUFJLENBQUMsWUFBWSxFQUFFO1FBQ2pCLEdBQUcsQ0FBQyxtQ0FBbUMsRUFBRSxTQUFTLEVBQUUsUUFBUSxDQUFDLENBQUE7S0FDOUQ7SUFFRCxPQUFPLFlBQTBDLENBQUE7QUFDbkQsQ0FBQyJ9
@@ -0,0 +1,70 @@
1
+ import fs from 'fs'
2
+ import { Readable } from 'stream'
3
+ import path from 'path'
4
+ import * as mime from 'mime-types'
5
+ import replaceStream from 'replacestream'
6
+ import { parsers } from '@rdfjs/formats-common'
7
+ import isAbsoluteUrl from 'is-absolute-url'
8
+ import log from './log.js'
9
+
10
+ function replacer(basePath: string, resourceUrl: string, s: string, e = s) {
11
+ return (_: unknown, match: string) => {
12
+ let absolute: string
13
+ if (isAbsoluteUrl(match)) {
14
+ return `${s}${match}${e}`
15
+ }
16
+
17
+ if (match.startsWith('/') && basePath !== '/') {
18
+ absolute = basePath + match
19
+ } else {
20
+ absolute = new URL(match, resourceUrl).toString()
21
+ if (!match.endsWith('/')) {
22
+ absolute = absolute.replace(/\/?$/, '')
23
+ }
24
+ }
25
+
26
+ const absoluteUrl = new URL(absolute, resourceUrl)
27
+ if (absoluteUrl.pathname === '/' && basePath !== '/') {
28
+ absoluteUrl.pathname = basePath
29
+ absolute = absoluteUrl.toString()
30
+ }
31
+
32
+ return `${s}${absolute}${e}`
33
+ }
34
+ }
35
+
36
+ const angleBracketTransform = (basePath: string, resourceUrl: string) => replaceStream(/<([^>]+)>(?=([^"\\]*(\\.|"([^"\\]*\\.)*[^"\\]*"))*[^"]*$)/g, replacer(basePath, resourceUrl, '<', '>'))
37
+ const jsonTransform = (basePath: string, resourceUrl: string) => replaceStream(/"([./][^"]+)"/g, replacer(basePath, resourceUrl, '"'))
38
+
39
+ const filePatchTransforms = new Map([
40
+ ['text/turtle', angleBracketTransform],
41
+ ['application/n-triples', angleBracketTransform],
42
+ ['application/n-quads', angleBracketTransform],
43
+ ['application/trig', angleBracketTransform],
44
+ ['application/ld+json', jsonTransform],
45
+ ])
46
+
47
+ export function getPatchedStream(file: string, cwd: string, api: string, resourceUrl: string): Readable | null {
48
+ const relative = path.relative(cwd, file)
49
+ const basePath = new URL(api).pathname
50
+ const mediaType = mime.lookup(file)
51
+ if (!mediaType) {
52
+ log('Could not determine media type for file %s', relative)
53
+ return null
54
+ }
55
+
56
+ let fileStream = fs.createReadStream(file)
57
+ if (filePatchTransforms.has(mediaType)) {
58
+ fileStream = fileStream.pipe(filePatchTransforms.get(mediaType)!(basePath, resourceUrl))
59
+ }
60
+ const parserStream = parsers.import(mediaType, fileStream, {
61
+ baseIRI: resourceUrl,
62
+ blankNodePrefix: '',
63
+ })
64
+
65
+ if (!parserStream) {
66
+ log('Unsupported media type %s from %s', mediaType, relative)
67
+ }
68
+
69
+ return parserStream as unknown as Readable | null
70
+ }
package/lib/log.d.ts ADDED
@@ -0,0 +1,2 @@
1
+ declare const _default: import("anylogger").Logger<import("anylogger").BaseLevels>;
2
+ export default _default;
package/lib/log.js ADDED
@@ -0,0 +1,3 @@
1
+ import Debugger from 'anylogger';
2
+ export default Debugger('talos');
3
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibG9nLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsibG9nLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sUUFBUSxNQUFNLFdBQVcsQ0FBQTtBQUVoQyxlQUFlLFFBQVEsQ0FBQyxPQUFPLENBQUMsQ0FBQSJ9
package/lib/log.ts ADDED
@@ -0,0 +1,3 @@
1
+ import Debugger from 'anylogger'
2
+
3
+ export default Debugger('talos')
package/lib/ns.d.ts ADDED
@@ -0,0 +1,3 @@
1
+ type TalosTerms = 'resources' | 'action' | 'default' | 'overwrite' | 'skip' | 'merge' | 'environmentRepresentation' | 'replace';
2
+ export declare const talosNs: import("@rdfjs/namespace").NamespaceBuilder<TalosTerms>;
3
+ export {};
package/lib/ns.js ADDED
@@ -0,0 +1,3 @@
1
+ import $rdf from '@zazuko/env';
2
+ export const talosNs = $rdf.namespace('urn:talos:');
3
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibnMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyJucy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLElBQUksTUFBTSxhQUFhLENBQUE7QUFZOUIsTUFBTSxDQUFDLE1BQU0sT0FBTyxHQUFHLElBQUksQ0FBQyxTQUFTLENBQWEsWUFBWSxDQUFDLENBQUEifQ==
package/lib/ns.ts ADDED
@@ -0,0 +1,13 @@
1
+ import $rdf from '@zazuko/env'
2
+
3
+ type TalosTerms =
4
+ 'resources' |
5
+ 'action' |
6
+ 'default' |
7
+ 'overwrite' |
8
+ 'skip' |
9
+ 'merge'|
10
+ 'environmentRepresentation' |
11
+ 'replace'
12
+
13
+ export const talosNs = $rdf.namespace<TalosTerms>('urn:talos:')
@@ -0,0 +1,2 @@
1
+ import { NamedNode } from 'rdf-js';
2
+ export declare function optionsFromPrefixes(options: Record<string, unknown>): (prefix: string, namespace: NamedNode) => void;
@@ -0,0 +1,11 @@
1
+ export function optionsFromPrefixes(options) {
2
+ return (prefix, namespace) => {
3
+ if (prefix === 'talos') {
4
+ const [key, value] = namespace.value.split(':');
5
+ if (key && value) {
6
+ options[key] = decodeURIComponent(value);
7
+ }
8
+ }
9
+ };
10
+ }
11
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicHJlZml4SGFuZGxlci5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbInByZWZpeEhhbmRsZXIudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBRUEsTUFBTSxVQUFVLG1CQUFtQixDQUFDLE9BQWdDO0lBQ2xFLE9BQU8sQ0FBQyxNQUFjLEVBQUUsU0FBb0IsRUFBRSxFQUFFO1FBQzlDLElBQUksTUFBTSxLQUFLLE9BQU8sRUFBRTtZQUN0QixNQUFNLENBQUMsR0FBRyxFQUFFLEtBQUssQ0FBQyxHQUFHLFNBQVMsQ0FBQyxLQUFLLENBQUMsS0FBSyxDQUFDLEdBQUcsQ0FBQyxDQUFBO1lBQy9DLElBQUksR0FBRyxJQUFJLEtBQUssRUFBRTtnQkFDaEIsT0FBTyxDQUFDLEdBQUcsQ0FBQyxHQUFHLGtCQUFrQixDQUFDLEtBQUssQ0FBQyxDQUFBO2FBQ3pDO1NBQ0Y7SUFDSCxDQUFDLENBQUE7QUFDSCxDQUFDIn0=
@@ -0,0 +1,12 @@
1
+ import { NamedNode } from 'rdf-js'
2
+
3
+ export function optionsFromPrefixes(options: Record<string, unknown>) {
4
+ return (prefix: string, namespace: NamedNode) => {
5
+ if (prefix === 'talos') {
6
+ const [key, value] = namespace.value.split(':')
7
+ if (key && value) {
8
+ options[key] = decodeURIComponent(value)
9
+ }
10
+ }
11
+ }
12
+ }
package/package.json ADDED
@@ -0,0 +1,33 @@
1
+ {
2
+ "name": "@hydrofoil/talos-core",
3
+ "version": "0.1.0",
4
+ "type": "module",
5
+ "main": "index.js",
6
+ "scripts": {
7
+ "prepublishOnly": "tsc",
8
+ "test": "mocha --package ../../package.json test"
9
+ },
10
+ "dependencies": {
11
+ "@fcostarodrigo/walk": "^5.0.1",
12
+ "@rdfjs/formats-common": "^3.1.0",
13
+ "@zazuko/env": "^1.3.1",
14
+ "anylogger": "^1.0.11",
15
+ "is-absolute-url": "^4.0.1",
16
+ "mime-types": "^2.1.35",
17
+ "rdf-dataset-ext": "^1.0.1",
18
+ "replacestream": "^4.0.3"
19
+ },
20
+ "devDependencies": {
21
+ "chai": "^4.3.8",
22
+ "get-stream": "^8.0.1",
23
+ "into-stream": "^8.0.0"
24
+ },
25
+ "repository": {
26
+ "type": "git",
27
+ "url": "https://github.com/hypermedia-app/talos",
28
+ "directory": "packages/cli"
29
+ },
30
+ "bugs": {
31
+ "url": "https://github.com/hypermedia-app/talos/issues"
32
+ }
33
+ }
@@ -0,0 +1,12 @@
1
+ exports["@hydrofoil/talos-core fromDirectories merges resources from multiple graph documents"] = "<https://example.com/#part> <http://schema.org/maxValue> \"100\"^^<http://www.w3.org/2001/XMLSchema#integer> <https://example.com> .\n<https://example.com/#part> <http://schema.org/minValue> \"10\"^^<http://www.w3.org/2001/XMLSchema#integer> <https://example.com> .\n<https://example.com> <http://schema.org/hasPart> <https://example.com/#part> <https://example.com> .\n<https://example.com> <http://schema.org/hasPart> _:c14n0 <https://example.com> .\n<https://example.com> <http://schema.org/hasPart> _:c14n1 <https://example.com> .\n<https://example.com> <http://schema.org/hasPart> _:c14n2 <https://example.com> .\n<https://example.com> <http://schema.org/name> \"Bar environment\" <https://example.com> .\n<https://example.com> <http://www.w3.org/1999/02/22-rdf-syntax-ns#type> <http://schema.org/Thing> <https://example.com> .\n_:c14n0 <http://schema.org/name> \"Bar part\" <https://example.com> .\n_:c14n0 <http://schema.org/name> \"Foo part\" <https://example.com> .\n_:c14n1 <http://schema.org/name> \"totally anonymous\" <https://example.com> .\n_:c14n2 <http://schema.org/name> \"totally anonymous\" <https://example.com> .\n";
2
+
3
+ exports["@hydrofoil/talos-core fromDirectories merges resources from dataset and graph documents"] = "<https://example.com/trig/users/john-doe> <http://www.w3.org/1999/02/22-rdf-syntax-ns#type> <http://xmlns.com/foaf/0.1/Person> <https://example.com/trig/users/john-doe> .\n<https://example.com/trig/users/john-doe> <http://www.w3.org/ns/earl#test> \"trig\" <https://example.com/trig/users/john-doe> .\n<https://example.com/trig/users/john-doe> <http://xmlns.com/foaf/0.1/name> \"John Doe\" <https://example.com/trig/users/john-doe> .\n";
4
+
5
+ exports["@hydrofoil/talos-core fromDirectories merges resources from multiple dataset documents"] = "<https://example.com/trig/users/jane-doe> <http://www.w3.org/1999/02/22-rdf-syntax-ns#type> <http://xmlns.com/foaf/0.1/Person> <https://example.com/trig/users/jane-doe> .\n<https://example.com/trig/users/jane-doe> <http://www.w3.org/ns/earl#test> \"trig\" <https://example.com/trig/users/jane-doe> .\n<https://example.com/trig/users/jane-doe> <http://xmlns.com/foaf/0.1/name> \"Jane Doe\" <https://example.com/trig/users/jane-doe> .\n";
6
+
7
+ exports["@hydrofoil/talos-core fromDirectories uses the last representation when is marked to replace other envs"] = "<https://example.com/only/one> <http://schema.org/name> \"bar\" <https://example.com/only/one> .\n";
8
+
9
+ exports["@hydrofoil/talos-core fromDirectories uses the last representation when is marked in another env"] = "<https://example.com/only/two> <http://schema.org/name> \"bar\" <https://example.com/only/two> .\n";
10
+
11
+ exports["@hydrofoil/talos-core fromDirectories uses the last representation when is marked in a dataset document"] = "<https://example.com/only/three> <http://schema.org/name> \"bar\" <https://example.com/only/three> .\n";
12
+
@@ -0,0 +1 @@
1
+ export {};
@@ -0,0 +1,58 @@
1
+ import path from 'path';
2
+ import url from 'url';
3
+ import { expect } from 'chai';
4
+ import toCanonical from 'rdf-dataset-ext/toCanonical.js';
5
+ import $rdf from '@zazuko/env';
6
+ import { talosNs } from '../lib/ns.js';
7
+ import { fromDirectories } from '../index.js';
8
+ const testDir = url.fileURLToPath(new URL('../../../test-resources', import.meta.url));
9
+ const ns = $rdf.namespace('https://example.com');
10
+ describe('@hydrofoil/talos-core', () => {
11
+ describe('fromDirectories', () => {
12
+ let dataset;
13
+ beforeEach(async () => {
14
+ const dirs = [
15
+ path.resolve(testDir, './resources'),
16
+ path.resolve(testDir, './resources.foo'),
17
+ path.resolve(testDir, './resources.bar'),
18
+ ];
19
+ dataset = await fromDirectories(dirs, ns().value);
20
+ });
21
+ it('merges resources from multiple graph documents', function () {
22
+ const resource = dataset.match(null, null, null, ns());
23
+ expect(toCanonical(resource)).to.matchSnapshot(this);
24
+ });
25
+ it('merges resources from dataset and graph documents', function () {
26
+ const resource = dataset.match(null, null, null, ns('/trig/users/john-doe'));
27
+ expect(toCanonical(resource)).to.matchSnapshot(this);
28
+ });
29
+ it('merges resources from multiple dataset documents', function () {
30
+ const resource = dataset.match(null, null, null, ns('/trig/users/jane-doe'));
31
+ expect(toCanonical(resource)).to.matchSnapshot(this);
32
+ });
33
+ it('marks a resource for "overwrite" by default', () => {
34
+ const [{ object: action }, ...more] = dataset.match(ns(), talosNs.action, null, talosNs.resources);
35
+ expect(action).to.deep.eq(talosNs.overwrite);
36
+ expect(more).to.be.empty;
37
+ });
38
+ it('marks a resource for "merge" when prefix is used', () => {
39
+ const datasetCore = dataset.match(ns('/project/creta/user.group/admins'), talosNs.action, null, talosNs.resources);
40
+ const [{ object: action }, ...more] = datasetCore;
41
+ expect(action).to.deep.eq(talosNs.merge);
42
+ expect(more).to.be.empty;
43
+ });
44
+ it('uses the last representation when is marked to replace other envs', function () {
45
+ const resource = dataset.match(null, null, null, ns('/only/one'));
46
+ expect(toCanonical(resource)).to.matchSnapshot(this);
47
+ });
48
+ it('uses the last representation when is marked in another env', function () {
49
+ const resource = dataset.match(null, null, null, ns('/only/two'));
50
+ expect(toCanonical(resource)).to.matchSnapshot(this);
51
+ });
52
+ it('uses the last representation when is marked in a dataset document', function () {
53
+ const resource = dataset.match(null, null, null, ns('/only/three'));
54
+ expect(toCanonical(resource)).to.matchSnapshot(this);
55
+ });
56
+ });
57
+ });
58
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXgudGVzdC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbImluZGV4LnRlc3QudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQ0EsT0FBTyxJQUFJLE1BQU0sTUFBTSxDQUFBO0FBQ3ZCLE9BQU8sR0FBRyxNQUFNLEtBQUssQ0FBQTtBQUNyQixPQUFPLEVBQUUsTUFBTSxFQUFFLE1BQU0sTUFBTSxDQUFBO0FBQzdCLE9BQU8sV0FBVyxNQUFNLGdDQUFnQyxDQUFBO0FBQ3hELE9BQU8sSUFBSSxNQUFNLGFBQWEsQ0FBQTtBQUM5QixPQUFPLEVBQUUsT0FBTyxFQUFFLE1BQU0sY0FBYyxDQUFBO0FBQ3RDLE9BQU8sRUFBRSxlQUFlLEVBQUUsTUFBTSxhQUFhLENBQUE7QUFFN0MsTUFBTSxPQUFPLEdBQUcsR0FBRyxDQUFDLGFBQWEsQ0FBQyxJQUFJLEdBQUcsQ0FBQyx5QkFBeUIsRUFBRSxNQUFNLENBQUMsSUFBSSxDQUFDLEdBQUcsQ0FBQyxDQUFDLENBQUE7QUFDdEYsTUFBTSxFQUFFLEdBQUcsSUFBSSxDQUFDLFNBQVMsQ0FBQyxxQkFBcUIsQ0FBQyxDQUFBO0FBRWhELFFBQVEsQ0FBQyx1QkFBdUIsRUFBRSxHQUFHLEVBQUU7SUFDckMsUUFBUSxDQUFDLGlCQUFpQixFQUFFLEdBQUcsRUFBRTtRQUMvQixJQUFJLE9BQW9CLENBQUE7UUFFeEIsVUFBVSxDQUFDLEtBQUssSUFBSSxFQUFFO1lBQ3BCLE1BQU0sSUFBSSxHQUFHO2dCQUNYLElBQUksQ0FBQyxPQUFPLENBQUMsT0FBTyxFQUFFLGFBQWEsQ0FBQztnQkFDcEMsSUFBSSxDQUFDLE9BQU8sQ0FBQyxPQUFPLEVBQUUsaUJBQWlCLENBQUM7Z0JBQ3hDLElBQUksQ0FBQyxPQUFPLENBQUMsT0FBTyxFQUFFLGlCQUFpQixDQUFDO2FBQ3pDLENBQUE7WUFDRCxPQUFPLEdBQUcsTUFBTSxlQUFlLENBQUMsSUFBSSxFQUFFLEVBQUUsRUFBRSxDQUFDLEtBQUssQ0FBQyxDQUFBO1FBQ25ELENB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
@@ -0,0 +1,77 @@
1
+ import { DatasetCore } from 'rdf-js'
2
+ import path from 'path'
3
+ import url from 'url'
4
+ import { expect } from 'chai'
5
+ import toCanonical from 'rdf-dataset-ext/toCanonical.js'
6
+ import $rdf from '@zazuko/env'
7
+ import { talosNs } from '../lib/ns.js'
8
+ import { fromDirectories } from '../index.js'
9
+
10
+ const testDir = url.fileURLToPath(new URL('../../../test-resources', import.meta.url))
11
+ const ns = $rdf.namespace('https://example.com')
12
+
13
+ describe('@hydrofoil/talos-core', () => {
14
+ describe('fromDirectories', () => {
15
+ let dataset: DatasetCore
16
+
17
+ beforeEach(async () => {
18
+ const dirs = [
19
+ path.resolve(testDir, './resources'),
20
+ path.resolve(testDir, './resources.foo'),
21
+ path.resolve(testDir, './resources.bar'),
22
+ ]
23
+ dataset = await fromDirectories(dirs, ns().value)
24
+ })
25
+
26
+ it('merges resources from multiple graph documents', function () {
27
+ const resource = dataset.match(null, null, null, ns())
28
+
29
+ expect(toCanonical(resource)).to.matchSnapshot(this)
30
+ })
31
+
32
+ it('merges resources from dataset and graph documents', function () {
33
+ const resource = dataset.match(null, null, null, ns('/trig/users/john-doe'))
34
+
35
+ expect(toCanonical(resource)).to.matchSnapshot(this)
36
+ })
37
+
38
+ it('merges resources from multiple dataset documents', function () {
39
+ const resource = dataset.match(null, null, null, ns('/trig/users/jane-doe'))
40
+
41
+ expect(toCanonical(resource)).to.matchSnapshot(this)
42
+ })
43
+
44
+ it('marks a resource for "overwrite" by default', () => {
45
+ const [{ object: action }, ...more] = dataset.match(ns(), talosNs.action, null, talosNs.resources)
46
+
47
+ expect(action).to.deep.eq(talosNs.overwrite)
48
+ expect(more).to.be.empty
49
+ })
50
+
51
+ it('marks a resource for "merge" when prefix is used', () => {
52
+ const datasetCore = dataset.match(ns('/project/creta/user.group/admins'), talosNs.action, null, talosNs.resources)
53
+ const [{ object: action }, ...more] = datasetCore
54
+
55
+ expect(action).to.deep.eq(talosNs.merge)
56
+ expect(more).to.be.empty
57
+ })
58
+
59
+ it('uses the last representation when is marked to replace other envs', function () {
60
+ const resource = dataset.match(null, null, null, ns('/only/one'))
61
+
62
+ expect(toCanonical(resource)).to.matchSnapshot(this)
63
+ })
64
+
65
+ it('uses the last representation when is marked in another env', function () {
66
+ const resource = dataset.match(null, null, null, ns('/only/two'))
67
+
68
+ expect(toCanonical(resource)).to.matchSnapshot(this)
69
+ })
70
+
71
+ it('uses the last representation when is marked in a dataset document', function () {
72
+ const resource = dataset.match(null, null, null, ns('/only/three'))
73
+
74
+ expect(toCanonical(resource)).to.matchSnapshot(this)
75
+ })
76
+ })
77
+ })
@@ -0,0 +1 @@
1
+ export {};
@@ -0,0 +1,36 @@
1
+ import { expect } from 'chai';
2
+ import { parsers } from '@rdfjs/formats-common';
3
+ import toStream from 'into-stream';
4
+ import getStream from 'get-stream';
5
+ import { optionsFromPrefixes } from '../../lib/prefixHandler.js';
6
+ describe('@hydrofoil/talos/lib/prefixHandler', () => {
7
+ function parse(str) {
8
+ return parsers.import('text/turtle', toStream(str));
9
+ }
10
+ describe('optionsFromPrefixes', () => {
11
+ it('sets options from parsed prefixes', async () => {
12
+ // given
13
+ const options = {};
14
+ // when
15
+ const stream = parse(`prefix talos: <foo:bar>
16
+ prefix talos: <foo:baz>
17
+ prefix talos: <another:also%20baz>`);
18
+ stream.on('prefix', optionsFromPrefixes(options));
19
+ await getStream(stream);
20
+ // then
21
+ expect(options).to.have.property('foo', 'baz');
22
+ expect(options).to.have.property('another', 'also baz');
23
+ });
24
+ it('ignores other prefixes', async () => {
25
+ // given
26
+ const options = {};
27
+ // when
28
+ const stream = parse('prefix schema: <http://schema.org/>');
29
+ stream.on('prefix', optionsFromPrefixes(options));
30
+ await getStream(stream);
31
+ // then
32
+ expect(options).to.deep.eq({});
33
+ });
34
+ });
35
+ });
36
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicHJlZml4SGFuZGxlci50ZXN0LmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsicHJlZml4SGFuZGxlci50ZXN0LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUNBLE9BQU8sRUFBRSxNQUFNLEVBQUUsTUFBTSxNQUFNLENBQUE7QUFDN0IsT0FBTyxFQUFFLE9BQU8sRUFBRSxNQUFNLHVCQUF1QixDQUFBO0FBQy9DLE9BQU8sUUFBUSxNQUFNLGFBQWEsQ0FBQTtBQUNsQyxPQUFPLFNBQVMsTUFBTSxZQUFZLENBQUE7QUFDbEMsT0FBTyxFQUFFLG1CQUFtQixFQUFFLE1BQU0sNEJBQTRCLENBQUE7QUFFaEUsUUFBUSxDQUFDLG9DQUFvQyxFQUFFLEdBQUcsRUFBRTtJQUNsRCxTQUFTLEtBQUssQ0FBQyxHQUFXO1FBQ3hCLE9BQU8sT0FBTyxDQUFDLE1BQU0sQ0FBQyxhQUFhLEVBQUUsUUFBUSxDQUFDLEdBQUcsQ0FBQyxDQUFzQixDQUFBO0lBQzFFLENBQUM7SUFFRCxRQUFRLENBQUMscUJBQXFCLEVBQUUsR0FBRyxFQUFFO1FBQ25DLEVBQUUsQ0FBQyxtQ0FBbUMsRUFBRSxLQUFLLElBQUksRUFBRTtZQUNqRCxRQUFRO1lBQ1IsTUFBTSxPQUFPLEdBQUcsRUFBRSxDQUFBO1lBRWxCLE9BQU87WUFDUCxNQUFNLE1BQU0sR0FBRyxLQUFLLENBQUM7O21DQUVRLENBQUMsQ0FBQTtZQUM5QixNQUFNLENBQUMsRUFBRSxDQUFDLFFBQVEsRUFBRSxtQkFBbUIsQ0FBQyxPQUFPLENBQUMsQ0FBQyxDQUFBO1lBQ2pELE1BQU0sU0FBUyxDQUFDLE1BQU0sQ0FBQyxDQUFBO1lBRXZCLE9BQU87WUFDUCxNQUFNLENBQUMsT0FBTyxDQUFDLENBQUMsRUFBRSxDQUFDLElBQUksQ0FBQyxRQUFRLENBQUMsS0FBSyxFQUFFLEtBQUssQ0FBQyxDQUFBO1lBQzlDLE1BQU0sQ0FBQyxPQUFPLENBQUMsQ0FBQyxFQUFFLENBQUMsSUFBSSxDQUFDLFFBQVEsQ0FBQyxTQUFTLEVBQUUsVUFBVSxDQUFDLENBQUE7UUFDekQsQ0FBQyxDQUFDLENBQUE7UUFFRixFQUFFLENBQUMsd0JBQXdCLEVBQUUsS0FBSyxJQUFJLEVBQUU7WUFDdEMsUUFBUTtZQUNSLE1BQU0sT0FBTyxHQUFHLEVBQUUsQ0FBQTtZQUVsQixPQUFPO1lBQ1AsTUFBTSxNQUFNLEdBQUcsS0FBSyxDQUFDLHFDQUFxQyxDQUFDLENBQUE7WUFDM0QsTUFBTSxDQUFDLEVBQUUsQ0FBQyxRQUFRLEVBQUUsbUJBQW1CLENBQUMsT0FBTyxDQUFDLENBQUMsQ0FBQTtZQUNqRCxNQUFNLFNBQVMsQ0FBQyxNQUFNLENBQUMsQ0FBQTtZQUV2QixPQUFPO1lBQ1AsTUFBTSxDQUFDLE9BQU8sQ0FBQyxDQUFDLEVBQUUsQ0FBQyxJQUFJLENBQUMsRUFBRSxDQUFDLEVBQUUsQ0FBQyxDQUFBO1FBQ2hDLENBQUMsQ0FBQyxDQUFBO0lBQ0osQ0FBQyxDQUFDLENBQUE7QUFDSixDQUFDLENBQUMsQ0FBQSJ9
@@ -0,0 +1,43 @@
1
+ import { Stream } from 'stream'
2
+ import { expect } from 'chai'
3
+ import { parsers } from '@rdfjs/formats-common'
4
+ import toStream from 'into-stream'
5
+ import getStream from 'get-stream'
6
+ import { optionsFromPrefixes } from '../../lib/prefixHandler.js'
7
+
8
+ describe('@hydrofoil/talos/lib/prefixHandler', () => {
9
+ function parse(str: string) {
10
+ return parsers.import('text/turtle', toStream(str)) as unknown as Stream
11
+ }
12
+
13
+ describe('optionsFromPrefixes', () => {
14
+ it('sets options from parsed prefixes', async () => {
15
+ // given
16
+ const options = {}
17
+
18
+ // when
19
+ const stream = parse(`prefix talos: <foo:bar>
20
+ prefix talos: <foo:baz>
21
+ prefix talos: <another:also%20baz>`)
22
+ stream.on('prefix', optionsFromPrefixes(options))
23
+ await getStream(stream)
24
+
25
+ // then
26
+ expect(options).to.have.property('foo', 'baz')
27
+ expect(options).to.have.property('another', 'also baz')
28
+ })
29
+
30
+ it('ignores other prefixes', async () => {
31
+ // given
32
+ const options = {}
33
+
34
+ // when
35
+ const stream = parse('prefix schema: <http://schema.org/>')
36
+ stream.on('prefix', optionsFromPrefixes(options))
37
+ await getStream(stream)
38
+
39
+ // then
40
+ expect(options).to.deep.eq({})
41
+ })
42
+ })
43
+ })
package/tsconfig.json ADDED
@@ -0,0 +1,3 @@
1
+ {
2
+ "extends": "../../tsconfig.json"
3
+ }