botserver/deploy/whatsapp.gblib/services/WhatsappDirectLine.ts

268 lines
10 KiB
TypeScript
Raw Normal View History

/*****************************************************************************\
| ( )_ _ |
| _ _ _ __ _ _ __ ___ ___ _ _ | ,_)(_) ___ ___ _ |
| ( '_`\ ( '__)/'_` ) /'_ `\/' _ ` _ `\ /'_` )| | | |/',__)/' _ `\ /'_`\ |
| | (_) )| | ( (_| |( (_) || ( ) ( ) |( (_| || |_ | |\__, \| ( ) |( (_) ) |
| | ,__/'(_) `\__,_)`\__ |(_) (_) (_)`\__,_)`\__)(_)(____/(_) (_)`\___/' |
| | | ( )_) | |
| (_) \___/' |
| |
| General Bots Copyright (c) Pragmatismo.io. All rights reserved. |
| Licensed under the AGPL-3.0. |
| |
| According to our dual licensing model, this program can be used either |
| under the terms of the GNU Affero General Public License, version 3, |
| or under a proprietary license. |
| |
| The texts of the GNU Affero General Public License with an additional |
| permission and of our proprietary license can be found at and |
| in the LICENSE file you have received along with this program. |
| |
| This program is distributed in the hope that it will be useful, |
2018-09-11 19:40:53 -03:00
| but WITHOUT ANY WARRANTY, without even the implied warranty of |
| MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the |
| GNU Affero General Public License for more details. |
| |
| "General Bots" is a registered trademark of Pragmatismo.io. |
| The licensing of the program under the AGPLv3 does not imply a |
| trademark license. Therefore any rights, title and interest in |
| our trademarks remain entirely with us. |
| |
\*****************************************************************************/
2018-09-11 19:40:53 -03:00
const Path = require("path")
const Fs = require("fs")
const _ = require("lodash")
const Parse = require("csv-parse")
const Async = require("async")
const UrlJoin = require("url-join")
const Walk = require("fs-walk")
const logger = require("../../../src/logger")
const Swagger = require('swagger-client')
const rp = require('request-promise')
import * as request from "request-promise-native"
2018-09-11 19:40:53 -03:00
import { GBServiceCallback, GBService, IGBInstance } from "botlib"
export class WhatsappDirectLine extends GBService {
2018-09-11 19:40:53 -03:00
pollInterval = 1000
directLineClientName = 'DirectLineClient'
directLineSpecUrl = 'https://docs.botframework.com/en-us/restapi/directline3/swagger.json'
2018-09-11 19:40:53 -03:00
directLineClient: any
whatsappServiceKey: string
whatsappServiceNumber: string
whatsappServiceUrl: string
whatsappServiceWebhookUrl: string
botId: string
watermark: string = null
2018-09-11 19:40:53 -03:00
conversationIds = {}
constructor(botId, directLineSecret, whatsappServiceKey, whatsappServiceNumber, whatsappServiceUrl, whatsappServiceWebhookUrl) {
2018-09-11 19:40:53 -03:00
super()
2018-09-11 19:40:53 -03:00
this.botId = botId
this.whatsappServiceKey = whatsappServiceKey
this.whatsappServiceNumber = whatsappServiceNumber
this.whatsappServiceUrl = whatsappServiceUrl
this.whatsappServiceWebhookUrl = whatsappServiceWebhookUrl
// TODO: Migrate to Swagger 3.
this.directLineClient = rp(this.directLineSpecUrl)
.then((spec) => {
return new Swagger({
spec: JSON.parse(spec.trim()),
usePromise: true
2018-09-11 19:40:53 -03:00
})
})
.then(async (client) => {
client.clientAuthorizations.add('AuthorizationBotConnector',
new Swagger.ApiKeyAuthorization('Authorization', 'Bearer ' +
2018-09-11 19:40:53 -03:00
directLineSecret, 'header'))
var options = {
method: 'POST',
url: UrlJoin(this.whatsappServiceUrl, "webhook"),
qs:
{
token: this.whatsappServiceKey,
webhookUrl: `${this.whatsappServiceWebhookUrl}/instances/${this.botId}/whatsapp`,
set: true
},
headers:
{
'cache-control': 'no-cache'
}
2018-09-11 19:40:53 -03:00
}
try {
2018-09-11 19:40:53 -03:00
const result = await request.post(options)
logger.info(result)
} catch (error) {
2018-09-11 19:40:53 -03:00
logger.error('Error initializing 3rd party Whatsapp provider.', error)
}
2018-09-11 19:40:53 -03:00
return client
})
.catch((err) => {
2018-09-11 19:40:53 -03:00
logger.error('Error initializing DirectLine client', err)
})
}
received(req, res) {
2018-09-11 19:40:53 -03:00
let text = req.body.messages[0].body
let from = req.body.messages[0].author.split('@')[0]
let fromName = req.body.messages[0].senderName
2018-05-11 23:27:00 -03:00
2018-05-13 18:28:24 -03:00
if (req.body.messages[0].fromMe) {
2018-09-11 19:40:53 -03:00
return // Exit here.
}
2018-05-13 18:28:24 -03:00
2018-09-11 19:40:53 -03:00
logger.info(`GBWhatsapp: Hook called. from: ${from}(${fromName}), text: ${text})`)
2018-09-11 19:40:53 -03:00
let conversationId = this.conversationIds[from]
this.directLineClient.then((client) => {
if (this.conversationIds[from] == null) {
2018-09-11 19:40:53 -03:00
logger.info(`GBWhatsapp: Starting new conversation on Bot.`)
client.Conversations.Conversations_StartConversation()
.then((response) => {
2018-09-11 19:40:53 -03:00
return response.obj.conversationId
})
.then((conversationId) => {
2018-09-11 19:40:53 -03:00
this.conversationIds[from] = conversationId
this.inputMessage(client, conversationId, text,
2018-09-11 19:40:53 -03:00
from, fromName)
2018-09-11 19:40:53 -03:00
this.pollMessages(client, conversationId, from, fromName)
})
.catch((err) => {
2018-09-11 19:40:53 -03:00
console.error('Error starting conversation', err)
})
} else {
this.inputMessage(client, conversationId, text,
2018-09-11 19:40:53 -03:00
from, fromName)
}
2018-09-11 19:40:53 -03:00
res.end()
})
}
inputMessage(client, conversationId, text, from, fromName) {
client.Conversations.Conversations_PostActivity(
{
conversationId: conversationId,
activity: {
textFormat: 'plain',
text: text,
type: 'message',
from: {
id: from,
name: fromName
},
replyToId: from
}
}).catch((err) => {
2018-09-11 19:40:53 -03:00
logger.error(`GBWhatsapp: Error receiving message: ${err}.`)
})
}
pollMessages(client, conversationId, from, fromName) {
logger.info(`GBWhatsapp: Starting polling message for conversationId:
2018-09-11 19:40:53 -03:00
${conversationId}.`)
setInterval(() => {
client.Conversations.Conversations_GetActivities({
conversationId:
2018-05-13 18:28:24 -03:00
conversationId, watermark: this.watermark
})
.then((response) => {
2018-09-11 19:40:53 -03:00
this.watermark = response.obj.watermark
return response.obj.activities
})
.then((activities) => {
2018-09-11 19:40:53 -03:00
this.printMessages(activities, conversationId, from, fromName)
})
}, this.pollInterval)
}
printMessages(activities, conversationId, from, fromName) {
if (activities && activities.length) {
// Ignore own messages.
2018-09-01 13:51:58 -03:00
// TODO: this.botId instead of "general-bot-9672a8d3"
2018-09-11 19:40:53 -03:00
activities = activities.filter((m) => { return (m.from.id === "GeneralBots") && m.type === "message" })
if (activities.length) {
// Print other messages.
2018-05-13 18:28:24 -03:00
activities.forEach(activity => {
2018-09-11 19:40:53 -03:00
this.printMessage(activity, conversationId, from, fromName)
})
}
}
}
printMessage(activity, conversationId, from, fromName) {
2018-09-11 19:40:53 -03:00
let output = ""
if (activity.text) {
2018-09-11 19:40:53 -03:00
logger.info(`GBWhatsapp: MSG: ${activity.text}`)
output = activity.text
}
if (activity.attachments) {
activity.attachments.forEach((attachment) => {
switch (attachment.contentType) {
case "application/vnd.microsoft.card.hero":
2018-09-11 19:40:53 -03:00
output += `\n${this.renderHeroCard(attachment)}`
break
case "image/png":
2018-09-11 19:40:53 -03:00
logger.info('Opening the requested image ' + attachment.contentUrl)
output += `\n${attachment.contentUrl}`
break
}
2018-09-11 19:40:53 -03:00
})
}
2018-09-11 19:40:53 -03:00
this.sendToDevice(conversationId, from, fromName, output)
}
renderHeroCard(attachment) {
2018-09-11 19:40:53 -03:00
return `${attachment.content.title} - ${attachment.content.text}`
}
async sendToDevice(conversationId, to, toName, msg) {
var options = {
method: 'POST',
url: UrlJoin(this.whatsappServiceUrl, 'message'),
qs:
{
token: this.whatsappServiceKey,
phone: to,
body: msg
},
headers:
{
'cache-control': 'no-cache'
}
2018-09-11 19:40:53 -03:00
}
2018-09-11 19:40:53 -03:00
const result = await request.get(options)
}
}