From 824eacdcb79f0c851d3c38a1bfdd9cdf0f8294a4 Mon Sep 17 00:00:00 2001 From: Marc Rabat Date: Tue, 19 Oct 2021 13:01:40 +0200 Subject: [PATCH 1/7] chore(dev-template): remove warnings in Actions and sort imports --- .../dev-template/bot/src/actions/Audio.jsx | 1 + .../dev-template/bot/src/actions/Buttons.jsx | 3 +- .../dev-template/bot/src/actions/Carousel.jsx | 12 ++++---- .../dev-template/bot/src/actions/Custom.jsx | 2 +- .../dev-template/bot/src/actions/Handoff.jsx | 30 +++++++++++++++++++ .../dev-template/bot/src/actions/Image.jsx | 1 + .../dev-template/bot/src/actions/Location.jsx | 1 + .../dev-template/bot/src/actions/Replies.jsx | 3 +- .../dev-template/bot/src/actions/Text.jsx | 3 +- .../dev-template/bot/src/actions/TextAll.jsx | 3 +- .../dev-template/bot/src/actions/Video.jsx | 1 + .../dev-template/bot/src/actions/Welcome.jsx | 11 +++++-- .../dev-template/bot/src/locales/es.js | 5 ++++ .../dev-template/bot/src/locales/index.js | 4 ++- .../dev-template/bot/src/routes.js | 27 +++++++++-------- 15 files changed, 81 insertions(+), 26 deletions(-) create mode 100644 packages/create-botonic-app/dev-template/bot/src/actions/Handoff.jsx create mode 100644 packages/create-botonic-app/dev-template/bot/src/locales/es.js diff --git a/packages/create-botonic-app/dev-template/bot/src/actions/Audio.jsx b/packages/create-botonic-app/dev-template/bot/src/actions/Audio.jsx index b0551bedde..774a4d3877 100644 --- a/packages/create-botonic-app/dev-template/bot/src/actions/Audio.jsx +++ b/packages/create-botonic-app/dev-template/bot/src/actions/Audio.jsx @@ -1,3 +1,4 @@ +// eslint-disable-next-line filenames/match-regex import { Audio, Button, Text } from '@botonic/react/src/experimental' import React from 'react' export default class extends React.Component { diff --git a/packages/create-botonic-app/dev-template/bot/src/actions/Buttons.jsx b/packages/create-botonic-app/dev-template/bot/src/actions/Buttons.jsx index e0ddcb6d41..fc45458600 100644 --- a/packages/create-botonic-app/dev-template/bot/src/actions/Buttons.jsx +++ b/packages/create-botonic-app/dev-template/bot/src/actions/Buttons.jsx @@ -1,5 +1,6 @@ +// eslint-disable-next-line filenames/match-regex +import { Button, Text } from '@botonic/react/src/experimental' import React from 'react' -import { Text, Button } from '@botonic/react/src/experimental' // import MyWebview from '../webviews/myWebview' export default class extends React.Component { diff --git a/packages/create-botonic-app/dev-template/bot/src/actions/Carousel.jsx b/packages/create-botonic-app/dev-template/bot/src/actions/Carousel.jsx index 64cd197025..9b2c3b1429 100644 --- a/packages/create-botonic-app/dev-template/bot/src/actions/Carousel.jsx +++ b/packages/create-botonic-app/dev-template/bot/src/actions/Carousel.jsx @@ -1,13 +1,14 @@ -import React from 'react' +// eslint-disable-next-line filenames/match-regex import { - Text, Button, Carousel, - Pic, Element, - Title, + Pic, Subtitle, + Text, + Title, } from '@botonic/react/src/experimental' +import React from 'react' export default class extends React.Component { render() { @@ -16,7 +17,8 @@ export default class extends React.Component { name: 'Pulp Fiction', desc: 'Le Big Mac', url: 'https://www.imdb.com/title/tt0110912', - pic: 'https://images-na.ssl-images-amazon.com/images/I/51Z95XQDHRL._SY445_.jpg', + pic: + 'https://images-na.ssl-images-amazon.com/images/I/51Z95XQDHRL._SY445_.jpg', }, { name: 'The Big Lebowski', diff --git a/packages/create-botonic-app/dev-template/bot/src/actions/Custom.jsx b/packages/create-botonic-app/dev-template/bot/src/actions/Custom.jsx index d10b4b49e6..09927a608f 100644 --- a/packages/create-botonic-app/dev-template/bot/src/actions/Custom.jsx +++ b/packages/create-botonic-app/dev-template/bot/src/actions/Custom.jsx @@ -1,6 +1,6 @@ +// eslint-disable-next-line filenames/match-regex import { Button, Reply, Text } from '@botonic/react/src/experimental' import React from 'react' - import CalendarMessage from 'webchat/custom-messages/calendar-message' export default class extends React.Component { diff --git a/packages/create-botonic-app/dev-template/bot/src/actions/Handoff.jsx b/packages/create-botonic-app/dev-template/bot/src/actions/Handoff.jsx new file mode 100644 index 0000000000..a4d0723548 --- /dev/null +++ b/packages/create-botonic-app/dev-template/bot/src/actions/Handoff.jsx @@ -0,0 +1,30 @@ +// eslint-disable-next-line filenames/match-regex +import { HandOffBuilder } from '@botonic/core' +import { Text } from '@botonic/react/src/experimental' +import React from 'react' + +export default class extends React.Component { + // TODO: Handoff logic in Botonic 1.0 + static async botonicInit({ input, botState, params, lastRoutePath }) { + let isHandOff = false + // if (openQueues.queues.indexOf('Customer Support') !== -1) { + const handOffBuilder = new HandOffBuilder(botState) + await handOffBuilder.handOff() + + isHandOff = true + // } + return { isHandOff } + } + + render() { + if (this.props.isHandOff) { + return You are being transferred to an agent! + } else { + return ( + + Sorry, right now we can't serve you... Please contact us later! + + ) + } + } +} diff --git a/packages/create-botonic-app/dev-template/bot/src/actions/Image.jsx b/packages/create-botonic-app/dev-template/bot/src/actions/Image.jsx index 8be89f91fc..8245ce8a2f 100644 --- a/packages/create-botonic-app/dev-template/bot/src/actions/Image.jsx +++ b/packages/create-botonic-app/dev-template/bot/src/actions/Image.jsx @@ -1,3 +1,4 @@ +// eslint-disable-next-line filenames/match-regex import { Button, Image, Text } from '@botonic/react/src/experimental' import React from 'react' export default class extends React.Component { diff --git a/packages/create-botonic-app/dev-template/bot/src/actions/Location.jsx b/packages/create-botonic-app/dev-template/bot/src/actions/Location.jsx index 6516627231..0df7ca2b8a 100644 --- a/packages/create-botonic-app/dev-template/bot/src/actions/Location.jsx +++ b/packages/create-botonic-app/dev-template/bot/src/actions/Location.jsx @@ -1,3 +1,4 @@ +// eslint-disable-next-line filenames/match-regex import { Button, Location, Text } from '@botonic/react/src/experimental' import React from 'react' export default class extends React.Component { diff --git a/packages/create-botonic-app/dev-template/bot/src/actions/Replies.jsx b/packages/create-botonic-app/dev-template/bot/src/actions/Replies.jsx index 3004db08fe..139d82fd85 100644 --- a/packages/create-botonic-app/dev-template/bot/src/actions/Replies.jsx +++ b/packages/create-botonic-app/dev-template/bot/src/actions/Replies.jsx @@ -1,5 +1,6 @@ +// eslint-disable-next-line filenames/match-regex +import { Reply, Text } from '@botonic/react/src/experimental' import React from 'react' -import { Text, Reply } from '@botonic/react/src/experimental' export default class extends React.Component { render() { diff --git a/packages/create-botonic-app/dev-template/bot/src/actions/Text.jsx b/packages/create-botonic-app/dev-template/bot/src/actions/Text.jsx index 082d815da3..1a6b8d579d 100644 --- a/packages/create-botonic-app/dev-template/bot/src/actions/Text.jsx +++ b/packages/create-botonic-app/dev-template/bot/src/actions/Text.jsx @@ -1,5 +1,6 @@ -import React from 'react' +// eslint-disable-next-line filenames/match-regex import { Text } from '@botonic/react/src/experimental' +import React from 'react' export default class extends React.Component { render() { diff --git a/packages/create-botonic-app/dev-template/bot/src/actions/TextAll.jsx b/packages/create-botonic-app/dev-template/bot/src/actions/TextAll.jsx index b467f676be..acf8b3eecd 100644 --- a/packages/create-botonic-app/dev-template/bot/src/actions/TextAll.jsx +++ b/packages/create-botonic-app/dev-template/bot/src/actions/TextAll.jsx @@ -1,5 +1,6 @@ +// eslint-disable-next-line filenames/match-regex +import { Button, Reply, Text } from '@botonic/react/src/experimental' import React from 'react' -import { Text, Reply, Button } from '@botonic/react/src/experimental' export default class extends React.Component { render() { diff --git a/packages/create-botonic-app/dev-template/bot/src/actions/Video.jsx b/packages/create-botonic-app/dev-template/bot/src/actions/Video.jsx index 514d026319..c3713253e0 100644 --- a/packages/create-botonic-app/dev-template/bot/src/actions/Video.jsx +++ b/packages/create-botonic-app/dev-template/bot/src/actions/Video.jsx @@ -1,3 +1,4 @@ +// eslint-disable-next-line filenames/match-regex import { Button, Text, Video } from '@botonic/react/src/experimental' import React from 'react' export default class extends React.Component { diff --git a/packages/create-botonic-app/dev-template/bot/src/actions/Welcome.jsx b/packages/create-botonic-app/dev-template/bot/src/actions/Welcome.jsx index 55281d4d64..2f2c8704c5 100644 --- a/packages/create-botonic-app/dev-template/bot/src/actions/Welcome.jsx +++ b/packages/create-botonic-app/dev-template/bot/src/actions/Welcome.jsx @@ -1,13 +1,18 @@ -import { Text } from '@botonic/react/src/experimental' +// eslint-disable-next-line filenames/match-regex +import { RequestContext, Text } from '@botonic/react/src/experimental' import React from 'react' export default class extends React.Component { + static contextType = RequestContext static async botonicInit() {} + render() { + this.context.setLocale('es') + let _ = this.context.getString return ( <> - Welcome to Botonic! - This works! + Welcome to Botonic! + From locale: {_('hello')} ) } diff --git a/packages/create-botonic-app/dev-template/bot/src/locales/es.js b/packages/create-botonic-app/dev-template/bot/src/locales/es.js new file mode 100644 index 0000000000..d706763dbe --- /dev/null +++ b/packages/create-botonic-app/dev-template/bot/src/locales/es.js @@ -0,0 +1,5 @@ +export default { + hello: ['Hola!', 'Buenas', 'Ep!'], + howAreYou: ['Que tal?', 'Como estas?'], + bye: ['Adios', 'Hasta luego!'], +} diff --git a/packages/create-botonic-app/dev-template/bot/src/locales/index.js b/packages/create-botonic-app/dev-template/bot/src/locales/index.js index 1fb527a896..04e7b667f8 100644 --- a/packages/create-botonic-app/dev-template/bot/src/locales/index.js +++ b/packages/create-botonic-app/dev-template/bot/src/locales/index.js @@ -1 +1,3 @@ -export const locales = {} +import es from './es' + +export const locales = { es } diff --git a/packages/create-botonic-app/dev-template/bot/src/routes.js b/packages/create-botonic-app/dev-template/bot/src/routes.js index 8ecf07ab5a..bb0abbfdf8 100644 --- a/packages/create-botonic-app/dev-template/bot/src/routes.js +++ b/packages/create-botonic-app/dev-template/bot/src/routes.js @@ -6,6 +6,7 @@ import ButtonsAction from './actions/Buttons' import CarouselAction from './actions/Carousel' import CustomAction from './actions/Custom' import DocumentAction from './actions/Document' +// import Handoff from './actions/Handoff' import ImageAction from './actions/Image' import LocationAction from './actions/Location' import RepliesAction from './actions/Replies' @@ -23,19 +24,21 @@ DefaultAction.botonicInit = ({ session }) => { } export const routes = [ - { text: /hi/i, action: Welcome }, - { text: 't', action: TextAction }, - { text: 'ta', action: TextAllAction }, - { text: 'b', action: ButtonsAction }, - { text: 'r', action: RepliesAction }, - { text: 'i', action: ImageAction }, - { text: 'a', action: AudioAction }, - { text: 'd', action: DocumentAction }, - { text: 'l', action: LocationAction }, - { text: 'v', action: VideoAction }, - { text: 'c', action: CarouselAction }, - { text: 'cus', action: CustomAction }, + // { path: 'handoff', text: 'handoff', action: Handoff }, + { path: 'welcome', text: /hi/i, action: Welcome }, + { path: 'text', text: 't', action: TextAction }, + { path: 'text-all', text: 'ta', action: TextAllAction }, + { path: 'buttons', text: 'b', action: ButtonsAction }, + { path: 'replies', text: 'r', action: RepliesAction }, + { path: 'image', text: 'i', action: ImageAction }, + { path: 'audio', text: 'a', action: AudioAction }, + { path: 'doc', text: 'd', action: DocumentAction }, + { path: 'location', text: 'l', action: LocationAction }, + { path: 'video', text: 'v', action: VideoAction }, + { path: 'carousel', text: 'c', action: CarouselAction }, + { path: 'custom', text: 'cus', action: CustomAction }, { + path: 'default-fallback', text: /.*/, action: DefaultAction, }, From 61685e69b9101df18d1ed359958bb15130d439ac Mon Sep 17 00:00:00 2001 From: Marc Rabat Date: Tue, 19 Oct 2021 13:02:28 +0200 Subject: [PATCH 2/7] chore(dev-template): project changes regarding botState data model --- .../api/src/handlers/botExecutor.js | 19 +++++++++++---- .../dev-template/api/src/websocket/onauth.js | 24 ++++++++++++------- 2 files changed, 30 insertions(+), 13 deletions(-) diff --git a/packages/create-botonic-app/dev-template/api/src/handlers/botExecutor.js b/packages/create-botonic-app/dev-template/api/src/handlers/botExecutor.js index e67934be6c..0ed469a0c3 100644 --- a/packages/create-botonic-app/dev-template/api/src/handlers/botExecutor.js +++ b/packages/create-botonic-app/dev-template/api/src/handlers/botExecutor.js @@ -7,14 +7,23 @@ import { handlers } from '.' const dataProvider = dataProviderFactory(process.env.DATA_PROVIDER_URL) -async function botExecutor({ input, session, lastRoutePath, websocketId }) { - const { messageEvents } = await bot.input({ - dataProvider, +async function botExecutor({ input, session, botState, websocketId }) { + const output = await bot.input({ input, session, - lastRoutePath, + botState, + dataProvider, + }) + + const events = [ + ...output.messageEvents, + { action: 'update_bot_state', ...output.botState }, + { action: 'update_session', ...output.session }, + ] + await handlers.run('sender', { + events, + websocketId, }) - await handlers.run('sender', { events: messageEvents, websocketId }) } // eslint-disable-next-line no-undef diff --git a/packages/create-botonic-app/dev-template/api/src/websocket/onauth.js b/packages/create-botonic-app/dev-template/api/src/websocket/onauth.js index f332e1b38a..b55c90f3f0 100644 --- a/packages/create-botonic-app/dev-template/api/src/websocket/onauth.js +++ b/packages/create-botonic-app/dev-template/api/src/websocket/onauth.js @@ -3,12 +3,21 @@ import { dataProviderFactory } from '@botonic/core/lib/esm/data-provider' import { decode } from 'jsonwebtoken' import { ulid } from 'ulid' +const initialBotState = { + botId: '1234', + lastRoutePath: null, + isFirstInteraction: true, + retries: 0, + locale: 'en', + isHandoff: false, + isShadowing: false, +} + export const onAuth = async ({ websocketId, data, send }) => { const { token } = JSON.parse(data) const { userId } = decode(token) const dp = dataProviderFactory(process.env.DATA_PROVIDER_URL) let user = await dp.getUser(userId) - // console.log('got user', { user }) await dp.saveEvent({ eventType: EventTypes.CONNECTION, userId, @@ -17,21 +26,20 @@ export const onAuth = async ({ websocketId, data, send }) => { status: ConnectionEventStatuses.CONNECTED, }) if (!user) { - // console.log('create user connection id', websocketId) - user = await dp.saveUser({ + const newUser = { id: userId, websocketId, isOnline: true, - route: '/', - session: JSON.stringify({}), - }) - // console.log('created user', { user }) + botState: initialBotState, + session: {}, + details: {}, // TODO: To be filled with geolocation info + } + user = await dp.saveUser(newUser) } else { // UPDATE USER CONNECTION user = await dp.updateUser({ ...user, websocketId, }) - // console.log('updated user', { user }) } } From 39fa063dc4a5541976b49782fb3e74b9d370fd6e Mon Sep 17 00:00:00 2001 From: Marc Rabat Date: Tue, 19 Oct 2021 13:05:50 +0200 Subject: [PATCH 3/7] chore(core): split data models into User, Session and BotState --- packages/botonic-core/src/models/bot-state.ts | 12 ++++++ .../models/events/message/message-event.ts | 2 + packages/botonic-core/src/models/index.ts | 2 + .../botonic-core/src/models/legacy-types.ts | 38 ++----------------- packages/botonic-core/src/models/session.ts | 3 ++ packages/botonic-core/src/models/user.ts | 15 ++++++-- 6 files changed, 34 insertions(+), 38 deletions(-) create mode 100644 packages/botonic-core/src/models/bot-state.ts create mode 100644 packages/botonic-core/src/models/session.ts diff --git a/packages/botonic-core/src/models/bot-state.ts b/packages/botonic-core/src/models/bot-state.ts new file mode 100644 index 0000000000..1046759adc --- /dev/null +++ b/packages/botonic-core/src/models/bot-state.ts @@ -0,0 +1,12 @@ +import { RoutePath } from './legacy-types' + +export interface BotState { + botId: string + isFirstInteraction: boolean + isHandoff: boolean + isShadowing: boolean + lastRoutePath: RoutePath + locale?: string + retries: number + botonicAction?: any +} diff --git a/packages/botonic-core/src/models/events/message/message-event.ts b/packages/botonic-core/src/models/events/message/message-event.ts index 193edd9b0d..14089163af 100644 --- a/packages/botonic-core/src/models/events/message/message-event.ts +++ b/packages/botonic-core/src/models/events/message/message-event.ts @@ -34,4 +34,6 @@ export interface BotonicMessageEvent extends BaseEvent { type: MessageEventTypes typing: number delay: number + // idFromChannel?:string references to msgId + // also channel } diff --git a/packages/botonic-core/src/models/index.ts b/packages/botonic-core/src/models/index.ts index ae35211424..d12ec9dd83 100644 --- a/packages/botonic-core/src/models/index.ts +++ b/packages/botonic-core/src/models/index.ts @@ -1,3 +1,5 @@ +export * from './bot-state' export * from './events' export * from './legacy-types' +export * from './session' export * from './user' diff --git a/packages/botonic-core/src/models/legacy-types.ts b/packages/botonic-core/src/models/legacy-types.ts index 5d25da156d..fc92086975 100644 --- a/packages/botonic-core/src/models/legacy-types.ts +++ b/packages/botonic-core/src/models/legacy-types.ts @@ -1,7 +1,8 @@ // TODO: This file contains all the legacy types we had in index.ts. After some refactors, we should be able to get rid of many of them. - import { DataProvider } from '../data-provider' +import { BotState } from './bot-state' import { BotonicEvent } from './events' +import { Session } from './session' export type CaseStatusType = | typeof CASE_STATUS.ATTENDING @@ -131,37 +132,6 @@ export type ProviderType = | typeof PROVIDER.WECHAT | typeof PROVIDER.WHATSAPP -export interface SessionUser { - id: string - // login - username?: string - // person name - name?: string - // whatsapp, telegram,... - provider: ProviderType - // The provider's user id - extra_data?: any - imp_id?: string - provider_id?: string -} - -// eslint-disable @typescript-eslint/naming-convention -export interface Session { - bot: { - id: string - name?: string - } - __locale?: string - __retries: number - is_first_interaction: boolean - last_session?: any - organization?: string - user: SessionUser - // after handoff - _hubtype_case_status?: CaseStatusType - _hubtype_case_typification?: string - _shadowing?: boolean -} // eslint-enable @typescript-eslint/naming-convention export type InputMatcher = (input: Input) => boolean @@ -204,8 +174,8 @@ export type Routes = R[] | ((_: BotRequest) => R[]) export interface BotRequest { input: Input - lastRoutePath: RoutePath session: Session + botState: BotState dataProvider?: DataProvider } @@ -259,7 +229,7 @@ export interface ProcessInputResult { action: Action emptyAction: Action fallbackAction: Action - lastRoutePath: RoutePath + botState: BotState params: Params } diff --git a/packages/botonic-core/src/models/session.ts b/packages/botonic-core/src/models/session.ts new file mode 100644 index 0000000000..671b64d61c --- /dev/null +++ b/packages/botonic-core/src/models/session.ts @@ -0,0 +1,3 @@ +export interface Session { + [key: string]: any +} diff --git a/packages/botonic-core/src/models/user.ts b/packages/botonic-core/src/models/user.ts index a67d9e90b2..a877bbeb6d 100644 --- a/packages/botonic-core/src/models/user.ts +++ b/packages/botonic-core/src/models/user.ts @@ -1,11 +1,18 @@ -import { Session } from './legacy-types' +import { BotState } from './bot-state' +import { Session } from './session' export interface User { id: string //TODO: UUID - providerId?: string - websocketId?: string + name?: string + userName?: string + channel: string + idFromChannel: string // providerId session: Session - route: string + botState: BotState + // functioning isOnline: boolean + websocketId?: string + // part of details? + // route: string inside botState locationInfo: string } From 3a55279d881b8b390006aec977bbc8a4e4656939 Mon Sep 17 00:00:00 2001 From: Marc Rabat Date: Tue, 19 Oct 2021 13:06:49 +0200 Subject: [PATCH 4/7] chore(api): changes regarding botState in botonic/api --- .../botonic-api/src/rest/routes/events.ts | 20 +++++++++---------- 1 file changed, 10 insertions(+), 10 deletions(-) diff --git a/packages/botonic-api/src/rest/routes/events.ts b/packages/botonic-api/src/rest/routes/events.ts index 153106a761..186f23f161 100644 --- a/packages/botonic-api/src/rest/routes/events.ts +++ b/packages/botonic-api/src/rest/routes/events.ts @@ -1,4 +1,4 @@ -import { BotonicEvent, MessageEventAck } from '@botonic/core' +import { BotonicEvent, MessageEventAck, PROVIDER } from '@botonic/core' import { dataProviderFactory } from '@botonic/core/lib/esm/data-provider' import { Router } from 'express' import jwt from 'express-jwt' @@ -76,25 +76,25 @@ export default function eventsRouter(args: any): Router { const { userId } = req.user const { message, sender } = req.body let user = await dp.getUser(userId) - user = await dp.updateUser({ - ...user, - session: JSON.stringify({ user: sender }), - }) - // TODO: Next iterations: We should receive an event with userId and eventId from frontend + const updatedUser = { ...user, ...sender } + user = await dp.updateUser(updatedUser) + // TODO: Only update ack for webchat + // TODO: Specific logic for webchat, move to webchat-events? const webchatMsgId = message.id await handlers.run('sender', { events: [ { action: 'update_message_info', - message: { id: webchatMsgId, ack: MessageEventAck.RECEIVED }, + id: webchatMsgId, + ack: MessageEventAck.RECEIVED, }, ], websocketId: user.websocketId, }) await handlers.run('botExecutor', { - input: message, - session: JSON.parse(user.session), - lastRoutePath: user.route, + input: { ...message, userId }, // To identify user executing the input + session: updatedUser.session, + botState: user.botState, websocketId: user.websocketId, }) } catch (e) { From 6901a41abc493a540f120d966df9cdad59a0b898 Mon Sep 17 00:00:00 2001 From: Marc Rabat Date: Tue, 19 Oct 2021 13:08:19 +0200 Subject: [PATCH 5/7] chore(core): core-bot and router logic to work with botState --- packages/botonic-core/src/core-bot.ts | 59 ++++++++++++------- .../src/data-provider/dynamodb-utils.ts | 8 ++- packages/botonic-core/src/handoff.ts | 21 +++---- packages/botonic-core/src/hubtype-service.ts | 8 +-- .../output-parser/botonic-output-parser.ts | 2 +- packages/botonic-core/src/plugins.ts | 15 +++-- packages/botonic-core/src/routing/router.ts | 45 ++++++++------ 7 files changed, 99 insertions(+), 59 deletions(-) diff --git a/packages/botonic-core/src/core-bot.ts b/packages/botonic-core/src/core-bot.ts index fd88b69868..d37eef5b83 100644 --- a/packages/botonic-core/src/core-bot.ts +++ b/packages/botonic-core/src/core-bot.ts @@ -6,6 +6,7 @@ import { BotonicEvent, BotRequest, BotResponse, + BotState, Locales, MessageEventAck, MessageEventFrom, @@ -81,26 +82,30 @@ export class CoreBot { ) } - getString(id: string, session: Session): string { - // @ts-ignore - return getString(this.locales, session.__locale, id) + getString(id: string, botState: BotState): string { + if (!botState.locale) { + console.error('Locale is not defined') + return '' + } + return getString(this.locales, botState.locale, id) } - setLocale(locale: string, session: Session): void { - session.__locale = locale + setLocale(locale: string, botState: BotState): void { + botState.locale = locale } async input({ input, session, - lastRoutePath, + botState, dataProvider, }: BotRequest): Promise { - session = session || {} - if (!session.__locale) session.__locale = 'en' + if (!botState.locale) botState.locale = 'en' + // @ts-ignore + const userId = input.userId + + const parsedUserEvent = this.botonicOutputParser.inputToBotonicEvent(input) - const parsedUserEvent = this.botonicOutputParser.parseFromUserInput(input) - const userId = session.user.id if (dataProvider) { // TODO: Next iterations. Review cycle of commited events to DB when messages change their ACK // @ts-ignore @@ -120,7 +125,7 @@ export class CoreBot { 'pre', input, session, - lastRoutePath, + botState, undefined, undefined, dataProvider @@ -133,7 +138,7 @@ export class CoreBot { ...(await getComputedRoutes(this.routes, { input, session, - lastRoutePath, + botState, })), ...this.defaultRoutes, ], @@ -144,18 +149,19 @@ export class CoreBot { const output = (this.router as Router).processInput( input, session, - lastRoutePath + botState ) + const request = { - getString: stringId => this.getString(stringId, session), - setLocale: locale => this.setLocale(locale, session), + getString: stringId => this.getString(stringId, botState), + setLocale: locale => this.setLocale(locale, botState), session: session || {}, params: output.params || {}, input: input, plugins: this.plugins, defaultTyping: this.defaultTyping, defaultDelay: this.defaultDelay, - lastRoutePath, + botState, dataProvider, } @@ -171,14 +177,15 @@ export class CoreBot { // console.error(e) } - lastRoutePath = output.lastRoutePath + botState.lastRoutePath = output.botState.lastRoutePath + if (this.plugins) { await runPlugins( this.plugins, 'post', input, session, - lastRoutePath, + botState, response, messageEvents, dataProvider @@ -200,13 +207,25 @@ export class CoreBot { } } - session.is_first_interaction = false + botState.isFirstInteraction = false + + if (dataProvider) { + const user = await dataProvider.getUser(userId) + if (!user) { + // throw error + } else { + const updatedUser = { ...user, session, botState } + // @ts-ignore + await dataProvider.updateUser(updatedUser) + } + } + // TODO: return also updatedUser? return { input, response, messageEvents, session, - lastRoutePath, + botState, dataProvider, } } diff --git a/packages/botonic-core/src/data-provider/dynamodb-utils.ts b/packages/botonic-core/src/data-provider/dynamodb-utils.ts index 2de283703e..41c041d33c 100644 --- a/packages/botonic-core/src/data-provider/dynamodb-utils.ts +++ b/packages/botonic-core/src/data-provider/dynamodb-utils.ts @@ -49,9 +49,15 @@ export function getUserEntity(table: Table): Entity { }, userId: [SORT_KEY_NAME, 0], websocketId: 'string', + name: 'string', + userName: 'string', + channel: 'string', + idFromChannel: 'string', isOnline: 'boolean', route: 'string', - session: 'string', + session: 'map', + botState: 'map', + details: 'map', }, table, }) diff --git a/packages/botonic-core/src/handoff.ts b/packages/botonic-core/src/handoff.ts index 3f45712467..81e35ee73b 100644 --- a/packages/botonic-core/src/handoff.ts +++ b/packages/botonic-core/src/handoff.ts @@ -57,7 +57,7 @@ export async function getOpenQueues( } export class HandOffBuilder { - _session: SessionWithBotonicAction + _botState: any _queue: string _onFinish: string _email: string @@ -66,8 +66,8 @@ export class HandOffBuilder { _caseInfo: string _shadowing: boolean - constructor(session: SessionWithBotonicAction) { - this._session = session + constructor(botState: any) { + this._botState = botState } withQueue(queueNameOrId: string): this { @@ -112,7 +112,7 @@ export class HandOffBuilder { async handOff(): Promise { return _humanHandOff( - this._session, + this._botState, this._queue, this._onFinish, this._email, @@ -154,7 +154,7 @@ interface HubtypeHandoffParams { on_finish?: string } async function _humanHandOff( - session: SessionWithBotonicAction, + botState: any, queueNameOrId = '', onFinish: string, agentEmail = '', @@ -185,7 +185,8 @@ async function _humanHandOff( if (onFinish) { params.on_finish = onFinish } - session._botonic_action = `create_case:${JSON.stringify(params)}` + botState.botonicAction = `create_case:${JSON.stringify(params)}` + botState.isHandoff = true } export async function storeCaseRating( @@ -256,14 +257,14 @@ export async function getAgentVacationRanges( } export function cancelHandoff( - session: SessionWithBotonicAction, + botState: any, typification: string | null = null ): void { let action = 'discard_case' if (typification) action = `${action}:${JSON.stringify({ typification })}` - session._botonic_action = action + botState.botonicAction = action } -export function deleteUser(session: SessionWithBotonicAction): void { - session._botonic_action = `delete_user` +export function deleteUser(botState: any): void { + botState.botonicAction = `delete_user` } diff --git a/packages/botonic-core/src/hubtype-service.ts b/packages/botonic-core/src/hubtype-service.ts index f19bce523a..75624e3cf1 100644 --- a/packages/botonic-core/src/hubtype-service.ts +++ b/packages/botonic-core/src/hubtype-service.ts @@ -2,7 +2,7 @@ import axios, { AxiosResponse } from 'axios' import Pusher, { AuthOptions, Channel } from 'pusher-js' import Channels from 'pusher-js/types/src/core/channels/channels' -import { Input, SessionUser } from './models' +import { Input } from './models' import { getWebpackEnvVar } from './utils' interface UnsentInput { @@ -21,7 +21,7 @@ interface ServerConfig { } interface HubtypeServiceArgs { appId: string - user: SessionUser + user: any lastMessageId: string lastMessageUpdateDate: string onEvent: any @@ -50,7 +50,7 @@ const PONG_TIMEOUT = 5 * 1000 // https://pusher.com/docs/channels/using_channels */ export class HubtypeService { appId: string - user: SessionUser + user: any lastMessageId: string lastMessageUpdateDate: string onEvent: any @@ -213,7 +213,7 @@ export class HubtypeService { /** * @return {Promise} */ - async postMessage(user: SessionUser, message: any): Promise { + async postMessage(user: any, message: any): Promise { try { // @ts-ignore await this.init(user) diff --git a/packages/botonic-core/src/output-parser/botonic-output-parser.ts b/packages/botonic-core/src/output-parser/botonic-output-parser.ts index 94586dccfa..d91b1ac586 100644 --- a/packages/botonic-core/src/output-parser/botonic-output-parser.ts +++ b/packages/botonic-core/src/output-parser/botonic-output-parser.ts @@ -28,7 +28,7 @@ export class BotonicOutputParser { * to be saved. This is, converting a botonic input like: '{id: 'msgId', data: 'rawData', payload: 'somePayload'}' * into a BotonicEvent with the expected properties. */ - parseFromUserInput(input: any): Partial { + inputToBotonicEvent(input: any): Partial { return this.factory.parse(input) } diff --git a/packages/botonic-core/src/plugins.ts b/packages/botonic-core/src/plugins.ts index 13b62f9d1d..ac74e3ee18 100644 --- a/packages/botonic-core/src/plugins.ts +++ b/packages/botonic-core/src/plugins.ts @@ -1,5 +1,12 @@ import { DataProvider } from './data-provider' -import { BotonicEvent, Input, PluginConfig, RoutePath, Session } from './models' +import { + BotonicEvent, + BotState, + Input, + PluginConfig, + RoutePath, + Session, +} from './models' type PluginMode = 'pre' | 'post' @@ -26,7 +33,7 @@ export async function runPlugins( mode: PluginMode, input: Input, session: Session, - lastRoutePath: RoutePath, + botState: BotState, response: string | null = null, messageEvents: Partial[] | null = null, dataProvider?: DataProvider @@ -35,12 +42,12 @@ export async function runPlugins( const p = await plugins[key] try { if (mode === 'pre') - await p.pre({ input, session, lastRoutePath, dataProvider, plugins }) + await p.pre({ input, session, botState, dataProvider, plugins }) if (mode === 'post') await p.post({ input, session, - lastRoutePath, + botState, response, messageEvents, dataProvider, diff --git a/packages/botonic-core/src/routing/router.ts b/packages/botonic-core/src/routing/router.ts index cf61b7e88d..f0f6985e97 100644 --- a/packages/botonic-core/src/routing/router.ts +++ b/packages/botonic-core/src/routing/router.ts @@ -1,6 +1,7 @@ import { NOT_FOUND_PATH } from '../constants' import { RouteInspector } from '../debug/inspector' import { + BotState, Input, MatchedValue, Matcher, @@ -19,7 +20,6 @@ import { getNotFoundAction, getPathParamsFromPathPayload, isPathPayload, - pathParamsToParams, } from './router-utils' export class Router { @@ -49,9 +49,9 @@ export class Router { processInput( input: Input, session: Session, - lastRoutePath: RoutePath = null + botState: BotState ): ProcessInputResult { - session.__retries = session?.__retries ?? 0 + botState.retries = botState.retries ?? 0 // 1. Getting the current routing state. const { @@ -59,7 +59,7 @@ export class Router { matchedRoute, params, isFlowBroken, - } = this.getRoutingState(input, session, lastRoutePath) + } = this.getRoutingState(input, session, botState) const currentRoutePath = currentRoute?.path ?? null const matchedRoutePath = matchedRoute?.path ?? null @@ -73,22 +73,24 @@ export class Router { * It has preference over ignoring retries. */ if (matchedRoute && matchedRoute.redirect) { - session.__retries = 0 + botState.retries = 0 const redirectionRoute = this.getRouteByPath(matchedRoute.redirect) if (redirectionRoute) { + botState.lastRoutePath = matchedRoute.redirect return { action: redirectionRoute.action, emptyAction: getEmptyAction(redirectionRoute.childRoutes), fallbackAction: null, - lastRoutePath: matchedRoute.redirect, + botState, params, } } + botState.lastRoutePath = null return { action: null, emptyAction: null, fallbackAction: getNotFoundAction(input, this.routes), - lastRoutePath: null, + botState, params, } } @@ -98,12 +100,13 @@ export class Router { * We have matched a route with an ignore retry, so we return directly the new bot state. The intent is to break the flow, so retries are set to 0. */ if (matchedRoute && matchedRoute.ignoreRetry) { - session.__retries = 0 + botState.retries = 0 + botState.lastRoutePath = matchedRoutePath return { action: matchedRoute.action, emptyAction: getEmptyAction(matchedRoute.childRoutes), fallbackAction: null, - lastRoutePath: matchedRoutePath, + botState, params, } } @@ -117,15 +120,16 @@ export class Router { isFlowBroken && currentRoute && currentRoute.retry && - session.__retries < currentRoute.retry + botState.retries < currentRoute.retry ) { - session.__retries = session.__retries !== 0 ? session.__retries + 1 : 1 + botState.retries = botState.retries !== 0 ? botState.retries + 1 : 1 + botState.lastRoutePath = currentRoutePath if (matchedRoute && matchedRoutePath !== NOT_FOUND_PATH) { return { action: currentRoute.action, emptyAction: getEmptyAction(matchedRoute.childRoutes), fallbackAction: matchedRoute.action, - lastRoutePath: currentRoutePath, + botState, params, } } @@ -133,7 +137,7 @@ export class Router { action: currentRoute.action ?? null, emptyAction: getEmptyAction(currentRoute.childRoutes), fallbackAction: getNotFoundAction(input, this.routes), - lastRoutePath: currentRoutePath, + botState, params, } } @@ -142,18 +146,19 @@ export class Router { * Default Scenario: * We have matched a route or not, but we don't need to execute retries logic, so retries stay to 0. */ - session.__retries = 0 + botState.retries = 0 /** * Matching Route Scenario: * We have matched a route, so we return the new bot state. */ if (matchedRoute && matchedRoutePath !== NOT_FOUND_PATH) { + botState.lastRoutePath = matchedRoutePath return { action: matchedRoute.action ?? null, emptyAction: getEmptyAction(matchedRoute.childRoutes), fallbackAction: null, - lastRoutePath: matchedRoutePath, + botState, params, } } @@ -162,12 +167,13 @@ export class Router { * 404 Scenario (No Route Found): * We have not matched any route, so we return the new bot state. */ + botState.lastRoutePath = currentRoutePath return { action: null, emptyAction: null, fallbackAction: getNotFoundAction(input, this.routes), params, - lastRoutePath: currentRoutePath, + botState, } } @@ -293,10 +299,11 @@ export class Router { getRoutingState( input: Input, session: Session, - lastRoutePath: RoutePath + botState: BotState ): RoutingState { - const currentRoute = this.getRouteByPath(lastRoutePath) - if (currentRoute && lastRoutePath) currentRoute.path = lastRoutePath + const currentRoute = this.getRouteByPath(botState.lastRoutePath) + if (currentRoute && botState.lastRoutePath) + currentRoute.path = botState.lastRoutePath if (typeof input.payload === 'string' && isPathPayload(input.payload)) { return this.getRoutingStateFromPathPayload(currentRoute, input.payload) } From ee9657a030d247d76729f397e12da7a28addd702 Mon Sep 17 00:00:00 2001 From: Marc Rabat Date: Tue, 19 Oct 2021 13:09:07 +0200 Subject: [PATCH 6/7] chore(core): adapt tests to work with botState model, disabling handoff tests for 1.0 --- packages/botonic-core/tests/handoff.test.ts | 2 +- .../botonic-core/tests/helpers/parsing.ts | 4 +- .../botonic-core/tests/helpers/routing.ts | 35 +- .../tests/parsing/user-events.test.tsx | 8 +- .../tests/routing/router.match-route.test.ts | 15 +- .../botonic-core/tests/routing/router.test.ts | 461 +++++++++--------- 6 files changed, 277 insertions(+), 248 deletions(-) diff --git a/packages/botonic-core/tests/handoff.test.ts b/packages/botonic-core/tests/handoff.test.ts index af919acb3d..db5e0904c7 100644 --- a/packages/botonic-core/tests/handoff.test.ts +++ b/packages/botonic-core/tests/handoff.test.ts @@ -2,7 +2,7 @@ import { PATH_PAYLOAD_IDENTIFIER } from '../src' import { HandOffBuilder, humanHandOff } from '../src/handoff' -describe('handOff', () => { +describe.skip('handOff', () => { test.each([ [ `create_case:{ diff --git a/packages/botonic-core/tests/helpers/parsing.ts b/packages/botonic-core/tests/helpers/parsing.ts index 6c472fa44c..f24c962916 100644 --- a/packages/botonic-core/tests/helpers/parsing.ts +++ b/packages/botonic-core/tests/helpers/parsing.ts @@ -15,8 +15,8 @@ export class BotonicOutputParserTester extends BotonicOutputParser { }) }) } - parseUserInputAndAssert(userInput, expected) { - const sut = this.parseFromUserInput(userInput) + inputToBotonicEventAndAssert(userInput, expected) { + const sut = this.inputToBotonicEvent(userInput) expect(sut).toEqual(expected) } } diff --git a/packages/botonic-core/tests/helpers/routing.ts b/packages/botonic-core/tests/helpers/routing.ts index bf6ada0b8b..650380e5c7 100644 --- a/packages/botonic-core/tests/helpers/routing.ts +++ b/packages/botonic-core/tests/helpers/routing.ts @@ -1,17 +1,42 @@ -import { PATH_PAYLOAD_IDENTIFIER, PROVIDER, Session } from '../../src' +import { BotState, PATH_PAYLOAD_IDENTIFIER, Session } from '../../src' export function testRoute(): any { return {} } export function testSession(): Session { + return {} +} + +type BotStateAttrs = { + isFirstInteraction?: boolean + lastRoutePath?: string + retries?: number +} +export function testBotState(botStateAttrs?: BotStateAttrs): BotState { return { - user: { id: 'userid', provider: PROVIDER.DEV }, - bot: { id: 'bot_id' }, - is_first_interaction: true, - __retries: 0, + botId: '1234', + isFirstInteraction: botStateAttrs?.isFirstInteraction + ? botStateAttrs?.isFirstInteraction + : true, + isHandoff: false, + isShadowing: false, + lastRoutePath: botStateAttrs?.lastRoutePath ?? null, + locale: undefined, + retries: botStateAttrs?.retries ?? 0, } } +export const botStateWithLastRoutePath = (lastRoutePath: any): BotState => { + return testBotState({ lastRoutePath }) +} + +export const botStateWithLastRoutePathAndRetries = ( + lastRoutePath: any, + retries: number +): BotState => { + return testBotState({ lastRoutePath, retries }) +} + export const createPathPayload = (pathWithParams: string): string => `${PATH_PAYLOAD_IDENTIFIER}${pathWithParams}` diff --git a/packages/botonic-core/tests/parsing/user-events.test.tsx b/packages/botonic-core/tests/parsing/user-events.test.tsx index 747a30caa4..6db2431f4b 100644 --- a/packages/botonic-core/tests/parsing/user-events.test.tsx +++ b/packages/botonic-core/tests/parsing/user-events.test.tsx @@ -18,7 +18,7 @@ describe('Parsing Text responses', () => { text: 't', markdown: true, } - tester.parseUserInputAndAssert(userInput, expected) + tester.inputToBotonicEventAndAssert(userInput, expected) }) it('TEST: Button clicked by user (no postback)', () => { @@ -37,7 +37,7 @@ describe('Parsing Text responses', () => { text: 'Button1', markdown: true, } - tester.parseUserInputAndAssert(userInput, expected) + tester.inputToBotonicEventAndAssert(userInput, expected) }) it('TEST: Postback sent by user', () => { @@ -48,7 +48,7 @@ describe('Parsing Text responses', () => { } const expected = { eventType: 'message', type: 'postback', payload: 'hi' } - tester.parseUserInputAndAssert(userInput, expected) + tester.inputToBotonicEventAndAssert(userInput, expected) }) it('TEST: Media attachment by user', () => { @@ -65,6 +65,6 @@ describe('Parsing Text responses', () => { src: '', } - tester.parseUserInputAndAssert(userInput, expected) + tester.inputToBotonicEventAndAssert(userInput, expected) }) }) diff --git a/packages/botonic-core/tests/routing/router.match-route.test.ts b/packages/botonic-core/tests/routing/router.match-route.test.ts index 811ce15aa3..5a83442c8f 100644 --- a/packages/botonic-core/tests/routing/router.match-route.test.ts +++ b/packages/botonic-core/tests/routing/router.match-route.test.ts @@ -1,6 +1,6 @@ import { BotRequest, Input } from '../../src' import { Router } from '../../src/routing' -import { testRoute, testSession } from '../helpers/routing' +import { testBotState, testRoute, testSession } from '../helpers/routing' const textInput: Input = { type: 'text', text: 'hi' } const textInputComplex: Input = { type: 'text', text: 'Cömplêx input &% 🚀' } @@ -26,8 +26,8 @@ const videoInput: Input = { const requestInput: BotRequest = { input: textInput, - session: { ...testSession(), organization: 'myOrg' }, - lastRoutePath: 'initial', + session: {}, + botState: { ...testBotState(), lastRoutePath: 'initial' }, } describe('TEST: Match route by MATCHER <> INPUT', () => { @@ -59,7 +59,7 @@ describe('TEST: Match route by MATCHER <> INPUT', () => { matcher, request.input, request.session, - request.lastRoutePath + request.botState.lastRoutePath ) it('text <> text', () => { expect(matchTextProp('hi', textInput)).toBeTruthy() @@ -140,13 +140,14 @@ describe('TEST: Match route by MATCHER <> INPUT', () => { matchPayloadProp(v => !v.startsWith('fo'), postbackInput) ).toBeFalsy() }) - it('function <> request', () => { + // TODO: Review how we adapt match route to receive botState + it.skip('function <> request', () => { expect( matchRequestProp( request => request.input.text === 'hi' && request.session.organization === 'myOrg' && - request.lastRoutePath === 'initial', + request.botState.lastRoutePath === 'initial', requestInput ) ).toBeTruthy() @@ -155,7 +156,7 @@ describe('TEST: Match route by MATCHER <> INPUT', () => { request => request.input.text === 'hello' && request.session.organization === 'myOrg' && - request.lastRoutePath === 'initial', + request.botState.lastRoutePath === 'initial', requestInput ) ).toBeFalsy() diff --git a/packages/botonic-core/tests/routing/router.test.ts b/packages/botonic-core/tests/routing/router.test.ts index 5551b11981..96fe3fe14e 100644 --- a/packages/botonic-core/tests/routing/router.test.ts +++ b/packages/botonic-core/tests/routing/router.test.ts @@ -1,24 +1,30 @@ // @ts-nocheck +import { BotState } from '../../src' import { getComputedRoutes, NoMatchingRouteError, Router, } from '../../src/routing' -import { createPathPayload, testSession } from '../helpers/routing' +import { + botStateWithLastRoutePath, + botStateWithLastRoutePathAndRetries, + createPathPayload, + testBotState, +} from '../helpers/routing' const textInput = { type: 'text', text: 'hi' } describe('TEST: Bad router initialization', () => { it('empty routes throw TypeError', () => { const router = new Router([]) - expect(() => router.processInput(textInput, testSession())).toThrow( + expect(() => router.processInput(textInput, {}, testBotState())).toThrow( NoMatchingRouteError ) }) it('null routes throw TypeError', () => { // @ts-ignore const router = new Router() - expect(() => router.processInput(textInput, testSession())).toThrow( + expect(() => router.processInput(textInput, {}, testBotState())).toThrow( TypeError ) }) @@ -29,7 +35,11 @@ const notFoundRoute = { path: '404', action: '404Action' } describe('TEST: Router initialization with default 404 route', () => { it('Router returns 404', () => { const router = new Router([notFoundRoute]) - const { fallbackAction } = router.processInput(textInput, testSession()) + const { fallbackAction } = router.processInput( + textInput, + {}, + testBotState() + ) expect(fallbackAction).toBe('404Action') }) }) @@ -92,14 +102,14 @@ describe('TEST: Root Level Accesses (lastRoutePath is null)', () => { expect( router.processInput( { type: 'text', text: 'hi', intent: 'greeting' }, - testSession(), - null + {}, + botStateWithLastRoutePath(null) ) ).toEqual({ action: 'Flow1', emptyAction: null, fallbackAction: null, - lastRoutePath: 'initial', + botState: botStateWithLastRoutePath('initial'), params: {}, }) }) @@ -108,14 +118,14 @@ describe('TEST: Root Level Accesses (lastRoutePath is null)', () => { expect( router.processInput( { type: 'postback', payload: 'help' }, - testSession(), - null + {}, + botStateWithLastRoutePath(null) ) ).toEqual({ action: 'Help', emptyAction: null, fallbackAction: null, - lastRoutePath: 'help', + botState: botStateWithLastRoutePath('help'), params: {}, }) }) @@ -124,14 +134,14 @@ describe('TEST: Root Level Accesses (lastRoutePath is null)', () => { expect( router.processInput( { type: 'text', text: 'not_found' }, - testSession(), - null + {}, + botStateWithLastRoutePath(null) ) ).toEqual({ action: null, emptyAction: null, fallbackAction: '404Action', - lastRoutePath: null, + botState: botStateWithLastRoutePath(null), params: {}, }) }) @@ -145,14 +155,14 @@ describe('TEST: Root Level Accesses (lastRoutePath is null)', () => { type: 'postback', payload: createPathPayload('initial'), }, - testSession(), - null + {}, + botStateWithLastRoutePath(null) ) ).toEqual({ action: 'Flow1', emptyAction: null, fallbackAction: null, - lastRoutePath: 'initial', + botState: botStateWithLastRoutePath('initial'), params: {}, }) }) @@ -161,14 +171,14 @@ describe('TEST: Root Level Accesses (lastRoutePath is null)', () => { expect( router.processInput( { type: 'postback', payload: createPathPayload('help') }, - testSession(), - null + {}, + botStateWithLastRoutePath(null) ) ).toEqual({ action: 'Help', emptyAction: null, fallbackAction: null, - lastRoutePath: 'help', + botState: botStateWithLastRoutePath('help'), params: {}, }) }) @@ -177,14 +187,14 @@ describe('TEST: Root Level Accesses (lastRoutePath is null)', () => { expect( router.processInput( { type: 'postback', payload: createPathPayload('404') }, - testSession(), - null + {}, + botStateWithLastRoutePath(null) ) ).toEqual({ action: null, emptyAction: null, fallbackAction: '404Action', - lastRoutePath: null, + botState: botStateWithLastRoutePath(null), params: {}, }) }) @@ -199,14 +209,14 @@ describe('TEST: Root Level Accesses (lastRoutePath is not null)', () => { expect( router.processInput( { type: 'text', text: 'hi', intent: 'greeting' }, - testSession(), - 'initial' + {}, + botStateWithLastRoutePath('initial') ) ).toEqual({ action: 'Flow1', emptyAction: null, fallbackAction: null, - lastRoutePath: 'initial', + botState: botStateWithLastRoutePath('initial'), params: {}, }) }) @@ -215,14 +225,14 @@ describe('TEST: Root Level Accesses (lastRoutePath is not null)', () => { expect( router.processInput( { type: 'text', text: 'hi', intent: 'greeting' }, - testSession(), - 'help' + {}, + botStateWithLastRoutePath('help') ) ).toEqual({ action: 'Flow1', emptyAction: null, fallbackAction: null, - lastRoutePath: 'initial', + botState: botStateWithLastRoutePath('initial'), params: {}, }) }) @@ -231,14 +241,14 @@ describe('TEST: Root Level Accesses (lastRoutePath is not null)', () => { expect( router.processInput( { type: 'postback', payload: 'help' }, - testSession(), - 'initial' + {}, + botStateWithLastRoutePath('initial') ) ).toEqual({ action: 'Help', emptyAction: null, fallbackAction: null, - lastRoutePath: 'help', + botState: botStateWithLastRoutePath('help'), params: {}, }) }) @@ -247,14 +257,14 @@ describe('TEST: Root Level Accesses (lastRoutePath is not null)', () => { expect( router.processInput( { type: 'postback', payload: 'help' }, - testSession(), - 'help' + {}, + botStateWithLastRoutePath('help') ) ).toEqual({ action: 'Help', emptyAction: null, fallbackAction: null, - lastRoutePath: 'help', + botState: botStateWithLastRoutePath('help'), params: {}, }) }) @@ -263,14 +273,14 @@ describe('TEST: Root Level Accesses (lastRoutePath is not null)', () => { expect( router.processInput( { type: 'postback', payload: 'unexisting' }, - testSession(), - 'initial' + {}, + botStateWithLastRoutePath('initial') ) ).toEqual({ action: null, emptyAction: null, fallbackAction: '404Action', - lastRoutePath: 'initial', + botState: botStateWithLastRoutePath('initial'), params: {}, }) }) @@ -279,14 +289,14 @@ describe('TEST: Root Level Accesses (lastRoutePath is not null)', () => { expect( router.processInput( { type: 'postback', payload: 'unexisting' }, - testSession(), - 'help' + {}, + botStateWithLastRoutePath('help') ) ).toEqual({ action: null, emptyAction: null, fallbackAction: '404Action', - lastRoutePath: 'help', + botState: botStateWithLastRoutePath('help'), params: {}, }) }) @@ -297,14 +307,14 @@ describe('TEST: Root Level Accesses (lastRoutePath is not null)', () => { expect( router.processInput( { type: 'postback', payload: createPathPayload('initial') }, - testSession(), - 'initial' + {}, + botStateWithLastRoutePath('initial') ) ).toEqual({ action: 'Flow1', emptyAction: null, fallbackAction: null, - lastRoutePath: 'initial', + botState: botStateWithLastRoutePath('initial'), params: {}, }) }) @@ -313,14 +323,14 @@ describe('TEST: Root Level Accesses (lastRoutePath is not null)', () => { expect( router.processInput( { type: 'postback', payload: createPathPayload('initial') }, - testSession(), - 'help' + {}, + botStateWithLastRoutePath('help') ) ).toEqual({ action: 'Flow1', emptyAction: null, fallbackAction: null, - lastRoutePath: 'initial', + botState: botStateWithLastRoutePath('initial'), params: {}, }) }) @@ -329,14 +339,14 @@ describe('TEST: Root Level Accesses (lastRoutePath is not null)', () => { expect( router.processInput( { type: 'postback', payload: createPathPayload('help') }, - testSession(), - 'initial' + {}, + botStateWithLastRoutePath('initial') ) ).toEqual({ action: 'Help', emptyAction: null, fallbackAction: null, - lastRoutePath: 'help', + botState: botStateWithLastRoutePath('help'), params: {}, }) }) @@ -345,14 +355,14 @@ describe('TEST: Root Level Accesses (lastRoutePath is not null)', () => { expect( router.processInput( { type: 'postback', payload: createPathPayload('help') }, - testSession(), - 'help' + {}, + botStateWithLastRoutePath('help') ) ).toEqual({ action: 'Help', emptyAction: null, fallbackAction: null, - lastRoutePath: 'help', + botState: botStateWithLastRoutePath('help'), params: {}, }) }) @@ -361,14 +371,14 @@ describe('TEST: Root Level Accesses (lastRoutePath is not null)', () => { expect( router.processInput( { type: 'postback', payload: createPathPayload('404') }, - testSession(), - 'initial' + {}, + botStateWithLastRoutePath('initial') ) ).toEqual({ action: null, emptyAction: null, fallbackAction: '404Action', - lastRoutePath: 'initial', + botState: botStateWithLastRoutePath('initial'), params: {}, }) }) @@ -377,14 +387,14 @@ describe('TEST: Root Level Accesses (lastRoutePath is not null)', () => { expect( router.processInput( { type: 'postback', payload: createPathPayload('404') }, - testSession(), - 'help' + {}, + botStateWithLastRoutePath('help') ) ).toEqual({ action: null, emptyAction: null, fallbackAction: '404Action', - lastRoutePath: 'help', + botState: botStateWithLastRoutePath('help'), params: {}, }) }) @@ -398,14 +408,14 @@ describe('TEST: 1st Level Accesses (lastRoutePath=initial)', () => { expect( router.processInput( { type: 'postback', payload: '1' }, - testSession(), - 'initial' + {}, + botStateWithLastRoutePath('initial') ) ).toEqual({ action: 'Flow1.1', emptyAction: null, fallbackAction: null, - lastRoutePath: 'initial/1', + botState: botStateWithLastRoutePath('initial/1'), params: {}, }) }) @@ -413,14 +423,14 @@ describe('TEST: 1st Level Accesses (lastRoutePath=initial)', () => { expect( router.processInput( { type: 'postback', payload: '2' }, - testSession(), - 'initial' + {}, + botStateWithLastRoutePath('initial') ) ).toEqual({ action: 'Flow1.2', emptyAction: 'Flow1.2.emptyAction', fallbackAction: null, - lastRoutePath: 'initial/2', + botState: botStateWithLastRoutePath('initial/2'), params: {}, }) }) @@ -428,14 +438,14 @@ describe('TEST: 1st Level Accesses (lastRoutePath=initial)', () => { expect( router.processInput( { type: 'postback', payload: '3' }, - testSession(), - 'initial' + {}, + botStateWithLastRoutePath('initial') ) ).toEqual({ action: 'Flow1.3', emptyAction: null, fallbackAction: null, - lastRoutePath: 'initial/3', + botState: botStateWithLastRoutePath('initial/3'), params: {}, }) }) @@ -444,14 +454,14 @@ describe('TEST: 1st Level Accesses (lastRoutePath=initial)', () => { expect( router.processInput( { type: 'postback', payload: 'help' }, - testSession(), - 'initial' + {}, + botStateWithLastRoutePath('initial') ) ).toEqual({ action: 'Help', emptyAction: null, fallbackAction: null, - lastRoutePath: 'help', + botState: botStateWithLastRoutePath('help'), params: {}, }) }) @@ -459,14 +469,14 @@ describe('TEST: 1st Level Accesses (lastRoutePath=initial)', () => { expect( router.processInput( { type: 'postback', payload: 'unexisting' }, - testSession(), - 'initial' + {}, + botStateWithLastRoutePath('initial') ) ).toEqual({ action: null, emptyAction: null, fallbackAction: '404Action', - lastRoutePath: 'initial', + botState: botStateWithLastRoutePath('initial'), params: {}, }) }) @@ -477,14 +487,14 @@ describe('TEST: 1st Level Accesses (lastRoutePath=initial)', () => { expect( router.processInput( { type: 'postback', payload: createPathPayload('1') }, - testSession(), - 'initial' + {}, + botStateWithLastRoutePath('initial') ) ).toEqual({ action: 'Flow1.1', emptyAction: null, fallbackAction: null, - lastRoutePath: 'initial/1', + botState: botStateWithLastRoutePath('initial/1'), params: {}, }) expect( @@ -493,14 +503,14 @@ describe('TEST: 1st Level Accesses (lastRoutePath=initial)', () => { type: 'postback', payload: createPathPayload('initial/1'), }, - testSession(), - 'initial' + {}, + botStateWithLastRoutePath('initial') ) ).toEqual({ action: 'Flow1.1', emptyAction: null, fallbackAction: null, - lastRoutePath: 'initial/1', + botState: botStateWithLastRoutePath('initial/1'), params: {}, }) }) @@ -509,27 +519,27 @@ describe('TEST: 1st Level Accesses (lastRoutePath=initial)', () => { expect( router.processInput( { type: 'postback', payload: createPathPayload('2') }, - testSession(), - 'initial' + {}, + botStateWithLastRoutePath('initial') ) ).toEqual({ action: 'Flow1.2', emptyAction: 'Flow1.2.emptyAction', fallbackAction: null, - lastRoutePath: 'initial/2', + botState: botStateWithLastRoutePath('initial/2'), params: {}, }) expect( router.processInput( { type: 'postback', payload: createPathPayload('initial/2') }, - testSession(), - 'initial' + {}, + botStateWithLastRoutePath('initial') ) ).toEqual({ action: 'Flow1.2', emptyAction: 'Flow1.2.emptyAction', fallbackAction: null, - lastRoutePath: 'initial/2', + botState: botStateWithLastRoutePath('initial/2'), params: {}, }) }) @@ -538,44 +548,43 @@ describe('TEST: 1st Level Accesses (lastRoutePath=initial)', () => { expect( router.processInput( { type: 'postback', payload: createPathPayload('3') }, - testSession(), - 'initial' + {}, + botStateWithLastRoutePath('initial') ) ).toEqual({ action: 'Flow1.3', emptyAction: null, fallbackAction: null, - lastRoutePath: 'initial/3', + botState: botStateWithLastRoutePath('initial/3'), params: {}, }) expect( router.processInput( { type: 'postback', payload: createPathPayload('initial/3') }, - testSession(), - 'initial' + {}, + botStateWithLastRoutePath('initial') ) ).toEqual({ action: 'Flow1.3', emptyAction: null, fallbackAction: null, - lastRoutePath: 'initial/3', + botState: botStateWithLastRoutePath('initial/3'), params: {}, }) }) - // HEHEHEHE it('4. help accessible from initial', () => { expect( router.processInput( { type: 'postback', payload: createPathPayload('help') }, - testSession(), - 'initial' + {}, + botStateWithLastRoutePath('initial') ) ).toEqual({ action: 'Help', emptyAction: null, fallbackAction: null, - lastRoutePath: 'help', + botState: botStateWithLastRoutePath('help'), params: {}, }) }) @@ -584,14 +593,14 @@ describe('TEST: 1st Level Accesses (lastRoutePath=initial)', () => { expect( router.processInput( { type: 'postback', payload: createPathPayload('unexisting') }, - testSession(), - 'initial' + {}, + botStateWithLastRoutePath('initial') ) ).toEqual({ action: null, fallbackAction: '404Action', emptyAction: null, - lastRoutePath: 'initial', + botState: botStateWithLastRoutePath('initial'), params: {}, }) }) @@ -605,14 +614,14 @@ describe('TEST: 1st Level Accesses (lastRoutePath=initial)', () => { type: 'postback', payload: createPathPayload('2/child'), }, - testSession(), - 'initial' + {}, + botStateWithLastRoutePath('initial') ) ).toEqual({ action: 'ChildAction', emptyAction: null, fallbackAction: null, - lastRoutePath: 'initial/2/child', + botState: botStateWithLastRoutePath('initial/2/child'), params: {}, }) expect( @@ -621,14 +630,14 @@ describe('TEST: 1st Level Accesses (lastRoutePath=initial)', () => { type: 'postback', payload: createPathPayload('initial/2/child'), }, - testSession(), - 'initial' + {}, + botStateWithLastRoutePath('initial') ) ).toEqual({ action: 'ChildAction', emptyAction: null, fallbackAction: null, - lastRoutePath: 'initial/2/child', + botState: botStateWithLastRoutePath('initial/2/child'), params: {}, }) }) @@ -642,14 +651,14 @@ describe('TEST: 2nd Level Accesses (lastRoutePath=initial/1)', () => { expect( router.processInput( { type: 'postback', payload: '1' }, - testSession(), - 'initial/1' + {}, + botStateWithLastRoutePath('initial/1') ) ).toEqual({ action: 'Flow1.1.1', emptyAction: null, fallbackAction: null, - lastRoutePath: 'initial/1/1', + botState: botStateWithLastRoutePath('initial/1/1'), params: {}, }) }) @@ -657,14 +666,14 @@ describe('TEST: 2nd Level Accesses (lastRoutePath=initial/1)', () => { expect( router.processInput( { type: 'text', text: 'whatever' }, - testSession(), - 'initial/1' + {}, + botStateWithLastRoutePath('initial/1') ) ).toEqual({ action: 'ChildRouteFallback', emptyAction: null, fallbackAction: null, - lastRoutePath: 'initial/1/fallback', + botState: botStateWithLastRoutePath('initial/1/fallback'), params: {}, }) }) @@ -672,14 +681,14 @@ describe('TEST: 2nd Level Accesses (lastRoutePath=initial/1)', () => { expect( router.processInput( { type: 'postback', payload: 'unexisting' }, - testSession(), - 'initial/1' + {}, + botStateWithLastRoutePath('initial/1') ) ).toEqual({ action: null, emptyAction: null, fallbackAction: '404Action', - lastRoutePath: 'initial/1', + botState: botStateWithLastRoutePath('initial/1'), params: {}, }) }) @@ -689,27 +698,27 @@ describe('TEST: 2nd Level Accesses (lastRoutePath=initial/1)', () => { expect( router.processInput( { type: 'postback', payload: createPathPayload('1') }, - testSession(), - 'initial/1' + {}, + botStateWithLastRoutePath('initial/1') ) ).toEqual({ action: 'Flow1.1.1', emptyAction: null, fallbackAction: null, - lastRoutePath: 'initial/1/1', + botState: botStateWithLastRoutePath('initial/1/1'), params: {}, }) expect( router.processInput( { type: 'postback', payload: createPathPayload('initial/1/1') }, - testSession(), - 'initial/1' + {}, + botStateWithLastRoutePath('initial/1') ) ).toEqual({ action: 'Flow1.1.1', emptyAction: null, fallbackAction: null, - lastRoutePath: 'initial/1/1', + botState: botStateWithLastRoutePath('initial/1/1'), params: {}, }) }) @@ -717,14 +726,14 @@ describe('TEST: 2nd Level Accesses (lastRoutePath=initial/1)', () => { expect( router.processInput( { type: 'postback', payload: createPathPayload('unexisting') }, - testSession(), - 'initial/1' + {}, + botStateWithLastRoutePath('initial/1') ) ).toEqual({ action: null, emptyAction: null, fallbackAction: '404Action', - lastRoutePath: 'initial/1', + botState: botStateWithLastRoutePath('initial/1'), params: {}, }) }) @@ -738,14 +747,14 @@ describe('TEST: 2nd Level Accesses (lastRoutePath=initial/2)', () => { expect( router.processInput( { type: 'text', text: 'child' }, - testSession(), - 'initial/2' + {}, + botStateWithLastRoutePath('initial/2') ) ).toEqual({ action: 'ChildAction', emptyAction: null, fallbackAction: null, - lastRoutePath: 'initial/2/child', + botState: botStateWithLastRoutePath('initial/2/child'), params: {}, }) }) @@ -753,14 +762,14 @@ describe('TEST: 2nd Level Accesses (lastRoutePath=initial/2)', () => { expect( router.processInput( { type: 'text', text: 'unexisting' }, - testSession(), - 'initial/2' + {}, + botStateWithLastRoutePath('initial/2') ) ).toEqual({ action: null, emptyAction: null, fallbackAction: '404Action', - lastRoutePath: 'initial/2', + botState: botStateWithLastRoutePath('initial/2'), params: {}, }) }) @@ -770,27 +779,27 @@ describe('TEST: 2nd Level Accesses (lastRoutePath=initial/2)', () => { expect( router.processInput( { type: 'postback', payload: createPathPayload('child') }, - testSession(), - 'initial/2' + {}, + botStateWithLastRoutePath('initial/2') ) ).toEqual({ action: 'ChildAction', emptyAction: null, fallbackAction: null, - lastRoutePath: 'initial/2/child', + botState: botStateWithLastRoutePath('initial/2/child'), params: {}, }) expect( router.processInput( { type: 'postback', payload: createPathPayload('initial/2/child') }, - testSession(), - 'initial/2' + {}, + botStateWithLastRoutePath('initial/2') ) ).toEqual({ action: 'ChildAction', emptyAction: null, fallbackAction: null, - lastRoutePath: 'initial/2/child', + botState: botStateWithLastRoutePath('initial/2/child'), params: {}, }) }) @@ -799,14 +808,14 @@ describe('TEST: 2nd Level Accesses (lastRoutePath=initial/2)', () => { expect( router.processInput( { type: 'postback', payload: createPathPayload('unexisting') }, - testSession(), - 'initial/2' + {}, + botStateWithLastRoutePath('initial/2') ) ).toEqual({ action: null, emptyAction: null, fallbackAction: '404Action', - lastRoutePath: 'initial/2', + botState: botStateWithLastRoutePath('initial/2'), params: {}, }) }) @@ -848,14 +857,14 @@ describe('TEST: Redirects', () => { expect( router.processInput( { type: 'text', text: 'redirectToEmptyAction' }, - testSession(), - null + {}, + botStateWithLastRoutePath(null) ) ).toEqual({ action: 'Flow1.2', emptyAction: 'Flow1.2.emptyAction', fallbackAction: null, - lastRoutePath: 'initial/2', + botState: botStateWithLastRoutePath('initial/2'), params: {}, }) }) @@ -864,14 +873,14 @@ describe('TEST: Redirects', () => { expect( router.processInput( { type: 'text', text: 'redirectToEmptyActionChildRoute' }, - testSession(), - null + {}, + botStateWithLastRoutePath(null) ) ).toEqual({ action: 'ChildAction', emptyAction: null, fallbackAction: null, - lastRoutePath: 'initial/2/child', + botState: botStateWithLastRoutePath('initial/2/child'), params: {}, }) }) @@ -880,14 +889,14 @@ describe('TEST: Redirects', () => { expect( router.processInput( { type: 'text', text: 'redirectToChildRoute' }, - testSession(), - null + {}, + botStateWithLastRoutePath(null) ) ).toEqual({ action: 'Flow1.3.2', emptyAction: null, fallbackAction: null, - lastRoutePath: 'initial/3/2', + botState: botStateWithLastRoutePath('initial/3/2'), params: {}, }) }) @@ -896,14 +905,14 @@ describe('TEST: Redirects', () => { expect( router.processInput( { type: 'text', text: 'wontBeResolved' }, - testSession(), - null + {}, + botStateWithLastRoutePath(null) ) ).toEqual({ action: null, emptyAction: null, fallbackAction: '404Action', - lastRoutePath: null, + botState: botStateWithLastRoutePath(null), params: {}, }) }) @@ -932,12 +941,12 @@ const routesWithRetries = [ ] describe('TEST: Retries', () => { - let retriesSession + let botState: BotState | null beforeEach(() => { - retriesSession = testSession() + botState = testBotState() }) afterEach(() => { - retriesSession = null + botState = null }) const router = new Router(routesWithRetries) @@ -945,94 +954,89 @@ describe('TEST: Retries', () => { expect( router.processInput( { type: 'postback', payload: 'final' }, - retriesSession, - null + {}, + botStateWithLastRoutePathAndRetries(null, 0) ) ).toEqual({ action: 'RetryFlow', emptyAction: null, fallbackAction: null, - lastRoutePath: 'retryFlow', + botState: botStateWithLastRoutePathAndRetries('retryFlow', 0), params: {}, }) }) it('Test retry flow in retryRoutes (2 mistakes)', () => { - expect(retriesSession.__retries).toEqual(0) + expect(botState.retries).toEqual(0) expect( router.processInput( { type: 'postback', payload: 'fail' }, - retriesSession, - 'retryFlow' + {}, + botStateWithLastRoutePathAndRetries('retryFlow', 0) ) ).toEqual({ action: 'RetryFlow', emptyAction: null, fallbackAction: '404Action', - lastRoutePath: 'retryFlow', + botState: botStateWithLastRoutePathAndRetries('retryFlow', 1), params: {}, }) - expect(retriesSession.__retries).toEqual(1) expect( router.processInput( { type: 'postback', payload: 'fail' }, - retriesSession, - 'retryFlow' + {}, + botStateWithLastRoutePathAndRetries('retryFlow', 1) ) ).toEqual({ action: 'RetryFlow', emptyAction: null, fallbackAction: '404Action', - lastRoutePath: 'retryFlow', + botState: botStateWithLastRoutePathAndRetries('retryFlow', 2), params: {}, }) - expect(retriesSession.__retries).toEqual(2) expect( router.processInput( { type: 'postback', payload: 'fail' }, - retriesSession, - 'retryFlow' + {}, + botStateWithLastRoutePathAndRetries('retryFlow', 2) ) ).toEqual({ action: null, emptyAction: null, fallbackAction: '404Action', - lastRoutePath: 'retryFlow', + botState: botStateWithLastRoutePathAndRetries('retryFlow', 0), params: {}, }) - expect(retriesSession.__retries).toEqual(0) }) it('Test retry flow in retryRoutes (with success)', () => { - expect(retriesSession.__retries).toEqual(0) + expect(botState.retries).toEqual(0) expect( router.processInput( { type: 'postback', payload: 'fail' }, - retriesSession, - 'retryFlow' + {}, + botStateWithLastRoutePathAndRetries('retryFlow', 0) ) ).toEqual({ action: 'RetryFlow', emptyAction: null, fallbackAction: '404Action', - lastRoutePath: 'retryFlow', + botState: botStateWithLastRoutePathAndRetries('retryFlow', 1), params: {}, }) - expect(retriesSession.__retries).toEqual(1) expect( router.processInput( { type: 'postback', payload: '1' }, - retriesSession, - 'retryFlow' + {}, + botStateWithLastRoutePathAndRetries('retryFlow', 1) ) ).toEqual({ action: 'FlowFinal1', emptyAction: null, fallbackAction: null, - lastRoutePath: 'retryFlow/1', + botState: botStateWithLastRoutePathAndRetries('retryFlow/1', 0), params: {}, }) - expect(retriesSession.__retries).toEqual(0) }) }) @@ -1056,12 +1060,12 @@ const routesWithEmptyActionRetries = [ ] describe('TEST: Retries (with empty action)', () => { - let retriesSession + let botState: BotState | null beforeEach(() => { - retriesSession = testSession() + botState = testBotState() }) afterEach(() => { - retriesSession = null + botState = null }) const router = new Router(routesWithEmptyActionRetries) @@ -1069,148 +1073,142 @@ describe('TEST: Retries (with empty action)', () => { expect( router.processInput( { type: 'postback', payload: 'final' }, - testSession(), - 'final' + {}, + botStateWithLastRoutePathAndRetries('final', 0) ) ).toEqual({ action: null, emptyAction: 'RetryFlowEmptyAction', fallbackAction: null, - lastRoutePath: 'retryFlowDA', + botState: botStateWithLastRoutePathAndRetries('retryFlowDA', 0), params: {}, }) }) it('Test retry flow in retryRoutes (1 mistakes)', () => { - expect(retriesSession.__retries).toEqual(0) + expect(botState.retries).toEqual(0) expect( router.processInput( { type: 'postback', payload: 'fail' }, - retriesSession, - 'retryFlowDA' + {}, + botStateWithLastRoutePathAndRetries('retryFlowDA', 0) ) ).toEqual({ action: null, emptyAction: 'RetryFlowEmptyAction', fallbackAction: '404Action', - lastRoutePath: 'retryFlowDA', + botState: botStateWithLastRoutePathAndRetries('retryFlowDA', 1), params: {}, }) - expect(retriesSession.__retries).toEqual(1) expect( router.processInput( { type: 'postback', payload: 'fail' }, - retriesSession, - 'retryFlowDA' + {}, + botStateWithLastRoutePathAndRetries('retryFlowDA', 1) ) ).toEqual({ action: null, emptyAction: null, fallbackAction: '404Action', - lastRoutePath: 'retryFlowDA', + botState: botStateWithLastRoutePathAndRetries('retryFlowDA', 0), params: {}, }) - expect(retriesSession.__retries).toEqual(0) }) }) describe('TEST: Retries (in childRoutes)', () => { - const retriesSession = testSession() + const botState: BotState = testBotState() + const router = new Router(routes) it('Test retry flow in childRoutes (3 mistakes, 1 goes to a fallback action which does not break flow)', () => { - expect(retriesSession.__retries).toEqual(0) + expect(botState.retries).toEqual(0) expect( router.processInput( { type: 'postback', payload: 'fail' }, - retriesSession, - 'initial/3' + {}, + botStateWithLastRoutePathAndRetries('initial/3', 0) ) ).toEqual({ action: 'Flow1.3', emptyAction: null, fallbackAction: '404Action', - lastRoutePath: 'initial/3', + botState: botStateWithLastRoutePathAndRetries('initial/3', 1), params: {}, }) - expect(retriesSession.__retries).toEqual(1) expect( router.processInput( { type: 'text', text: 'fuck' }, - retriesSession, - 'initial/3' + {}, + botStateWithLastRoutePathAndRetries('initial/3', 1) ) ).toEqual({ action: 'Flow1.3', emptyAction: null, fallbackAction: 'Insult', - lastRoutePath: 'initial/3', + botState: botStateWithLastRoutePathAndRetries('initial/3', 2), params: {}, }) - expect(retriesSession.__retries).toEqual(2) expect( router.processInput( { type: 'postback', payload: 'fail' }, - retriesSession, - 'initial/3' + {}, + botStateWithLastRoutePathAndRetries('initial/3', 2) ) ).toEqual({ action: 'Flow1.3', emptyAction: null, fallbackAction: '404Action', - lastRoutePath: 'initial/3', + botState: botStateWithLastRoutePathAndRetries('initial/3', 3), params: {}, }) - expect(retriesSession.__retries).toEqual(3) expect( router.processInput( { type: 'postback', payload: 'kk' }, - retriesSession, - 'initial/3' + {}, + botStateWithLastRoutePathAndRetries('initial/3', 3) ) ).toEqual({ action: null, emptyAction: null, fallbackAction: '404Action', - lastRoutePath: 'initial/3', + botState: botStateWithLastRoutePathAndRetries('initial/3', 0), params: {}, }) - expect(retriesSession.__retries).toEqual(0) }) }) describe('TEST: Retries (in childRoutes, ignoreRetry)', () => { - const retriesSession = testSession() + const botState: BotState = testBotState() const router = new Router(routes) it('Test retry flow in childRoutes (1 mistake and go to an action which break flow)', () => { - expect(retriesSession.__retries).toEqual(0) + expect(botState.retries).toEqual(0) expect( router.processInput( { type: 'postback', payload: 'fail' }, - retriesSession, - 'initial/3' + {}, + botStateWithLastRoutePathAndRetries('initial/3', 0) ) ).toEqual({ action: 'Flow1.3', emptyAction: null, fallbackAction: '404Action', - lastRoutePath: 'initial/3', + botState: botStateWithLastRoutePathAndRetries('initial/3', 1), params: {}, }) - expect(retriesSession.__retries).toEqual(1) expect( router.processInput( { type: 'text', payload: 'help' }, - retriesSession, - 'initial/3' + {}, + botStateWithLastRoutePathAndRetries('initial/3', 1) ) ).toEqual({ action: 'Help', emptyAction: null, fallbackAction: null, - lastRoutePath: 'help', + botState: botStateWithLastRoutePathAndRetries('help', 0), params: {}, }) - expect(retriesSession.__retries).toEqual(0) + expect(botState.retries).toEqual(0) }) }) @@ -1260,8 +1258,8 @@ describe('TEST: Converting Functional Routes to Routes', () => { // eslint-disable-next-line jest/valid-describe describe('TEST: Functional Router process input', () => { it('Resolves correctly the dynamic routes and incoming input', async () => { - const routes = async ({ input, session }) => { - if (session.is_first_interaction) { + const routes = async ({ input, session, botState }) => { + if (botState.isFirstInteraction) { return [{ text: /.*/, action: 'Hi' }] } else { return [ @@ -1272,20 +1270,23 @@ describe('TEST: Functional Router process input', () => { } const args = { input: { type: 'text', text: 'hi' }, - session: testSession(), + session: {}, + botState: testBotState(), } let computedRoutes = await getComputedRoutes(routes, args) let router = new Router(computedRoutes) expect(computedRoutes).toEqual([{ text: /.*/, action: 'Hi' }]) - expect(router.processInput(args.input, args.session, null)).toEqual({ + expect( + router.processInput(args.input, args.session, args.botState) + ).toEqual({ action: 'Hi', emptyAction: null, fallbackAction: null, - lastRoutePath: null, + botState: botStateWithLastRoutePath(null), params: {}, }) - // Now modifying args to process an input when is not first interaction - args.session.is_first_interaction = false + // // Now modifying args to process an input when is not first interaction + args.botState.isFirstInteraction = false args.input.text = 'help' computedRoutes = await getComputedRoutes(routes, args) router = new Router(computedRoutes) @@ -1293,11 +1294,13 @@ describe('TEST: Functional Router process input', () => { { path: 'help', text: 'help', action: 'Help' }, { path: '404', action: 'NotFound' }, ]) - expect(router.processInput(args.input, args.session, null)).toEqual({ + expect( + router.processInput(args.input, args.session, args.botState) + ).toEqual({ action: 'Help', emptyAction: null, fallbackAction: null, - lastRoutePath: 'help', + botState: { ...args.botState, lastRoutePath: 'help' }, params: {}, }) }) From e85f900d2a13328f4fcdc9dfd2c32dab80b725c8 Mon Sep 17 00:00:00 2001 From: Marc Rabat Pla <35448568+vanbasten17@users.noreply.github.com> Date: Fri, 29 Oct 2021 17:19:04 +0200 Subject: [PATCH 7/7] Full stack/refactor internal session vars - adapt frontend code to botState data model (#1967) * chore(react): changes in webchat regarding move to botState model * refactor(core/api/dev-template): remove initial locale and minor fixes * refactor(react): remove locale from initial session and make use of merge func --- .../botonic-api/src/rest/routes/events.ts | 2 +- packages/botonic-core/src/handoff.ts | 7 +- packages/botonic-core/src/hubtype-service.ts | 2 +- .../src/experimental/dev-app.jsx | 6 +- .../botonic-react/src/experimental/index.js | 29 ++-- .../src/experimental/util/webchat.js | 13 +- .../src/experimental/webchat-app.jsx | 41 ++++-- .../src/experimental/webchat/actions.jsx | 2 + .../src/experimental/webchat/hooks.js | 50 ++++++- .../src/experimental/webchat/session-view.jsx | 12 +- .../src/experimental/webchat/webchat-dev.jsx | 38 ++--- .../experimental/webchat/webchat-reducer.js | 11 +- .../src/experimental/webchat/webchat.jsx | 136 ++++++++++-------- packages/botonic-react/src/webchat-app.jsx | 2 +- .../dev-template/api/src/websocket/onauth.js | 1 - 15 files changed, 225 insertions(+), 127 deletions(-) diff --git a/packages/botonic-api/src/rest/routes/events.ts b/packages/botonic-api/src/rest/routes/events.ts index 186f23f161..c5a0a70c2a 100644 --- a/packages/botonic-api/src/rest/routes/events.ts +++ b/packages/botonic-api/src/rest/routes/events.ts @@ -93,7 +93,7 @@ export default function eventsRouter(args: any): Router { }) await handlers.run('botExecutor', { input: { ...message, userId }, // To identify user executing the input - session: updatedUser.session, + session: user.session, botState: user.botState, websocketId: user.websocketId, }) diff --git a/packages/botonic-core/src/handoff.ts b/packages/botonic-core/src/handoff.ts index 81e35ee73b..acc2886408 100644 --- a/packages/botonic-core/src/handoff.ts +++ b/packages/botonic-core/src/handoff.ts @@ -1,7 +1,7 @@ import axios from 'axios' import { PATH_PAYLOAD_IDENTIFIER } from './constants' -import { Session } from './models' +import { BotState, Session } from './models' const HUBTYPE_API_URL = 'https://api.hubtype.com' @@ -257,14 +257,15 @@ export async function getAgentVacationRanges( } export function cancelHandoff( - botState: any, + botState: BotState, typification: string | null = null ): void { let action = 'discard_case' if (typification) action = `${action}:${JSON.stringify({ typification })}` botState.botonicAction = action + botState.isHandoff = false // TODO: Review handoff functionalities } -export function deleteUser(botState: any): void { +export function deleteUser(botState: BotState): void { botState.botonicAction = `delete_user` } diff --git a/packages/botonic-core/src/hubtype-service.ts b/packages/botonic-core/src/hubtype-service.ts index 75624e3cf1..a21c08afb4 100644 --- a/packages/botonic-core/src/hubtype-service.ts +++ b/packages/botonic-core/src/hubtype-service.ts @@ -185,7 +185,7 @@ export class HubtypeService { } handleConnectionChange(online: boolean): void { - this.onPusherEvent({ action: 'connectionChange', online }) + this.onPusherEvent({ action: 'connection_change', online }) } onPusherEvent(event: any): void { diff --git a/packages/botonic-react/src/experimental/dev-app.jsx b/packages/botonic-react/src/experimental/dev-app.jsx index 4389a7507d..89f308d0b7 100644 --- a/packages/botonic-react/src/experimental/dev-app.jsx +++ b/packages/botonic-react/src/experimental/dev-app.jsx @@ -102,8 +102,10 @@ export class DevApp extends WebchatApp { enableAnimations={enableAnimations} storage={storage} storageKey={storageKey} - getString={(stringId, session) => this.bot.getString(stringId, session)} - setLocale={(locale, session) => this.bot.setLocale(locale, session)} + getString={(stringId, botState) => + this.bot.getString(stringId, botState) + } + setLocale={(locale, botState) => this.bot.setLocale(locale, botState)} onInit={(...args) => this.onInitWebchat(...args)} onOpen={(...args) => this.onOpenWebchat(...args)} onClose={(...args) => this.onCloseWebchat(...args)} diff --git a/packages/botonic-react/src/experimental/index.js b/packages/botonic-react/src/experimental/index.js index b96c46ca3f..ce971d10a2 100644 --- a/packages/botonic-react/src/experimental/index.js +++ b/packages/botonic-react/src/experimental/index.js @@ -38,9 +38,9 @@ class WebsocketBackendService { // On Event Received... this.wsClient.addEventListener('message', event => { console.log(event, this.onEvent) - const message = JSON.parse(decode(event.data)) + const eventData = JSON.parse(decode(event.data)) if (this.onEvent && typeof this.onEvent === 'function') - this.onEvent({ message }) + this.onEvent(eventData) }) } async doAuthAndUpdateJwt() { @@ -72,7 +72,7 @@ class WebsocketBackendService { `${REST_API_URL}events/`, { message, - sender: user, + sender: user, // TODO: Really needed or we should pass user information through JWT? }, { headers: { Authorization: 'Bearer ' + this.jwt } } // Note: Do not use string template as it will convert the token with commas, which will be invalid ) @@ -85,14 +85,14 @@ class WebsocketBackendService { if (hasErrors) { // TODO: Handle rest of errors await this.doAuthAndUpdateJwt() - await this.postMessage(user, message) + // await this.postMessage(user, message) // Temporary, avoid infinite events loop } } } export class FullstackProdApp extends WebchatApp { - async onUserInput({ user, input }) { - this.onMessage && this.onMessage(this, { from: 'user', message: input }) + async onUserInput({ user, input, session, botState }) { + this.onMessage && this.onMessage(this, { from: input.from, message: input }) this.backendService.postMessage(user, input) } @@ -100,7 +100,7 @@ export class FullstackProdApp extends WebchatApp { return await this.backendService.doAuth({ userId }) } - onStateChange({ session: { user }, messagesJSON, jwt, updateJwt }) { + onStateChange({ user, messagesJSON, jwt, updateJwt }) { if (!this.backendService && user) { const lastMessage = messagesJSON[messagesJSON.length - 1] this.backendService = new WebsocketBackendService({ @@ -121,8 +121,8 @@ export class FullstackDevApp extends DevApp { console.log('FullstackDevApp ', args.playgroundCode) } - async onUserInput({ user, input }) { - this.onMessage && this.onMessage(this, { from: 'user', message: input }) + async onUserInput({ user, input, session, botState }) { + this.onMessage && this.onMessage(this, { from: input.from, message: input }) this.backendService && this.backendService.postMessage(user, input) } @@ -179,8 +179,10 @@ export class FullstackDevApp extends DevApp { storageKey={storageKey} playgroundCode={this.playgroundCode} onStateChange={webchatState => this.onStateChange(webchatState)} - getString={(stringId, session) => this.bot.getString(stringId, session)} - setLocale={(locale, session) => this.bot.setLocale(locale, session)} + getString={(stringId, botState) => + this.bot.getString(stringId, botState) + } + setLocale={(locale, botState) => this.bot.setLocale(locale, botState)} onInit={(...args) => this.onInitWebchat(...args)} onOpen={(...args) => this.onOpenWebchat(...args)} onClose={(...args) => this.onCloseWebchat(...args)} @@ -194,7 +196,7 @@ export class FullstackDevApp extends DevApp { return await this.backendService.doAuth({ userId }) } - onStateChange({ session: { user }, messagesJSON, jwt, updateJwt }) { + onStateChange({ user, messagesJSON, jwt, updateJwt }) { if (!this.backendService && user) { const lastMessage = messagesJSON[messagesJSON.length - 1] this.backendService = new WebsocketBackendService({ @@ -260,6 +262,7 @@ export class BrowserProdApp extends WebchatApp { ...botOptions, }) } + // TODO: Review how this be done for only browser versions async onUserInput({ input, session, lastRoutePath }) { this.onMessage && this.onMessage(this, { from: 'user', message: input }) const resp = await this.bot.input({ input, session, lastRoutePath }) @@ -280,10 +283,10 @@ export { ShareButton } from '../components/share-button' export { Subtitle } from '../components/subtitle' export { Title } from '../components/title' export { WebchatSettings } from '../components/webchat-settings' -export { RequestContext, WebchatContext } from '../contexts' export { staticAsset } from '../util/environment' export { getBotonicApp } from '../webchat' export { WebviewApp } from '../webview' +export { RequestContext, WebchatContext } from './contexts' // Experimental export { Audio } from './components/audio' export { Carousel } from './components/carousel' diff --git a/packages/botonic-react/src/experimental/util/webchat.js b/packages/botonic-react/src/experimental/util/webchat.js index 217f38da15..806aa4339f 100644 --- a/packages/botonic-react/src/experimental/util/webchat.js +++ b/packages/botonic-react/src/experimental/util/webchat.js @@ -1,3 +1,4 @@ +import { PROVIDER } from '@botonic/core' import merge from 'lodash.merge' import UAParser from 'ua-parser-js' import { v4 as uuidv4 } from 'uuid' @@ -34,14 +35,14 @@ export const createUser = () => { return { id: uuidv4(), name, + channel: PROVIDER.DEV, } } -export const initSession = session => { - if (!session) session = {} - const hasUserId = session.user && session.user.id !== undefined - if (!session.user || Object.keys(session.user).length === 0 || !hasUserId) - session.user = !hasUserId ? merge(session.user, createUser()) : createUser() - return session + +export const initUser = user => { + if (!user) return createUser() + if (user && !user.id) return merge(user, createUser()) + return user } export const shouldKeepSessionOnReload = ({ diff --git a/packages/botonic-react/src/experimental/webchat-app.jsx b/packages/botonic-react/src/experimental/webchat-app.jsx index d268a768e0..8b310392cb 100644 --- a/packages/botonic-react/src/experimental/webchat-app.jsx +++ b/packages/botonic-react/src/experimental/webchat-app.jsx @@ -139,22 +139,27 @@ export class WebchatApp { } onServiceEvent(event) { - if (event.action === 'connectionChange') + const { action, ...eventData } = event + if (action === 'connection_change') this.webchatRef.current.setOnline(event.online) - // TODO: Temporary solution, decide how we will send these events in next iterations - else if (event.message.action === 'update_message_info') { - const { message } = event.message - this.updateMessageInfo(message.id, message) - } else if (event.action === 'update_message_info') - this.updateMessageInfo(event.message.id, event.message) - else if (event.message.type === 'update_webchat_settings') - this.updateWebchatSettings(event.message.data) - else if (event.message.type === 'sender_action') - this.setTyping(event.message.data === 'typing_on') - else { + else if (action === 'update_message_info') { + this.updateMessageInfo(eventData.id, eventData) + } else if (action === 'update_user') { + this.updateUser(eventData) + } else if (action === 'update_session') { + this.updateSession(eventData) + } else if (action === 'update_bot_state') { + this.updateBotState(eventData) + } + // TODO: Discuss how this updates to be done + else if (eventData.type === 'update_webchat_settings') + this.updateWebchatSettings(event.data) + else if (eventData.type === 'sender_action') + this.setTyping(event.data === 'typing_on') + else if (eventData.eventType === 'message') { this.onMessage && - this.onMessage(this, { from: SENDERS.bot, message: event.message }) - this.addBotMessage(event.message) + this.onMessage(this, { from: SENDERS.bot, message: eventData }) + this.addBotMessage(eventData) } } @@ -162,6 +167,14 @@ export class WebchatApp { this.webchatRef.current.updateUser(user) } + updateSession(session) { + this.webchatRef.current.updateSession(session) + } + + updateBotState(botState) { + this.webchatRef.current.updateBotState(botState) + } + addBotMessage(message) { this.webchatRef.current.addBotResponse({ response: msgToBotonic( diff --git a/packages/botonic-react/src/experimental/webchat/actions.jsx b/packages/botonic-react/src/experimental/webchat/actions.jsx index 2c11e7acc9..c2dad111b3 100644 --- a/packages/botonic-react/src/experimental/webchat/actions.jsx +++ b/packages/botonic-react/src/experimental/webchat/actions.jsx @@ -20,3 +20,5 @@ export const UPDATE_LAST_MESSAGE_DATE = 'updateLastMessageDate' export const SET_CURRENT_ATTACHMENT = 'setCurrentAttachment' export const SET_ONLINE = 'setOnline' export const UPDATE_JWT = 'updateJwt' +export const UPDATE_USER = 'updateUser' +export const UPDATE_BOT_STATE = 'updateBotState' diff --git a/packages/botonic-react/src/experimental/webchat/hooks.js b/packages/botonic-react/src/experimental/webchat/hooks.js index ea0d5067d8..47a2c2aa31 100644 --- a/packages/botonic-react/src/experimental/webchat/hooks.js +++ b/packages/botonic-react/src/experimental/webchat/hooks.js @@ -13,6 +13,7 @@ import { TOGGLE_EMOJI_PICKER, TOGGLE_PERSISTENT_MENU, TOGGLE_WEBCHAT, + UPDATE_BOT_STATE, UPDATE_DEV_SETTINGS, UPDATE_HANDOFF, UPDATE_JWT, @@ -24,10 +25,31 @@ import { UPDATE_SESSION, UPDATE_THEME, UPDATE_TYPING, + UPDATE_USER, UPDATE_WEBVIEW, } from './actions' import { webchatReducer } from './webchat-reducer' +export const initialUser = { + id: undefined, + name: undefined, + userName: undefined, + channel: undefined, + idFromChannel: undefined, + isOnline: true, +} + +const initialBotState = { + botId: undefined, + lastRoutePath: null, + isFirstInteraction: true, + retries: 0, + isHandoff: false, + isShadowing: false, +} + +const initialSession = {} + export const webchatInitialState = { width: WEBCHAT.DEFAULTS.WIDTH, height: WEBCHAT.DEFAULTS.HEIGHT, @@ -38,9 +60,9 @@ export const webchatInitialState = { typing: false, webview: null, webviewParams: null, - session: { user: null }, - lastRoutePath: null, - handoff: false, + // session: { user: null }, + // lastRoutePath: null, + // handoff: false, theme: { headerTitle: WEBCHAT.DEFAULTS.TITLE, brandColor: COLORS.BOTONIC_BLUE, @@ -53,7 +75,7 @@ export const webchatInitialState = { }, themeUpdates: {}, error: {}, - online: true, + isWebchatOnline: true, devSettings: { keepSessionOnReload: false }, isWebchatOpen: false, isEmojiPickerOpen: false, @@ -62,6 +84,9 @@ export const webchatInitialState = { lastMessageUpdate: undefined, currentAttachment: undefined, jwt: null, + user: initialUser, + session: initialSession, + botState: initialBotState, } export function useWebchat() { @@ -87,12 +112,23 @@ export function useWebchat() { type: UPDATE_WEBVIEW, payload: { webview, webviewParams: params }, }) - const updateSession = session => { + const updateSession = session => webchatDispatch({ type: UPDATE_SESSION, payload: session, }) - } + + const updateUser = user => + webchatDispatch({ + type: UPDATE_USER, + payload: user, + }) + + const updateBotState = botState => + webchatDispatch({ + type: UPDATE_BOT_STATE, + payload: botState, + }) const updateLastRoutePath = path => webchatDispatch({ @@ -198,6 +234,8 @@ export function useWebchat() { updateLastMessageDate, setCurrentAttachment, updateJwt, + updateBotState, + updateUser, } } diff --git a/packages/botonic-react/src/experimental/webchat/session-view.jsx b/packages/botonic-react/src/experimental/webchat/session-view.jsx index 85d254b703..cbf8d2a7b5 100644 --- a/packages/botonic-react/src/experimental/webchat/session-view.jsx +++ b/packages/botonic-react/src/experimental/webchat/session-view.jsx @@ -100,7 +100,9 @@ const KeepSessionContainer = styled.div` export const SessionView = props => { // eslint-disable-next-line react-hooks/rules-of-hooks const { webchatState, updateDevSettings } = props.webchatHooks || useWebchat() - const { latestInput: input, session, lastRoutePath } = webchatState + const { latestInput: input, session, botState } = webchatState + const { type, id, ...latestInputData } = input + const toggleSessionView = () => updateDevSettings({ ...webchatState.devSettings, @@ -122,7 +124,7 @@ export const SessionView = props => { label='INPUT:' value={ input && Object.keys(input).length - ? `[${input.type}] ${input.data || ''}` + ? `[${type}] ${JSON.stringify(latestInputData) || ''}` : '' } /> @@ -137,8 +139,12 @@ export const SessionView = props => { /> + + + + diff --git a/packages/botonic-react/src/experimental/webchat/webchat-dev.jsx b/packages/botonic-react/src/experimental/webchat/webchat-dev.jsx index 57c4b20d37..c4a633efc1 100644 --- a/packages/botonic-react/src/experimental/webchat/webchat-dev.jsx +++ b/packages/botonic-react/src/experimental/webchat/webchat-dev.jsx @@ -5,7 +5,6 @@ import React, { forwardRef, useEffect, useState } from 'react' import { createPortal } from 'react-dom' import styled from 'styled-components' -import { SessionView } from '../../webchat/session-view' import MessengerLogo from './assets/messenger.svg' import Open from './assets/open.svg' import OpenNewWindow from './assets/open-new-window.svg' @@ -13,6 +12,7 @@ import TelegramLogo from './assets/telegram.svg' import WebchatLogo from './assets/webchat.svg' import WhatsappLogo from './assets/whatsapp.svg' import { useWebchat } from './hooks' +import { SessionView } from './session-view' import { Webchat } from './webchat' export const DebugTab = styled.div` @@ -341,22 +341,24 @@ export const PlaygroundPortal = props => document.body ) -const initialSession = { - is_first_interaction: true, - last_session: {}, - user: { - id: '000001', - username: 'johndoe', - name: 'John Doe', - provider: PROVIDER.DEV, - provider_id: '0000000', - extra_data: {}, - }, - organization: '', - bot: { - id: '0000000', - name: 'botName', - }, +const initialUser = { + id: '000001', + name: 'John Doe', + username: 'johndoe', + channel: PROVIDER.DEV, + idFromChannel: '0000000', + details: {}, +} + +const initialSession = {} + +const initialBotState = { + botId: '0000000', + isFirstInteraction: true, + retries: 0, + lastRoutePath: null, + isHandoff: false, + isShadowing: false, } // eslint-disable-next-line react/display-name @@ -377,7 +379,9 @@ export const WebchatDev = forwardRef((props, ref) => { {...props} ref={ref} webchatHooks={webchatHooks} + initialUser={initialUser} initialSession={initialSession} + initialBotState={initialBotState} initialDevSettings={{ keepSessionOnReload: webchatState.devSettings.keepSessionOnReload, showSessionView: webchatState.devSettings.showSessionView, diff --git a/packages/botonic-react/src/experimental/webchat/webchat-reducer.js b/packages/botonic-react/src/experimental/webchat/webchat-reducer.js index 06e9a0e4d4..c42ed9f5ea 100644 --- a/packages/botonic-react/src/experimental/webchat/webchat-reducer.js +++ b/packages/botonic-react/src/experimental/webchat/webchat-reducer.js @@ -6,6 +6,7 @@ import { TOGGLE_EMOJI_PICKER, TOGGLE_PERSISTENT_MENU, TOGGLE_WEBCHAT, + UPDATE_BOT_STATE, UPDATE_DEV_SETTINGS, UPDATE_HANDOFF, UPDATE_JWT, @@ -14,16 +15,16 @@ import { UPDATE_SESSION, UPDATE_THEME, UPDATE_TYPING, + UPDATE_USER, UPDATE_WEBVIEW, } from './actions' import { messagesReducer } from './messages-reducer' +// eslint-disable-next-line complexity export function webchatReducer(state, action) { switch (action.type) { case UPDATE_WEBVIEW: return { ...state, ...action.payload } - case UPDATE_SESSION: - return { ...state, session: { ...action.payload } } case UPDATE_TYPING: return { ...state, typing: action.payload } case UPDATE_THEME: @@ -55,6 +56,12 @@ export function webchatReducer(state, action) { return { ...state, currentAttachment: action.payload } case UPDATE_JWT: return { ...state, jwt: action.payload } + case UPDATE_USER: + return { ...state, user: action.payload } + case UPDATE_SESSION: + return { ...state, session: action.payload } + case UPDATE_BOT_STATE: + return { ...state, botState: action.payload } default: return messagesReducer(state, action) } diff --git a/packages/botonic-react/src/experimental/webchat/webchat.jsx b/packages/botonic-react/src/experimental/webchat/webchat.jsx index d54cafdb68..63e2488162 100644 --- a/packages/botonic-react/src/experimental/webchat/webchat.jsx +++ b/packages/botonic-react/src/experimental/webchat/webchat.jsx @@ -42,12 +42,6 @@ import { scrollToBottom } from '../../util/dom' import { isDev, resolveImage } from '../../util/environment' import { ConditionalWrapper } from '../../util/react' import { deserializeRegex, stringifyWithRegexs } from '../../util/regexs' -import { - _getThemeProperty, - getServerErrorMessage, - initSession, - shouldKeepSessionOnReload, -} from '../../util/webchat' import { Attachment } from '../../webchat/components/attachment' import { EmojiPicker, @@ -61,12 +55,6 @@ import { SendButton } from '../../webchat/components/send-button' import { TypingIndicator } from '../../webchat/components/typing-indicator' import { DeviceAdapter } from '../../webchat/devices/device-adapter' import { StyledWebchatHeader } from '../../webchat/header' -import { - useComponentWillMount, - usePrevious, - useTyping, - useWebchat, -} from '../../webchat/hooks' import { WebchatMessageList } from '../../webchat/message-list' import { WebchatReplies } from '../../webchat/replies' import { useStorageState } from '../../webchat/use-storage-state-hook' @@ -74,6 +62,18 @@ import { WebviewContainer } from '../../webchat/webview' import { Audio, Document, Image, Video } from '../components' import { Text } from '../components/text' import { msgToBotonic } from '../msg-to-botonic' +import { + _getThemeProperty, + getServerErrorMessage, + initUser, + shouldKeepSessionOnReload, +} from '../util/webchat' +import { + useComponentWillMount, + usePrevious, + useTyping, + useWebchat, +} from '../webchat/hooks' export const getParsedAction = botonicAction => { const splittedAction = botonicAction.split('create_case:') if (splittedAction.length <= 1) return undefined @@ -181,7 +181,6 @@ export const Webchat = forwardRef((props, ref) => { updateLatestInput, updateTyping, updateWebview, - updateSession, updateLastRoutePath, updateHandoff, updateTheme, @@ -198,14 +197,23 @@ export const Webchat = forwardRef((props, ref) => { updateLastMessageDate, setCurrentAttachment, updateJwt, + updateUser, + updateSession, + updateBotState, // eslint-disable-next-line react-hooks/rules-of-hooks } = props.webchatHooks || useWebchat() const firstUpdate = useRef(true) - const isOnline = () => webchatState.online + const isOnline = () => webchatState.isWebchatOnline const currentDateString = () => new Date().toISOString() const theme = merge(webchatState.theme, props.theme) - const { initialSession, initialDevSettings, onStateChange } = props + const { + initialUser, + initialSession, + initialBotState, + initialDevSettings, + onStateChange, + } = props const getThemeProperty = _getThemeProperty(theme) const storage = props.storage === undefined ? localStorage : props.storage @@ -227,14 +235,13 @@ export const Webchat = forwardRef((props, ref) => { JSON.parse( stringifyWithRegexs({ messages: webchatState.messagesJSON, - session: webchatState.session, - botState: webchatState.botState, - user: webchatState.user, - lastRoutePath: webchatState.lastRoutePath, devSettings: webchatState.devSettings, lastMessageUpdate: webchatState.lastMessageUpdate, themeUpdates: webchatState.themeUpdates, jwt: webchatState.jwt, + user: webchatState.user, + session: webchatState.session, + botState: webchatState.botState, }) ) ) @@ -261,16 +268,16 @@ export const Webchat = forwardRef((props, ref) => { const sendUserInput = async input => { input = { - ...input, ack: MessageEventAck.DRAFT, from: MessageEventFrom.USER, + ...input, } props.onUserInput && props.onUserInput({ - user: webchatState.session.user, + user: webchatState.user, input, session: webchatState.session, - lastRoutePath: webchatState.lastRoutePath, + botState: webchatState.botState, }) } @@ -304,16 +311,18 @@ export const Webchat = forwardRef((props, ref) => { // Load initial state from storage useEffect(() => { - let { + const { messages, - session, lastRoutePath, devSettings, lastMessageUpdate, themeUpdates, + user, + session, + botState, } = botonicState || {} - session = initSession(session) - updateSession(session) + updateUser(merge(initialUser, initUser(user))) + if (shouldKeepSessionOnReload({ initialDevSettings, devSettings })) { if (messages) { messages.forEach(m => { @@ -326,12 +335,15 @@ export const Webchat = forwardRef((props, ref) => { if (newComponent) addMessageComponent(newComponent) }) } - if (initialSession) updateSession(merge(initialSession, session)) - if (lastRoutePath) updateLastRoutePath(lastRoutePath) + if (initialSession) { + updateSession(merge(initialSession, session)) + } + if (initialBotState) { + updateBotState(merge(initialBotState, botState)) + } } else { - session.__retries = 0 - session.is_first_interaction = true updateSession(merge(initialSession, session)) + updateBotState(merge(initialBotState, botState)) } if (devSettings) updateDevSettings(devSettings) else if (initialDevSettings) updateDevSettings(initialDevSettings) @@ -348,21 +360,26 @@ export const Webchat = forwardRef((props, ref) => { }, [webchatState.isWebchatOpen]) useEffect(() => { - if (onStateChange && typeof onStateChange === 'function') { + if ( + onStateChange && + typeof onStateChange === 'function' && + webchatState.user.id + ) { onStateChange({ ...webchatState, updateJwt }) } saveWebchatState(webchatState) }, [ webchatState.messagesJSON, - webchatState.session, - webchatState.lastRoutePath, webchatState.devSettings, webchatState.lastMessageUpdate, webchatState.jwt, + webchatState.user, + webchatState.session, + webchatState.botState, ]) useAsyncEffect(async () => { - if (!webchatState.online) { + if (!webchatState.isWebchatOnline) { setError({ message: getServerErrorMessage(props.server), }) @@ -371,7 +388,7 @@ export const Webchat = forwardRef((props, ref) => { setError(undefined) } } - }, [webchatState.online]) + }, [webchatState.isWebchatOnline]) useTyping({ webchatState, updateTyping, updateMessage, host }) @@ -555,27 +572,28 @@ export const Webchat = forwardRef((props, ref) => { https://stackoverflow.com/questions/37949981/call-child-method-from-parent */ - const updateSessionWithUser = userToUpdate => - updateSession(merge(webchatState.session, { user: userToUpdate })) + const mergeAndUpdateUser = userToUpdate => + updateUser(merge(webchatState.user, userToUpdate)) useImperativeHandle(ref, () => ({ addBotResponse: ({ response, session, lastRoutePath }) => { updateTyping(false) if (Array.isArray(response)) response.map(r => addMessageComponent(r)) else if (response) addMessageComponent(response) - if (session) { - updateSession(merge(session, { user: webchatState.session.user })) - const action = session._botonic_action || '' - const handoff = action.startsWith('create_case') - if (handoff && isDev) addMessageComponent() - updateHandoff(handoff) - } - if (lastRoutePath) updateLastRoutePath(lastRoutePath) updateLastMessageDate(currentDateString()) }, setTyping: typing => updateTyping(typing), addUserMessage: message => sendInput(message), - updateUser: updateSessionWithUser, + updateUser: mergeAndUpdateUser, + updateBotState: botState => { + // TODO: Review handoff logic for 1.0 + // const action = botState.botonicAction || '' + // // const isHandoff = action.startsWith('create_case') + if (botState.isHandoff && isDev) addMessageComponent() + updateBotState(botState) + // updateHandoff(botState.isHandoff) + }, + updateSession: updateSession, openWebchat: () => toggleWebchat(true), closeWebchat: () => toggleWebchat(false), toggleWebchat: () => toggleWebchat(!webchatState.isWebchatOpen), @@ -608,22 +626,26 @@ export const Webchat = forwardRef((props, ref) => { })) const resolveCase = () => { - updateHandoff(false) - updateSession({ ...webchatState.session, _botonic_action: null }) + // updateHandoff(false) + updateBotState({ + ...webchatState.botState, + isHandoff: false, + botonicAction: null, + }) } - const prevSession = usePrevious(webchatState.session) + const previousBotState = usePrevious(webchatState.botState) useEffect(() => { // Resume conversation after handoff if ( - prevSession && - prevSession._botonic_action && - !webchatState.session._botonic_action + previousBotState && + previousBotState.botonicAction && + !webchatState.botState.botonicAction ) { - const action = getParsedAction(prevSession._botonic_action) + const action = getParsedAction(previousBotState.botonicAction) if (action && action.on_finish) sendPayload(action.on_finish) } - }, [webchatState.session._botonic_action]) + }, [webchatState.botState.botonicAction]) const sendText = async (text, payload) => { if (!text) return @@ -663,8 +685,8 @@ export const Webchat = forwardRef((props, ref) => { } const webviewRequestContext = { - getString: stringId => props.getString(stringId, webchatState.session), - setLocale: locale => props.getString(locale, webchatState.session), + getString: stringId => props.getString(stringId, webchatState.botState), + setLocale: locale => props.getString(locale, webchatState.botState), session: webchatState.session || {}, params: webchatState.webviewParams || {}, closeWebview: closeWebview, @@ -924,7 +946,7 @@ export const Webchat = forwardRef((props, ref) => { updateMessage, updateReplies, updateLatestInput, - updateUser: updateSessionWithUser, + updateUser: mergeAndUpdateUser, updateWebchatDevSettings: updateWebchatDevSettings, }} > diff --git a/packages/botonic-react/src/webchat-app.jsx b/packages/botonic-react/src/webchat-app.jsx index f8c3c3602c..0d952c05d6 100644 --- a/packages/botonic-react/src/webchat-app.jsx +++ b/packages/botonic-react/src/webchat-app.jsx @@ -141,7 +141,7 @@ export class WebchatApp { } onServiceEvent(event) { - if (event.action === 'connectionChange') { + if (event.action === 'connection_change') { this.onConnectionChange && this.onConnectionChange(this, event.online) this.webchatRef.current.setOnline(event.online) } else if (event.action === 'update_message_info') diff --git a/packages/create-botonic-app/dev-template/api/src/websocket/onauth.js b/packages/create-botonic-app/dev-template/api/src/websocket/onauth.js index b55c90f3f0..84784ab8c9 100644 --- a/packages/create-botonic-app/dev-template/api/src/websocket/onauth.js +++ b/packages/create-botonic-app/dev-template/api/src/websocket/onauth.js @@ -8,7 +8,6 @@ const initialBotState = { lastRoutePath: null, isFirstInteraction: true, retries: 0, - locale: 'en', isHandoff: false, isShadowing: false, }