mirror of
https://github.com/n8n-io/n8n.git
synced 2024-11-10 06:34:05 -08:00
🔀 Merge branch 'RicardoE105-feature/segment-node'
This commit is contained in:
commit
a6d148b0cc
|
@ -57,7 +57,7 @@ import ParameterInputFull from '@/components/ParameterInputFull.vue';
|
|||
import ParameterInputList from '@/components/ParameterInputList.vue';
|
||||
import NodeCredentials from '@/components/NodeCredentials.vue';
|
||||
import NodeWebhooks from '@/components/NodeWebhooks.vue';
|
||||
import { get, set } from 'lodash';
|
||||
import { get, set, unset } from 'lodash';
|
||||
|
||||
import { genericHelpers } from '@/components/mixins/genericHelpers';
|
||||
import { nodeHelpers } from '@/components/mixins/nodeHelpers';
|
||||
|
@ -369,9 +369,12 @@ export default mixins(
|
|||
Vue.set(nodeParameters as object, path, data);
|
||||
}
|
||||
} else {
|
||||
// For everything else
|
||||
if (newValue === undefined) {
|
||||
unset(nodeParameters as object, parameterPath);
|
||||
} else {
|
||||
set(nodeParameters as object, parameterPath, newValue);
|
||||
}
|
||||
}
|
||||
|
||||
// Get the parameters with the now new defaults according to the
|
||||
// from the user actually defined parameters
|
||||
|
|
|
@ -30,6 +30,9 @@ export default Vue
|
|||
},
|
||||
computed: {
|
||||
isMultiLineParameter () {
|
||||
if (this.level > 4) {
|
||||
return true;
|
||||
}
|
||||
const rows = this.getArgument('rows');
|
||||
if (rows !== undefined && rows > 1) {
|
||||
return true;
|
||||
|
@ -37,6 +40,9 @@ export default Vue
|
|||
|
||||
return false;
|
||||
},
|
||||
level (): number {
|
||||
return this.path.split('.').length;
|
||||
},
|
||||
},
|
||||
props: [
|
||||
'displayOptions',
|
||||
|
|
17
packages/nodes-base/credentials/SegmentApi.credentials.ts
Normal file
17
packages/nodes-base/credentials/SegmentApi.credentials.ts
Normal file
|
@ -0,0 +1,17 @@
|
|||
import {
|
||||
ICredentialType,
|
||||
NodePropertyTypes,
|
||||
} from 'n8n-workflow';
|
||||
|
||||
export class SegmentApi implements ICredentialType {
|
||||
name = 'segmentApi';
|
||||
displayName = 'Segment API';
|
||||
properties = [
|
||||
{
|
||||
displayName: 'Write Key',
|
||||
name: 'writekey',
|
||||
type: 'string' as NodePropertyTypes,
|
||||
default: '',
|
||||
},
|
||||
];
|
||||
}
|
36
packages/nodes-base/nodes/Segment/GenericFunctions.ts
Normal file
36
packages/nodes-base/nodes/Segment/GenericFunctions.ts
Normal file
|
@ -0,0 +1,36 @@
|
|||
import { OptionsWithUri } from 'request';
|
||||
import {
|
||||
IExecuteFunctions,
|
||||
IExecuteSingleFunctions,
|
||||
IHookFunctions,
|
||||
ILoadOptionsFunctions,
|
||||
IWebhookFunctions,
|
||||
} from 'n8n-core';
|
||||
import { IDataObject } from 'n8n-workflow';
|
||||
|
||||
export async function segmentApiRequest(this: IHookFunctions | IExecuteFunctions | IExecuteSingleFunctions | ILoadOptionsFunctions | IWebhookFunctions, method: string, resource: string, body: any = {}, qs: IDataObject = {}, uri?: string, option: IDataObject = {}): Promise<any> { // tslint:disable-line:no-any
|
||||
const credentials = this.getCredentials('segmentApi');
|
||||
if (credentials === undefined) {
|
||||
throw new Error('No credentials got returned!');
|
||||
}
|
||||
const base64Key = Buffer.from(`${credentials.writekey}:`).toString('base64');
|
||||
const options: OptionsWithUri = {
|
||||
headers: {
|
||||
Authorization: `Basic ${base64Key}`,
|
||||
'Content-Type': 'application/json',
|
||||
},
|
||||
method,
|
||||
qs,
|
||||
body,
|
||||
uri: uri ||`https://api.segment.io/v1${resource}`,
|
||||
json: true
|
||||
};
|
||||
if (!Object.keys(body).length) {
|
||||
delete options.body;
|
||||
}
|
||||
try {
|
||||
return await this.helpers.request!(options);
|
||||
} catch (error) {
|
||||
throw new Error('Segment Error: ' + error);
|
||||
}
|
||||
}
|
508
packages/nodes-base/nodes/Segment/IdentifyDescription.ts
Normal file
508
packages/nodes-base/nodes/Segment/IdentifyDescription.ts
Normal file
|
@ -0,0 +1,508 @@
|
|||
import { INodeProperties } from 'n8n-workflow';
|
||||
|
||||
export const identifyOperations = [
|
||||
{
|
||||
displayName: 'Operation',
|
||||
name: 'operation',
|
||||
type: 'options',
|
||||
displayOptions: {
|
||||
show: {
|
||||
resource: [
|
||||
'identify',
|
||||
],
|
||||
},
|
||||
},
|
||||
options: [
|
||||
{
|
||||
name: 'Create',
|
||||
value: 'create',
|
||||
description: 'Create an identity',
|
||||
},
|
||||
],
|
||||
default: 'create',
|
||||
description: 'The operation to perform.',
|
||||
},
|
||||
] as INodeProperties[];
|
||||
|
||||
export const identifyFields = [
|
||||
|
||||
/* -------------------------------------------------------------------------- */
|
||||
/* identify:create */
|
||||
/* -------------------------------------------------------------------------- */
|
||||
{
|
||||
displayName: 'User ID',
|
||||
name: 'userId',
|
||||
type: 'string',
|
||||
default: '',
|
||||
displayOptions: {
|
||||
show: {
|
||||
resource: [
|
||||
'identify',
|
||||
],
|
||||
operation: [
|
||||
'create',
|
||||
],
|
||||
},
|
||||
},
|
||||
required: false,
|
||||
},
|
||||
{
|
||||
displayName: 'Traits',
|
||||
name: 'traits',
|
||||
placeholder: 'Add Trait',
|
||||
type: 'fixedCollection',
|
||||
typeOptions: {
|
||||
multipleValues: false,
|
||||
},
|
||||
displayOptions: {
|
||||
show: {
|
||||
resource: [
|
||||
'identify',
|
||||
],
|
||||
operation: [
|
||||
'create',
|
||||
],
|
||||
},
|
||||
},
|
||||
default: {},
|
||||
options: [
|
||||
{
|
||||
name: 'traitsUi',
|
||||
displayName: 'Trait',
|
||||
values: [
|
||||
{
|
||||
displayName: 'Email',
|
||||
name: 'email',
|
||||
type: 'string',
|
||||
default: '',
|
||||
description: 'Email address of a user',
|
||||
},
|
||||
{
|
||||
displayName: 'First Name',
|
||||
name: 'firstname',
|
||||
type: 'string',
|
||||
default: '',
|
||||
description: 'First name of a user',
|
||||
},
|
||||
{
|
||||
displayName: 'Last Name',
|
||||
name: 'lastname',
|
||||
type: 'string',
|
||||
default: '',
|
||||
description: 'Last name of a user',
|
||||
},
|
||||
{
|
||||
displayName: 'Gender',
|
||||
name: 'gender',
|
||||
type: 'string',
|
||||
default: '',
|
||||
description: 'Gender of a user',
|
||||
},
|
||||
{
|
||||
displayName: 'Phone',
|
||||
name: 'phone',
|
||||
type: 'string',
|
||||
default: '',
|
||||
description: 'Phone number of a user',
|
||||
},
|
||||
{
|
||||
displayName: 'Username',
|
||||
name: 'username',
|
||||
type: 'string',
|
||||
default: '',
|
||||
description: 'User’s username',
|
||||
},
|
||||
{
|
||||
displayName: 'Website',
|
||||
name: 'website',
|
||||
type: 'string',
|
||||
default: '',
|
||||
description: 'Website of a user',
|
||||
},
|
||||
{
|
||||
displayName: 'Age',
|
||||
name: 'age',
|
||||
type: 'number',
|
||||
default: 1,
|
||||
description: 'Age of a user',
|
||||
},
|
||||
{
|
||||
displayName: 'Avatar',
|
||||
name: 'avatar',
|
||||
type: 'string',
|
||||
default: '',
|
||||
description: 'URL to an avatar image for the user',
|
||||
},
|
||||
{
|
||||
displayName: 'Birthday',
|
||||
name: 'birthday',
|
||||
type: 'dateTime',
|
||||
default: '',
|
||||
description: 'User’s birthday',
|
||||
},
|
||||
{
|
||||
displayName: 'Created At',
|
||||
name: 'createdAt',
|
||||
type: 'dateTime',
|
||||
default: '',
|
||||
description: 'Date the user’s account was first created',
|
||||
},
|
||||
{
|
||||
displayName: 'Description',
|
||||
name: 'description',
|
||||
type: 'string',
|
||||
typeOptions: {
|
||||
alwaysOpenEditWindow: true,
|
||||
},
|
||||
default: '',
|
||||
description: 'Description of the user',
|
||||
},
|
||||
{
|
||||
displayName: 'ID',
|
||||
name: 'id',
|
||||
type: 'string',
|
||||
default: '',
|
||||
description: 'Unique ID in your database for a user',
|
||||
},
|
||||
{
|
||||
displayName: 'Company',
|
||||
name: 'company',
|
||||
placeholder: 'Add Company',
|
||||
type: 'fixedCollection',
|
||||
typeOptions: {
|
||||
multipleValues: false,
|
||||
},
|
||||
default: {},
|
||||
options: [
|
||||
{
|
||||
name: 'companyUi',
|
||||
displayName: 'Company',
|
||||
values: [
|
||||
{
|
||||
displayName: 'ID',
|
||||
name: 'id',
|
||||
type: 'string',
|
||||
default: '',
|
||||
},
|
||||
{
|
||||
displayName: 'Name',
|
||||
name: 'name',
|
||||
type: 'string',
|
||||
default: '',
|
||||
},
|
||||
{
|
||||
displayName: 'Industry',
|
||||
name: 'industry',
|
||||
type: 'string',
|
||||
default: '',
|
||||
},
|
||||
{
|
||||
displayName: 'Employee Count',
|
||||
name: 'employeeCount',
|
||||
type: 'number',
|
||||
default: 1,
|
||||
},
|
||||
{
|
||||
displayName: 'Plan',
|
||||
name: 'plan',
|
||||
type: 'string',
|
||||
default: '',
|
||||
},
|
||||
]
|
||||
},
|
||||
],
|
||||
},
|
||||
{
|
||||
displayName: 'Address',
|
||||
name: 'address',
|
||||
placeholder: 'Add Address',
|
||||
type: 'fixedCollection',
|
||||
typeOptions: {
|
||||
multipleValues: false,
|
||||
},
|
||||
default: {},
|
||||
options: [
|
||||
{
|
||||
name: 'addressUi',
|
||||
displayName: 'Address',
|
||||
values: [
|
||||
{
|
||||
displayName: 'Street',
|
||||
name: 'street',
|
||||
type: 'string',
|
||||
default: '',
|
||||
},
|
||||
{
|
||||
displayName: 'City',
|
||||
name: 'city',
|
||||
type: 'string',
|
||||
default: '',
|
||||
},
|
||||
{
|
||||
displayName: 'State',
|
||||
name: 'state',
|
||||
type: 'string',
|
||||
default: '',
|
||||
},
|
||||
{
|
||||
displayName: 'Postal Code',
|
||||
name: 'postalCode',
|
||||
type: 'string',
|
||||
default: '',
|
||||
},
|
||||
{
|
||||
displayName: 'Country',
|
||||
name: 'country',
|
||||
type: 'string',
|
||||
default: '',
|
||||
},
|
||||
]
|
||||
},
|
||||
],
|
||||
},
|
||||
]
|
||||
},
|
||||
],
|
||||
},
|
||||
{
|
||||
displayName: 'Context',
|
||||
name: 'context',
|
||||
placeholder: 'Add Context',
|
||||
type: 'fixedCollection',
|
||||
typeOptions: {
|
||||
multipleValues: false,
|
||||
},
|
||||
displayOptions: {
|
||||
show: {
|
||||
resource: [
|
||||
'identify',
|
||||
],
|
||||
operation: [
|
||||
'create',
|
||||
],
|
||||
},
|
||||
},
|
||||
default: {},
|
||||
options: [
|
||||
{
|
||||
name: 'contextUi',
|
||||
displayName: 'Context',
|
||||
values: [
|
||||
{
|
||||
displayName: 'Active',
|
||||
name: 'active',
|
||||
type: 'boolean',
|
||||
default: '',
|
||||
description: 'Whether a user is active',
|
||||
},
|
||||
{
|
||||
displayName: 'IP',
|
||||
name: 'ip',
|
||||
type: 'string',
|
||||
default: '',
|
||||
description: 'Current user’s IP address.',
|
||||
},
|
||||
{
|
||||
displayName: 'Locale',
|
||||
name: 'locate',
|
||||
type: 'string',
|
||||
default: '',
|
||||
description: 'Locale string for the current user, for example en-US.',
|
||||
},
|
||||
{
|
||||
displayName: 'Page',
|
||||
name: 'page',
|
||||
type: 'string',
|
||||
default: '',
|
||||
description: 'Dictionary of information about the current page in the browser, containing hash, path, referrer, search, title and url',
|
||||
},
|
||||
{
|
||||
displayName: 'Timezone',
|
||||
name: 'timezone',
|
||||
type: 'string',
|
||||
default: '',
|
||||
description: 'Timezones are sent as tzdata strings to add user timezone information which might be stripped from the timestamp, for example America/New_York',
|
||||
},
|
||||
{
|
||||
displayName: 'App',
|
||||
name: 'app',
|
||||
placeholder: 'Add App',
|
||||
type: 'fixedCollection',
|
||||
typeOptions: {
|
||||
multipleValues: false,
|
||||
},
|
||||
default: {},
|
||||
options: [
|
||||
{
|
||||
name: 'appUi',
|
||||
displayName: 'App',
|
||||
values: [
|
||||
{
|
||||
displayName: 'Name',
|
||||
name: 'name',
|
||||
type: 'string',
|
||||
default: '',
|
||||
},
|
||||
{
|
||||
displayName: 'Version',
|
||||
name: 'version',
|
||||
type: 'string',
|
||||
default: '',
|
||||
},
|
||||
{
|
||||
displayName: 'Build',
|
||||
name: 'build',
|
||||
type: 'string',
|
||||
default: '',
|
||||
},
|
||||
]
|
||||
},
|
||||
],
|
||||
},
|
||||
{
|
||||
displayName: 'Campaign',
|
||||
name: 'campaign',
|
||||
placeholder: 'Campaign App',
|
||||
type: 'fixedCollection',
|
||||
typeOptions: {
|
||||
multipleValues: false,
|
||||
},
|
||||
default: {},
|
||||
options: [
|
||||
{
|
||||
name: 'campaignUi',
|
||||
displayName: 'Campaign',
|
||||
values: [
|
||||
{
|
||||
displayName: 'Name',
|
||||
name: 'name',
|
||||
type: 'string',
|
||||
default: '',
|
||||
},
|
||||
{
|
||||
displayName: 'Source',
|
||||
name: 'source',
|
||||
type: 'string',
|
||||
default: '',
|
||||
},
|
||||
{
|
||||
displayName: 'Medium',
|
||||
name: 'medium',
|
||||
type: 'string',
|
||||
default: '',
|
||||
},
|
||||
{
|
||||
displayName: 'Term',
|
||||
name: 'term',
|
||||
type: 'string',
|
||||
default: '',
|
||||
},
|
||||
{
|
||||
displayName: 'Content',
|
||||
name: 'content',
|
||||
type: 'string',
|
||||
default: '',
|
||||
},
|
||||
]
|
||||
},
|
||||
],
|
||||
},
|
||||
{
|
||||
displayName: 'Device',
|
||||
name: 'device',
|
||||
placeholder: 'Add Device',
|
||||
type: 'fixedCollection',
|
||||
typeOptions: {
|
||||
multipleValues: false,
|
||||
},
|
||||
default: {},
|
||||
options: [
|
||||
{
|
||||
name: 'deviceUi',
|
||||
displayName: 'Device',
|
||||
values: [
|
||||
{
|
||||
displayName: 'ID',
|
||||
name: 'id',
|
||||
type: 'string',
|
||||
default: '',
|
||||
},
|
||||
{
|
||||
displayName: 'Manufacturer',
|
||||
name: 'manufacturer',
|
||||
type: 'string',
|
||||
default: '',
|
||||
},
|
||||
{
|
||||
displayName: 'Model',
|
||||
name: 'model',
|
||||
type: 'string',
|
||||
default: '',
|
||||
},
|
||||
{
|
||||
displayName: 'Name',
|
||||
name: 'name',
|
||||
type: 'string',
|
||||
default: '',
|
||||
},
|
||||
{
|
||||
displayName: 'Type',
|
||||
name: 'type',
|
||||
type: 'string',
|
||||
default: '',
|
||||
},
|
||||
{
|
||||
displayName: 'Version',
|
||||
name: 'version',
|
||||
type: 'string',
|
||||
default: '',
|
||||
},
|
||||
],
|
||||
},
|
||||
],
|
||||
},
|
||||
]
|
||||
},
|
||||
],
|
||||
},
|
||||
{
|
||||
displayName: 'Integration',
|
||||
name: 'integrations',
|
||||
placeholder: 'Add Integration',
|
||||
type: 'fixedCollection',
|
||||
typeOptions: {
|
||||
multipleValues: false,
|
||||
},
|
||||
displayOptions: {
|
||||
show: {
|
||||
resource: [
|
||||
'identify',
|
||||
],
|
||||
operation: [
|
||||
'create',
|
||||
],
|
||||
},
|
||||
},
|
||||
default: {},
|
||||
options: [
|
||||
{
|
||||
name: 'integrationsUi',
|
||||
displayName: 'Integration',
|
||||
values: [
|
||||
{
|
||||
displayName: 'All',
|
||||
name: 'all',
|
||||
type: 'boolean',
|
||||
default: true,
|
||||
},
|
||||
{
|
||||
displayName: 'Salesforce',
|
||||
name: 'salesforce',
|
||||
type: 'boolean',
|
||||
default: false,
|
||||
},
|
||||
],
|
||||
},
|
||||
],
|
||||
},
|
||||
] as INodeProperties[];
|
10
packages/nodes-base/nodes/Segment/IdentifyInterface.ts
Normal file
10
packages/nodes-base/nodes/Segment/IdentifyInterface.ts
Normal file
|
@ -0,0 +1,10 @@
|
|||
import { IDataObject } from "n8n-workflow";
|
||||
|
||||
export interface IIdentify {
|
||||
userId?: string;
|
||||
anonymousId?: string;
|
||||
traits?: IDataObject;
|
||||
context?: IDataObject;
|
||||
integrations?: IDataObject;
|
||||
timestamp?: string;
|
||||
}
|
774
packages/nodes-base/nodes/Segment/Segment.node.ts
Normal file
774
packages/nodes-base/nodes/Segment/Segment.node.ts
Normal file
|
@ -0,0 +1,774 @@
|
|||
import {
|
||||
IExecuteFunctions,
|
||||
} from 'n8n-core';
|
||||
import {
|
||||
IDataObject,
|
||||
INodeExecutionData,
|
||||
INodeType,
|
||||
INodeTypeDescription,
|
||||
} from 'n8n-workflow';
|
||||
import {
|
||||
segmentApiRequest,
|
||||
} from './GenericFunctions';
|
||||
import {
|
||||
identifyFields,
|
||||
identifyOperations,
|
||||
} from './IdentifyDescription';
|
||||
import {
|
||||
IIdentify,
|
||||
} from './IdentifyInterface';
|
||||
import {
|
||||
trackOperations,
|
||||
trackFields,
|
||||
} from './TrackDescription';
|
||||
import { ITrack } from './TrackInterface';
|
||||
import * as uuid from 'uuid/v4';
|
||||
|
||||
export class Segment implements INodeType {
|
||||
description: INodeTypeDescription = {
|
||||
displayName: 'Segment',
|
||||
name: 'segment',
|
||||
icon: 'file:segment.png',
|
||||
group: ['output'],
|
||||
version: 1,
|
||||
subtitle: '={{$parameter["operation"] + ":" + $parameter["resource"]}}',
|
||||
description: 'Consume Segment API',
|
||||
defaults: {
|
||||
name: 'Segment',
|
||||
color: '#6ebb99',
|
||||
},
|
||||
inputs: ['main'],
|
||||
outputs: ['main'],
|
||||
credentials: [
|
||||
{
|
||||
name: 'segmentApi',
|
||||
required: true,
|
||||
}
|
||||
],
|
||||
properties: [
|
||||
{
|
||||
displayName: 'Resource',
|
||||
name: 'resource',
|
||||
type: 'options',
|
||||
options: [
|
||||
{
|
||||
name: 'Identify',
|
||||
value: 'identify',
|
||||
description: 'Identify lets you tie a user to their actions.'
|
||||
},
|
||||
{
|
||||
name: 'Track',
|
||||
value: 'track',
|
||||
description: 'Track lets you record events',
|
||||
},
|
||||
],
|
||||
default: 'identify',
|
||||
description: 'Resource to consume.',
|
||||
},
|
||||
...identifyOperations,
|
||||
...trackOperations,
|
||||
...identifyFields,
|
||||
...trackFields,
|
||||
],
|
||||
};
|
||||
|
||||
async execute(this: IExecuteFunctions): Promise<INodeExecutionData[][]> {
|
||||
const items = this.getInputData();
|
||||
const returnData: IDataObject[] = [];
|
||||
const length = items.length as unknown as number;
|
||||
const qs: IDataObject = {};
|
||||
let responseData;
|
||||
const resource = this.getNodeParameter('resource', 0) as string;
|
||||
const operation = this.getNodeParameter('operation', 0) as string;
|
||||
for (let i = 0; i < length; i++) {
|
||||
if (resource === 'identify') {
|
||||
//https://segment.com/docs/connections/sources/catalog/libraries/server/http-api/#identify
|
||||
if (operation === 'create') {
|
||||
const userId = this.getNodeParameter('userId', i) as string;
|
||||
const traits = (this.getNodeParameter('traits', i) as IDataObject).traitsUi as IDataObject;
|
||||
const context = (this.getNodeParameter('context', i) as IDataObject).contextUi as IDataObject;
|
||||
const integrations = (this.getNodeParameter('integrations', i) as IDataObject).integrationsUi as IDataObject;
|
||||
const body: IIdentify = {
|
||||
traits: {
|
||||
company: {},
|
||||
address: {},
|
||||
},
|
||||
context: {
|
||||
app: {},
|
||||
campaign: {},
|
||||
device: {},
|
||||
},
|
||||
integrations: {},
|
||||
};
|
||||
if (userId) {
|
||||
body.userId = userId as string;
|
||||
} else {
|
||||
body.anonymousId = uuid();
|
||||
}
|
||||
if (traits) {
|
||||
if (traits.email) {
|
||||
body.traits!.email = traits.email as string;
|
||||
}
|
||||
if (traits.firstname) {
|
||||
body.traits!.firstname = traits.firstname as string;
|
||||
}
|
||||
if (traits.lastname) {
|
||||
body.traits!.lastname = traits.lastname as string;
|
||||
}
|
||||
if (traits.gender) {
|
||||
body.traits!.gender = traits.gender as string;
|
||||
}
|
||||
if (traits.phone) {
|
||||
body.traits!.phone = traits.phone as string;
|
||||
}
|
||||
if (traits.username) {
|
||||
body.traits!.username = traits.username as string;
|
||||
}
|
||||
if (traits.website) {
|
||||
body.traits!.website = traits.website as string;
|
||||
}
|
||||
if (traits.age) {
|
||||
body.traits!.age = traits.age as number;
|
||||
}
|
||||
if (traits.avatar) {
|
||||
body.traits!.avatar = traits.avatar as string;
|
||||
}
|
||||
if (traits.birthday) {
|
||||
body.traits!.birthday = traits.birthday as string;
|
||||
}
|
||||
if (traits.createdAt) {
|
||||
body.traits!.createdAt = traits.createdAt as string;
|
||||
}
|
||||
if (traits.description) {
|
||||
body.traits!.description = traits.description as string;
|
||||
}
|
||||
if (traits.id) {
|
||||
body.traits!.id = traits.id as string;
|
||||
}
|
||||
if (traits.company) {
|
||||
const company = (traits.company as IDataObject).companyUi as IDataObject;
|
||||
if (company) {
|
||||
if (company.id) {
|
||||
//@ts-ignore
|
||||
body.traits.company.id = company.id as string;
|
||||
}
|
||||
if (company.name) {
|
||||
//@ts-ignore
|
||||
body.traits.company.name = company.name as string;
|
||||
}
|
||||
if (company.industry) {
|
||||
//@ts-ignore
|
||||
body.traits.company.industry = company.industry as string;
|
||||
}
|
||||
if (company.employeeCount) {
|
||||
//@ts-ignore
|
||||
body.traits.company.employeeCount = company.employeeCount as number;
|
||||
}
|
||||
if (company.plan) {
|
||||
//@ts-ignore
|
||||
body.traits.company.plan = company.plan as string;
|
||||
}
|
||||
}
|
||||
}
|
||||
if (traits.address) {
|
||||
const address = (traits.address as IDataObject).addressUi as IDataObject;
|
||||
if (address) {
|
||||
if (address.street) {
|
||||
//@ts-ignore
|
||||
body.traits.address.street = address.street as string;
|
||||
}
|
||||
if (address.city) {
|
||||
//@ts-ignore
|
||||
body.traits.address.city = address.city as string;
|
||||
}
|
||||
if (address.state) {
|
||||
//@ts-ignore
|
||||
body.traits.address.state = address.state as string;
|
||||
}
|
||||
if (address.postalCode) {
|
||||
//@ts-ignore
|
||||
body.traits.address.postalCode = address.postalCode as string;
|
||||
}
|
||||
if (address.country) {
|
||||
//@ts-ignore
|
||||
body.traits.address.country = address.country as string;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
if (context) {
|
||||
if (context.active) {
|
||||
body.context!.active = context.active as boolean;
|
||||
}
|
||||
if (context.ip) {
|
||||
body.context!.ip = context.ip as string;
|
||||
}
|
||||
if (context.locate) {
|
||||
body.context!.locate = context.locate as string;
|
||||
}
|
||||
if (context.page) {
|
||||
body.context!.page = context.page as string;
|
||||
}
|
||||
if (context.timezone) {
|
||||
body.context!.timezone = context.timezone as string;
|
||||
}
|
||||
if (context.timezone) {
|
||||
body.context!.timezone = context.timezone as string;
|
||||
}
|
||||
if (context.app) {
|
||||
const app = (context.app as IDataObject).appUi as IDataObject;
|
||||
if (app) {
|
||||
if (app.name) {
|
||||
//@ts-ignore
|
||||
body.context.app.name = app.name as string;
|
||||
}
|
||||
if (app.version) {
|
||||
//@ts-ignore
|
||||
body.context.app.version = app.version as string;
|
||||
}
|
||||
if (app.build) {
|
||||
//@ts-ignore
|
||||
body.context.app.build = app.build as string;
|
||||
}
|
||||
}
|
||||
}
|
||||
if (context.campaign) {
|
||||
const campaign = (context.campaign as IDataObject).campaignUi as IDataObject;
|
||||
if (campaign) {
|
||||
if (campaign.name) {
|
||||
//@ts-ignore
|
||||
body.context.campaign.name = campaign.name as string;
|
||||
}
|
||||
if (campaign.source) {
|
||||
//@ts-ignore
|
||||
body.context.campaign.source = campaign.source as string;
|
||||
}
|
||||
if (campaign.medium) {
|
||||
//@ts-ignore
|
||||
body.context.campaign.medium = campaign.medium as string;
|
||||
}
|
||||
if (campaign.term) {
|
||||
//@ts-ignore
|
||||
body.context.campaign.term = campaign.term as string;
|
||||
}
|
||||
if (campaign.content) {
|
||||
//@ts-ignore
|
||||
body.context.campaign.content = campaign.content as string;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
if (context.device) {
|
||||
const device = (context.device as IDataObject).deviceUi as IDataObject;
|
||||
if (device) {
|
||||
if (device.id) {
|
||||
//@ts-ignore
|
||||
body.context.device.id = device.id as string;
|
||||
}
|
||||
if (device.manufacturer) {
|
||||
//@ts-ignore
|
||||
body.context.device.manufacturer = device.manufacturer as string;
|
||||
}
|
||||
if (device.model) {
|
||||
//@ts-ignore
|
||||
body.context.device.model = device.model as string;
|
||||
}
|
||||
if (device.type) {
|
||||
//@ts-ignore
|
||||
body.context.device.type = device.type as string;
|
||||
}
|
||||
if (device.version) {
|
||||
//@ts-ignore
|
||||
body.context.device.version = device.version as string;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
if (integrations) {
|
||||
if (integrations.all) {
|
||||
body.integrations!.all = integrations.all as boolean;
|
||||
}
|
||||
if (integrations.salesforce) {
|
||||
body.integrations!.salesforce = integrations.salesforce as boolean;
|
||||
}
|
||||
}
|
||||
responseData = await segmentApiRequest.call(this, 'POST', '/identify', body);
|
||||
}
|
||||
}
|
||||
if (resource === 'track') {
|
||||
//https://segment.com/docs/connections/sources/catalog/libraries/server/http-api/#track
|
||||
if (operation === 'event') {
|
||||
const userId = this.getNodeParameter('userId', i) as string;
|
||||
const event = this.getNodeParameter('event', i) as string;
|
||||
const traits = (this.getNodeParameter('traits', i) as IDataObject).traitsUi as IDataObject;
|
||||
const context = (this.getNodeParameter('context', i) as IDataObject).contextUi as IDataObject;
|
||||
const integrations = (this.getNodeParameter('integrations', i) as IDataObject).integrationsUi as IDataObject;
|
||||
const properties = (this.getNodeParameter('properties', i) as IDataObject).propertiesUi as IDataObject;
|
||||
const body: ITrack = {
|
||||
event,
|
||||
traits: {
|
||||
company: {},
|
||||
address: {},
|
||||
},
|
||||
context: {
|
||||
app: {},
|
||||
campaign: {},
|
||||
device: {},
|
||||
},
|
||||
integrations: {},
|
||||
properties: {},
|
||||
};
|
||||
if (userId) {
|
||||
body.userId = userId as string;
|
||||
} else {
|
||||
body.anonymousId = uuid();
|
||||
}
|
||||
if (traits) {
|
||||
if (traits.email) {
|
||||
body.traits!.email = traits.email as string;
|
||||
}
|
||||
if (traits.firstname) {
|
||||
body.traits!.firstname = traits.firstname as string;
|
||||
}
|
||||
if (traits.lastname) {
|
||||
body.traits!.lastname = traits.lastname as string;
|
||||
}
|
||||
if (traits.gender) {
|
||||
body.traits!.gender = traits.gender as string;
|
||||
}
|
||||
if (traits.phone) {
|
||||
body.traits!.phone = traits.phone as string;
|
||||
}
|
||||
if (traits.username) {
|
||||
body.traits!.username = traits.username as string;
|
||||
}
|
||||
if (traits.website) {
|
||||
body.traits!.website = traits.website as string;
|
||||
}
|
||||
if (traits.age) {
|
||||
body.traits!.age = traits.age as number;
|
||||
}
|
||||
if (traits.avatar) {
|
||||
body.traits!.avatar = traits.avatar as string;
|
||||
}
|
||||
if (traits.birthday) {
|
||||
body.traits!.birthday = traits.birthday as string;
|
||||
}
|
||||
if (traits.createdAt) {
|
||||
body.traits!.createdAt = traits.createdAt as string;
|
||||
}
|
||||
if (traits.description) {
|
||||
body.traits!.description = traits.description as string;
|
||||
}
|
||||
if (traits.id) {
|
||||
body.traits!.id = traits.id as string;
|
||||
}
|
||||
if (traits.company) {
|
||||
const company = (traits.company as IDataObject).companyUi as IDataObject;
|
||||
if (company) {
|
||||
if (company.id) {
|
||||
//@ts-ignore
|
||||
body.traits.company.id = company.id as string;
|
||||
}
|
||||
if (company.name) {
|
||||
//@ts-ignore
|
||||
body.traits.company.name = company.name as string;
|
||||
}
|
||||
if (company.industry) {
|
||||
//@ts-ignore
|
||||
body.traits.company.industry = company.industry as string;
|
||||
}
|
||||
if (company.employeeCount) {
|
||||
//@ts-ignore
|
||||
body.traits.company.employeeCount = company.employeeCount as number;
|
||||
}
|
||||
if (company.plan) {
|
||||
//@ts-ignore
|
||||
body.traits.company.plan = company.plan as string;
|
||||
}
|
||||
}
|
||||
}
|
||||
if (traits.address) {
|
||||
const address = (traits.address as IDataObject).addressUi as IDataObject;
|
||||
if (address) {
|
||||
if (address.street) {
|
||||
//@ts-ignore
|
||||
body.traits.address.street = address.street as string;
|
||||
}
|
||||
if (address.city) {
|
||||
//@ts-ignore
|
||||
body.traits.address.city = address.city as string;
|
||||
}
|
||||
if (address.state) {
|
||||
//@ts-ignore
|
||||
body.traits.address.state = address.state as string;
|
||||
}
|
||||
if (address.postalCode) {
|
||||
//@ts-ignore
|
||||
body.traits.address.postalCode = address.postalCode as string;
|
||||
}
|
||||
if (address.country) {
|
||||
//@ts-ignore
|
||||
body.traits.address.country = address.country as string;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
if (context) {
|
||||
if (context.active) {
|
||||
body.context!.active = context.active as boolean;
|
||||
}
|
||||
if (context.ip) {
|
||||
body.context!.ip = context.ip as string;
|
||||
}
|
||||
if (context.locate) {
|
||||
body.context!.locate = context.locate as string;
|
||||
}
|
||||
if (context.page) {
|
||||
body.context!.page = context.page as string;
|
||||
}
|
||||
if (context.timezone) {
|
||||
body.context!.timezone = context.timezone as string;
|
||||
}
|
||||
if (context.timezone) {
|
||||
body.context!.timezone = context.timezone as string;
|
||||
}
|
||||
if (context.app) {
|
||||
const app = (context.app as IDataObject).appUi as IDataObject;
|
||||
if (app) {
|
||||
if (app.name) {
|
||||
//@ts-ignore
|
||||
body.context.app.name = app.name as string;
|
||||
}
|
||||
if (app.version) {
|
||||
//@ts-ignore
|
||||
body.context.app.version = app.version as string;
|
||||
}
|
||||
if (app.build) {
|
||||
//@ts-ignore
|
||||
body.context.app.build = app.build as string;
|
||||
}
|
||||
}
|
||||
}
|
||||
if (context.campaign) {
|
||||
const campaign = (context.campaign as IDataObject).campaignUi as IDataObject;
|
||||
if (campaign) {
|
||||
if (campaign.name) {
|
||||
//@ts-ignore
|
||||
body.context.campaign.name = campaign.name as string;
|
||||
}
|
||||
if (campaign.source) {
|
||||
//@ts-ignore
|
||||
body.context.campaign.source = campaign.source as string;
|
||||
}
|
||||
if (campaign.medium) {
|
||||
//@ts-ignore
|
||||
body.context.campaign.medium = campaign.medium as string;
|
||||
}
|
||||
if (campaign.term) {
|
||||
//@ts-ignore
|
||||
body.context.campaign.term = campaign.term as string;
|
||||
}
|
||||
if (campaign.content) {
|
||||
//@ts-ignore
|
||||
body.context.campaign.content = campaign.content as string;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
if (context.device) {
|
||||
const device = (context.device as IDataObject).deviceUi as IDataObject;
|
||||
if (device) {
|
||||
if (device.id) {
|
||||
//@ts-ignore
|
||||
body.context.device.id = device.id as string;
|
||||
}
|
||||
if (device.manufacturer) {
|
||||
//@ts-ignore
|
||||
body.context.device.manufacturer = device.manufacturer as string;
|
||||
}
|
||||
if (device.model) {
|
||||
//@ts-ignore
|
||||
body.context.device.model = device.model as string;
|
||||
}
|
||||
if (device.type) {
|
||||
//@ts-ignore
|
||||
body.context.device.type = device.type as string;
|
||||
}
|
||||
if (device.version) {
|
||||
//@ts-ignore
|
||||
body.context.device.version = device.version as string;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
if (integrations) {
|
||||
if (integrations.all) {
|
||||
body.integrations!.all = integrations.all as boolean;
|
||||
}
|
||||
if (integrations.salesforce) {
|
||||
body.integrations!.salesforce = integrations.salesforce as boolean;
|
||||
}
|
||||
}
|
||||
if (properties) {
|
||||
if (properties.revenue) {
|
||||
body.properties!.revenue = properties.revenue as number;
|
||||
}
|
||||
if (properties.currency) {
|
||||
body.properties!.currency = properties.currency as string;
|
||||
}
|
||||
if (properties.value) {
|
||||
body.properties!.value = properties.value as string;
|
||||
}
|
||||
}
|
||||
responseData = await segmentApiRequest.call(this, 'POST', '/track', body);
|
||||
}
|
||||
//https://segment.com/docs/connections/sources/catalog/libraries/server/http-api/#page
|
||||
if (operation === 'page') {
|
||||
const userId = this.getNodeParameter('userId', i) as string;
|
||||
const event = this.getNodeParameter('event', i) as string;
|
||||
const traits = (this.getNodeParameter('traits', i) as IDataObject).traitsUi as IDataObject;
|
||||
const context = (this.getNodeParameter('context', i) as IDataObject).contextUi as IDataObject;
|
||||
const integrations = (this.getNodeParameter('integrations', i) as IDataObject).integrationsUi as IDataObject;
|
||||
const properties = (this.getNodeParameter('properties', i) as IDataObject).propertiesUi as IDataObject;
|
||||
const body: ITrack = {
|
||||
event,
|
||||
traits: {
|
||||
company: {},
|
||||
address: {},
|
||||
},
|
||||
context: {
|
||||
app: {},
|
||||
campaign: {},
|
||||
device: {},
|
||||
},
|
||||
integrations: {},
|
||||
properties: {},
|
||||
};
|
||||
if (userId) {
|
||||
body.userId = userId as string;
|
||||
} else {
|
||||
body.anonymousId = uuid();
|
||||
}
|
||||
if (traits) {
|
||||
if (traits.email) {
|
||||
body.traits!.email = traits.email as string;
|
||||
}
|
||||
if (traits.firstname) {
|
||||
body.traits!.firstname = traits.firstname as string;
|
||||
}
|
||||
if (traits.lastname) {
|
||||
body.traits!.lastname = traits.lastname as string;
|
||||
}
|
||||
if (traits.gender) {
|
||||
body.traits!.gender = traits.gender as string;
|
||||
}
|
||||
if (traits.phone) {
|
||||
body.traits!.phone = traits.phone as string;
|
||||
}
|
||||
if (traits.username) {
|
||||
body.traits!.username = traits.username as string;
|
||||
}
|
||||
if (traits.website) {
|
||||
body.traits!.website = traits.website as string;
|
||||
}
|
||||
if (traits.age) {
|
||||
body.traits!.age = traits.age as number;
|
||||
}
|
||||
if (traits.avatar) {
|
||||
body.traits!.avatar = traits.avatar as string;
|
||||
}
|
||||
if (traits.birthday) {
|
||||
body.traits!.birthday = traits.birthday as string;
|
||||
}
|
||||
if (traits.createdAt) {
|
||||
body.traits!.createdAt = traits.createdAt as string;
|
||||
}
|
||||
if (traits.description) {
|
||||
body.traits!.description = traits.description as string;
|
||||
}
|
||||
if (traits.id) {
|
||||
body.traits!.id = traits.id as string;
|
||||
}
|
||||
if (traits.company) {
|
||||
const company = (traits.company as IDataObject).companyUi as IDataObject;
|
||||
if (company) {
|
||||
if (company.id) {
|
||||
//@ts-ignore
|
||||
body.traits.company.id = company.id as string;
|
||||
}
|
||||
if (company.name) {
|
||||
//@ts-ignore
|
||||
body.traits.company.name = company.name as string;
|
||||
}
|
||||
if (company.industry) {
|
||||
//@ts-ignore
|
||||
body.traits.company.industry = company.industry as string;
|
||||
}
|
||||
if (company.employeeCount) {
|
||||
//@ts-ignore
|
||||
body.traits.company.employeeCount = company.employeeCount as number;
|
||||
}
|
||||
if (company.plan) {
|
||||
//@ts-ignore
|
||||
body.traits.company.plan = company.plan as string;
|
||||
}
|
||||
}
|
||||
}
|
||||
if (traits.address) {
|
||||
const address = (traits.address as IDataObject).addressUi as IDataObject;
|
||||
if (address) {
|
||||
if (address.street) {
|
||||
//@ts-ignore
|
||||
body.traits.address.street = address.street as string;
|
||||
}
|
||||
if (address.city) {
|
||||
//@ts-ignore
|
||||
body.traits.address.city = address.city as string;
|
||||
}
|
||||
if (address.state) {
|
||||
//@ts-ignore
|
||||
body.traits.address.state = address.state as string;
|
||||
}
|
||||
if (address.postalCode) {
|
||||
//@ts-ignore
|
||||
body.traits.address.postalCode = address.postalCode as string;
|
||||
}
|
||||
if (address.country) {
|
||||
//@ts-ignore
|
||||
body.traits.address.country = address.country as string;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
if (context) {
|
||||
if (context.active) {
|
||||
body.context!.active = context.active as boolean;
|
||||
}
|
||||
if (context.ip) {
|
||||
body.context!.ip = context.ip as string;
|
||||
}
|
||||
if (context.locate) {
|
||||
body.context!.locate = context.locate as string;
|
||||
}
|
||||
if (context.page) {
|
||||
body.context!.page = context.page as string;
|
||||
}
|
||||
if (context.timezone) {
|
||||
body.context!.timezone = context.timezone as string;
|
||||
}
|
||||
if (context.timezone) {
|
||||
body.context!.timezone = context.timezone as string;
|
||||
}
|
||||
if (context.app) {
|
||||
const app = (context.app as IDataObject).appUi as IDataObject;
|
||||
if (app) {
|
||||
if (app.name) {
|
||||
//@ts-ignore
|
||||
body.context.app.name = app.name as string;
|
||||
}
|
||||
if (app.version) {
|
||||
//@ts-ignore
|
||||
body.context.app.version = app.version as string;
|
||||
}
|
||||
if (app.build) {
|
||||
//@ts-ignore
|
||||
body.context.app.build = app.build as string;
|
||||
}
|
||||
}
|
||||
}
|
||||
if (context.campaign) {
|
||||
const campaign = (context.campaign as IDataObject).campaignUi as IDataObject;
|
||||
if (campaign) {
|
||||
if (campaign.name) {
|
||||
//@ts-ignore
|
||||
body.context.campaign.name = campaign.name as string;
|
||||
}
|
||||
if (campaign.source) {
|
||||
//@ts-ignore
|
||||
body.context.campaign.source = campaign.source as string;
|
||||
}
|
||||
if (campaign.medium) {
|
||||
//@ts-ignore
|
||||
body.context.campaign.medium = campaign.medium as string;
|
||||
}
|
||||
if (campaign.term) {
|
||||
//@ts-ignore
|
||||
body.context.campaign.term = campaign.term as string;
|
||||
}
|
||||
if (campaign.content) {
|
||||
//@ts-ignore
|
||||
body.context.campaign.content = campaign.content as string;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
if (context.device) {
|
||||
const device = (context.device as IDataObject).deviceUi as IDataObject;
|
||||
if (device) {
|
||||
if (device.id) {
|
||||
//@ts-ignore
|
||||
body.context.device.id = device.id as string;
|
||||
}
|
||||
if (device.manufacturer) {
|
||||
//@ts-ignore
|
||||
body.context.device.manufacturer = device.manufacturer as string;
|
||||
}
|
||||
if (device.model) {
|
||||
//@ts-ignore
|
||||
body.context.device.model = device.model as string;
|
||||
}
|
||||
if (device.type) {
|
||||
//@ts-ignore
|
||||
body.context.device.type = device.type as string;
|
||||
}
|
||||
if (device.version) {
|
||||
//@ts-ignore
|
||||
body.context.device.version = device.version as string;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
if (integrations) {
|
||||
if (integrations.all) {
|
||||
body.integrations!.all = integrations.all as boolean;
|
||||
}
|
||||
if (integrations.salesforce) {
|
||||
body.integrations!.salesforce = integrations.salesforce as boolean;
|
||||
}
|
||||
}
|
||||
if (properties) {
|
||||
if (properties.name) {
|
||||
body.properties!.name = properties.name as number;
|
||||
}
|
||||
if (properties.path) {
|
||||
body.properties!.path = properties.path as string;
|
||||
}
|
||||
if (properties.referrer) {
|
||||
body.properties!.referrer = properties.referrer as string;
|
||||
}
|
||||
if (properties.search) {
|
||||
body.properties!.search = properties.search as string;
|
||||
}
|
||||
if (properties.title) {
|
||||
body.properties!.title = properties.title as string;
|
||||
}
|
||||
if (properties.url) {
|
||||
body.properties!.url = properties.url as string;
|
||||
}
|
||||
if (properties.keywords) {
|
||||
body.properties!.keywords = properties.keywords as string;
|
||||
}
|
||||
}
|
||||
responseData = await segmentApiRequest.call(this, 'POST', '/page', body);
|
||||
}
|
||||
}
|
||||
if (Array.isArray(responseData)) {
|
||||
returnData.push.apply(returnData, responseData as IDataObject[]);
|
||||
} else {
|
||||
returnData.push(responseData as IDataObject);
|
||||
}
|
||||
}
|
||||
return [this.helpers.returnJsonArray(returnData)];
|
||||
}
|
||||
}
|
1155
packages/nodes-base/nodes/Segment/TrackDescription.ts
Normal file
1155
packages/nodes-base/nodes/Segment/TrackDescription.ts
Normal file
File diff suppressed because it is too large
Load diff
13
packages/nodes-base/nodes/Segment/TrackInterface.ts
Normal file
13
packages/nodes-base/nodes/Segment/TrackInterface.ts
Normal file
|
@ -0,0 +1,13 @@
|
|||
import { IDataObject } from "n8n-workflow";
|
||||
|
||||
export interface ITrack {
|
||||
event?: string;
|
||||
userId?: string;
|
||||
name?: string;
|
||||
anonymousId?: string;
|
||||
traits?: IDataObject;
|
||||
context?: IDataObject;
|
||||
timestamp?: string;
|
||||
properties?: IDataObject;
|
||||
integrations?: IDataObject;
|
||||
}
|
BIN
packages/nodes-base/nodes/Segment/segment.png
Normal file
BIN
packages/nodes-base/nodes/Segment/segment.png
Normal file
Binary file not shown.
After Width: | Height: | Size: 1.1 KiB |
|
@ -75,6 +75,7 @@
|
|||
"dist/credentials/Smtp.credentials.js",
|
||||
"dist/credentials/StripeApi.credentials.js",
|
||||
"dist/credentials/SalesmateApi.credentials.js",
|
||||
"dist/credentials/SegmentApi.credentials.js",
|
||||
"dist/credentials/TelegramApi.credentials.js",
|
||||
"dist/credentials/TodoistApi.credentials.js",
|
||||
"dist/credentials/TrelloApi.credentials.js",
|
||||
|
@ -174,6 +175,7 @@
|
|||
"dist/nodes/Stripe/StripeTrigger.node.js",
|
||||
"dist/nodes/Switch.node.js",
|
||||
"dist/nodes/Salesmate/Salesmate.node.js",
|
||||
"dist/nodes/Segment/Segment.node.js",
|
||||
"dist/nodes/Telegram/Telegram.node.js",
|
||||
"dist/nodes/Telegram/TelegramTrigger.node.js",
|
||||
"dist/nodes/Todoist/Todoist.node.js",
|
||||
|
@ -208,6 +210,7 @@
|
|||
"@types/nodemailer": "^4.6.5",
|
||||
"@types/redis": "^2.8.11",
|
||||
"@types/request-promise-native": "~1.0.15",
|
||||
"@types/uuid": "^3.4.6",
|
||||
"@types/xml2js": "^0.4.3",
|
||||
"gulp": "^4.0.0",
|
||||
"jest": "^24.9.0",
|
||||
|
@ -239,6 +242,7 @@
|
|||
"redis": "^2.8.0",
|
||||
"rhea": "^1.0.11",
|
||||
"rss-parser": "^3.7.0",
|
||||
"uuid": "^3.4.0",
|
||||
"vm2": "^3.6.10",
|
||||
"xlsx": "^0.14.3",
|
||||
"xml2js": "^0.4.22"
|
||||
|
|
Loading…
Reference in a new issue