209 lines
5.9 KiB

import { models } from './models'
import * as md5 from 'md5'
import * as network from './network'
const constants = require('../config/constants.json');
const findOrCreateChat = async (params) => {
const { chat_id, owner_id, recipient_id } = params
let chat
let date = new Date();
date.setMilliseconds(0)
if (chat_id) {
chat = await models.Chat.findOne({ where: { id: chat_id } })
// console.log('findOrCreateChat: chat_id exists')
} else {
console.log("chat does not exists, create new")
const owner = await models.Contact.findOne({ where: { id: owner_id } })
const recipient = await models.Contact.findOne({ where: { id: recipient_id } })
const uuid = md5([owner.publicKey, recipient.publicKey].sort().join("-"))
// find by uuid
chat = await models.Chat.findOne({ where:{uuid} })
if(!chat){ // no chat! create new
chat = await models.Chat.create({
uuid: uuid,
contactIds: JSON.stringify([parseInt(owner_id), parseInt(recipient_id)]),
createdAt: date,
updatedAt: date,
type: constants.chat_types.conversation
})
}
}
return chat
}
const sendContactKeys = async (args) => {
const { type, contactIds, contactPubKey, sender, success, failure } = args
const msg = newkeyexchangemsg(type, sender)
let yes:any = null
let no:any = null
let cids = contactIds
if(!contactIds) cids = [null] // nully
await asyncForEach(cids, async contactId => {
let destination_key:string
if(!contactId){ // nully
destination_key = contactPubKey
} else {
if (contactId == sender.id) {
return
}
const contact = await models.Contact.findOne({ where: { id: contactId } })
destination_key = contact.publicKey
}
performKeysendMessage({
sender,
destination_key,
amount: 3,
msg,
success: (data) => {
yes = data
},
failure: (error) => {
no = error
}
})
})
if(no && failure){
failure(no)
}
if(!no && yes && success){
success(yes)
}
}
const performKeysendMessage = async ({ destination_key, amount, msg, success, failure, sender }) => {
const opts = {
dest: destination_key,
data: msg || {},
amt: Math.max(amount, 3)
}
try {
const r = await network.signAndSend(opts, sender.publicKey)
console.log("=> external keysend")
if (success) success(r)
} catch (e) {
console.log("MESSAGE ERROR", e)
if (failure) failure(e)
}
}
async function findOrCreateContactByPubkey(senderPubKey) {
let sender = await models.Contact.findOne({ where: { publicKey: senderPubKey } })
if (!sender) {
sender = await models.Contact.create({
publicKey: senderPubKey,
alias: "Unknown",
status: 1
})
const owner = await models.Contact.findOne({ where: { isOwner: true } })
sendContactKeys({
contactIds: [sender.id],
sender: owner,
type: constants.message_types.contact_key,
})
}
return sender
}
async function findOrCreateChatByUUID(chat_uuid, contactIds) {
let chat = await models.Chat.findOne({ where: { uuid: chat_uuid } })
if (!chat) {
var date = new Date();
date.setMilliseconds(0)
chat = await models.Chat.create({
uuid: chat_uuid,
contactIds: JSON.stringify(contactIds || []),
createdAt: date,
updatedAt: date,
type: 0 // conversation
})
}
return chat
}
async function sleep(ms) {
return new Promise(resolve => setTimeout(resolve, ms))
}
async function parseReceiveParams(payload) {
const dat = payload.content || payload
const sender_pub_key = dat.sender.pub_key
const sender_alias = dat.sender.alias
const chat_uuid = dat.chat.uuid
const chat_type = dat.chat.type
const chat_members: { [k: string]: any } = dat.chat.members || {}
const chat_name = dat.chat.name
const chat_key = dat.chat.groupKey
const chat_host = dat.chat.host
const amount = dat.message.amount
const content = dat.message.content
const remote_content = dat.message.remoteContent
const mediaToken = dat.message.mediaToken
const originalMuid = dat.message.originalMuid
const msg_id = dat.message.id||0
const msg_uuid = dat.message.uuid||''
const mediaKey = dat.message.mediaKey
const mediaType = dat.message.mediaType
const date_string = dat.message.date
const skip_payment_processing = dat.message.skipPaymentProcessing
const reply_uuid = dat.message.replyUuid
const purchaser_id = dat.message.purchaser
const isTribeOwner = dat.isTribeOwner?true:false
const isConversation = !chat_type || (chat_type && chat_type == constants.chat_types.conversation)
let sender
let chat
const owner = await models.Contact.findOne({ where: { isOwner: true } })
if (isConversation) {
sender = await findOrCreateContactByPubkey(sender_pub_key)
chat = await findOrCreateChatByUUID(
chat_uuid, [parseInt(owner.id), parseInt(sender.id)]
)
if(sender.fromGroup) { // if a private msg received, update the contact
await sender.update({fromGroup:false})
}
} else { // group
sender = await models.Contact.findOne({ where: { publicKey: sender_pub_key } })
// inject a "sender" with an alias
if(!sender && chat_type == constants.chat_types.tribe){
sender = {id:0, alias:sender_alias}
}
chat = await models.Chat.findOne({ where: { uuid: chat_uuid } })
}
return { owner, sender, chat, sender_pub_key, sender_alias, isTribeOwner, chat_uuid, amount, content, mediaToken, mediaKey, mediaType, originalMuid, chat_type, msg_id, chat_members, chat_name, chat_host, chat_key, remote_content, msg_uuid, date_string, reply_uuid, skip_payment_processing, purchaser_id }
}
export {
findOrCreateChat,
sendContactKeys,
findOrCreateContactByPubkey,
findOrCreateChatByUUID,
sleep,
parseReceiveParams,
performKeysendMessage
}
async function asyncForEach(array, callback) {
for (let index = 0; index < array.length; index++) {
await callback(array[index], index, array);
}
}
function newkeyexchangemsg(type, sender){
const includePhotoUrl = sender && sender.photoUrl && !sender.privatePhoto
return {
type: type,
sender: {
// pub_key: sender.publicKey,
contact_key: sender.contactKey,
...sender.alias && {alias: sender.alias},
...includePhotoUrl && {photoUrl: sender.photoUrl}
}
}
}