2013-01-13 16:45:07 +01:00
|
|
|
var _ = require('underscore')._;
|
|
|
|
|
2013-08-13 01:14:44 +02:00
|
|
|
var spelling = function(dbot) {
|
|
|
|
this.last = {};
|
|
|
|
|
|
|
|
this.api = {
|
|
|
|
'allGroupings': function(arr) {
|
|
|
|
if (arr.length == 0) {
|
|
|
|
return []; /* short-circuit the empty-array case */
|
|
|
|
}
|
|
|
|
var groupings = [];
|
|
|
|
for(var n=1;n<=arr.length;n++) {
|
|
|
|
for(var i=0;i<(arr.length-(n-1));i++) {
|
|
|
|
groupings.push(arr.slice(i, i+n));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return groupings;
|
|
|
|
},
|
|
|
|
|
|
|
|
'distance': function(s1, s2) {
|
|
|
|
// Calculate Levenshtein distance between two strings
|
|
|
|
//
|
|
|
|
// version: 1109.2015
|
|
|
|
// discuss at: http://phpjs.org/functions/levenshtein
|
|
|
|
// + original by: Carlos R. L. Rodrigues (http://www.jsfromhell.com)
|
|
|
|
// + bugfixed by: Onno Marsman
|
|
|
|
// + revised by: Andrea Giammarchi (http://webreflection.blogspot.com)
|
|
|
|
// + reimplemented by: Brett Zamir (http://brett-zamir.me)
|
|
|
|
// + reimplemented by: Alexander M Beedie
|
|
|
|
if (s1 == s2) {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
var s1_len = s1.length;
|
|
|
|
var s2_len = s2.length;
|
|
|
|
if (s1_len === 0) {
|
|
|
|
return s2_len; }
|
|
|
|
if (s2_len === 0) {
|
|
|
|
return s1_len;
|
|
|
|
}
|
|
|
|
// BEGIN STATIC
|
|
|
|
var split = false;
|
|
|
|
try {
|
|
|
|
split = !('0')[0];
|
|
|
|
} catch (e) {
|
|
|
|
split = true; // Earlier IE may not support access by string index
|
|
|
|
}
|
|
|
|
// END STATIC
|
|
|
|
if (split) {
|
|
|
|
s1 = s1.split(''); s2 = s2.split('');
|
|
|
|
}
|
|
|
|
|
|
|
|
var v0 = new Array(s1_len + 1);
|
|
|
|
var v1 = new Array(s1_len + 1);
|
|
|
|
var s1_idx = 0,
|
|
|
|
s2_idx = 0,
|
|
|
|
cost = 0;
|
|
|
|
for (s1_idx = 0; s1_idx < s1_len + 1; s1_idx++) { v0[s1_idx] = s1_idx;
|
2013-01-13 16:45:07 +01:00
|
|
|
}
|
2013-08-13 01:14:44 +02:00
|
|
|
var char_s1 = '',
|
|
|
|
char_s2 = '';
|
|
|
|
for (s2_idx = 1; s2_idx <= s2_len; s2_idx++) { v1[0] = s2_idx;
|
|
|
|
char_s2 = s2[s2_idx - 1];
|
|
|
|
|
|
|
|
for (s1_idx = 0; s1_idx < s1_len; s1_idx++) {
|
|
|
|
char_s1 = s1[s1_idx]; cost = (char_s1 == char_s2) ? 0 : 1;
|
|
|
|
var m_min = v0[s1_idx + 1] + 1;
|
|
|
|
var b = v1[s1_idx] + 1;
|
|
|
|
var c = v0[s1_idx] + cost;
|
|
|
|
if (b < m_min) { m_min = b;
|
|
|
|
}
|
|
|
|
if (c < m_min) {
|
|
|
|
m_min = c;
|
|
|
|
} v1[s1_idx + 1] = m_min;
|
|
|
|
}
|
|
|
|
var v_tmp = v0;
|
|
|
|
v0 = v1;
|
|
|
|
v1 = v_tmp; }
|
|
|
|
return v0[s1_len];
|
2013-01-13 16:45:07 +01:00
|
|
|
}
|
2013-08-13 01:14:44 +02:00
|
|
|
};
|
2013-01-13 16:45:07 +01:00
|
|
|
|
2013-01-15 17:05:04 +01:00
|
|
|
this.internalAPI = {};
|
|
|
|
this.internalAPI.correct = function (event, correction, candidate, output_callback) {
|
2013-08-13 01:14:44 +02:00
|
|
|
var rawCandidates = this.api.allGroupings(this.last[event.channel.name][candidate].split(' '));
|
2013-01-13 16:45:07 +01:00
|
|
|
|
2012-12-11 17:04:52 +01:00
|
|
|
var candidates = [];
|
|
|
|
for(var i=0;i<rawCandidates.length;i++) {
|
|
|
|
candidates.push(rawCandidates[i].join(' '));
|
|
|
|
}
|
|
|
|
var winner = false;
|
|
|
|
var winnerDistance = Infinity;
|
|
|
|
|
|
|
|
for(var i=0;i<candidates.length;i++) {
|
2013-08-13 01:14:44 +02:00
|
|
|
var d = this.api.distance(correction.toLowerCase(), candidates[i].toLowerCase());
|
2013-01-13 16:45:07 +01:00
|
|
|
if((d < winnerDistance) && (d > 0)) {
|
2012-12-11 17:04:52 +01:00
|
|
|
winner = candidates[i];
|
2013-01-13 16:45:07 +01:00
|
|
|
winnerDistance = d;
|
2012-12-11 17:04:52 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if(winnerDistance < Math.ceil(winner.length * 1.33)) {
|
|
|
|
if(winner !== correction) {
|
2013-01-15 17:05:04 +01:00
|
|
|
var fix = this.last[event.channel.name][candidate].replace(winner, correction);
|
2012-12-11 17:04:52 +01:00
|
|
|
if (/^.ACTION/.test(fix)) {
|
|
|
|
fix = fix.replace(/^.ACTION/, '/me');
|
2013-03-28 22:20:41 +01:00
|
|
|
fix = fix.replace("\x01", '');
|
2012-12-11 17:04:52 +01:00
|
|
|
}
|
2013-01-15 17:05:04 +01:00
|
|
|
this.last[event.channel.name][candidate] = fix;
|
2012-12-11 17:04:52 +01:00
|
|
|
var output = {
|
|
|
|
'fix': fix,
|
|
|
|
'correcter': event.user,
|
|
|
|
'candidate': candidate
|
|
|
|
};
|
|
|
|
output_callback(output);
|
|
|
|
}
|
|
|
|
}
|
2013-01-15 17:05:04 +01:00
|
|
|
}.bind(this);
|
2012-12-11 17:04:52 +01:00
|
|
|
|
2013-01-15 17:05:04 +01:00
|
|
|
this.listener = function(event) {
|
|
|
|
var q = event.message.valMatch(/^(?:\*\*?([\d\w\s']*)|([\d\w\s']*)\*\*?)$/, 3);
|
2013-01-23 10:36:46 +01:00
|
|
|
var otherQ = event.message.valMatch(/^([\d\w\s]*)[:|,] (?:\*\*?([\d\w\s']*)|([\d\w\s']*)\*\*?)$/, 4);
|
2013-01-15 17:05:04 +01:00
|
|
|
if(q) {
|
|
|
|
this.internalAPI.correct(event, q[1] || q[2], event.user, function (e) {
|
|
|
|
event.reply(dbot.t('spelling_self', e));
|
|
|
|
});
|
|
|
|
} else if(otherQ) {
|
|
|
|
this.internalAPI.correct(event, otherQ[2] || otherQ[3], otherQ[1], function (e) {
|
|
|
|
event.reply(dbot.t('spelling_other', e));
|
|
|
|
});
|
|
|
|
} else {
|
|
|
|
if(_.has(this.last, event.channel.name)) {
|
|
|
|
this.last[event.channel.name][event.user] = event.message;
|
2012-12-11 17:04:52 +01:00
|
|
|
} else {
|
2013-01-15 17:05:04 +01:00
|
|
|
this.last[event.channel.name] = { };
|
|
|
|
this.last[event.channel.name][event.user] = event.message;
|
2012-12-11 17:04:52 +01:00
|
|
|
}
|
2013-01-15 17:05:04 +01:00
|
|
|
}
|
|
|
|
}.bind(this);
|
|
|
|
this.on = 'PRIVMSG';
|
2012-12-11 17:04:52 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
exports.fetch = function(dbot) {
|
2013-01-15 17:05:04 +01:00
|
|
|
return new spelling(dbot);
|
2012-12-11 17:04:52 +01:00
|
|
|
};
|