2024-09-30 06:42:37 -07:00
import type { BaseChatMemory } from '@langchain/community/memory/chat_memory' ;
import { pick } from 'lodash' ;
2024-12-12 04:54:44 -08:00
import { Node , NodeConnectionType } from 'n8n-workflow' ;
2024-07-09 04:45:41 -07:00
import type {
IDataObject ,
IWebhookFunctions ,
IWebhookResponseData ,
INodeTypeDescription ,
MultiPartFormData ,
INodeExecutionData ,
IBinaryData ,
INodeProperties ,
2024-01-09 03:11:39 -08:00
} from 'n8n-workflow' ;
2024-07-09 04:45:41 -07:00
2024-01-09 03:11:39 -08:00
import { validateAuth } from './GenericFunctions' ;
2024-09-30 06:42:37 -07:00
import { createPage } from './templates' ;
2024-01-09 03:11:39 -08:00
import type { LoadPreviousSessionChatOption } from './types' ;
const CHAT_TRIGGER_PATH_IDENTIFIER = 'chat' ;
2024-07-09 04:45:41 -07:00
const allowFileUploadsOption : INodeProperties = {
displayName : 'Allow File Uploads' ,
name : 'allowFileUploads' ,
type : 'boolean' ,
default : false ,
description : 'Whether to allow file uploads in the chat' ,
} ;
const allowedFileMimeTypeOption : INodeProperties = {
displayName : 'Allowed File Mime Types' ,
name : 'allowedFilesMimeTypes' ,
type : 'string' ,
default : '*' ,
placeholder : 'e.g. image/*, text/*, application/pdf' ,
description :
'Allowed file types for upload. Comma-separated list of <a href="https://developer.mozilla.org/en-US/docs/Web/HTTP/Basics_of_HTTP/MIME_types/Common_types" target="_blank">MIME types</a>.' ,
} ;
2024-01-09 03:11:39 -08:00
2024-07-09 04:45:41 -07:00
export class ChatTrigger extends Node {
2024-01-09 03:11:39 -08:00
description : INodeTypeDescription = {
displayName : 'Chat Trigger' ,
name : 'chatTrigger' ,
icon : 'fa:comments' ,
2024-06-06 04:34:30 -07:00
iconColor : 'black' ,
2024-01-09 03:11:39 -08:00
group : [ 'trigger' ] ,
2024-07-09 04:45:41 -07:00
version : [ 1 , 1.1 ] ,
2024-01-09 03:11:39 -08:00
description : 'Runs the workflow when an n8n generated webchat is submitted' ,
defaults : {
2024-06-04 08:24:18 -07:00
name : 'When chat message received' ,
2024-01-09 03:11:39 -08:00
} ,
codex : {
categories : [ 'Core Nodes' ] ,
resources : {
primaryDocumentation : [
{
url : 'https://docs.n8n.io/integrations/builtin/core-nodes/n8n-nodes-langchain.chattrigger/' ,
} ,
] ,
} ,
} ,
maxNodes : 1 ,
inputs : ` ={{ (() => {
if ( ! [ 'hostedChat' , 'webhook' ] . includes ( $parameter . mode ) ) {
return [ ] ;
}
if ( $parameter . options ? . loadPreviousSession !== 'memory' ) {
return [ ] ;
}
return [
{
displayName : 'Memory' ,
maxConnections : 1 ,
type : '${NodeConnectionType.AiMemory}' ,
required : true ,
}
] ;
} ) ( ) } } ` ,
2024-08-29 06:55:53 -07:00
outputs : [ NodeConnectionType . Main ] ,
2024-01-09 03:11:39 -08:00
credentials : [
{
// eslint-disable-next-line n8n-nodes-base/node-class-description-credentials-name-unsuffixed
name : 'httpBasicAuth' ,
required : true ,
displayOptions : {
show : {
authentication : [ 'basicAuth' ] ,
} ,
} ,
} ,
] ,
webhooks : [
{
name : 'setup' ,
httpMethod : 'GET' ,
responseMode : 'onReceived' ,
path : CHAT_TRIGGER_PATH_IDENTIFIER ,
ndvHideUrl : true ,
} ,
{
name : 'default' ,
httpMethod : 'POST' ,
responseMode : '={{$parameter.options?.["responseMode"] || "lastNode" }}' ,
path : CHAT_TRIGGER_PATH_IDENTIFIER ,
ndvHideMethod : true ,
ndvHideUrl : '={{ !$parameter.public }}' ,
} ,
] ,
eventTriggerDescription : 'Waiting for you to submit the chat' ,
activationMessage : 'You can now make calls to your production chat URL.' ,
triggerPanel : false ,
properties : [
/ * *
* @note If we change this property , also update it in ChatEmbedModal . vue
* /
{
displayName : 'Make Chat Publicly Available' ,
name : 'public' ,
type : 'boolean' ,
default : false ,
description :
'Whether the chat should be publicly available or only accessible through the manual chat interface' ,
} ,
{
displayName : 'Mode' ,
name : 'mode' ,
type : 'options' ,
options : [
{
name : 'Hosted Chat' ,
value : 'hostedChat' ,
description : 'Chat on a page served by n8n' ,
} ,
{
name : 'Embedded Chat' ,
value : 'webhook' ,
description : 'Chat through a widget embedded in another page, or by calling a webhook' ,
} ,
] ,
default : 'hostedChat' ,
displayOptions : {
show : {
public : [ true ] ,
} ,
} ,
} ,
{
displayName :
'Chat will be live at the URL above once you activate this workflow. Live executions will show up in the ‘ executions’ tab' ,
name : 'hostedChatNotice' ,
type : 'notice' ,
displayOptions : {
show : {
mode : [ 'hostedChat' ] ,
public : [ true ] ,
} ,
} ,
default : '' ,
} ,
{
displayName :
'Follow the instructions <a href="https://www.npmjs.com/package/@n8n/chat" target="_blank">here</a> to embed chat in a webpage (or just call the webhook URL at the top of this section). Chat will be live once you activate this workflow' ,
name : 'embeddedChatNotice' ,
type : 'notice' ,
displayOptions : {
show : {
mode : [ 'webhook' ] ,
public : [ true ] ,
} ,
} ,
default : '' ,
} ,
{
displayName : 'Authentication' ,
name : 'authentication' ,
type : 'options' ,
displayOptions : {
show : {
public : [ true ] ,
} ,
} ,
options : [
{
name : 'Basic Auth' ,
value : 'basicAuth' ,
description : 'Simple username and password (the same one for all users)' ,
} ,
{
// eslint-disable-next-line n8n-nodes-base/node-param-display-name-miscased
name : 'n8n User Auth' ,
value : 'n8nUserAuth' ,
description : 'Require user to be logged in with their n8n account' ,
} ,
{
name : 'None' ,
value : 'none' ,
} ,
] ,
default : 'none' ,
description : 'The way to authenticate' ,
} ,
{
displayName : 'Initial Message(s)' ,
name : 'initialMessages' ,
type : 'string' ,
displayOptions : {
show : {
mode : [ 'hostedChat' ] ,
public : [ true ] ,
} ,
} ,
typeOptions : {
rows : 3 ,
} ,
default : 'Hi there! 👋\nMy name is Nathan. How can I assist you today?' ,
description : 'Default messages shown at the start of the chat, one per line' ,
} ,
2024-07-09 04:45:41 -07:00
{
displayName : 'Options' ,
name : 'options' ,
type : 'collection' ,
displayOptions : {
show : {
public : [ false ] ,
'@version' : [ { _cnd : { gte : 1.1 } } ] ,
} ,
} ,
placeholder : 'Add Field' ,
default : { } ,
options : [ allowFileUploadsOption , allowedFileMimeTypeOption ] ,
} ,
2024-01-09 03:11:39 -08:00
{
displayName : 'Options' ,
name : 'options' ,
type : 'collection' ,
displayOptions : {
show : {
mode : [ 'hostedChat' , 'webhook' ] ,
public : [ true ] ,
} ,
} ,
placeholder : 'Add Field' ,
default : { } ,
options : [
2024-09-30 06:42:37 -07:00
// CORS parameters are only valid for when chat is used in hosted or webhook mode
2024-12-12 04:54:44 -08:00
{
displayName : 'Allowed Origins (CORS)' ,
name : 'allowedOrigins' ,
type : 'string' ,
default : '*' ,
description :
'Comma-separated list of URLs allowed for cross-origin non-preflight requests. Use * (default) to allow all origins.' ,
2024-09-30 06:42:37 -07:00
displayOptions : {
show : {
'/mode' : [ 'hostedChat' , 'webhook' ] ,
} ,
} ,
2024-12-12 04:54:44 -08:00
} ,
2024-07-09 04:45:41 -07:00
{
. . . allowFileUploadsOption ,
displayOptions : {
show : {
'/mode' : [ 'hostedChat' ] ,
} ,
} ,
} ,
{
. . . allowedFileMimeTypeOption ,
displayOptions : {
show : {
'/mode' : [ 'hostedChat' ] ,
} ,
} ,
} ,
2024-01-09 03:11:39 -08:00
{
displayName : 'Input Placeholder' ,
name : 'inputPlaceholder' ,
type : 'string' ,
displayOptions : {
show : {
'/mode' : [ 'hostedChat' ] ,
} ,
} ,
default : 'Type your question..' ,
placeholder : 'e.g. Type your message here' ,
description : 'Shown as placeholder text in the chat input field' ,
} ,
{
displayName : 'Load Previous Session' ,
name : 'loadPreviousSession' ,
type : 'options' ,
options : [
{
name : 'Off' ,
value : 'notSupported' ,
description : 'Loading messages of previous session is turned off' ,
} ,
{
name : 'From Memory' ,
value : 'memory' ,
description : 'Load session messages from memory' ,
} ,
{
name : 'Manually' ,
value : 'manually' ,
description : 'Manually return messages of session' ,
} ,
] ,
default : 'notSupported' ,
description : 'If loading messages of a previous session should be enabled' ,
} ,
{
displayName : 'Response Mode' ,
name : 'responseMode' ,
type : 'options' ,
options : [
{
name : 'When Last Node Finishes' ,
value : 'lastNode' ,
description : 'Returns data of the last-executed node' ,
} ,
{
name : "Using 'Respond to Webhook' Node" ,
value : 'responseNode' ,
description : 'Response defined in that node' ,
} ,
] ,
default : 'lastNode' ,
description : 'When and how to respond to the webhook' ,
} ,
{
displayName : 'Require Button Click to Start Chat' ,
name : 'showWelcomeScreen' ,
type : 'boolean' ,
displayOptions : {
show : {
'/mode' : [ 'hostedChat' ] ,
} ,
} ,
default : false ,
description : 'Whether to show the welcome screen at the start of the chat' ,
} ,
{
displayName : 'Start Conversation Button Text' ,
name : 'getStarted' ,
type : 'string' ,
displayOptions : {
show : {
showWelcomeScreen : [ true ] ,
'/mode' : [ 'hostedChat' ] ,
} ,
} ,
default : 'New Conversation' ,
placeholder : 'e.g. New Conversation' ,
description : 'Shown as part of the welcome screen, in the middle of the chat window' ,
} ,
{
displayName : 'Subtitle' ,
name : 'subtitle' ,
type : 'string' ,
displayOptions : {
show : {
'/mode' : [ 'hostedChat' ] ,
} ,
} ,
default : "Start a chat. We're here to help you 24/7." ,
placeholder : "e.g. We're here for you" ,
description : 'Shown at the top of the chat, under the title' ,
} ,
{
displayName : 'Title' ,
name : 'title' ,
type : 'string' ,
displayOptions : {
show : {
'/mode' : [ 'hostedChat' ] ,
} ,
} ,
default : 'Hi there! 👋' ,
placeholder : 'e.g. Welcome' ,
description : 'Shown at the top of the chat' ,
} ,
] ,
} ,
] ,
} ;
2024-07-09 04:45:41 -07:00
private async handleFormData ( context : IWebhookFunctions ) {
const req = context . getRequestObject ( ) as MultiPartFormData . Request ;
const options = context . getNodeParameter ( 'options' , { } ) as IDataObject ;
const { data , files } = req . body ;
const returnItem : INodeExecutionData = {
json : data ,
} ;
if ( files && Object . keys ( files ) . length ) {
returnItem . json . files = [ ] as Array < Omit < IBinaryData , ' data ' > > ;
returnItem . binary = { } ;
const count = 0 ;
for ( const fileKey of Object . keys ( files ) ) {
const processedFiles : MultiPartFormData.File [ ] = [ ] ;
if ( Array . isArray ( files [ fileKey ] ) ) {
processedFiles . push ( . . . files [ fileKey ] ) ;
} else {
processedFiles . push ( files [ fileKey ] ) ;
}
let fileIndex = 0 ;
for ( const file of processedFiles ) {
let binaryPropertyName = 'data' ;
2024-01-09 03:11:39 -08:00
2024-07-09 04:45:41 -07:00
// Remove the '[]' suffix from the binaryPropertyName if it exists
if ( binaryPropertyName . endsWith ( '[]' ) ) {
binaryPropertyName = binaryPropertyName . slice ( 0 , - 2 ) ;
}
if ( options . binaryPropertyName ) {
binaryPropertyName = ` ${ options . binaryPropertyName . toString ( ) } ${ count } ` ;
}
const binaryFile = await context . nodeHelpers . copyBinaryFile (
file . filepath ,
file . originalFilename ? ? file . newFilename ,
file . mimetype ,
) ;
const binaryKey = ` ${ binaryPropertyName } ${ fileIndex } ` ;
const binaryInfo = {
. . . pick ( binaryFile , [ 'fileName' , 'fileSize' , 'fileType' , 'mimeType' , 'fileExtension' ] ) ,
binaryKey ,
} ;
returnItem . binary = Object . assign ( returnItem . binary ? ? { } , {
[ ` ${ binaryKey } ` ] : binaryFile ,
} ) ;
returnItem . json . files = [
. . . ( returnItem . json . files as Array < Omit < IBinaryData , ' data ' > > ) ,
binaryInfo ,
] ;
fileIndex += 1 ;
}
}
}
return returnItem ;
}
async webhook ( ctx : IWebhookFunctions ) : Promise < IWebhookResponseData > {
const res = ctx . getResponseObject ( ) ;
const isPublic = ctx . getNodeParameter ( 'public' , false ) as boolean ;
const nodeMode = ctx . getNodeParameter ( 'mode' , 'hostedChat' ) as string ;
2024-01-09 03:11:39 -08:00
if ( ! isPublic ) {
res . status ( 404 ) . end ( ) ;
return {
noWebhookResponse : true ,
} ;
}
2024-07-09 04:45:41 -07:00
const options = ctx . getNodeParameter ( 'options' , { } ) as {
2024-01-09 03:11:39 -08:00
getStarted? : string ;
inputPlaceholder? : string ;
loadPreviousSession? : LoadPreviousSessionChatOption ;
showWelcomeScreen? : boolean ;
subtitle? : string ;
title? : string ;
2024-07-09 04:45:41 -07:00
allowFileUploads? : boolean ;
allowedFilesMimeTypes? : string ;
2024-01-09 03:11:39 -08:00
} ;
2024-07-09 04:45:41 -07:00
const req = ctx . getRequestObject ( ) ;
const webhookName = ctx . getWebhookName ( ) ;
const mode = ctx . getMode ( ) === 'manual' ? 'test' : 'production' ;
const bodyData = ctx . getBodyData ( ) ? ? { } ;
2024-09-09 08:33:52 -07:00
try {
await validateAuth ( ctx ) ;
} catch ( error ) {
if ( error ) {
res . writeHead ( ( error as IDataObject ) . responseCode as number , {
'www-authenticate' : 'Basic realm="Webhook"' ,
} ) ;
res . end ( ( error as IDataObject ) . message as string ) ;
return { noWebhookResponse : true } ;
2024-01-09 03:11:39 -08:00
}
2024-09-09 08:33:52 -07:00
throw error ;
}
if ( nodeMode === 'hostedChat' ) {
2024-01-09 03:11:39 -08:00
// Show the chat on GET request
if ( webhookName === 'setup' ) {
2024-07-09 04:45:41 -07:00
const webhookUrlRaw = ctx . getNodeWebhookUrl ( 'default' ) as string ;
2024-01-09 03:11:39 -08:00
const webhookUrl =
mode === 'test' ? webhookUrlRaw . replace ( '/webhook' , '/webhook-test' ) : webhookUrlRaw ;
2024-07-09 04:45:41 -07:00
const authentication = ctx . getNodeParameter ( 'authentication' ) as
2024-01-09 03:11:39 -08:00
| 'none'
| 'basicAuth'
| 'n8nUserAuth' ;
2024-07-09 04:45:41 -07:00
const initialMessagesRaw = ctx . getNodeParameter ( 'initialMessages' , '' ) as string ;
2024-01-09 03:11:39 -08:00
const initialMessages = initialMessagesRaw
. split ( '\n' )
. filter ( ( line ) = > line )
. map ( ( line ) = > line . trim ( ) ) ;
2024-07-09 04:45:41 -07:00
const instanceId = ctx . getInstanceId ( ) ;
2024-01-09 03:11:39 -08:00
const i18nConfig = pick ( options , [ 'getStarted' , 'inputPlaceholder' , 'subtitle' , 'title' ] ) ;
const page = createPage ( {
i18n : {
en : i18nConfig ,
} ,
showWelcomeScreen : options.showWelcomeScreen ,
loadPreviousSession : options.loadPreviousSession ,
initialMessages ,
webhookUrl ,
mode ,
instanceId ,
authentication ,
2024-07-09 04:45:41 -07:00
allowFileUploads : options.allowFileUploads ,
allowedFilesMimeTypes : options.allowedFilesMimeTypes ,
2024-01-09 03:11:39 -08:00
} ) ;
res . status ( 200 ) . send ( page ) . end ( ) ;
return {
noWebhookResponse : true ,
} ;
}
}
if ( bodyData . action === 'loadPreviousSession' ) {
if ( options ? . loadPreviousSession === 'memory' ) {
2024-07-09 04:45:41 -07:00
const memory = ( await ctx . getInputConnectionData ( NodeConnectionType . AiMemory , 0 ) ) as
2024-01-09 03:11:39 -08:00
| BaseChatMemory
| undefined ;
2024-01-15 00:13:54 -08:00
const messages = ( ( await memory ? . chatHistory . getMessages ( ) ) ? ? [ ] )
. filter ( ( message ) = > ! message ? . additional_kwargs ? . hideFromUI )
. map ( ( message ) = > message ? . toJSON ( ) ) ;
2024-01-09 03:11:39 -08:00
return {
webhookResponse : { data : messages } ,
} ;
} else if ( options ? . loadPreviousSession === 'notSupported' ) {
// If messages of a previous session should not be loaded, simply return an empty array
return {
webhookResponse : { data : [ ] } ,
} ;
}
}
2024-07-09 04:45:41 -07:00
let returnData : INodeExecutionData [ ] ;
2024-01-09 03:11:39 -08:00
const webhookResponse : IDataObject = { status : 200 } ;
2024-07-09 04:45:41 -07:00
if ( req . contentType === 'multipart/form-data' ) {
returnData = [ await this . handleFormData ( ctx ) ] ;
return {
webhookResponse ,
workflowData : [ returnData ] ,
} ;
} else {
returnData = [ { json : bodyData } ] ;
}
2024-01-09 03:11:39 -08:00
return {
webhookResponse ,
2024-07-09 04:45:41 -07:00
workflowData : [ ctx . helpers . returnJsonArray ( returnData ) ] ,
2024-01-09 03:11:39 -08:00
} ;
}
}