Skip to content
This repository has been archived by the owner on Feb 16, 2020. It is now read-only.

Commands and remote logging #2103

Closed
wants to merge 4 commits into from
Closed
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Jump to
Jump to file
Failed to load files.
Diff view
Diff view
33 changes: 31 additions & 2 deletions core/log.js
Expand Up @@ -23,6 +23,7 @@ var sendToParent = function() {
error: send('error'),
warn: send('warn'),
info: send('info'),
status: send('status'),
write: send('write')
}
}
Expand All @@ -35,18 +36,39 @@ var Log = function() {
this.output = console;
else if(this.env === 'child-process')
this.output = sendToParent();

this.remoteLoggers = [];
};

Log.prototype = {
_write: function(method, args, name) {

var config = util.getConfig();
var silent = config.silent;

if(silent)
return;

if(!name)
name = method.toUpperCase();

var message = moment().format('YYYY-MM-DD HH:mm:ss');
message += ' (' + name + '):\t';
message += fmt.apply(null, args);
var rawMessage = fmt.apply(null, args);
message += rawMessage;

this.output[method](message);
if (method == 'remote')
{
// mirror remote output to info
this.output['info'](message);
_.each(this.remoteLoggers, function(logger) {
logger.logRemote(rawMessage);
});
}
else
{
this.output[method](message);
}
},
error: function() {
this._write('error', arguments);
Expand All @@ -57,10 +79,16 @@ Log.prototype = {
info: function() {
this._write('info', arguments);
},
remote: function() {
this._write('remote', arguments);
},
write: function() {
var args = _.toArray(arguments);
var message = fmt.apply(null, args);
this.output.info(message);
},
addRemoteLogger: function(logger) {
this.remoteLoggers.push(logger);
}
}

Expand All @@ -77,6 +105,7 @@ if(silent) {
Log.prototype.warn = _.noop;
Log.prototype.error = _.noop;
Log.prototype.write = _.noop;
Log.prototype.remote = _.noop;
}

module.exports = new Log;
1 change: 1 addition & 0 deletions plugins.js
Expand Up @@ -58,6 +58,7 @@ var plugins = [
slug: 'telegrambot',
async: false,
modes: ['realtime'],
emits: ['command'],
dependencies: [{
module: 'node-telegram-bot-api',
version: '0.24.0'
Expand Down
203 changes: 74 additions & 129 deletions plugins/telegrambot.js
@@ -1,158 +1,103 @@
var log = require('../core/log');
var moment = require('moment');
var _ = require('lodash');
var config = require('../core/util').getConfig();
var telegrambot = config.telegrambot;
var util = require('../core/util');
var config = util.getConfig();
var utc = moment.utc;

var telegram = require("node-telegram-bot-api");
var telegramfancy = require("tgfancy");

var Actor = function() {
var Actor = function () {
_.bindAll(this);

this.advice = 'Dont got one yet :(';
this.adviceTime = utc();

this.price = 'Dont know yet :(';
this.priceTime = utc();

this.commands = {
'/advice': 'emitAdvice',
'/price': 'emitPrice',
'/donate': 'emitDonation',
'/realadvice': 'emitRealAdvice', //Without space
'/help': 'emitHelp'
};

this.rawCommands = _.keys(this.commands);

this.chatId = null;
this.bot = new telegram(telegrambot.token, { polling: true });
this.bot = new telegramfancy(config.telegrambot.token, { polling: true });
this.bot.onText(/(.+)/, this.verifyQuestion);
}

Actor.prototype.processCandle = function(candle, done) {
this.price = candle.close;
this.priceTime = candle.start;

done();
};

Actor.prototype.processAdvice = function(advice) {
if (telegrambot.muteSoft && advice.recommendation === 'soft') return;
this.advice = advice.recommendation;
this.adviceTime = utc();

if (telegrambot.emitUpdates)
this.newAdvice();
};

Actor.prototype.verifyQuestion = function(msg, text) {
this.chatId = msg.chat.id;
if (text[1].toLowerCase() in this.commands)
this[this.commands[text[1].toLowerCase()]]();
else
this.bot.sendMessage(this.chatId, "Hello!");
log.addRemoteLogger(this);
}

Actor.prototype.newAdvice = function() {
util.makeEventEmitter(Actor);

Actor.prototype.processAdvice = function (advice) {
if (this.chatId) {
this.bot.sendMessage(this.chatId, 'Important news!');
this.emitAdvice();
var message;
if (this.advice == 'long') {
message = "BUY order received, sending to exchange";
}
else {
message = "SELL order received, sending to exchange";
}

this.bot.sendMessage(this.chatId, message);
}
}

// sent advice to the last chat
Actor.prototype.emitAdvice = function() {
var message = [
'Advice for ',
config.watch.exchange,
' ',
config.watch.currency,
'/',
config.watch.asset,
' using ',
config.tradingAdvisor.method,
' at ',
config.tradingAdvisor.candleSize,
' minute candles, is:\n',
this.advice,
' ',
config.watch.asset,
' (from ',
this.adviceTime.fromNow(),
')'
].join('');

if (this.chatId)
this.bot.sendMessage(this.chatId, message);
};

// sent price over to the last chat
Actor.prototype.emitPrice = function() {
var message = [
'Current price at ',
config.watch.exchange,
' ',
config.watch.currency,
'/',
config.watch.asset,
' is ',
this.price,
' ',
config.watch.currency,
' (from ',
this.priceTime.fromNow(),
')'
].join('');

if (this.chatId)
this.bot.sendMessage(this.chatId, message);
};
Actor.prototype.processTrade = function (trade) {
//{
// action: [either "buy" or "sell"],
// price: [number, price that was sold at],
// date: [moment object, exchange time trade completed at],
// portfolio: [object containing amount in currency and asset],
// balance: [number, total worth of portfolio]
//}

// sent donation info over to the IRC channel
Actor.prototype.emitDonation = function() {
var message = 'You want to donate? How nice of you! You can send your coins here:';
message += '\nBTC:\t13r1jyivitShUiv9FJvjLH7Nh1ZZptumwW';

if (this.chatId)
this.bot.sendMessage(this.chatId, message);
};
if (this.chatId) {
this.bot.sendMessage(this.chatId, "Trade completed!");
this.bot.sendMessage(this.chatId, trade.date.toDate() + ": " + trade.action + " at " + trade.price.toFixed(2));
// emit portfolio command to get results of trade
this.emit('command', {
command: 'portfolio',
arguments: [null],
handled: false,
response: null
});
}
}

Actor.prototype.emitHelp = function() {
var message = _.reduce(
this.rawCommands,
function(message, command) {
return message + ' ' + command + ',';
},
'possible commands are:'
);
Actor.prototype.verifyQuestion = function (msg, text) {
this.chatId = msg.chat.id;

message = message.substr(0, _.size(message) - 1) + '.';
// simple parsing that supports a command and single argument
var tokens = text[1].split(" ");

if (this.chatId)
this.bot.sendMessage(this.chatId, message);
if (tokens.length == 1 || tokens.length == 2) {
var command = tokens[0].toLowerCase();
var arg = tokens.length == 2 ? tokens[1].toLowerCase() : null;
this.emitCommand(command, arg);
}
else {
this.bot.sendMessage(this.chatId, "'" + text[1] + "' - syntax error");
}
}

Actor.prototype.emitRealAdvice = function() {
// http://www.examiner.com/article/uncaged-a-look-at-the-top-10-quotes-of-gordon-gekko
// http://elitedaily.com/money/memorable-gordon-gekko-quotes/
var realAdvice = [
'I don\'t throw darts at a board. I bet on sure things. Read Sun-tzu, The Art of War. Every battle is won before it is ever fought.',
'Ever wonder why fund managers can\'t beat the S&P 500? \'Cause they\'re sheep, and sheep get slaughtered.',
'If you\'re not inside, you\'re outside!',
'The most valuable commodity I know of is information.',
'It\'s not a question of enough, pal. It\'s a zero sum game, somebody wins, somebody loses. Money itself isn\'t lost or made, it\'s simply transferred from one perception to another.',
'What\'s worth doing is worth doing for money. (Wait, wasn\'t I a free and open source bot?)',
'When I get a hold of the son of a bitch who leaked this, I\'m gonna tear his eyeballs out and I\'m gonna suck his fucking skull.'
];

if (this.chatId)
this.bot.sendMessage(this.chatId, _.first(_.shuffle(realAdvice)));
Actor.prototype.emitCommand = function(command, arg) {
var cmd = {
command: command.replace('/',''),
arguments: [arg],
handled: false,
response: null
};

this.emit('command', cmd);
if (cmd.handled) {
if (cmd.response) {
this.bot.sendMessage(this.chatId, cmd.response);
}
}
else {
this.bot.sendMessage(this.chatId, "'" + cmd.command + "' - unrecognised command");
}
}

Actor.prototype.logError = function(message) {
Actor.prototype.logError = function (message) {
log.error('Telegram ERROR:', message);
};

Actor.prototype.logRemote = function (message) {
if (this.chatId) {
this.bot.sendMessage(this.chatId, message);
}
}

module.exports = Actor;