2024-11-20 12:57:27 -08:00
import get from 'lodash/get' ;
2023-03-09 09:13:15 -08:00
import type {
IDataObject ,
IExecuteFunctions ,
ILoadOptionsFunctions ,
IOAuth2Options ,
2024-02-14 07:29:09 -08:00
IHttpRequestMethods ,
IRequestOptions ,
2024-05-15 05:54:32 -07:00
IWebhookFunctions ,
2023-03-09 09:13:15 -08:00
} from 'n8n-workflow' ;
2024-06-18 03:50:44 -07:00
import { NodeOperationError } from 'n8n-workflow' ;
2020-07-25 10:58:38 -07:00
2024-11-20 12:57:27 -08:00
import type { SendAndWaitMessageBody } from './MessageInterface' ;
2024-10-07 06:45:22 -07:00
import { getSendAndWaitConfig } from '../../../utils/sendAndWait/utils' ;
2020-03-05 15:25:18 -08:00
2022-08-17 08:50:24 -07:00
export async function slackApiRequest (
2024-05-15 05:54:32 -07:00
this : IExecuteFunctions | ILoadOptionsFunctions | IWebhookFunctions ,
2024-02-14 07:29:09 -08:00
method : IHttpRequestMethods ,
2022-08-17 08:50:24 -07:00
resource : string ,
body : object = { } ,
2024-02-14 07:29:09 -08:00
query : IDataObject = { } ,
2023-02-03 08:04:37 -08:00
headers : { } | undefined = undefined ,
option : { } = { } ,
// tslint:disable-next-line:no-any
2022-08-17 08:50:24 -07:00
) : Promise < any > {
2020-03-05 15:25:18 -08:00
const authenticationMethod = this . getNodeParameter ( 'authentication' , 0 , 'accessToken' ) as string ;
2024-02-14 07:29:09 -08:00
let options : IRequestOptions = {
2020-03-05 15:25:18 -08:00
method ,
2023-02-03 08:04:37 -08:00
headers : headers ? ? {
2020-10-22 06:46:03 -07:00
'Content-Type' : 'application/json; charset=utf-8' ,
2020-03-05 15:25:18 -08:00
} ,
body ,
qs : query ,
2024-05-08 02:40:24 -07:00
uri : resource.startsWith ( 'https' ) ? resource : ` https://slack.com/api ${ resource } ` ,
2020-10-22 06:46:03 -07:00
json : true ,
2020-03-05 15:25:18 -08:00
} ;
2020-03-08 15:22:33 -07:00
options = Object . assign ( { } , options , option ) ;
2020-03-05 15:25:18 -08:00
if ( Object . keys ( body ) . length === 0 ) {
delete options . body ;
}
if ( Object . keys ( query ) . length === 0 ) {
delete options . qs ;
}
2020-10-01 23:50:37 -07:00
2022-04-19 03:36:01 -07:00
const oAuth2Options : IOAuth2Options = {
tokenType : 'Bearer' ,
property : 'authed_user.access_token' ,
} ;
2020-07-25 10:58:38 -07:00
2023-02-03 08:04:37 -08:00
const credentialType = authenticationMethod === 'accessToken' ? 'slackApi' : 'slackOAuth2Api' ;
const response = await this . helpers . requestWithAuthentication . call (
this ,
credentialType ,
options ,
{
oauth2 : oAuth2Options ,
} ,
) ;
2021-11-03 17:55:04 -07:00
2023-02-03 08:04:37 -08:00
if ( response . ok === false ) {
if ( response . error === 'paid_teams_only' ) {
2022-08-17 08:50:24 -07:00
throw new NodeOperationError (
this . getNode ( ) ,
2023-02-03 08:04:37 -08:00
` Your current Slack plan does not include the resource ' ${
this . getNodeParameter ( 'resource' , 0 ) as string
} ' ` ,
{
description :
2023-02-16 02:45:03 -08:00
'Hint: Upgrade to a Slack plan that includes the functionality you want to use.' ,
2023-12-07 07:57:02 -08:00
level : 'warning' ,
2023-02-03 08:04:37 -08:00
} ,
) ;
} else if ( response . error === 'missing_scope' ) {
throw new NodeOperationError (
this . getNode ( ) ,
'Your Slack credential is missing required Oauth Scopes' ,
{
description : ` Add the following scope(s) to your Slack App: ${ response . needed } ` ,
2023-12-07 07:57:02 -08:00
level : 'warning' ,
2023-02-03 08:04:37 -08:00
} ,
2022-08-17 08:50:24 -07:00
) ;
2024-10-08 05:35:37 -07:00
} else if ( response . error === 'not_admin' ) {
throw new NodeOperationError (
this . getNode ( ) ,
'Need higher Role Level for this Operation (e.g. Owner or Admin Rights)' ,
{
description :
'Hint: Check the Role of your Slack App Integration. For more information see the Slack Documentation - https://slack.com/help/articles/360018112273-Types-of-roles-in-Slack' ,
level : 'warning' ,
} ,
) ;
2020-10-01 23:50:37 -07:00
}
2024-05-08 02:40:24 -07:00
2023-02-03 08:04:37 -08:00
throw new NodeOperationError (
this . getNode ( ) ,
'Slack error response: ' + JSON . stringify ( response . error ) ,
) ;
}
2024-10-08 05:35:37 -07:00
2023-02-03 08:04:37 -08:00
if ( response . ts !== undefined ) {
Object . assign ( response , { message_timestamp : response.ts } ) ;
delete response . ts ;
2020-03-05 15:25:18 -08:00
}
2024-05-15 05:54:32 -07:00
2023-02-03 08:04:37 -08:00
return response ;
2020-03-05 15:25:18 -08:00
}
2022-08-17 08:50:24 -07:00
export async function slackApiRequestAllItems (
this : IExecuteFunctions | ILoadOptionsFunctions ,
propertyName : string ,
2024-02-14 07:29:09 -08:00
method : IHttpRequestMethods ,
2022-08-17 08:50:24 -07:00
endpoint : string ,
2023-02-03 08:04:37 -08:00
// tslint:disable-next-line:no-any
2022-08-17 08:50:24 -07:00
body : any = { } ,
query : IDataObject = { } ,
2023-02-03 08:04:37 -08:00
// tslint:disable-next-line:no-any
2022-08-17 08:50:24 -07:00
) : Promise < any > {
2020-03-05 15:25:18 -08:00
const returnData : IDataObject [ ] = [ ] ;
let responseData ;
2020-03-08 15:22:33 -07:00
query . page = 1 ;
2021-10-18 21:04:58 -07:00
//if the endpoint uses legacy pagination use count
//https://api.slack.com/docs/pagination#classic
if ( endpoint . includes ( 'files.list' ) ) {
query . count = 100 ;
} else {
2021-10-19 20:52:53 -07:00
query . limit = 100 ;
2021-10-18 21:04:58 -07:00
}
2020-03-05 15:25:18 -08:00
do {
2023-02-27 19:39:43 -08:00
responseData = await slackApiRequest . call ( this , method , endpoint , body as IDataObject , query ) ;
2023-02-23 07:16:05 -08:00
query . cursor = get ( responseData , 'response_metadata.next_cursor' ) ;
2020-03-08 15:22:33 -07:00
query . page ++ ;
2023-02-03 08:04:37 -08:00
returnData . push . apply (
returnData ,
2023-02-27 19:39:43 -08:00
( responseData [ propertyName ] . matches as IDataObject [ ] ) ? ? responseData [ propertyName ] ,
2023-02-03 08:04:37 -08:00
) ;
2020-03-05 15:25:18 -08:00
} while (
2022-12-02 12:54:28 -08:00
( responseData . response_metadata ? . next_cursor !== undefined &&
2020-12-13 01:47:52 -08:00
responseData . response_metadata . next_cursor !== '' &&
responseData . response_metadata . next_cursor !== null ) ||
2022-12-02 12:54:28 -08:00
( responseData . paging ? . pages !== undefined &&
2020-12-13 01:47:52 -08:00
responseData . paging . page !== undefined &&
2023-02-03 08:04:37 -08:00
responseData . paging . page < responseData . paging . pages ) ||
( responseData [ propertyName ] . paging ? . pages !== undefined &&
responseData [ propertyName ] . paging . page !== undefined &&
responseData [ propertyName ] . paging . page < responseData [ propertyName ] . paging . pages )
2020-03-05 15:25:18 -08:00
) ;
return returnData ;
}
2020-05-05 10:22:02 -07:00
2023-10-03 01:18:59 -07:00
export function getMessageContent (
this : IExecuteFunctions | ILoadOptionsFunctions ,
i : number ,
nodeVersion : number ,
instanceId? : string ,
) {
2023-07-10 06:03:21 -07:00
const includeLinkToWorkflow = this . getNodeParameter (
'otherOptions.includeLinkToWorkflow' ,
i ,
nodeVersion >= 2.1 ? true : false ,
) as IDataObject ;
const { id } = this . getWorkflow ( ) ;
2023-10-03 01:18:59 -07:00
const automatedMessage = ` _Automated with this < ${ this . getInstanceBaseUrl ( ) } workflow/ ${ id } ?utm_source=n8n-internal&utm_medium=powered_by&utm_campaign= ${ encodeURIComponent (
'n8n-nodes-base.slack' ,
) } $ { instanceId ? '_' + instanceId : '' } | n8n workflow > _ ` ;
2023-07-10 06:03:21 -07:00
const messageType = this . getNodeParameter ( 'messageType' , i ) as string ;
let content : IDataObject = { } ;
const text = this . getNodeParameter ( 'text' , i , '' ) as string ;
switch ( messageType ) {
case 'text' :
content = {
text : includeLinkToWorkflow ? ` ${ text } \ n ${ automatedMessage } ` : text ,
} ;
break ;
case 'block' :
2024-06-18 03:50:44 -07:00
content = this . getNodeParameter ( 'blocksUi' , i , { } , { ensureType : 'object' } ) as IDataObject ;
2023-07-10 06:03:21 -07:00
if ( includeLinkToWorkflow && Array . isArray ( content . blocks ) ) {
content . blocks . push ( {
type : 'section' ,
text : {
type : 'mrkdwn' ,
text : automatedMessage ,
} ,
} ) ;
}
if ( text ) {
content . text = text ;
}
break ;
case 'attachment' :
2024-02-01 03:26:04 -08:00
const attachmentsUI = this . getNodeParameter ( 'attachments' , i ) as IDataObject [ ] ;
const attachments : IDataObject [ ] = [ ] ;
for ( const attachment of attachmentsUI ) {
if ( attachment . fields !== undefined ) {
if ( ( attachment ? . fields as IDataObject ) ? . item ) {
attachment . fields = ( attachment ? . fields as IDataObject ) ? . item as IDataObject [ ] ;
}
}
attachments . push ( attachment ) ;
}
content = { attachments } as IDataObject ;
2023-07-10 06:03:21 -07:00
if ( includeLinkToWorkflow && Array . isArray ( content . attachments ) ) {
content . attachments . push ( {
text : automatedMessage ,
} ) ;
}
break ;
default :
throw new NodeOperationError (
this . getNode ( ) ,
` The message type " ${ messageType } " is not known! ` ,
) ;
}
return content ;
}
2023-02-03 08:04:37 -08:00
// tslint:disable-next-line:no-any
2022-08-17 08:50:24 -07:00
export function validateJSON ( json : string | undefined ) : any {
2020-05-05 10:22:02 -07:00
let result ;
try {
result = JSON . parse ( json ! ) ;
} catch ( exception ) {
result = undefined ;
}
return result ;
}
2024-10-07 06:45:22 -07:00
export function getTarget (
context : IExecuteFunctions ,
itemIndex : number ,
idType : 'user' | 'channel' ,
) : string {
let target = '' ;
if ( idType === 'channel' ) {
target = context . getNodeParameter ( 'channelId' , itemIndex , undefined , {
extractValue : true ,
} ) as string ;
} else {
target = context . getNodeParameter ( 'user' , itemIndex , undefined , {
extractValue : true ,
} ) as string ;
}
if (
idType === 'user' &&
( context . getNodeParameter ( 'user' , itemIndex ) as IDataObject ) . mode === 'username'
) {
target = target . slice ( 0 , 1 ) === '@' ? target : ` @ ${ target } ` ;
}
return target ;
}
export function createSendAndWaitMessageBody ( context : IExecuteFunctions ) {
const select = context . getNodeParameter ( 'select' , 0 ) as 'user' | 'channel' ;
const target = getTarget ( context , 0 , select ) ;
const config = getSendAndWaitConfig ( context ) ;
2024-11-20 12:57:27 -08:00
const body : SendAndWaitMessageBody = {
2024-10-07 06:45:22 -07:00
channel : target ,
blocks : [
{
type : 'divider' ,
} ,
{
type : 'section' ,
text : {
2024-11-20 12:57:27 -08:00
type : context . getNode ( ) . typeVersion > 2.2 ? 'mrkdwn' : 'plain_text' ,
2024-10-07 06:45:22 -07:00
text : config.message ,
emoji : true ,
} ,
} ,
{
type : 'section' ,
text : {
type : 'plain_text' ,
text : ' ' ,
} ,
} ,
{
type : 'divider' ,
} ,
{
type : 'actions' ,
elements : config.options.map ( ( option ) = > {
return {
type : 'button' ,
style : option.style === 'primary' ? 'primary' : undefined ,
text : {
type : 'plain_text' ,
text : option.label ,
emoji : true ,
} ,
url : ` ${ config . url } ?approved= ${ option . value } ` ,
} ;
} ) ,
} ,
] ,
} ;
2024-11-20 12:57:27 -08:00
if ( context . getNode ( ) . typeVersion > 2.2 && body . blocks ? . [ 1 ] ? . type === 'section' ) {
delete body . blocks [ 1 ] . text . emoji ;
}
2024-10-07 06:45:22 -07:00
return body ;
}