Skip to content

Commit 810515f

Browse files
author
Jesús Seijas
authored
Feature/version3 4 (axa-group#227)
* feat: remove handlebars * feat: Adding default intent and score when score is less than threshold * feat: update deps * feat: using decay learning rate * test: adjust tests
1 parent 71a626a commit 810515f

17 files changed

+849
-329
lines changed

lib/classifiers/neural-network.js

+4-1
Original file line numberDiff line numberDiff line change
@@ -150,14 +150,17 @@ class NeuralNetwork {
150150
const { learningRate, leakyReluAlpha: alpha, momentum } = this.trainOpts;
151151
const numOutputs = this.sizes[1];
152152
let error = 0;
153+
const decayLearningRate =
154+
learningRate / (1 + 0.001 * this.status.iterations);
153155
for (let node = 0, l = numOutputs; node < l; node += 1) {
154156
const perceptron = this.perceptrons[node];
155157
const { changes, weights } = perceptron;
156158
const output = outputs[node];
157159
const currentError = target[node] - output;
158160
if (currentError) {
159161
error += currentError ** 2;
160-
const delta = (output > 0 ? 1 : alpha) * currentError * learningRate;
162+
const delta =
163+
(output > 0 ? 1 : alpha) * currentError * decayLearningRate;
161164
for (let k = 0; k < incoming.length; k += 1) {
162165
const change = delta * incoming[k] + momentum * changes[k];
163166
changes[k] = change;

lib/index.js

+2-1
Original file line numberDiff line numberDiff line change
@@ -51,7 +51,7 @@ const {
5151
} = require('./ner');
5252
const { SentimentAnalyzer, SentimentManager } = require('./sentiment');
5353
const { SlotManager } = require('./slot');
54-
const { Evaluator, SimilarSearch } = require('./util');
54+
const { Evaluator, Handlebars, SimilarSearch } = require('./util');
5555
const { XTableUtils, XTable, XDoc } = require('./xtables');
5656
const {
5757
ConversationContext,
@@ -84,6 +84,7 @@ const exportClasses = {
8484
SentimentManager,
8585
SlotManager,
8686
Evaluator,
87+
Handlebars,
8788
SimilarSearch,
8889
XTableUtils,
8990
XTable,

lib/nlp/nlp-manager.js

+14-1
Original file line numberDiff line numberDiff line change
@@ -22,7 +22,7 @@
2222
*/
2323

2424
const fs = require('fs');
25-
const Handlebars = require('handlebars');
25+
const { Handlebars } = require('../util');
2626
const NluManager = require('../nlu/nlu-manager');
2727
const NerManager = require('../ner/ner-manager');
2828
const { SentimentManager } = require('../sentiment');
@@ -61,6 +61,12 @@ class NlpManager {
6161
this.sentiment = new SentimentManager(this.settings.sentiment);
6262
this.slotManager = new SlotManager();
6363
this.utteranceDict = this.settings.utteranceDict || { '?': 'help' };
64+
this.defaultThreshold = this.settings.defaultThreshold;
65+
if (this.settings.defaultThreshold === undefined) {
66+
this.defaultThreshold = 0.5;
67+
}
68+
this.defaultIntent = this.settings.defaultIntent || 'None';
69+
this.defaultScore = this.settings.defaultScore || 1;
6470
if (this.settings.languages) {
6571
this.addLanguage(this.settings.languages);
6672
}
@@ -392,6 +398,13 @@ class NlpManager {
392398
result = optional;
393399
}
394400
}
401+
if (
402+
this.settings.defaultThreshold > 0 &&
403+
result.score < this.settings.defaultThreshold
404+
) {
405+
result.intent = this.defaultIntent;
406+
result.score = this.defaultScore;
407+
}
395408
result.entities = await this.nerManager.findEntities(
396409
result.utterance,
397410
result.localeIso2,

lib/nlp/nlp-util.js

+33
Original file line numberDiff line numberDiff line change
@@ -351,6 +351,39 @@ NlpUtil.useAlternative = {
351351
tr: false,
352352
};
353353

354+
NlpUtil.useNoneFeature = {
355+
en: true,
356+
fa: false,
357+
fr: true,
358+
ru: true,
359+
es: true,
360+
gl: true,
361+
it: true,
362+
nl: true,
363+
no: true,
364+
pt: true,
365+
pl: true,
366+
sv: true,
367+
tl: true,
368+
id: true,
369+
ja: false,
370+
ar: false,
371+
hy: false,
372+
eu: true,
373+
ca: true,
374+
cs: true,
375+
da: true,
376+
fi: true,
377+
de: true,
378+
hu: true,
379+
ga: true,
380+
ro: true,
381+
sl: true,
382+
ta: false,
383+
tr: true,
384+
zh: false,
385+
};
386+
354387
NlpUtil.tokenizers = {};
355388

356389
module.exports = NlpUtil;

lib/nlu/brain-nlu.js

+5
Original file line numberDiff line numberDiff line change
@@ -23,6 +23,7 @@
2323

2424
const BaseNLU = require('./base-nlu');
2525
const BrainClassifier = require('../classifiers/brain-classifier');
26+
const NlpUtil = require('../nlp/nlp-util');
2627

2728
/**
2829
* Class for the Logistic Regression NLU
@@ -34,6 +35,10 @@ class BrainNLU extends BaseNLU {
3435
*/
3536
constructor(settings) {
3637
super(settings);
38+
if (this.settings.useNoneFeature === undefined) {
39+
this.settings.useNoneFeature =
40+
NlpUtil.useNoneFeature[this.settings.language] || false;
41+
}
3742
this.classifier = this.settings.classifier || new BrainClassifier(settings);
3843
}
3944

lib/nlu/domain-manager.js

+1-48
Original file line numberDiff line numberDiff line change
@@ -43,7 +43,7 @@ class DomainManager {
4343
: this.settings.useMasterDomain;
4444
this.trainByDomain =
4545
this.settings.trainByDomain === undefined
46-
? true
46+
? false
4747
: this.settings.trainByDomain;
4848
this.stemmer = this.settings.stemmer || NlpUtil.getStemmer(this.language);
4949
this.keepStopwords =
@@ -242,53 +242,6 @@ class DomainManager {
242242
return this.getClassifications(stems, 'master_domain');
243243
}
244244

245-
// getClassifications(utterance) {
246-
// const stems = this.tokenizeAndStem(utterance);
247-
// if (this.useStemDict) {
248-
// const stemKey = this.generateStemKey(stems);
249-
// if (this.stemDict[stemKey]) {
250-
// const classifications = [];
251-
// classifications.push({
252-
// label: this.stemDict[stemKey].intent,
253-
// value: 1,
254-
// });
255-
// Object.keys(this.intentDict).forEach(intent => {
256-
// if (intent !== this.stemDict[stemKey].intent) {
257-
// if (
258-
// !this.trainByDomain ||
259-
// this.stemDict[stemKey].domain === this.intentDict[intent]
260-
// ) {
261-
// classifications.push({ label: intent, value: 0 });
262-
// }
263-
// }
264-
// });
265-
// return { domain: this.stemDict[stemKey].domain, classifications };
266-
// }
267-
// }
268-
// if (this.trainByDomain) {
269-
// let domain;
270-
// if (Object.keys(this.domains).length > 2) {
271-
// const master = this.addDomain('master_domain');
272-
// const domainClassifications = master.getClassifications(stems);
273-
// domain = domainClassifications[0].label;
274-
// } else {
275-
// [domain] = Object.keys(this.domains).filter(x => x !== 'master_domain');
276-
// }
277-
// if (domain) {
278-
// const nlu = this.addDomain(domain);
279-
// const classifications = nlu.getClassifications(stems);
280-
// return { domain, classifications };
281-
// }
282-
// return { domain, classifications: [] };
283-
// }
284-
// const nlu = this.addDomain('master_domain');
285-
// const classifications = nlu.getClassifications(stems);
286-
// return {
287-
// domain: this.intentDict[classifications[0].label],
288-
// classifications,
289-
// };
290-
// }
291-
292245
/**
293246
* Exports object properties.
294247
* @returns {Object} Object properties.

lib/util/handlebars.js

+20
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,20 @@
1+
const Evaluator = require('./evaluator');
2+
3+
const evaluator = new Evaluator();
4+
5+
const dictionary = {};
6+
7+
function compile(str) {
8+
if (dictionary[str] === undefined) {
9+
dictionary[str] = str.match(/{{\s*([^}]+)\s*}}/g) || [];
10+
}
11+
const matches = dictionary[str];
12+
return (context = {}) => {
13+
return matches.reduce((p, c) => {
14+
const solution = evaluator.evaluate(c.substr(2, c.length - 4), context);
15+
return solution ? p.replace(c, solution) : p;
16+
}, str);
17+
};
18+
}
19+
20+
module.exports = { compile };

lib/util/index.js

+2
Original file line numberDiff line numberDiff line change
@@ -23,8 +23,10 @@
2323

2424
const SimilarSearch = require('./similar-search');
2525
const Evaluator = require('./evaluator');
26+
const Handlebars = require('./handlebars');
2627

2728
module.exports = {
2829
SimilarSearch,
2930
Evaluator,
31+
Handlebars,
3032
};

0 commit comments

Comments
 (0)