481 lines
17 KiB

"use strict";
var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, generator) {
function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }
return new (P || (P = Promise))(function (resolve, reject) {
function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }
function rejected(value) { try { step(generator["throw"](value)); } catch (e) { reject(e); } }
function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); }
step((generator = generator.apply(thisArg, _arguments || [])).next());
});
};
Object.defineProperty(exports, "__esModule", { value: true });
const models_1 = require("../models");
const socket = require("../utils/socket");
const jsonUtils = require("../utils/json");
const resUtils = require("../utils/res");
const helpers = require("../helpers");
const hub_1 = require("../hub");
const lightning_1 = require("../utils/lightning");
const rp = require("request-promise");
const lightning_2 = require("../utils/lightning");
const ldat_1 = require("../utils/ldat");
const cron_1 = require("cron");
const zbase32 = require("../utils/zbase32");
const schemas = require("./schemas");
const env = process.env.NODE_ENV || 'development';
const config = require(__dirname + '/../../config/app.json')[env];
const constants = require(__dirname + '/../../config/constants.json');
/*
TODO line 233: parse that from token itself, dont use getMediaInfo at all
"attachment": sends a message to a chat with a signed receipt for a file, which can be accessed from sphinx-meme server
If the attachment has a price, then the media must be purchased to get the receipt
"purchase" sends sats.
if the amount matches the price, the media owner
will respond ("purchase_accept" or "purchase_deny" type)
with the signed token, which can only be used by the buyer
purchase_accept should update the original attachment message with the terms and receipt
(both Relay and client need to do this) or make new???
purchase_deny returns the sats
*/
const sendAttachmentMessage = (req, res) => __awaiter(void 0, void 0, void 0, function* () {
// try {
// schemas.attachment.validateSync(req.body)
// } catch(e) {
// return resUtils.failure(res, e.message)
// }
const { chat_id, contact_id, muid, text, remote_text, remote_text_map, media_key_map, media_type, file_name, ttl, price, } = req.body;
console.log('[send attachment]', req.body);
const owner = yield models_1.models.Contact.findOne({ where: { isOwner: true } });
const chat = yield helpers.findOrCreateChat({
chat_id,
owner_id: owner.id,
recipient_id: contact_id
});
let TTL = ttl;
if (ttl) {
TTL = parseInt(ttl);
}
if (!TTL)
TTL = 31536000; // default year
const amt = price || 0;
// generate media token for self!
const myMediaToken = yield ldat_1.tokenFromTerms({
muid, ttl: TTL, host: '',
pubkey: owner.publicKey,
meta: Object.assign(Object.assign({}, amt && { amt }), { ttl })
});
const date = new Date();
date.setMilliseconds(0);
const myMediaKey = (media_key_map && media_key_map[owner.id]) || '';
const mediaType = media_type || '';
const remoteMessageContent = remote_text_map ? JSON.stringify(remote_text_map) : remote_text;
const message = yield models_1.models.Message.create({
chatId: chat.id,
sender: owner.id,
type: constants.message_types.attachment,
status: constants.statuses.pending,
messageContent: text || file_name || '',
remoteMessageContent,
mediaToken: myMediaToken,
mediaKey: myMediaKey,
mediaType: mediaType,
date,
createdAt: date,
updatedAt: date
});
saveMediaKeys(muid, media_key_map, chat.id, message.id);
const mediaTerms = {
muid, ttl: TTL,
meta: Object.assign({}, amt && { amt }),
skipSigning: amt ? true : false // only sign if its free
};
const msg = {
mediaTerms,
id: message.id,
content: remote_text_map || remote_text || text || file_name || '',
mediaKey: media_key_map,
mediaType: mediaType,
};
helpers.sendMessage({
chat: chat,
sender: owner,
type: constants.message_types.attachment,
message: msg,
success: (data) => __awaiter(void 0, void 0, void 0, function* () {
console.log('attachment sent', { data });
resUtils.success(res, jsonUtils.messageToJson(message, chat));
}),
failure: error => resUtils.failure(res, error.message),
});
});
exports.sendAttachmentMessage = sendAttachmentMessage;
function saveMediaKeys(muid, mediaKeyMap, chatId, messageId) {
if (typeof mediaKeyMap !== 'object') {
console.log('wrong type for mediaKeyMap');
return;
}
var date = new Date();
date.setMilliseconds(0);
for (let [contactId, key] of Object.entries(mediaKeyMap)) {
models_1.models.MediaKey.create({
muid, chatId, contactId, key, messageId,
createdAt: date,
});
}
}
const purchase = (req, res) => __awaiter(void 0, void 0, void 0, function* () {
const { chat_id, contact_id, amount, mediaToken, } = req.body;
var date = new Date();
date.setMilliseconds(0);
try {
schemas.purchase.validateSync(req.body);
}
catch (e) {
return resUtils.failure(res, e.message);
}
console.log('purchase!');
const owner = yield models_1.models.Contact.findOne({ where: { isOwner: true } });
const chat = yield helpers.findOrCreateChat({
chat_id,
owner_id: owner.id,
recipient_id: contact_id
});
const message = yield models_1.models.Message.create({
sender: owner.id,
type: constants.message_types.purchase,
mediaToken: mediaToken,
date: date,
createdAt: date,
updatedAt: date
});
const msg = {
amount, mediaToken, id: message.id,
};
helpers.sendMessage({
chat: Object.assign(Object.assign({}, chat), { contactIds: [contact_id] }),
sender: owner,
type: constants.message_types.purchase,
message: msg,
success: (data) => __awaiter(void 0, void 0, void 0, function* () {
console.log('purchase sent', { data });
resUtils.success(res, jsonUtils.messageToJson(message));
}),
failure: error => resUtils.failure(res, error.message),
});
});
exports.purchase = purchase;
/* RECEIVERS */
const receivePurchase = (payload) => __awaiter(void 0, void 0, void 0, function* () {
console.log('received purchase', { payload });
var date = new Date();
date.setMilliseconds(0);
const { owner, sender, chat, amount, mediaToken } = yield helpers.parseReceiveParams(payload);
if (!owner || !sender || !chat) {
return console.log('=> group chat not found!');
}
yield models_1.models.Message.create({
chatId: chat.id,
sender: sender.id,
type: constants.message_types.purchase,
mediaToken: mediaToken,
date: date,
createdAt: date,
updatedAt: date
});
const muid = mediaToken && mediaToken.split('.').length && mediaToken.split('.')[1];
if (!muid) {
return console.log('no muid');
}
const ogMessage = models_1.models.Message.findOne({
where: { mediaToken }
});
if (!ogMessage) {
return console.log('no original message');
}
// find mediaKey for who sent
const mediaKey = models_1.models.MediaKey.findOne({ where: {
muid, receiver: sender.id,
} });
const terms = ldat_1.parseLDAT(mediaToken);
// get info
let TTL = terms.meta && terms.meta.ttl;
let price = terms.meta && terms.meta.amt;
if (!TTL || !price) {
const media = yield getMediaInfo(muid);
console.log("GOT MEDIA", media);
if (media) {
TTL = media.ttl && parseInt(media.ttl);
price = media.price;
}
if (!TTL)
TTL = 31536000;
if (!price)
price = 0;
}
if (amount < price) { // didnt pay enough
return helpers.sendMessage({
chat: Object.assign(Object.assign({}, chat), { contactIds: [sender.id] }),
sender: owner,
amount: amount,
type: constants.message_types.purchase_deny,
message: { amount, content: 'Payment Denied' },
success: (data) => __awaiter(void 0, void 0, void 0, function* () {
console.log('purchase_deny sent', { data });
}),
failure: error => console.log('=> couldnt send purcahse deny', error),
});
}
const acceptTerms = {
muid, ttl: TTL,
meta: { amt: amount },
};
helpers.sendMessage({
chat: Object.assign(Object.assign({}, chat), { contactIds: [sender.id] }),
sender: owner,
type: constants.message_types.purchase_accept,
message: {
mediaTerms: acceptTerms,
mediaKey: mediaKey.key,
mediaType: ogMessage.mediaType,
},
success: (data) => __awaiter(void 0, void 0, void 0, function* () {
console.log('purchase_accept sent', { data });
}),
failure: error => console.log('=> couldnt send purchase accept', error),
});
});
exports.receivePurchase = receivePurchase;
const receivePurchaseAccept = (payload) => __awaiter(void 0, void 0, void 0, function* () {
var date = new Date();
date.setMilliseconds(0);
const { owner, sender, chat, mediaToken, mediaKey, mediaType } = yield helpers.parseReceiveParams(payload);
if (!owner || !sender || !chat) {
return console.log('=> no group chat!');
}
const termsArray = mediaToken.split('.');
// const host = termsArray[0]
const muid = termsArray[1];
if (!muid) {
return console.log('wtf no muid');
}
// const attachmentMessage = await models.Message.findOne({where:{
// mediaToken: {$like: `${host}.${muid}%`}
// }})
// if(attachmentMessage){
// console.log('=> updated msg!')
// attachmentMessage.update({
// mediaToken, mediaKey
// })
// }
const msg = yield models_1.models.Message.create({
chatId: chat.id,
sender: sender.id,
type: constants.message_types.purchase_accept,
status: constants.statuses.received,
mediaToken,
mediaKey,
mediaType,
date: date,
createdAt: date,
updatedAt: date
});
socket.sendJson({
type: 'purchase_accept',
response: jsonUtils.messageToJson(msg, chat)
});
});
exports.receivePurchaseAccept = receivePurchaseAccept;
const receivePurchaseDeny = (payload) => __awaiter(void 0, void 0, void 0, function* () {
var date = new Date();
date.setMilliseconds(0);
const { owner, sender, chat, amount, mediaToken } = yield helpers.parseReceiveParams(payload);
if (!owner || !sender || !chat) {
return console.log('=> no group chat!');
}
const msg = yield models_1.models.Message.create({
chatId: chat.id,
sender: sender.id,
type: constants.message_types.purchase_deny,
status: constants.statuses.received,
messageContent: 'Purchase has been denied and sats returned to you',
amount: amount,
amountMsat: parseFloat(amount) * 1000,
mediaToken,
date: date,
createdAt: date,
updatedAt: date
});
socket.sendJson({
type: 'purchase_deny',
response: jsonUtils.messageToJson(msg, chat)
});
});
exports.receivePurchaseDeny = receivePurchaseDeny;
const receiveAttachment = (payload) => __awaiter(void 0, void 0, void 0, function* () {
console.log('received attachment', { payload });
var date = new Date();
date.setMilliseconds(0);
const { owner, sender, chat, mediaToken, mediaKey, mediaType, content, msg_id } = yield helpers.parseReceiveParams(payload);
if (!owner || !sender || !chat) {
return console.log('=> no group chat!');
}
const msg = {
chatId: chat.id,
type: constants.message_types.attachment,
sender: sender.id,
date: date,
createdAt: date,
updatedAt: date
};
if (content)
msg.messageContent = content;
if (mediaToken)
msg.mediaToken = mediaToken;
if (mediaKey)
msg.mediaKey = mediaKey;
if (mediaType)
msg.mediaType = mediaType;
const message = yield models_1.models.Message.create(msg);
console.log('saved attachment', message.dataValues);
socket.sendJson({
type: 'attachment',
response: jsonUtils.messageToJson(message, chat)
});
hub_1.sendNotification(chat, sender.alias, 'message');
sendConfirmation({ chat, sender: owner, msg_id });
});
exports.receiveAttachment = receiveAttachment;
const sendConfirmation = ({ chat, sender, msg_id }) => {
helpers.sendMessage({
chat,
sender,
message: { id: msg_id },
type: constants.message_types.confirmation,
});
};
function signer(req, res) {
return __awaiter(this, void 0, void 0, function* () {
if (!req.params.challenge)
return resUtils.failure(res, "no challenge");
try {
const sig = yield lightning_1.signBuffer(Buffer.from(req.params.challenge, 'base64'));
const sigBytes = zbase32.decode(sig);
const sigBase64 = ldat_1.urlBase64FromBytes(sigBytes);
resUtils.success(res, {
sig: sigBase64
});
}
catch (e) {
resUtils.failure(res, e);
}
});
}
exports.signer = signer;
function verifier(msg, sig) {
return __awaiter(this, void 0, void 0, function* () {
try {
const res = yield lightning_1.verifyMessage(msg, sig);
return res;
}
catch (e) {
console.log(e);
}
});
}
exports.verifier = verifier;
function getMyPubKey() {
return __awaiter(this, void 0, void 0, function* () {
return new Promise((resolve, reject) => {
const lightning = lightning_2.loadLightning();
var request = {};
lightning.getInfo(request, function (err, response) {
if (err)
reject(err);
if (!response.identity_pubkey)
reject('no pub key');
else
resolve(response.identity_pubkey);
});
});
});
}
function cycleMediaToken() {
return __awaiter(this, void 0, void 0, function* () {
try {
if (process.env.TEST_LDAT)
ldat_1.testLDAT();
const mt = yield getMediaToken(null);
if (mt)
console.log('=> [meme] authed!');
new cron_1.CronJob('1 * * * *', function () {
getMediaToken(true);
});
}
catch (e) {
console.log(e.message);
}
});
}
exports.cycleMediaToken = cycleMediaToken;
const mediaURL = 'http://' + config.media_host + '/';
let mediaToken;
function getMediaToken(force) {
return __awaiter(this, void 0, void 0, function* () {
if (!force && mediaToken)
return mediaToken;
yield helpers.sleep(3000);
try {
const res = yield rp.get(mediaURL + 'ask');
const r = JSON.parse(res);
if (!(r && r.challenge && r.id)) {
throw new Error('no challenge');
}
const sig = yield lightning_1.signBuffer(Buffer.from(r.challenge, 'base64'));
if (!sig)
throw new Error('no signature');
const pubkey = yield getMyPubKey();
if (!pubkey) {
throw new Error('no pub key!');
}
const sigBytes = zbase32.decode(sig);
const sigBase64 = ldat_1.urlBase64FromBytes(sigBytes);
const bod = yield rp.post(mediaURL + 'verify', {
form: { id: r.id, sig: sigBase64, pubkey }
});
const body = JSON.parse(bod);
if (!(body && body.token)) {
throw new Error('no token');
}
mediaToken = body.token;
return body.token;
}
catch (e) {
throw e;
}
});
}
exports.getMediaToken = getMediaToken;
function getMediaInfo(muid) {
return __awaiter(this, void 0, void 0, function* () {
try {
const token = yield getMediaToken(null);
const res = yield rp.get(mediaURL + 'mymedia/' + muid, {
headers: {
'Authorization': `Bearer ${token}`,
'Content-Type': 'application/json'
},
json: true
});
return res;
}
catch (e) {
return null;
}
});
}
//# sourceMappingURL=media.js.map