257 lines
7.0 KiB

import {models} from './models'
import * as fetch from 'node-fetch'
import { Op } from 'sequelize'
import * as socket from './utils/socket'
import * as jsonUtils from './utils/json'
import * as helpers from './helpers'
import {nodeinfo} from './utils/nodeinfo'
import { loadLightning } from './utils/lightning'
const constants = require(__dirname + '/../config/constants.json');
const env = process.env.NODE_ENV || 'development';
const config = require('../config/app.json')[env];
const checkInviteHub = async (params = {}) => {
if (env != "production") {
return
}
const owner = await models.Contact.findOne({ where: { isOwner: true }})
//console.log('[hub] checking invites ping')
const inviteStrings = await models.Invite.findAll({ where: { status: { [Op.notIn]: [constants.invite_statuses.complete, constants.invite_statuses.expired] } } }).map(invite => invite.inviteString)
fetch(config.hub_api_url + '/invites/check', {
method: 'POST' ,
body: JSON.stringify({ invite_strings: inviteStrings }),
headers: { 'Content-Type': 'application/json' }
})
.then(res => res.json())
.then(json => {
if (json.object) {
json.object.invites.map(async object => {
const invite = object.invite
const pubkey = object.pubkey
const price = object.price
const invoice = object.invoice
const dbInvite = await models.Invite.findOne({ where: { inviteString: invite.pin }})
const contact = await models.Contact.findOne({ where: { id: dbInvite.contactId } })
console.log('got INVITE, status:',invite.invite_status)
console.log('existing status: ',dbInvite.status)
if (dbInvite.status != invite.invite_status) {
const updateObj:{[k:string]:any} = { status: invite.invite_status, price: price }
if(invoice) updateObj.invoice = invoice
console.log('update invite!', updateObj)
dbInvite.update(updateObj)
socket.sendJson({
type: 'invite',
response: jsonUtils.inviteToJson(dbInvite)
})
if (dbInvite.status == constants.invite_statuses.ready && contact) {
sendNotification(-1, contact.alias, 'invite')
}
}
if (pubkey && dbInvite.status == constants.invite_statuses.complete && contact) {
contact.update({ publicKey: pubkey, status: constants.contact_statuses.confirmed })
var contactJson = jsonUtils.contactToJson(contact)
contactJson.invite = jsonUtils.inviteToJson(dbInvite)
socket.sendJson({
type: 'contact',
response: contactJson
})
helpers.sendContactKeys({
contactIds: [contact.id],
sender: owner,
type: constants.message_types.contact_key,
})
}
})
}
})
.catch(error => {
console.log('[hub error]', error)
})
}
const pingHub = async (params = {}) => {
if (env != "production") {
return
}
const node = await nodeinfo()
sendHubCall({ ...params, node })
}
const sendHubCall = (params) => {
// console.log('[hub] sending ping')
fetch(config.hub_api_url + '/ping', {
method: 'POST',
body: JSON.stringify(params),
headers: { 'Content-Type': 'application/json' }
})
.then(res => res.json())
.then(json => {
// ?
})
.catch(error => {
console.log('[hub error]', error)
})
}
const pingHubInterval = (ms) => {
setInterval(pingHub, ms)
}
const checkInvitesHubInterval = (ms) => {
setInterval(checkInviteHub, ms)
}
export function sendInvoice(payReq) {
console.log('[hub] sending invoice')
fetch(config.hub_api_url + '/invoices', {
method: 'POST',
body: JSON.stringify({invoice:payReq}),
headers: { 'Content-Type': 'application/json' }
})
.then(res => res.json())
.then(json => {
// ?
})
.catch(error => {
console.log('[hub error]', error)
})
}
const finishInviteInHub = (params, onSuccess, onFailure) => {
fetch(config.hub_api_url + '/invites/finish', {
method: 'POST' ,
body: JSON.stringify(params),
headers: { 'Content-Type': 'application/json' }
})
.then(res => res.json())
.then(json => {
if (json.object) {
console.log('[hub] finished invite to hub')
onSuccess(json)
} else {
console.log('[hub] fail to finish invite in hub')
onFailure(json)
}
})
}
const payInviteInHub = (invite_string, params, onSuccess, onFailure) => {
fetch(config.hub_api_url + '/invites/' + invite_string + '/pay', {
method: 'POST' ,
body: JSON.stringify(params),
headers: { 'Content-Type': 'application/json' }
})
.then(res => res.json())
.then(json => {
if (json.object) {
console.log('[hub] finished pay to hub')
onSuccess(json)
} else {
console.log('[hub] fail to pay invite in hub')
onFailure(json)
}
})
}
async function payInviteInvoice(invoice, onSuccess, onFailure) {
const lightning = await loadLightning()
var call = lightning.sendPayment({})
call.on('data', async response => {
onSuccess(response)
})
call.on('error', async err => {
onFailure(err)
})
call.write({ payment_request:invoice })
}
const createInviteInHub = (params, onSuccess, onFailure) => {
fetch(config.hub_api_url + '/invites_new', {
method: 'POST' ,
body: JSON.stringify(params),
headers: { 'Content-Type': 'application/json' }
})
.then(res => res.json())
.then(json => {
if (json.object) {
console.log('[hub] sent invite to be created to hub')
onSuccess(json)
} else {
console.log('[hub] fail to create invite in hub')
onFailure(json)
}
})
}
const sendNotification = async (chat, name, type) => {
let message = `You have a new message from ${name}`
if(type==='invite'){
message = `Your invite to ${name} is ready`
}
if(type==='group'){
message = `You have been added to group ${name}`
}
if(type==='message' && chat.type==constants.chat_types.group && chat.name && chat.name.length){
message += ` on ${chat.name}`
}
console.log('[send notification]', { chat_id:chat.id, message })
if (chat.isMuted) {
console.log('[send notification] skipping. chat is muted.')
return
}
const owner = await models.Contact.findOne({ where: { isOwner: true }})
if (!owner.deviceId) {
console.log('[send notification] skipping. owner.deviceId not set.')
return
}
const unseenMessages = await models.Message.findAll({ where: { sender: { [Op.ne]: owner.id }, seen: false } })
const params = {
device_id: owner.deviceId,
notification: {
chat_id: chat.id,
message,
badge: unseenMessages.length
}
}
fetch("http://hub.sphinx.chat/api/v1/nodes/notify", {
method: 'POST' ,
body: JSON.stringify(params),
headers: { 'Content-Type': 'application/json' }
})
.then(res => res.json())
.then(json => console.log('[hub notification]', json))
}
export {
pingHubInterval,
checkInvitesHubInterval,
sendHubCall,
sendNotification,
createInviteInHub,
finishInviteInHub,
payInviteInHub,
payInviteInvoice
}