import { success, failure } from '../utils/res' import * as path from 'path' import * as network from '../network' import { models } from '../models' import * as short from 'short-uuid' import * as rsa from '../crypto/rsa' import * as crypto from 'crypto' import * as jsonUtils from '../utils/json' import * as tribes from '../utils/tribes' /* hexdump -n 8 -e '4/4 "%08X" 1 "\n"' /dev/random hexdump -n 16 -e '4/4 "%08X" 1 "\n"' /dev/random */ const constants = require(path.join(__dirname, '../../config/constants.json')) export const getBots = async (req, res) => { try { const bots = await models.Bot.findAll() success(res, { bots: bots.map(b=> jsonUtils.botToJson(b)) }) } catch(e) { failure(res,'no bots') } } export const getBotsForTribe = async (req, res) => { const chat_id = req.params.chat_id const chatId = parseInt(chat_id) if(!chatId) return failure(res,'no chat id') try { const bots = await models.Bot.findAll({where:{chatId}}) success(res, { bots: bots.map(b=> jsonUtils.botToJson(b)) }) } catch(e) { failure(res,'no bots') } } export const createBot = async (req, res) => { const { chat_id, name, } = req.body const chatId = parseInt(chat_id) const chat = await models.Chat.findOne({where:{id:chatId}}) if(!chat) return failure(res,'no chat') const owner = await models.Contact.findOne({where: {isOwner:true}}) const isTribeOwner = owner.publicKey===chat.ownerPubkey if(!isTribeOwner) return failure(res, 'not tribe owner') const uuid = await tribes.genSignedTimestamp() const newBot = { uuid, id: crypto.randomBytes(16).toString('hex').toUpperCase(), chatId: chat_id, name: name, secret: crypto.randomBytes(16).toString('hex').toUpperCase() } try { const theBot = await models.Bot.create(newBot) success(res, jsonUtils.botToJson(theBot)) } catch (e) { failure(res, 'bot creation failed') } } export const deleteBot = async (req, res) => { const id = req.params.id if (!id) return try { models.Bot.destroy({ where: { id } }) success(res, true) } catch (e) { console.log('ERROR deleteBot', e) failure(res, e) } } export interface Action { action: string chatID: number, botName?: string amount?: number pubkey?: string text?: string } export async function processAction(req, res) { let body = req.body if (body.data && typeof body.data === 'string' && body.data[1] === "'") { try { // parse out body from "data" for github webhook action const dataBody = JSON.parse(body.data.replace(/'/g, '"')) if (dataBody) body = dataBody } catch (e) { console.log(e) return failure(res, 'failed to parse webhook body json') } } const { action, bot_id, bot_secret, pubkey, amount, text } = body const bot = await models.Bot.findOne({ where: { id: bot_id } }) if (!bot) return failure(res, 'no bot') if (!(bot.secret && bot.secret === bot_secret)) { return failure(res, 'wrong secret') } if (!action) { return failure(res, 'no action') } const a:Action = { action, pubkey, text, amount, botName:bot.name, chatID: bot.chatId } try { const r = await finalActionProcess(a) success(res, r) } catch(e) { failure(res, e) } } export async function finalActionProcess(a:Action){ const {action,pubkey,amount,text,botName,chatID} = a if (action === 'keysend') { console.log('=> BOT KEYSEND') if (!(pubkey && pubkey.length === 66 && amount)) { throw 'wrong params' } const MIN_SATS = 3 const destkey = pubkey const opts = { dest: destkey, data: {}, amt: Math.max((amount || 0), MIN_SATS) } try { await network.signAndSend(opts) return ({ success: true }) } catch (e) { throw e } } else if (action === 'broadcast') { console.log('=> BOT BROADCAST') if (!chatID || !text) throw 'no chatID or text' const owner = await models.Contact.findOne({ where: { isOwner: true } }) const theChat = await models.Chat.findOne({ where: { id: chatID } }) if (!theChat || !owner) throw 'no chat' if (!theChat.type === constants.chat_types.tribe) throw 'not a tribe' const encryptedForMeText = rsa.encrypt(owner.contactKey, text) const encryptedText = rsa.encrypt(theChat.groupKey, text) const textMap = { 'chat': encryptedText } var date = new Date(); date.setMilliseconds(0) const alias = botName || 'Bot' const msg: { [k: string]: any } = { chatId: theChat.id, uuid: short.generate(), type: constants.message_types.message, sender: owner.id, amount: amount || 0, date: date, messageContent: encryptedForMeText, remoteMessageContent: JSON.stringify(textMap), status: constants.statuses.confirmed, createdAt: date, updatedAt: date, senderAlias: alias, } const message = await models.Message.create(msg) await network.sendMessage({ chat: theChat, sender: { ...owner.dataValues, alias }, message: { content: textMap, id: message.id, uuid: message.uuid }, type: constants.message_types.message, success: () => ({ success: true }), failure: (e) => { throw e } }) } else { throw 'no action' } }