forked from typedb/typedb-driver-examples
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathqueries.js
358 lines (312 loc) · 10.8 KB
/
queries.js
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
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
const GraknClient = require("grakn-client");
const readline = require("readline");
// to add a new query implementation:
// 1. add the question and function to the approriate list of dictionaries:
// current lists are: getQsFunc, aggregateQsFunc and computeQsFunc
// example:
// getQsFunc = [
// ...
// {
// "question": "new question?"
// "queryFunction": executeQuery#
// }
// ]
// 2. add the function and its implementation:
// use the template below
// the template for executeQuery functions
// function executeQueryFormat(question) {
// printToLog("Question: ", question)
// // queries are written as a list for better readibility
// const query = [
// "each line;",
// "as an element;",
// "ends with simocolon;",
// ];
// // join the query list elements with a new line before console.loging
// printToLog("Query:", query.join("\n"));
// // join the query list elements to obtain the quer as a string to be executed
// const query = query.join("");
// const iterator = await transaction.query(query);
// // ... retrieve the answers
// const result = "example result";
// printToLog("Result:", result);
// }
const getQsFunc = [
{
question:
"From 2018-09-10 onwards, which customers called the person with phone number +86 921 547 9004?",
queryFunction: executeQuery1
},
{
question:
"Who are the people aged under 20 who have received at least one phone call from a Cambridge customer aged over 50?",
queryFunction: executeQuery2
},
{
question:
"Who are the common contacts of customers with phone numbers +7 171 898 0853 and +370 351 224 5176?",
queryFunction: executeQuery3
},
{
question:
"Who are the customers who 1) have all called each other and 2) have all called person with phone number +48 894 777 5173 at least once?",
queryFunction: executeQuery4
}
];
const aggregateQsFunc = [
{
question:
"How does the average call duration among customers aged under 20 compare those aged over 40?",
queryFunction: executeQuery5
}
];
const queryExamples = getQsFunc.concat(aggregateQsFunc);
// utils
const log = console.log;
function printToLog(title, content) {
log(title);
log("");
log(content);
log("\n");
}
// From 2018-09-10 onwards, which customers called person with phone number +86 921 547 9004?
async function executeQuery1(question, transaction) {
printToLog("Question: ", question);
let query = [
"match",
" $customer isa person, has phone-number $phone-number;",
' $company isa company, has name "Telecom";',
" (customer: $customer, provider: $company) isa contract;",
' $target isa person, has phone-number "+86 921 547 9004";',
" (caller: $customer, callee: $target) isa call, has started-at $started-at;",
" $min-date == 2018-09-14T17:18:49; $started-at > $min-date;",
"get $phone-number;"
];
printToLog("Query:", query.join("\n"));
query = query.join("");
const iterator = await transaction.query(query);
const answers = await iterator.collect();
const result = await Promise.all(
answers.map(answer =>
answer.map()
.get("phone-number")
.value()
)
);
printToLog("Result:", result);
return result;
}
// who are the people aged under 20 who have received at least one phone call from a Cambridge customer aged over 60?
async function executeQuery2(question, transaction) {
printToLog("Question: ", question);
let query = [
"match ",
' $suspect isa person, has city "London", has age > 50;',
' $company isa company, has name "Telecom";',
" (customer: $suspect, provider: $company) isa contract;",
" $pattern-callee isa person, has age < 20;",
" (caller: $suspect, callee: $pattern-callee) isa call, has started-at $pattern-call-date;",
" $target isa person, has phone-number $phone-number;",
" not { (customer: $target, provider: $company) isa contract; };",
" (caller: $suspect, callee: $target) isa call, has started-at $target-call-date;",
" $target-call-date > $pattern-call-date;",
"get $phone-number;"
];
printToLog("Query:", query.join("\n"));
query = query.join("");
const iterator = await transaction.query(query);
const answers = await iterator.collect();
const result = await Promise.all(
answers.map(answer =>
answer.map()
.get("phone-number")
.value()
)
);
printToLog("Result:", result);
return result;
}
// "Who are the common contacts of customers with phone numbers +7 171 898 0853 and +370 351 224 5176?
async function executeQuery3(question, transaction) {
printToLog("Question: ", question);
let query = [
"match ",
" $common-contact isa person, has phone-number $phone-number;",
' $customer-a isa person, has phone-number "+7 171 898 0853";',
' $customer-b isa person, has phone-number "+370 351 224 5176";',
" (caller: $customer-a, callee: $common-contact) isa call;",
" (caller: $customer-b, callee: $common-contact) isa call;",
"get $phone-number;"
];
printToLog("Query:", query.join("\n"));
query = query.join("");
const iterator = await transaction.query(query);
const answers = await iterator.collect();
const result = await Promise.all(
answers.map(answer =>
answer.map()
.get("phone-number")
.value()
)
);
printToLog("Result:", result);
return result;
}
// Who are the customers who 1) have all called each other and 2) have all called person with phone number +48 894 777 5173 at least once?",
async function executeQuery4(question, transaction) {
printToLog("Question: ", question);
let query = [
"match ",
' $target isa person, has phone-number "+48 894 777 5173";',
' $company isa company, has name "Telecom";',
" $customer-a isa person, has phone-number $phone-number-a;",
" (customer: $customer-a, provider: $company) isa contract;",
" (caller: $customer-a, callee: $target) isa call;",
" $customer-b isa person, has phone-number $phone-number-b;",
" (customer: $customer-b, provider: $company) isa contract;",
" (caller: $customer-b, callee: $target) isa call;",
" (caller: $customer-a, callee: $customer-b) isa call;",
"get $phone-number-a, $phone-number-b;"
];
printToLog("Query:", query.join("\n"));
query = query.join("");
const iterator = await transaction.query(query);
const answers = await iterator.collect();
const result = await Promise.all(
answers.map(answer =>
answer.map()
.get("phone-number-a")
.value()
)
);
printToLog("Result:", result);
return result;
}
// How does the average call duration among customers aged under 20 compare those aged over 40?
async function executeQuery5(question, transaction) {
printToLog("Question: ", question);
firstQuery = [
'match',
' $customer isa person, has age < 20;',
' $company isa company, has name "Telecom";',
' (customer: $customer, provider: $company) isa contract;',
' (caller: $customer, callee: $anyone) isa call, has duration $duration;',
'get $duration; mean $duration;'
];
printToLog("Query:", firstQuery.join("\n"));
firstQuery = firstQuery.join("");
result = [];
const firstIterator = await transaction.query(firstQuery);
const firstAnswer = await firstIterator.collect();
let firstResult = 0;
if(firstAnswer.length > 0) {
firstResult = firstAnswer[0].number();
}
let output =
"Customers aged under 20 have made calls with average duration of " +
Math.round(firstResult) +
" seconds.\n";
result.push(firstResult);
secondQuery = [
'match ' +
' $customer isa person, has age > 40;',
' $company isa company, has name "Telecom";',
' (customer: $customer, provider: $company) isa contract;',
' (caller: $customer, callee: $anyone) isa call, has duration $duration;',
'get $duration; mean $duration;'
];
printToLog("Query:", secondQuery.join("\n"));
secondQuery = secondQuery.join("");
const secondIterator = await transaction.query(secondQuery);
const secondAnswer = await secondIterator.collect();
let secondResult = 0;
if(secondAnswer.length > 0) {
secondResult = secondAnswer[0].number();
}
output +=
"Customers aged over 40 have made calls with average duration of " +
Math.round(secondResult) +
" seconds.\n";
result.push(secondResult);
printToLog("Result:", output);
return result;
}
//
async function executeQuery6(question, transaction) {
printToLog("Question: ", question);
}
//
async function executeQuery7(question, transaction) {
printToLog("Question: ", question);
}
// execute all queries for all questions
async function executeAllQueries(transaction) {
for (queryExample of queryExamples) {
question = queryExample["question"];
queryFunction = queryExample["queryFunction"];
await queryFunction(question, transaction);
log("\n - - - - - - - - - - - - \n");
}
}
/**
* this is the main function
* prints the questions and asks for the input, based on which the corresponding function
* to execute the query is called
*/
async function executeQueries() {
// print questions
log("\nSelect a question for which you'd like to execute the query?\n");
for (let [index, queryExample] of queryExamples.entries())
log(index + 1 + ". " + queryExample["question"]);
log("");
// get user's selection and call the function for it
const rl = readline.createInterface({
input: process.stdin,
output: process.stdout
});
executeBasedOnSelection(rl);
}
/**
* a recursive function that terminates after receiving a valid input
* otherwise keeps asking
* @param {object} rl the readline insterface to receive input via console
* exists on completion
*/
function executeBasedOnSelection(rl) {
const question = "choose a number (0 for to answer all questions): ";
rl.question(question, async function (answer, rl) {
if (answer >= 0 && answer < queryExamples.length + 1) {
await processSelection(answer, "phone_calls");
process.exit(0);
}
executeBasedOnSelection(rl);
});
}
/**
* 1. create an instance of Grakn, connecting to the server
* 2. create a session of the instance, connecting to the keyspace phone_calls
* 3. create a transaction, off the session
* 4. call the function corresponding to the selected question
* 5. close the transaction
* 6. close the session
* 7. closes the client
* @param {integer} qsNumber the (question) number selected by the user
*/
async function processSelection(qsNumber, keyspaceName) {
const client = new GraknClient("localhost:48555"); // 1
const session = await client.session((keyspace = keyspaceName)); // 2
const transaction = await session.transaction().read(); // 3
if (qsNumber == 0) {
await executeAllQueries(transaction); // 4
} else {
const question = queryExamples[qsNumber - 1]["question"];
const queryFunction = queryExamples[qsNumber - 1]["queryFunction"];
await queryFunction(question, transaction); // 4
}
await transaction.close(); // 5
await session.close(); // 6
client.close(); // 7
}
module.exports.processSelection = processSelection;
module.exports.queryExamples = queryExamples;
module.exports.init = executeQueries;