mirror of
https://github.com/n8n-io/n8n.git
synced 2025-01-26 12:01:16 -08:00
ce066a160f
* introduce analytics * add user survey backend * add user survey backend * set answers on survey submit Co-authored-by: Mutasem Aldmour <4711238+mutdmour@users.noreply.github.com> * change name to personalization * lint Co-authored-by: Mutasem Aldmour <4711238+mutdmour@users.noreply.github.com> * N8n 2495 add personalization modal (#2280) * update modals * add onboarding modal * implement questions * introduce analytics * simplify impl * implement survey handling * add personalized cateogry * update modal behavior * add thank you view * handle empty cases * rename modal * standarize modal names * update image, add tags to headings * remove unused file * remove unused interfaces * clean up footer spacing * introduce analytics * refactor to fix bug * update endpoint * set min height * update stories * update naming from questions to survey * remove spacing after core categories * fix bug in logic * sort nodes * rename types * merge with be * rename userSurvey * clean up rest api * use constants for keys * use survey keys * clean up types * move personalization to its own file Co-authored-by: ahsan-virani <ahsan.virani@gmail.com> * update parameter inputs to be multiline * update spacing * Survey new options (#2300) * split up options * fix quotes * remove unused import * refactor node credentials * add user created workflow event (#2301) * update multi params * simplify env vars * fix versionCli on FE * update personalization env * clean up node detail settings * fix event User opened Credentials panel * fix font sizes across modals * clean up input spacing * fix select modal spacing * increase spacing * fix input copy * fix webhook, tab spacing, retry button * fix button sizes * fix button size * add mini xlarge sizes * fix webhook spacing * fix nodes panel event * fix workflow id in workflow execute event * improve telemetry error logging * fix config and stop process events * add flush call on n8n stop * ready for release * fix input error highlighting * revert change * update toggle spacing * fix delete positioning * keep tooltip while focused * set strict size * increase left spacing * fix sort icons * remove unnessasry <br/> * remove unnessary break * remove unnessary margin * clean unused functionality * remove unnessary css * remove duplicate tracking * only show tooltip when hovering over label * remove extra space * add br * remove extra space * clean up commas * clean up commas * remove extra space * remove extra space * rewrite desc * add commas * add space * remove extra space * add space * add dot * update credentials section * use includes Co-authored-by: ahsan-virani <ahsan.virani@gmail.com> Co-authored-by: Jan Oberhauser <jan.oberhauser@gmail.com>
794 lines
17 KiB
TypeScript
794 lines
17 KiB
TypeScript
import {
|
|
IExecuteFunctions,
|
|
} from 'n8n-core';
|
|
import {
|
|
IDataObject,
|
|
INodeExecutionData,
|
|
INodeType,
|
|
INodeTypeDescription,
|
|
NodeOperationError,
|
|
} from 'n8n-workflow';
|
|
|
|
import { disqusApiRequest, disqusApiRequestAllItems } from './GenericFunctions';
|
|
|
|
|
|
export class Disqus implements INodeType {
|
|
description: INodeTypeDescription = {
|
|
displayName: 'Disqus',
|
|
name: 'disqus',
|
|
icon: 'file:disqus.png',
|
|
group: ['input'],
|
|
version: 1,
|
|
subtitle: '={{$parameter["operation"] + ": " + $parameter["resource"]}}',
|
|
description: 'Access data on Disqus',
|
|
defaults: {
|
|
name: 'Disqus',
|
|
color: '#22BB44',
|
|
},
|
|
inputs: ['main'],
|
|
outputs: ['main'],
|
|
credentials: [
|
|
{
|
|
name: 'disqusApi',
|
|
required: true,
|
|
},
|
|
],
|
|
properties: [
|
|
{
|
|
displayName: 'Resource',
|
|
name: 'resource',
|
|
type: 'options',
|
|
options: [
|
|
{
|
|
name: 'Forum',
|
|
value: 'forum',
|
|
},
|
|
],
|
|
default: 'forum',
|
|
description: 'The resource to operate on.',
|
|
},
|
|
|
|
// ----------------------------------
|
|
// forum
|
|
// ----------------------------------
|
|
{
|
|
displayName: 'Operation',
|
|
name: 'operation',
|
|
type: 'options',
|
|
displayOptions: {
|
|
show: {
|
|
resource: [
|
|
'forum',
|
|
],
|
|
},
|
|
},
|
|
options: [
|
|
{
|
|
name: 'Get',
|
|
value: 'get',
|
|
description: 'Return forum details',
|
|
},
|
|
{
|
|
name: 'Get All Categories',
|
|
value: 'getCategories',
|
|
description: 'Return a list of categories within a forum',
|
|
},
|
|
{
|
|
name: 'Get All Threads',
|
|
value: 'getThreads',
|
|
description: 'Return a list of threads within a forum',
|
|
},
|
|
{
|
|
name: 'Get All Posts',
|
|
value: 'getPosts',
|
|
description: 'Return a list of posts within a forum',
|
|
},
|
|
],
|
|
default: 'get',
|
|
description: 'The operation to perform.',
|
|
},
|
|
|
|
// ----------------------------------
|
|
// forum:get
|
|
// ----------------------------------
|
|
{
|
|
displayName: 'Forum name',
|
|
name: 'id',
|
|
type: 'string',
|
|
default: '',
|
|
required: true,
|
|
displayOptions: {
|
|
show: {
|
|
operation: [
|
|
'get',
|
|
],
|
|
resource: [
|
|
'forum',
|
|
],
|
|
},
|
|
},
|
|
description: 'The short name(aka ID) of the forum to get.',
|
|
},
|
|
{
|
|
displayName: 'Additional Fields',
|
|
name: 'additionalFields',
|
|
type: 'collection',
|
|
placeholder: 'Add Field',
|
|
displayOptions: {
|
|
show: {
|
|
operation: [
|
|
'get',
|
|
],
|
|
resource: [
|
|
'forum',
|
|
],
|
|
},
|
|
},
|
|
default: {},
|
|
options: [
|
|
{
|
|
displayName: 'Attach',
|
|
name: 'attach',
|
|
type: 'multiOptions',
|
|
options: [
|
|
{
|
|
name: 'counters',
|
|
value: 'counters',
|
|
},
|
|
{
|
|
name: 'followsForum',
|
|
value: 'followsForum',
|
|
},
|
|
{
|
|
name: 'forumCanDisableAds',
|
|
value: 'forumCanDisableAds',
|
|
},
|
|
{
|
|
name: 'forumDaysAlive',
|
|
value: 'forumDaysAlive',
|
|
},
|
|
{
|
|
name: 'forumFeatures',
|
|
value: 'forumFeatures',
|
|
},
|
|
{
|
|
name: 'forumForumCategory',
|
|
value: 'forumForumCategory',
|
|
},
|
|
{
|
|
name: 'forumIntegration',
|
|
value: 'forumIntegration',
|
|
},
|
|
{
|
|
name: 'forumNewPolicy',
|
|
value: 'forumNewPolicy',
|
|
},
|
|
{
|
|
name: 'forumPermissions',
|
|
value: 'forumPermissions',
|
|
},
|
|
],
|
|
default: [],
|
|
description: 'The resource to operate on.',
|
|
},
|
|
{
|
|
displayName: 'Related',
|
|
name: 'related',
|
|
type: 'multiOptions',
|
|
options: [
|
|
{
|
|
name: 'author',
|
|
value: 'author',
|
|
},
|
|
],
|
|
default: [],
|
|
description: 'You may specify relations to include with your response',
|
|
},
|
|
],
|
|
},
|
|
|
|
// ----------------------------------
|
|
// forum:getPosts
|
|
// ----------------------------------
|
|
{
|
|
displayName: 'Forum name',
|
|
name: 'id',
|
|
type: 'string',
|
|
default: '',
|
|
required: true,
|
|
displayOptions: {
|
|
show: {
|
|
operation: [
|
|
'getPosts',
|
|
],
|
|
resource: [
|
|
'forum',
|
|
],
|
|
},
|
|
},
|
|
description: 'The short name(aka ID) of the forum to get.',
|
|
},
|
|
{
|
|
displayName: 'Return All',
|
|
name: 'returnAll',
|
|
type: 'boolean',
|
|
displayOptions: {
|
|
show: {
|
|
resource: [
|
|
'forum',
|
|
],
|
|
operation: [
|
|
'getPosts',
|
|
],
|
|
},
|
|
},
|
|
default: false,
|
|
description: 'If all results should be returned or only up to a given limit.',
|
|
},
|
|
{
|
|
displayName: 'Limit',
|
|
name: 'limit',
|
|
type: 'number',
|
|
displayOptions: {
|
|
show: {
|
|
resource: [
|
|
'forum',
|
|
],
|
|
operation: [
|
|
'getPosts',
|
|
],
|
|
returnAll: [
|
|
false,
|
|
],
|
|
},
|
|
},
|
|
typeOptions: {
|
|
minValue: 1,
|
|
maxValue: 100,
|
|
},
|
|
default: 100,
|
|
description: 'How many results to return.',
|
|
},
|
|
{
|
|
displayName: 'Additional Fields',
|
|
name: 'additionalFields',
|
|
type: 'collection',
|
|
placeholder: 'Add Field',
|
|
displayOptions: {
|
|
show: {
|
|
operation: [
|
|
'getPosts',
|
|
],
|
|
resource: [
|
|
'forum',
|
|
],
|
|
},
|
|
},
|
|
default: {},
|
|
options: [
|
|
{
|
|
displayName: 'Filters',
|
|
name: 'filters',
|
|
type: 'multiOptions',
|
|
options: [
|
|
{
|
|
name: 'Has_Bad_Word',
|
|
value: 'Has_Bad_Word',
|
|
},
|
|
{
|
|
name: 'Has_Link',
|
|
value: 'Has_Link',
|
|
},
|
|
{
|
|
name: 'Has_Low_Rep_Author',
|
|
value: 'Has_Low_Rep_Author',
|
|
},
|
|
{
|
|
name: 'Has_Media',
|
|
value: 'Has_Media',
|
|
},
|
|
{
|
|
name: 'Is_Anonymous',
|
|
value: 'Is_Anonymous',
|
|
},
|
|
{
|
|
name: 'Is_Flagged',
|
|
value: 'Is_Flagged',
|
|
},
|
|
{
|
|
name: 'No_Issue',
|
|
value: 'No_Issue',
|
|
},
|
|
{
|
|
name: 'Is_At_Flag_Limit',
|
|
value: 'Is_At_Flag_Limit',
|
|
},
|
|
{
|
|
name: 'Is_Toxic',
|
|
value: 'Is_Toxic',
|
|
},
|
|
{
|
|
name: 'Modified_By_Rule',
|
|
value: 'Modified_By_Rule',
|
|
},
|
|
{
|
|
name: 'Shadow_Banned',
|
|
value: 'Shadow_Banned',
|
|
},
|
|
],
|
|
default: [],
|
|
description: 'You may specify filters for your response.',
|
|
},
|
|
{
|
|
displayName: 'Include',
|
|
name: 'include',
|
|
type: 'multiOptions',
|
|
options: [
|
|
{
|
|
name: 'approved',
|
|
value: 'approved',
|
|
},
|
|
],
|
|
default: [],
|
|
description: 'You may specify relations to include with your response.',
|
|
},
|
|
{
|
|
displayName: 'Order',
|
|
name: 'order',
|
|
type: 'options',
|
|
options: [
|
|
{
|
|
name: 'ASC',
|
|
value: 'asc',
|
|
},
|
|
{
|
|
name: 'DESC',
|
|
value: 'desc',
|
|
},
|
|
],
|
|
default: 'asc',
|
|
description: 'You may specify order to sort your response.',
|
|
},
|
|
{
|
|
displayName: 'Query',
|
|
name: 'query',
|
|
type: 'string',
|
|
default: '',
|
|
description: 'You may specify query forChoices: asc, desc your response.',
|
|
},
|
|
{
|
|
displayName: 'Related',
|
|
name: 'related',
|
|
type: 'multiOptions',
|
|
options: [
|
|
{
|
|
name: 'thread',
|
|
value: 'thread',
|
|
},
|
|
],
|
|
default: [],
|
|
description: 'You may specify relations to include with your response',
|
|
},
|
|
{
|
|
displayName: 'Since',
|
|
name: 'since',
|
|
type: 'dateTime',
|
|
default: '',
|
|
description: 'Unix timestamp (or ISO datetime standard)',
|
|
},
|
|
],
|
|
},
|
|
|
|
// ----------------------------------
|
|
// forum:getCategories
|
|
// ----------------------------------
|
|
{
|
|
displayName: 'Forum name',
|
|
name: 'id',
|
|
type: 'string',
|
|
default: '',
|
|
required: true,
|
|
displayOptions: {
|
|
show: {
|
|
operation: [
|
|
'getCategories',
|
|
],
|
|
resource: [
|
|
'forum',
|
|
],
|
|
},
|
|
},
|
|
description: 'The short name(aka ID) of the forum to get Categories.',
|
|
},
|
|
{
|
|
displayName: 'Return All',
|
|
name: 'returnAll',
|
|
type: 'boolean',
|
|
displayOptions: {
|
|
show: {
|
|
resource: [
|
|
'forum',
|
|
],
|
|
operation: [
|
|
'getCategories',
|
|
],
|
|
},
|
|
},
|
|
default: false,
|
|
description: 'If all results should be returned or only up to a given limit.',
|
|
},
|
|
{
|
|
displayName: 'Limit',
|
|
name: 'limit',
|
|
type: 'number',
|
|
displayOptions: {
|
|
show: {
|
|
resource: [
|
|
'forum',
|
|
],
|
|
operation: [
|
|
'getCategories',
|
|
],
|
|
returnAll: [
|
|
false,
|
|
],
|
|
},
|
|
},
|
|
typeOptions: {
|
|
minValue: 1,
|
|
maxValue: 100,
|
|
},
|
|
default: 100,
|
|
description: 'How many results to return.',
|
|
},
|
|
{
|
|
displayName: 'Additional Fields',
|
|
name: 'additionalFields',
|
|
type: 'collection',
|
|
placeholder: 'Add Field',
|
|
displayOptions: {
|
|
show: {
|
|
operation: [
|
|
'getCategories',
|
|
],
|
|
resource: [
|
|
'forum',
|
|
],
|
|
},
|
|
},
|
|
default: {},
|
|
options: [
|
|
{
|
|
displayName: 'Order',
|
|
name: 'order',
|
|
type: 'options',
|
|
options: [
|
|
{
|
|
name: 'ASC',
|
|
value: 'asc',
|
|
},
|
|
{
|
|
name: 'DESC',
|
|
value: 'desc',
|
|
},
|
|
],
|
|
default: 'asc',
|
|
description: 'You may specify order to sort your response.',
|
|
},
|
|
],
|
|
},
|
|
|
|
// ----------------------------------
|
|
// forum:getThreads
|
|
// ----------------------------------
|
|
{
|
|
displayName: 'Forum name',
|
|
name: 'id',
|
|
type: 'string',
|
|
default: '',
|
|
required: true,
|
|
displayOptions: {
|
|
show: {
|
|
operation: [
|
|
'getThreads',
|
|
],
|
|
resource: [
|
|
'forum',
|
|
],
|
|
},
|
|
},
|
|
description: 'The short name(aka ID) of the forum to get Threads.',
|
|
},
|
|
{
|
|
displayName: 'Return All',
|
|
name: 'returnAll',
|
|
type: 'boolean',
|
|
displayOptions: {
|
|
show: {
|
|
resource: [
|
|
'forum',
|
|
],
|
|
operation: [
|
|
'getThreads',
|
|
],
|
|
},
|
|
},
|
|
default: false,
|
|
description: 'If all results should be returned or only up to a given limit.',
|
|
},
|
|
{
|
|
displayName: 'Limit',
|
|
name: 'limit',
|
|
type: 'number',
|
|
displayOptions: {
|
|
show: {
|
|
resource: [
|
|
'forum',
|
|
],
|
|
operation: [
|
|
'getThreads',
|
|
],
|
|
returnAll: [
|
|
false,
|
|
],
|
|
},
|
|
},
|
|
typeOptions: {
|
|
minValue: 1,
|
|
maxValue: 100,
|
|
},
|
|
default: 100,
|
|
description: 'How many results to return.',
|
|
},
|
|
{
|
|
displayName: 'Additional Fields',
|
|
name: 'additionalFields',
|
|
type: 'collection',
|
|
placeholder: 'Add Field',
|
|
displayOptions: {
|
|
show: {
|
|
operation: [
|
|
'getThreads',
|
|
],
|
|
resource: [
|
|
'forum',
|
|
],
|
|
},
|
|
},
|
|
default: {},
|
|
options: [
|
|
{
|
|
displayName: 'Related',
|
|
name: 'related',
|
|
type: 'multiOptions',
|
|
options: [
|
|
{
|
|
name: 'author',
|
|
value: 'author',
|
|
},
|
|
{
|
|
name: 'forum',
|
|
value: 'forum',
|
|
},
|
|
],
|
|
default: [],
|
|
description: 'You may specify relations to include with your response',
|
|
},
|
|
{
|
|
displayName: 'Include',
|
|
name: 'include',
|
|
type: 'multiOptions',
|
|
options: [
|
|
{
|
|
name: 'closed',
|
|
value: 'closed',
|
|
},
|
|
{
|
|
name: 'open',
|
|
value: 'open',
|
|
},
|
|
{
|
|
name: 'killed',
|
|
value: 'killed',
|
|
},
|
|
],
|
|
default: [],
|
|
description: 'You may specify relations to include with your response.',
|
|
},
|
|
{
|
|
displayName: 'Order',
|
|
name: 'order',
|
|
type: 'options',
|
|
options: [
|
|
{
|
|
name: 'ASC',
|
|
value: 'asc',
|
|
},
|
|
{
|
|
name: 'DESC',
|
|
value: 'desc',
|
|
},
|
|
],
|
|
default: 'asc',
|
|
description: 'You may specify order to sort your response.',
|
|
},
|
|
{
|
|
displayName: 'Since',
|
|
name: 'since',
|
|
type: 'dateTime',
|
|
default: '',
|
|
description: 'Unix timestamp (or ISO datetime standard)',
|
|
},
|
|
{
|
|
displayName: 'Thread',
|
|
name: 'threadId',
|
|
type: 'string',
|
|
default: '',
|
|
description: 'Looks up a thread by ID. You may pass us the "ident" query type instead of an ID by including "forum". You may pass us the "link" query type to filter by URL. You must pass the "forum" if you do not have the Pro API Access addon.',
|
|
},
|
|
],
|
|
},
|
|
],
|
|
};
|
|
|
|
|
|
async execute(this: IExecuteFunctions): Promise<INodeExecutionData[][]> {
|
|
const items = this.getInputData();
|
|
const returnData: IDataObject[] = [];
|
|
|
|
|
|
const resource = this.getNodeParameter('resource', 0) as string;
|
|
const operation = this.getNodeParameter('operation', 0) as string;
|
|
|
|
let endpoint = '';
|
|
let requestMethod = '';
|
|
let body: IDataObject | Buffer;
|
|
let qs: IDataObject;
|
|
|
|
|
|
for (let i = 0; i < items.length; i++) {
|
|
try {
|
|
body = {};
|
|
qs = {};
|
|
|
|
if (resource === 'forum') {
|
|
if (operation === 'get') {
|
|
// ----------------------------------
|
|
// get
|
|
// ----------------------------------
|
|
|
|
requestMethod = 'GET';
|
|
|
|
endpoint = 'forums/details.json';
|
|
|
|
const id = this.getNodeParameter('id', i) as string;
|
|
qs.forum = id;
|
|
|
|
const additionalFields = this.getNodeParameter('additionalFields', i) as IDataObject;
|
|
|
|
Object.assign(qs, additionalFields);
|
|
|
|
try {
|
|
const responseData = await disqusApiRequest.call(this, requestMethod, qs, endpoint);
|
|
returnData.push(responseData.response);
|
|
} catch (error) {
|
|
throw error;
|
|
}
|
|
|
|
} else if (operation === 'getPosts') {
|
|
// ----------------------------------
|
|
// getPosts
|
|
// ----------------------------------
|
|
|
|
requestMethod = 'GET';
|
|
|
|
endpoint = 'forums/listPosts.json';
|
|
|
|
const id = this.getNodeParameter('id', i) as string;
|
|
const additionalFields = this.getNodeParameter('additionalFields', i) as IDataObject;
|
|
Object.assign(qs, additionalFields);
|
|
|
|
const returnAll = this.getNodeParameter('returnAll', i) as boolean;
|
|
|
|
qs.forum = id;
|
|
qs.limit = 100;
|
|
|
|
try {
|
|
let responseData: IDataObject = {};
|
|
if(returnAll) {
|
|
responseData.response = await disqusApiRequestAllItems.call(this, requestMethod, qs, endpoint);
|
|
} else {
|
|
const limit = this.getNodeParameter('limit', i) as string;
|
|
qs.limit = limit;
|
|
responseData = await disqusApiRequest.call(this, requestMethod, qs, endpoint);
|
|
}
|
|
returnData.push.apply(returnData, responseData.response as IDataObject[]);
|
|
} catch (error) {
|
|
throw error;
|
|
}
|
|
|
|
} else if (operation === 'getCategories') {
|
|
// ----------------------------------
|
|
// getCategories
|
|
// ----------------------------------
|
|
|
|
requestMethod = 'GET';
|
|
|
|
endpoint = 'forums/listCategories.json';
|
|
|
|
const id = this.getNodeParameter('id', i) as string;
|
|
const returnAll = this.getNodeParameter('returnAll', i) as boolean;
|
|
const additionalFields = this.getNodeParameter('additionalFields', i) as IDataObject;
|
|
Object.assign(qs, additionalFields);
|
|
|
|
qs.forum = id;
|
|
qs.limit = 100;
|
|
|
|
try {
|
|
let responseData: IDataObject = {};
|
|
|
|
if(returnAll) {
|
|
responseData.response = await disqusApiRequestAllItems.call(this, requestMethod, qs, endpoint);
|
|
} else {
|
|
const limit = this.getNodeParameter('limit', i) as string;
|
|
qs.limit = limit;
|
|
responseData = await disqusApiRequest.call(this, requestMethod, qs, endpoint) as IDataObject;
|
|
}
|
|
returnData.push.apply(returnData, responseData.response as IDataObject[]) ;
|
|
} catch (error) {
|
|
throw error;
|
|
}
|
|
|
|
} else if (operation === 'getThreads') {
|
|
// ----------------------------------
|
|
// getThreads
|
|
// ----------------------------------
|
|
|
|
requestMethod = 'GET';
|
|
|
|
endpoint = 'forums/listThreads.json';
|
|
|
|
const id = this.getNodeParameter('id', i) as string;
|
|
const returnAll = this.getNodeParameter('returnAll', i) as boolean;
|
|
|
|
qs.forum = id;
|
|
qs.limit = 100;
|
|
|
|
const additionalFields = this.getNodeParameter('additionalFields', i) as IDataObject;
|
|
|
|
Object.assign(qs, additionalFields);
|
|
|
|
try {
|
|
let responseData: IDataObject = {};
|
|
if(returnAll) {
|
|
responseData.response = await disqusApiRequestAllItems.call(this, requestMethod, qs, endpoint);
|
|
} else {
|
|
const limit = this.getNodeParameter('limit', i) as string;
|
|
qs.limit = limit;
|
|
responseData = await disqusApiRequest.call(this, requestMethod, qs, endpoint);
|
|
}
|
|
returnData.push.apply(returnData, responseData.response as IDataObject[]);
|
|
} catch (error) {
|
|
throw error;
|
|
}
|
|
|
|
} else {
|
|
throw new NodeOperationError(this.getNode(), `The operation "${operation}" is not known!`);
|
|
}
|
|
|
|
} else {
|
|
throw new NodeOperationError(this.getNode(), `The resource "${resource}" is not known!`);
|
|
}
|
|
} catch (error) {
|
|
if (this.continueOnFail()) {
|
|
returnData.push({ error: error.message });
|
|
continue;
|
|
}
|
|
throw error;
|
|
}
|
|
}
|
|
|
|
return [this.helpers.returnJsonArray(returnData)];
|
|
}
|
|
}
|