3
0
mirror of https://github.com/reality/dbot.git synced 2025-01-04 00:52:35 +01:00
dbot/modules/link/link.js

230 lines
8.1 KiB
JavaScript
Raw Permalink Normal View History

/**
* Module Name: Link
* Description: Stores recent channel links, with commands to retrieve
* information about links.
*/
2013-01-13 16:30:53 +01:00
var request = require('request'),
2013-03-09 18:02:33 +01:00
_ = require('underscore')._,
ent = require('ent');
2013-01-13 16:30:53 +01:00
var link = function(dbot) {
2013-01-15 00:11:50 +01:00
this.urlRegex = /(\b(https?|ftp|file):\/\/[-A-Z0-9+&@#\/%?=~_|!:,.;]*[-A-Z0-9+&@#\/%=~_|])/ig;
this.links = {};
this.handlers = [];
this.api = {
'addHandler': function(name, regex, handler) {
this.handlers.push({
'name': name,
'regex': regex,
'callback': handler
});
},
'getTitle': function(link, callback) {
var limit = 1000000,
size = 0,
page = request(link.replace('https', 'http'), function(error, response, body) {
if(!error && response.statusCode == 200) {
2015-02-04 19:52:35 +01:00
body = body.replace(/(\r\n|\n\r|\n)/gim, " ");
var title = body.valMatch(/<title>(.*?)<\\?\/title>/, 2);
if(title && title.length < 140) {
callback(ent.decode(title[1]).trim());
}
}
});
page.on('data', function(chunk) {
size += chunk.length;
if(size > limit) {
page.abort();
}
});
},
'udLookup': function(query, callback) {
var reqUrl = 'http://api.urbandictionary.com/v0/define?term=' +
encodeURI(query);
request(reqUrl, function(error, response, body) {
console.log(reqUrl);
try {
var result = JSON.parse(body);
if(_.has(result, 'list') && result.list.length > 0) {
callback(result.list[0].word, result.list[0].definition.split('\n')[0]);
} else {
callback(false);
}
} catch(err) {
console.log(err);
callback(false);
}
});
},
'parseLink': function(link, callback) {
var handler = false;
for(var i=0;i<this.handlers.length;i++) {
var matches = this.handlers[i].regex.exec(link);
if(matches) {
2013-08-06 00:11:25 +02:00
console.log(this.handlers[i].name);
handler = this.handlers[i];
break;
}
}
if(handler) {
this.handlers[i].callback(matches, this.handlers[i].name, function(parsed) {
callback(parsed);
});
} else {
this.api.getTitle(link, function(title) {
callback(dbot.t('link', { 'link': title } ));
});
}
}
};
var commands = {
'~title': function(event) {
2013-01-15 00:11:50 +01:00
var link = this.links[event.channel.name];
if(!_.isUndefined(event.params[1])) {
var urlMatches = event.params[1].match(this.urlRegex);
if(urlMatches !== null) {
link = urlMatches[0];
}
}
2013-05-20 17:24:24 +02:00
this.api.getTitle(link, function(title) {
2013-05-21 19:45:58 +02:00
event.reply(dbot.t('link', { 'link': title} ));
});
2013-01-22 20:52:26 +01:00
},
'~xkcd': function(event) {
//var comicId = event.params[1] || "";
var comicId = event.params.slice(1).join(' ');
if(comicId == "*") {
request("http://xkcd.com/info.0.json", function(error, response, body){
try {
if(response.statusCode == "200") {
data = JSON.parse(body);
event.params[1] = (Math.floor(Math.random() * data.num) + 1);
dbot.commands['~xkcd'](event);
}
} catch(err) { };
});
} else {
if (isNaN(parseInt(comicId))) {
request({
url: 'http://www.explainxkcd.com/wiki/api.php',
qs: {
action: 'query',
format: 'json',
generator: 'search',
gsrwhat: 'text',
gsrsearch: comicId,
prop: 'info|categories',
gsrlimit: 50
},
json: true
}, function(err, res, body) {
if(!body) {
event.reply(dbot.t("no-hits"));
return;
}
var pages = _.values(body.query.pages);
// page titles must be of the format "####: $$$$$$"
pages = _.filter(pages, p => p.title.indexOf(':') > 0);
if (pages.length > 0) {
// See if any of these matches are exact title matches
var match = false;
_.each(pages, function(p) {
var title = p.title.slice(p.title.indexOf(':')+2).trim();
if(title.toLowerCase() == comicId.toLowerCase()) {
match = p;
}
});
if (match) {
// We got a match! Get the ID and let's get tf out of here.
comicId = match.title.slice(0, match.title.indexOf(':'));
} else {
comicId = pages[0].title.slice(0, pages[0].title.indexOf(':'));
}
var link = "http://xkcd.com/"+comicId+"/info.0.json";
request(link, function(error, response, body) {
try {
if (response.statusCode == "200") {
data = JSON.parse(body);
event.reply(dbot.t("xkcd", data));
} else {
event.reply(dbot.t("no-hits"));
}
} catch(err) { };
});
} else {
event.reply(dbot.t("no-hits"));
}
});
} else {
if(comicId !== "") {
comicId = comicId + "/";
}
var link = "http://xkcd.com/"+comicId+"info.0.json";
request(link, function(error, response, body) {
try {
if (response.statusCode == "200") {
data = JSON.parse(body);
event.reply(dbot.t("xkcd", data));
} else {
event.reply(dbot.t("no-hits"));
}
} catch(err) { };
});
}
}
},
2013-01-22 20:52:26 +01:00
'~ud': function(event) {
var query = event.input[1];
this.api.udLookup(query, function(word, definition) {
if(word) {
event.reply(word + ': ' + definition);
} else {
event.reply(event.user + ': No definition found.');
}
2013-01-22 20:52:26 +01:00
});
}
};
2013-12-29 19:38:24 +01:00
commands['~ud'].regex = [/ud (.+)/, 2];
2013-01-15 00:11:50 +01:00
this.commands = commands;
2013-01-15 00:11:50 +01:00
this.listener = function(event) {
var urlMatches = event.message.match(this.urlRegex);
if(urlMatches !== null) {
this.links[event.channel.name] = urlMatches[0];
2013-08-06 00:11:25 +02:00
console.log('DEBUG: got a link');
if(this.config.autoTitle == true) {
this.api.parseLink(urlMatches[0], function(result) {
event.reply(result);
});
}
2013-01-15 00:11:50 +01:00
}
}.bind(this);
this.on = 'PRIVMSG';
};
exports.fetch = function(dbot) {
2013-01-15 00:11:50 +01:00
return new link(dbot);
};