@aj-archipelago/cortex 0.0.9 → 0.0.11

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.
Files changed (43) hide show
  1. package/config.js +17 -11
  2. package/graphql/chunker.js +97 -107
  3. package/graphql/graphql.js +19 -22
  4. package/graphql/parser.js +1 -1
  5. package/graphql/pathwayPrompter.js +8 -9
  6. package/graphql/pathwayResolver.js +12 -14
  7. package/graphql/pathwayResponseParser.js +2 -2
  8. package/graphql/plugins/azureTranslatePlugin.js +2 -2
  9. package/graphql/plugins/modelPlugin.js +67 -25
  10. package/graphql/plugins/openAiChatPlugin.js +3 -3
  11. package/graphql/plugins/openAiCompletionPlugin.js +5 -4
  12. package/graphql/plugins/openAiWhisperPlugin.js +7 -6
  13. package/graphql/prompt.js +1 -1
  14. package/graphql/pubsub.js +2 -2
  15. package/graphql/requestState.js +1 -1
  16. package/graphql/resolver.js +4 -4
  17. package/graphql/subscriptions.js +5 -4
  18. package/graphql/typeDef.js +53 -53
  19. package/index.js +5 -5
  20. package/lib/fileChunker.js +15 -11
  21. package/lib/keyValueStorageClient.js +5 -5
  22. package/lib/promiser.js +2 -2
  23. package/lib/request.js +11 -9
  24. package/lib/requestMonitor.js +2 -2
  25. package/package.json +15 -5
  26. package/pathways/basePathway.js +5 -4
  27. package/pathways/bias.js +2 -2
  28. package/pathways/chat.js +3 -2
  29. package/pathways/complete.js +4 -2
  30. package/pathways/edit.js +3 -2
  31. package/pathways/entities.js +3 -2
  32. package/pathways/index.js +25 -12
  33. package/pathways/lc_test.mjs +99 -0
  34. package/pathways/paraphrase.js +3 -2
  35. package/pathways/sentiment.js +3 -2
  36. package/pathways/summary.js +27 -10
  37. package/pathways/transcribe.js +4 -2
  38. package/pathways/translate.js +3 -2
  39. package/start.js +5 -2
  40. package/tests/chunkfunction.test.js +125 -0
  41. package/tests/chunking.test.js +25 -19
  42. package/tests/main.test.js +52 -38
  43. package/tests/translate.test.js +13 -10
@@ -2,6 +2,7 @@
2
2
  // Paraphrasing module
3
3
  // This module exports a prompt that takes an input text and rewrites it in a different way while maintaining the original meaning.
4
4
 
5
- module.exports = {
5
+ export default {
6
6
  prompt: `Rewrite the following:\n\n{{{text}}}`
7
- }
7
+ };
8
+
@@ -2,6 +2,7 @@
2
2
  // Sentiment detection module
3
3
  // This module exports a prompt that takes an input text and asks how it makes the AI feel.
4
4
 
5
- module.exports = {
5
+ export default {
6
6
  prompt: `How does the text below make you feel?\n\n{{text}}`,
7
- }
7
+ };
8
+
@@ -3,12 +3,12 @@
3
3
  // This module exports a prompt that takes an input text and generates a summary using a custom resolver.
4
4
 
5
5
  // Import required modules
6
- const { semanticTruncate } = require('../graphql/chunker');
7
- const { PathwayResolver } = require('../graphql/pathwayResolver');
6
+ import { semanticTruncate } from '../graphql/chunker.js';
7
+ import { PathwayResolver } from '../graphql/pathwayResolver.js';
8
8
 
9
- module.exports = {
9
+ export default {
10
10
  // The main prompt function that takes the input text and asks to generate a summary.
11
- prompt: `{{{text}}}\n\nWrite a summary of the above text:\n\n`,
11
+ prompt: `{{{text}}}\n\nWrite a summary of the above text. If the text is in a language other than english, make sure the summary is written in the same language:\n\n`,
12
12
 
13
13
  // Define input parameters for the prompt, such as the target length of the summary.
14
14
  inputParameters: {
@@ -26,7 +26,7 @@ module.exports = {
26
26
  return await pathwayResolver.resolve(args);
27
27
  }
28
28
 
29
- const errorMargin = 0.2;
29
+ const errorMargin = 0.1;
30
30
  const lowTargetLength = originalTargetLength * (1 - errorMargin);
31
31
  const targetWords = Math.round(originalTargetLength / 6.6);
32
32
 
@@ -39,17 +39,32 @@ module.exports = {
39
39
  let summary = '';
40
40
  let pathwayResolver = new PathwayResolver({ config, pathway, args, requestState });
41
41
 
42
-
43
42
  // Modify the prompt to be words-based instead of characters-based.
44
- pathwayResolver.pathwayPrompt = `{{{text}}}\n\nWrite a summary of the above text in exactly ${targetWords} words:\n\n`
43
+ pathwayResolver.pathwayPrompt = `Write a summary of all of the text below. If the text is in a language other than english, make sure the summary is written in the same language. Your summary should be ${targetWords} words in length.\n\nText:\n\n{{{text}}}\n\nSummary:\n\n`
45
44
 
46
45
  let i = 0;
47
- // Reprompt if summary is too long or too short.
48
- while (((summary.length > originalTargetLength) || (summary.length < lowTargetLength)) && i < MAX_ITERATIONS) {
46
+ // Make sure it's long enough to start
47
+ while ((summary.length < lowTargetLength) && i < MAX_ITERATIONS) {
49
48
  summary = await pathwayResolver.resolve(args);
50
49
  i++;
51
50
  }
52
51
 
52
+ // If it's too long, it could be because the input text was chunked
53
+ // and now we have all the chunks together. We can summarize that
54
+ // to get a comprehensive summary.
55
+ if (summary.length > originalTargetLength) {
56
+ pathwayResolver.pathwayPrompt = `Write a summary of all of the text below. If the text is in a language other than english, make sure the summary is written in the same language. Your summary should be ${targetWords} words in length.\n\nText:\n\n${summary}\n\nSummary:\n\n`
57
+ summary = await pathwayResolver.resolve(args);
58
+ i++;
59
+
60
+ // Now make sure it's not too long
61
+ while ((summary.length > originalTargetLength) && i < MAX_ITERATIONS) {
62
+ pathwayResolver.pathwayPrompt = `${summary}\n\nIs that less than ${targetWords} words long? If not, try again using a length of no more than ${targetWords} words.\n\n`;
63
+ summary = await pathwayResolver.resolve(args);
64
+ i++;
65
+ }
66
+ }
67
+
53
68
  // If the summary is still too long, truncate it.
54
69
  if (summary.length > originalTargetLength) {
55
70
  return semanticTruncate(summary, originalTargetLength);
@@ -57,4 +72,6 @@ module.exports = {
57
72
  return summary;
58
73
  }
59
74
  }
60
- }
75
+ };
76
+
77
+
@@ -1,8 +1,10 @@
1
- module.exports = {
1
+ export default {
2
2
  prompt: `{{text}}`,
3
3
  model: `oai-whisper`,
4
4
  inputParameters: {
5
5
  file: ``,
6
6
  },
7
7
  timeout: 600, // in seconds
8
- }
8
+ };
9
+
10
+
@@ -2,7 +2,7 @@
2
2
  // Translation module
3
3
  // This module exports a prompt that takes an input text and translates it from one language to another.
4
4
 
5
- module.exports = {
5
+ export default {
6
6
  // Set the temperature to 0 to favor more deterministic output when generating translations.
7
7
  temperature: 0,
8
8
 
@@ -15,4 +15,5 @@ module.exports = {
15
15
 
16
16
  // Set the timeout for the translation process, in seconds.
17
17
  timeout: 300,
18
- }
18
+ };
19
+
package/start.js CHANGED
@@ -1,3 +1,6 @@
1
- const { startServer } = require('./index')();
1
+ import startServerFactory from './index.js';
2
2
 
3
- startServer && startServer();
3
+ (async () => {
4
+ const { startServer } = await startServerFactory();
5
+ startServer && startServer();
6
+ })();
@@ -0,0 +1,125 @@
1
+ import test from 'ava';
2
+ import { getSemanticChunks } from '../graphql/chunker.js';
3
+ import { encode } from 'gpt-3-encoder';
4
+
5
+ const testText = `Lorem ipsum dolor sit amet, consectetur adipiscing elit. In id erat sem. Phasellus ac dapibus purus, in fermentum nunc. Mauris quis rutrum magna. Quisque rutrum, augue vel blandit posuere, augue magna convallis turpis, nec elementum augue mauris sit amet nunc. Aenean sit amet leo est. Nunc ante ex, blandit et felis ut, iaculis lacinia est. Phasellus dictum orci id libero ullamcorper tempor.
6
+
7
+ Vivamus id pharetra odio. Sed consectetur leo sed tortor dictum venenatis.Donec gravida libero non accumsan suscipit.Donec lectus turpis, ullamcorper eu pulvinar iaculis, ornare ut risus.Phasellus aliquam, turpis quis viverra condimentum, risus est pretium metus, in porta ipsum tortor vitae elit.Pellentesque id finibus erat. In suscipit, sapien non posuere dignissim, augue nisl ultrices tortor, sit amet eleifend nibh elit at risus.
8
+
9
+ Donec diam ligula, sagittis ut nisl tincidunt, porta sodales magna.Vestibulum ut dui arcu.Fusce at dolor ex.Aliquam eu justo non libero volutpat pulvinar at id urna.Donec nec purus sed elit bibendum faucibus.Pellentesque habitant morbi tristique senectus et netus et malesuada fames ac turpis egestas.Vivamus iaculis mattis velit, ut lacinia massa lacinia quis.Phasellus porttitor gravida ex, id aliquet eros rhoncus quis. Ut fringilla, lectus a vehicula luctus, diam odio convallis dolor, sodales pharetra nulla ex dictum justo.Ut faucibus, augue quis dictum iaculis, diam leo maximus sapien, sit amet vulputate eros quam sed sem.Cras malesuada, sapien sit amet iaculis euismod, nunc odio lacinia est, dictum iaculis ante nisi in est.Fusce vehicula lorem tellus.Nullam a tempus nisi .
10
+
11
+ Sed ut lectus nec ligula blandit tempus.Donec faucibus turpis id urna vehicula imperdiet.Duis tempor vitae orci interdum dignissim.Phasellus sed efficitur sem.Nullam accumsan, turpis vitae consectetur ullamcorper, lectus purus tincidunt nisi, in pulvinar leo tortor at sem.Donec at feugiat dui, nec rhoncus nibh.Nam faucibus ultrices nisl at lobortis.Morbi congue, nisl vel fermentum tristique, dui ipsum rhoncus massa, non varius nibh massa in turpis.Vestibulum vulputate, felis quis lacinia porta, nulla ex volutpat lorem, non rhoncus neque erat quis arcu.Morbi massa nisl, hendrerit eget tortor condimentum, lobortis dapibus sem.Aliquam ut dapibus elit.Sed porta dignissim ante.Nullam interdum ligula et massa vehicula, vel gravida diam laoreet.Vivamus et enim eget turpis pellentesque laoreet.Vivamus pellentesque neque et mauris imperdiet pulvinar.
12
+
13
+ Aliquam eget ligula congue, tincidunt magna eu, rutrum urna.Sed consequat orci est, vel laoreet magna tincidunt sit amet.Curabitur eget condimentum odio, vitae condimentum elit.Duis viverra lobortis magna.Pellentesque habitant morbi tristique senectus et netus et malesuada fames ac turpis egestas.Sed facilisis mi eu scelerisque pharetra.Cras et massa odio.Praesent quis nulla vitae mi blandit egestas ac vitae libero.Cras ultricies ex non consequat scelerisque.Nulla et est ac sem placerat convallis ac vitae massa.Phasellus lobortis mauris vel est vehicula lobortis.Curabitur ipsum ipsum, ullamcorper eget placerat sit amet, dapibus iaculis dui.Phasellus facilisis rutrum metus nec euismod.
14
+
15
+ Nam viverra est ac orci rhoncus, mollis mattis mi lobortis.Maecenas lectus ex, pulvinar vel mauris vel, egestas ornare massa.Nam placerat, tellus vel ullamcorper ullamcorper, enim felis egestas tellus, eu dictum augue tortor vel libero.Integer vel nunc felis.Nulla vehicula et enim non luctus.Vestibulum non odio magna.Donec vitae ipsum et nisl vestibulum maximus eu at augue.Morbi ac tristique quam.Suspendisse vestibulum nec dui et consectetur.Aliquam a dapibus dolor, sit amet fringilla eros.Nam id lorem nulla.
16
+
17
+ Proin vulputate risus purus, id tincidunt magna eleifend vel.Pellentesque et commodo leo, sit amet molestie nunc.Nunc purus lectus, interdum ut mauris ac, varius pretium magna.Etiam sollicitudin eros at pretium molestie.Cras fermentum sagittis elit at egestas.Fusce auctor lacinia nisl ac ullamcorper.Interdum et malesuada fames ac ante ipsum primis in faucibus.Fusce commodo pretium urna vel consequat.In finibus tellus vitae magna pharetra, porttitor egestas libero cursus.Donec eget tincidunt dolor, ac tristique diam.Etiam interdum dictum ex suscipit tempus.In hac habitasse platea dictumst.Nulla ornare libero a leo mollis, sed gravida leo finibus.Nunc ornare, dolor ac convallis varius, quam ipsum ultricies dui, non vehicula est eros eget ipsum.Mauris vel rhoncus ligula, non porta metus.
18
+
19
+ Ut non felis pretium leo viverra tincidunt.Vivamus et ligula commodo dolor faucibus gravida.Quisque eu dolor ac metus pretium pharetra.Integer mattis efficitur libero, sed condimentum nulla ultricies eu.Donec turpis orci, fermentum vitae imperdiet nec, luctus quis lectus.Nunc viverra ornare libero.Vestibulum elementum tempus tortor id semper.
20
+
21
+ Aliquam in dapibus risus.Praesent vitae condimentum elit, sodales pellentesque diam.Curabitur luctus pellentesque nunc, ut eleifend urna dictum ac.Aenean rhoncus lacinia quam a suscipit.Proin purus metus, egestas a pretium eu, tempus ut ante.Sed tellus turpis, hendrerit consequat porta id, porttitor non dolor.Proin volutpat massa a dui dictum facilisis a vel eros.Fusce eu efficitur odio.Aliquam interdum metus id ex dapibus dapibus.Nullam porttitor non sapien nec rhoncus.Cras iaculis fringilla cursus.Praesent at leo orci.Sed eget vulputate eros, eget auctor sapien.Nulla auctor, lectus ut tincidunt rhoncus, ante lorem volutpat eros, ac tincidunt enim ipsum at ex.Fusce dolor arcu, pretium eget elementum vel, semper at ipsum.
22
+
23
+ Integer rhoncus fringilla felis ac tincidunt.Phasellus eu ultricies tellus.Sed pharetra, eros sed dignissim mattis, mi lectus blandit elit, vitae euismod ipsum sapien a eros.Aliquam lobortis tellus venenatis, sagittis lorem non, eleifend odio.Duis ultrices urna vel commodo varius.Sed ultricies mauris ut velit dignissim, eu lobortis ex tempor.Praesent vitae iaculis nisl.Vestibulum id convallis tellus.Vivamus eu consectetur erat.Curabitur interdum est non nibh malesuada ullamcorper.Phasellus mollis arcu a pharetra lacinia.Praesent sit amet sem non dui iaculis tincidunt.Aliquam vitae libero viverra metus feugiat volutpat ut eget sem.Nam facilisis pulvinar urna, ut venenatis ligula accumsan finibus.Maecenas nec aliquam nulla.Maecenas mattis magna erat.
24
+
25
+ Nunc a nulla sed ante sollicitudin ultrices a a ante.Sed feugiat scelerisque leo, eget venenatis orci cursus eget.Ut pretium leo et nunc sodales, in luctus erat faucibus.Interdum et malesuada fames ac ante ipsum primis in faucibus.Mauris facilisis lorem quis turpis commodo, id vulputate metus molestie.Fusce id neque vestibulum, pretium elit in, ultrices justo.Praesent turpis dui, ullamcorper in vulputate ut, posuere at sapien.Etiam laoreet ultrices felis, id venenatis purus.Sed nec mauris pharetra, rhoncus sem non, interdum justo.Nulla sed tincidunt nisi.Suspendisse luctus viverra volutpat.Duis arcu nulla, euismod eu scelerisque in, vulputate eget quam.
26
+
27
+ Vestibulum ante ipsum primis in faucibus orci luctus et ultrices posuere cubilia curae; Fusce at dignissim quam.Suspendisse eget metus nec sem accumsan sagittis.Suspendisse non mollis lacus.Donec ac consectetur ante.Nullam luctus, nibh ac imperdiet porta, sem felis bibendum nibh, ut sollicitudin libero nulla a sapien.Sed tristique odio enim, eget tempor enim cursus vel.Morbi tristique lobortis tortor, nec placerat lorem suscipit ac.Nullam sed sodales diam, sed tincidunt est.Quisque semper velit sed risus dictum pretium.Proin condimentum, nisi a vulputate tristique, tellus erat scelerisque nisi, tincidunt viverra est neque non magna.Quisque nibh augue, interdum non justo et, varius rutrum erat.Pellentesque habitant morbi tristique senectus et netus et malesuada fames ac turpis egestas.
28
+
29
+ Vestibulum et lorem auctor, vestibulum nisl id, elementum metus.Pellentesque quis mi a augue consectetur cursus.Morbi sodales risus et faucibus dictum.Ut in lobortis nisl, et euismod nisl.Donec ornare tellus placerat, blandit justo quis, pharetra nisl.Nulla scelerisque magna at nisi suscipit commodo.Fusce pellentesque in elit et consequat.Phasellus vehicula accumsan enim, vitae pellentesque nulla.Nullam id arcu vitae nunc consectetur mattis.Fusce ac sapien vel mi congue fringilla.Nulla mattis consectetur fringilla.Morbi orci elit, tempor in rhoncus eget, fringilla eget erat.
30
+
31
+ Phasellus nec lorem lectus.Donec in cursus elit.In dictum elementum odio a scelerisque.Phasellus ac sapien eget velit accumsan elementum.Mauris odio eros, vulputate eu aliquet a, bibendum in quam.Integer euismod libero ac massa imperdiet, ullamcorper cursus risus auctor.Nam rutrum erat eget tortor suscipit semper sit amet nec mauris.Pellentesque nec semper neque.Nunc fringilla nisl erat, a sollicitudin tortor accumsan finibus.
32
+
33
+ Integer vulputate ex dui, vitae scelerisque purus viverra vel.Cras ultricies purus in nibh dapibus, non hendrerit nulla aliquam.Fusce vitae gravida urna.Mauris eleifend rutrum ex, at fermentum enim fringilla quis.Suspendisse dignissim est eget tempus condimentum.Fusce scelerisque, felis et malesuada dictum, mauris dolor cursus ex, eget pulvinar sem nulla id diam.Ut volutpat tincidunt efficitur.Nunc vel risus fringilla, lacinia urna vitae, aliquet nulla.Nunc sed pulvinar dolor, eu fermentum velit.Curabitur a pretium quam, ut consectetur neque.Nunc ultricies, ex sed mattis efficitur, nulla nunc convallis odio, sit amet pellentesque orci tortor ut sapien.Vivamus felis orci, ultricies eget lacinia at, blandit vitae quam.In lacinia dui nec tincidunt maximus.Donec feugiat consectetur bibendum.Aenean eget vestibulum lacus.
34
+
35
+ Suspendisse vel molestie magna, et viverra justo.Aenean nec mi felis.Nam lacinia purus et congue facilisis.Pellentesque eget odio sed sem tincidunt imperdiet.Proin finibus ex nec placerat aliquet.Phasellus quis sapien nunc.Mauris eu augue aliquam sem suscipit vehicula a luctus augue.Phasellus ac scelerisque nibh.Nullam eleifend eleifend sapien eget convallis.
36
+
37
+ Nunc vitae metus risus.Ut iaculis dolor accumsan bibendum posuere.Morbi vitae odio sed velit dictum consequat.Aliquam vel erat vitae lacus luctus cursus vel ut risus.Aliquam a nunc eu lorem consequat finibus.Sed non enim vestibulum, ornare dui id, dignissim turpis.Etiam fermentum rutrum porttitor.Maecenas id nisl sodales, ornare turpis placerat, tincidunt dui.
38
+
39
+ Nulla aliquam purus at leo fringilla euismod.Praesent condimentum augue nibh, sed scelerisque mauris bibendum vitae.Vivamus maximus enim non massa commodo gravida.Cras iaculis elit ac est dapibus convallis.Quisque in tortor tincidunt, placerat turpis pulvinar, rhoncus orci.In vel risus et lacus lacinia volutpat.Maecenas facilisis fermentum dictum.Lorem ipsum dolor sit amet, consectetur adipiscing elit.Praesent aliquam pretium pellentesque.In eleifend leo eros, in lobortis eros elementum maximus.Fusce in orci ut massa vehicula mollis vitae non nibh.Sed ac porttitor urna.Nulla ac venenatis sapien, eget vulputate metus.
40
+
41
+ Mauris hendrerit lacus quam, vel mollis ligula porttitor ac.Nulla ornare libero at faucibus dictum.Donec tincidunt viverra sapien nec tincidunt.Donec leo sapien, rutrum quis dui a, auctor sodales nisi.Fusce condimentum eros sit amet ligula viverra, eget ullamcorper erat dapibus.Suspendisse dignissim ligula sed luctus aliquet.Aenean consectetur enim non nibh semper volutpat.
42
+
43
+ Mauris diam dolor, maximus et ultrices sed, semper sed felis.Morbi ac eros tellus.Maecenas eget ex vitae quam lacinia eleifend non nec leo.Donec condimentum consectetur nunc, quis luctus elit commodo eu.Nunc tincidunt condimentum neque, sed porta ligula porttitor et.Suspendisse scelerisque id massa sit amet placerat.Sed eleifend aliquet facilisis.Donec ac purus nec metus vestibulum euismod.Maecenas sollicitudin consequat ornare.Suspendisse pharetra vehicula eros nec malesuada.`;
44
+
45
+ test('should return identical text that chunker was passed, given large chunk size (1812)', async t => {
46
+ const maxChunkToken = 1812;
47
+ const chunks = getSemanticChunks(testText, maxChunkToken);
48
+ t.true(chunks.length > 0); //check chunking
49
+ t.true(chunks.every(chunk => encode(chunk).length <= maxChunkToken)); //check chunk size
50
+ const recomposedText = chunks.reduce((acc, chunk) => acc + chunk, '');
51
+ t.is(recomposedText, testText); //check recomposition
52
+ });
53
+
54
+ test('should return identical text that chunker was passed, given medium chunk size (500)', async t => {
55
+ const maxChunkToken = 500;
56
+ const chunks = getSemanticChunks(testText, maxChunkToken);
57
+ t.true(chunks.length > 1); //check chunking
58
+ t.true(chunks.every(chunk => encode(chunk).length <= maxChunkToken)); //check chunk size
59
+ const recomposedText = chunks.reduce((acc, chunk) => acc + chunk, '');
60
+ t.is(recomposedText, testText); //check recomposition
61
+ });
62
+
63
+ test('should return identical text that chunker was passed, given tiny chunk size (1)', async t => {
64
+ const maxChunkToken = 1;
65
+ const chunks = getSemanticChunks(testText, maxChunkToken);
66
+ t.true(chunks.length > 1); //check chunking
67
+ t.true(chunks.every(chunk => encode(chunk).length <= maxChunkToken)); //check chunk size
68
+ const recomposedText = chunks.reduce((acc, chunk) => acc + chunk, '');
69
+ t.is(recomposedText, testText); //check recomposition
70
+ });
71
+
72
+ /*
73
+ it('should return identical text that chunker was passed, given tiny chunk size (1)', () => {
74
+ const maxChunkToken = 1;
75
+ const chunks = getSemanticChunks(testText, maxChunkToken);
76
+ expect(chunks.length).toBeGreaterThan(1); //check chunking
77
+ expect(chunks.every(chunk => encode(chunk).length <= maxChunkToken)).toBe(true); //check chunk size
78
+ const recomposedText = chunks.reduce((acc, chunk) => acc + chunk, '');
79
+ expect(recomposedText).toBe(testText); //check recomposition
80
+ });
81
+
82
+ it('should return identical text that chunker was passed, given huge chunk size (32000)', () => {
83
+ const maxChunkToken = 32000;
84
+ const chunks = getSemanticChunks(testText, maxChunkToken);
85
+ expect(chunks.length).toBe(1); //check chunking
86
+ expect(chunks.every(chunk => encode(chunk).length <= maxChunkToken)).toBe(true); //check chunk size
87
+ const recomposedText = chunks.reduce((acc, chunk) => acc + chunk, '');
88
+ expect(recomposedText).toBe(testText); //check recomposition
89
+ });
90
+
91
+ const testTextNoSpaces = `Loremipsumdolorsitamet,consecteturadipiscingelit.Inideratsem.Phasellusacdapibuspurus,infermentumnunc.Maurisquisrutrummagna.Quisquerutrum,auguevelblanditposuere,auguemagnacon vallisturpis,necelementumauguemaurissitametnunc.Aeneansitametleoest.Nuncanteex,blanditetfelisut,iaculislaciniaest.Phasellusdictumorciidliberoullamcorpertempor.Vivamusidpharetraodioq.Sedconsecteturleosedtortordictumvenenatis.Donecgravidaliberononaccumsansuscipit.Doneclectusturpis,ullamcorpereupulvinariaculis,ornareutrisus.Phasellusaliquam,turpisquisviverracondimentum,risusestpretiummetus,inportaips umtortorvita elit.Pellentesqueidfinibuserat.Insuscipit,sapiennonposueredignissim,auguenisl ultricestortor,sitameteleifendnibhelitatrisus.`;
92
+
93
+ it('should return identical text that chunker was passed, given no spaces and small chunks(5)', () => {
94
+ const maxChunkToken = 5;
95
+ const chunks = getSemanticChunks(testTextNoSpaces, maxChunkToken);
96
+ expect(chunks.length).toBeGreaterThan(0); //check chunking
97
+ expect(chunks.every(chunk => encode(chunk).length <= maxChunkToken)).toBe(true); //check chunk size
98
+ const recomposedText = chunks.reduce((acc, chunk) => acc + chunk, '');
99
+ expect(recomposedText).toBe(testTextNoSpaces); //check recomposition
100
+ });
101
+
102
+ const testTextShortWeirdSpaces=`Lorem ipsum dolor sit amet, consectetur adipiscing elit. In id erat sem. Phasellus ac dapibus purus, in fermentum nunc.............................. Mauris quis rutrum magna. Quisque rutrum, augue vel blandit posuere, augue magna convallis turpis, nec elementum augue mauris sit amet nunc. Aenean sit a;lksjdf 098098- -23 eln ;lkn l;kn09 oij[0u ,,,,,,,,,,,,,,,,,,,,, amet leo est. Nunc ante ex, blandit et felis ut, iaculis lacinia est. Phasellus dictum orci id libero ullamcorper tempor.
103
+
104
+
105
+
106
+
107
+ Vivamus id pharetra odio. Sed consectetur leo sed tortor dictum venenatis.Donec gravida libero non accumsan suscipit.Donec lectus turpis, ullamcorper eu pulvinar iaculis, ornare ut risus.Phasellus aliquam, turpis quis viverra condimentum, risus est pretium metus, in porta ipsum tortor vitae elit.Pellentesque id finibus erat. In suscipit, sapien non posuere dignissim, augue nisl ultrices tortor, sit amet eleifend nibh elit at risus.`;
108
+
109
+ it('should return identical text that chunker was passed, given weird spaces and tiny chunks(1)', () => {
110
+ const maxChunkToken = 1;
111
+ const chunks = getSemanticChunks(testTextShortWeirdSpaces, maxChunkToken);
112
+ expect(chunks.length).toBeGreaterThan(0); //check chunking
113
+ expect(chunks.every(chunk => encode(chunk).length <= maxChunkToken)).toBe(true); //check chunk size
114
+ const recomposedText = chunks.reduce((acc, chunk) => acc + chunk, '');
115
+ expect(recomposedText).toBe(testTextShortWeirdSpaces); //check recomposition
116
+ });
117
+
118
+ it('should return identical text that chunker was passed, given weird spaces and small chunks(10)', () => {
119
+ const maxChunkToken = 1;
120
+ const chunks = getSemanticChunks(testTextShortWeirdSpaces, maxChunkToken);
121
+ expect(chunks.length).toBeGreaterThan(0); //check chunking
122
+ expect(chunks.every(chunk => encode(chunk).length <= maxChunkToken)).toBe(true); //check chunk size
123
+ const recomposedText = chunks.reduce((acc, chunk) => acc + chunk, '');
124
+ expect(recomposedText).toBe(testTextShortWeirdSpaces); //check recomposition
125
+ });*/
@@ -1,15 +1,19 @@
1
- const { getTestServer } = require('./main.test');
1
+ import test from 'ava';
2
+ import { getTestServer, initTypeDefsResolvers } from './main.test.js';
2
3
 
3
- jest.setTimeout(1800000);
4
+ let testServer;
4
5
 
5
- const testServer = getTestServer();
6
+ test.before(async () => {
7
+ await initTypeDefsResolvers();
8
+ testServer = getTestServer();
9
+ });
6
10
 
7
- //stop server after all tests
8
- afterAll(async () => {
11
+ test.after.always(async () => {
9
12
  await testServer.stop();
10
13
  });
11
14
 
12
- it('chunking test of translate endpoint with huge text', async () => {
15
+ test('chunking test of translate endpoint with huge text', async t => {
16
+ t.timeout(180000);
13
17
  const response = await testServer.executeOperation({
14
18
  query: 'query translate($text: String!) { translate(text: $text) { result } }',
15
19
  variables: {
@@ -54,12 +58,12 @@ Mauris hendrerit lacus quam, vel mollis ligula porttitor ac.Nulla ornare libero
54
58
  Mauris diam dolor, maximus et ultrices sed, semper sed felis.Morbi ac eros tellus.Maecenas eget ex vitae quam lacinia eleifend non nec leo.Donec condimentum consectetur nunc, quis luctus elit commodo eu.Nunc tincidunt condimentum neque, sed porta ligula porttitor et.Suspendisse scelerisque id massa sit amet placerat.Sed eleifend aliquet facilisis.Donec ac purus nec metus vestibulum euismod.Maecenas sollicitudin consequat ornare.Suspendisse pharetra vehicula eros nec malesuada.` },
55
59
  });
56
60
 
57
- expect(response.errors).toBeUndefined();
58
- expect(response.data?.translate.result.length).toBeGreaterThan(1000); //check return length huge
61
+ t.is(response.errors, undefined);
62
+ t.true(response.data?.translate.result.length > 1000);
59
63
  });
60
64
 
61
-
62
- it('chunking test of translate endpoint with single long text sentence', async () => {
65
+ test('chunking test of translate endpoint with single long text sentence', async t => {
66
+ t.timeout(180000);
63
67
  const response = await testServer.executeOperation({
64
68
  query: 'query translate($text: String!) { translate(text: $text) { result } }',
65
69
  variables: {
@@ -67,11 +71,12 @@ it('chunking test of translate endpoint with single long text sentence', async (
67
71
  }
68
72
  });
69
73
 
70
- expect(response.errors).toBeUndefined();
71
- expect(response.data?.translate.result.length).toBeGreaterThan(200); //check return length huge
74
+ t.is(response.errors, undefined);
75
+ t.true(response.data?.translate.result.length > 200);
72
76
  });
73
77
 
74
- it('chunking test of translate endpoint with two long text sentence', async () => {
78
+ test('chunking test of translate endpoint with two long text sentence', async t => {
79
+ t.timeout(180000);
75
80
  const response = await testServer.executeOperation({
76
81
  query: 'query translate($text: String!) { translate(text: $text) { result } }',
77
82
  variables: {
@@ -79,11 +84,12 @@ it('chunking test of translate endpoint with two long text sentence', async () =
79
84
  }
80
85
  });
81
86
 
82
- expect(response.errors).toBeUndefined();
83
- expect(response.data?.translate.result.length).toBeGreaterThan(500); //check return length huge
87
+ t.is(response.errors, undefined);
88
+ t.true(response.data?.translate.result.length > 500);
84
89
  });
85
90
 
86
- it('chunking test...', async () => {
91
+ test('chunking test...', async t => {
92
+ t.timeout(180000);
87
93
  const response = await testServer.executeOperation({
88
94
  query: 'query translate($text: String!) { translate(text: $text) { result } }',
89
95
  variables: {
@@ -144,6 +150,6 @@ it('chunking test...', async () => {
144
150
  }
145
151
  });
146
152
 
147
- expect(response.errors).toBeUndefined();
148
- expect(response.data?.translate.result.length).toBeGreaterThan(500); //check return length huge
149
- });
153
+ t.is(response.errors, undefined);
154
+ t.true(response.data?.translate.result.length > 500);
155
+ });
@@ -1,98 +1,112 @@
1
- const { ApolloServer } = require('apollo-server');
2
- const { config } = require('../config');
3
- const { typeDefs, resolvers } = require('../index')();
1
+ import test from 'ava';
2
+ import { ApolloServer } from 'apollo-server';
3
+ import { config } from '../config.js';
4
+ import typeDefsresolversFactory from '../index.js';
4
5
 
5
- jest.setTimeout(60000);
6
+ let typeDefs;
7
+ let resolvers;
6
8
 
7
- const getTestServer = () => {
8
- return new ApolloServer({
9
- typeDefs,
10
- resolvers,
11
- context: () => ({ config, requestState: {} }),
12
- });
13
- }
9
+ const initTypeDefsResolvers = async () => {
10
+ const result = await typeDefsresolversFactory();
11
+ typeDefs = result.typeDefs;
12
+ resolvers = result.resolvers;
13
+ };
14
14
 
15
- const testServer = getTestServer();
15
+ const getTestServer = () => {
16
+ return new ApolloServer({
17
+ typeDefs,
18
+ resolvers,
19
+ context: () => ({ config, requestState: {} }),
20
+ });
21
+ };
22
+
23
+ let testServer;
24
+
25
+ test.before(async () => {
26
+ await initTypeDefsResolvers();
27
+ testServer = getTestServer();
28
+ });
16
29
 
17
30
  //stop server after all tests
18
- afterAll(async () => {
19
- await testServer.stop();
31
+ test.after.always('cleanup', async () => {
32
+ await testServer.stop();
20
33
  });
21
34
 
22
- it('validates bias endpoint', async () => {
35
+ test('validates bias endpoint', async (t) => {
23
36
  const response = await testServer.executeOperation({
24
37
  query: 'query bias($text: String!) { bias(text: $text) { result } }',
25
38
  variables: { text: 'hello there my dear world!' },
26
39
  });
27
40
 
28
- expect(response.errors).toBeUndefined();
29
- expect(response.data?.bias?.result).toMatch(/(yes|no|bias)/i)
41
+ t.is(response.errors, undefined);
42
+ t.regex(response.data?.bias?.result, /(yes|no|bias)/i);
30
43
  });
31
44
 
32
- it('validates completion endpoint', async () => {
45
+ test('validates completion endpoint', async (t) => {
33
46
  const response = await testServer.executeOperation({
34
47
  query: 'query complete($text: String!) { complete(text: $text) { result } }',
35
48
  variables: { text: 'hello there my dear world!' },
36
49
  });
37
50
 
38
- expect(response.errors).toBeUndefined();
39
- expect(response.data?.complete?.result.length).toBeGreaterThan(0);
51
+ t.is(response.errors, undefined);
52
+ t.true(response.data?.complete?.result.length > 0);
40
53
  });
41
54
 
42
- it('validates entities endpoint with given num of count return', async () => {
55
+ test('validates entities endpoint with given num of count return', async (t) => {
43
56
  const response = await testServer.executeOperation({
44
57
  query: 'query entities($text: String!, $count: Int) { entities(text: $text, count: $count){ result { name, definition } } }',
45
58
  variables: { text: 'hello there my dear world!', count: 3 },
46
59
  });
47
60
 
48
- expect(response.errors).toBeUndefined();
49
- expect(response.data?.entities.result.length).toBe(3);
61
+ t.is(response.errors, undefined);
62
+ t.is(response.data?.entities.result.length, 3);
50
63
  response.data?.result?.entities.forEach((entity) => {
51
- expect(entity.name).toBeDefined();
52
- expect(entity.definition).toBeDefined();
64
+ t.truthy(entity.name);
65
+ t.truthy(entity.definition);
53
66
  });
54
67
  });
55
68
 
56
- it('validates paraphrase endpoint', async () => {
69
+ test('validates paraphrase endpoint', async (t) => {
57
70
  const response = await testServer.executeOperation({
58
71
  query: 'query paraphrase($text: String!) { paraphrase(text: $text) { result } }',
59
72
  variables: { text: 'hello there my dear world!' },
60
73
  });
61
74
 
62
- expect(response.errors).toBeUndefined();
63
- expect(response.data?.paraphrase?.result).toBeDefined();
75
+ t.is(response.errors, undefined);
76
+ t.truthy(response.data?.paraphrase?.result);
64
77
  });
65
78
 
66
- it('validates sentiment endpoint', async () => {
79
+ test('validates sentiment endpoint', async (t) => {
67
80
  const response = await testServer.executeOperation({
68
81
  query: 'query sentiment($text: String!) { sentiment(text: $text) { result } }',
69
82
  variables: { text: 'hello there my dear world!' },
70
83
  });
71
84
 
72
- expect(response.errors).toBeUndefined();
73
- expect(response.data?.sentiment.result).toBeDefined();
85
+ t.is(response.errors, undefined);
86
+ t.truthy(response.data?.sentiment.result);
74
87
  });
75
88
 
76
- it('validates edit endpoint', async () => {
89
+ test('validates edit endpoint', async (t) => {
77
90
  const response = await testServer.executeOperation({
78
91
  query: 'query edit($text: String!) { edit(text: $text) { result } }',
79
92
  variables: { text: 'helo there my dear worldd!' },
80
93
  });
81
94
 
82
- expect(response.errors).toBeUndefined();
83
- expect(response.data?.edit.result).toMatch(/hello.*world/i);
95
+ t.is(response.errors, undefined);
96
+ t.regex(response.data?.edit.result, /hello.*world/i);
84
97
  });
85
98
 
86
- it('validates summary endpoint', async () => {
99
+ test('validates summary endpoint', async (t) => {
87
100
  const response = await testServer.executeOperation({
88
101
  query: 'query summary($text: String!) { summary(text: $text) { result } }',
89
102
  variables: { text: 'hello there my dear world!' },
90
103
  });
91
104
 
92
- expect(response.errors).toBeUndefined();
93
- expect(response.data?.summary.result).toBeDefined();
105
+ t.is(response.errors, undefined);
106
+ t.truthy(response.data?.summary.result);
94
107
  });
95
108
 
96
- module.exports = {
109
+ export {
110
+ initTypeDefsResolvers,
97
111
  getTestServer,
98
112
  };
@@ -1,15 +1,18 @@
1
- const { getTestServer } = require('./main.test');
1
+ import test from 'ava';
2
+ import { getTestServer, initTypeDefsResolvers } from './main.test.js';
2
3
 
3
- jest.setTimeout(1800000);
4
+ let testServer;
4
5
 
5
- const testServer = getTestServer();
6
+ test.before(async () => {
7
+ await initTypeDefsResolvers();
8
+ testServer = getTestServer();
9
+ });
6
10
 
7
- //stop server after all tests
8
- afterAll(async () => {
11
+ test.after.always(async () => {
9
12
  await testServer.stop();
10
13
  });
11
14
 
12
- it('test translate endpoint with huge arabic text english translation and check return non-arabic/english', async () => {
15
+ test('test translate endpoint with huge arabic text english translation and check return non-arabic/english', async t => {
13
16
  const response = await testServer.executeOperation({
14
17
  query: 'query translate($text: String!, $to:String) { translate(text: $text, to:$to) { result } }',
15
18
  variables: {
@@ -115,9 +118,9 @@ it('test translate endpoint with huge arabic text english translation and check
115
118
  ` },
116
119
  });
117
120
 
118
- expect(response.errors).toBeUndefined();
119
- expect(response.data?.translate.result.length).toBeGreaterThan(1000); //check return length huge
120
- //check return only contains non-Arabic characters
121
- expect(response.data?.translate.result).not.toMatch(/[ء-ي]/);
121
+ t.falsy(response.errors);
122
+ t.true(response.data?.translate.result.length > 1000); // check return length huge
123
+ // check return only contains non-Arabic characters
124
+ t.notRegex(response.data?.translate.result, /[ء-ي]/);
122
125
  });
123
126