mirror of
https://github.com/Abdulazizzn/n8n-enterprise-unlocked.git
synced 2025-12-17 10:02:05 +00:00
✨ Add SecurityScorecard node (#1371)
* ✨Add SecurityScorecard node * Move portfolio:edit fields to the main view. The API request uses PUT so it will replace undefined fields if they are not set. * Style improvements * ⚡ Improvements to #1247 * ⚡ Improvements * ⚡ Minor improvements on SecurityScorecard Node Co-authored-by: Mika Luhta <12100880+mluhta@users.noreply.github.com> Co-authored-by: Jan Oberhauser <jan.oberhauser@gmail.com>
This commit is contained in:
@@ -0,0 +1,568 @@
|
||||
import {
|
||||
IExecuteFunctions
|
||||
} from 'n8n-core';
|
||||
|
||||
import {
|
||||
IDataObject,
|
||||
INodeExecutionData,
|
||||
INodeType,
|
||||
INodeTypeDescription,
|
||||
} from 'n8n-workflow';
|
||||
|
||||
import {
|
||||
companyFields,
|
||||
companyOperations,
|
||||
} from './descriptions/CompanyDescription';
|
||||
|
||||
import {
|
||||
industryFields,
|
||||
industryOperations,
|
||||
} from './descriptions/IndustryDescription';
|
||||
|
||||
import {
|
||||
inviteFields,
|
||||
inviteOperations,
|
||||
} from './descriptions/InviteDescription';
|
||||
|
||||
import {
|
||||
portfolioFields,
|
||||
portfolioOperations,
|
||||
} from './descriptions/PortfolioDescription';
|
||||
|
||||
import {
|
||||
portfolioCompanyFields,
|
||||
portfolioCompanyOperations,
|
||||
} from './descriptions/PortfolioCompanyDescription';
|
||||
|
||||
import {
|
||||
reportFields,
|
||||
reportOperations,
|
||||
} from './descriptions/ReportDescription';
|
||||
|
||||
import {
|
||||
scorecardApiRequest,
|
||||
simplify,
|
||||
} from './GenericFunctions';
|
||||
|
||||
import * as moment from 'moment';
|
||||
|
||||
export class SecurityScorecard implements INodeType {
|
||||
description: INodeTypeDescription = {
|
||||
displayName: 'SecurityScorecard',
|
||||
name: 'securityScorecard',
|
||||
icon: 'file:securityScorecard.svg',
|
||||
group: ['transform'],
|
||||
subtitle: '={{$parameter["operation"]}} : {{$parameter["resource"]}}',
|
||||
version: 1,
|
||||
description: 'Consume SecurityScorecard API',
|
||||
defaults: {
|
||||
name: 'SecurityScorecard',
|
||||
color: '#619e73',
|
||||
},
|
||||
inputs: ['main'],
|
||||
outputs: ['main'],
|
||||
credentials: [
|
||||
{
|
||||
name: 'securityScorecardApi',
|
||||
required: true,
|
||||
},
|
||||
],
|
||||
properties: [
|
||||
{
|
||||
displayName: 'Resource',
|
||||
name: 'resource',
|
||||
type: 'options',
|
||||
required: true,
|
||||
options: [
|
||||
{
|
||||
name: 'Company',
|
||||
value: 'company',
|
||||
},
|
||||
{
|
||||
name: 'Industry',
|
||||
value: 'industry',
|
||||
},
|
||||
{
|
||||
name: 'Invite',
|
||||
value: 'invite',
|
||||
},
|
||||
{
|
||||
name: 'Portfolio',
|
||||
value: 'portfolio',
|
||||
},
|
||||
{
|
||||
name: 'Portfolio Company',
|
||||
value: 'portfolioCompany',
|
||||
},
|
||||
{
|
||||
name: 'Report',
|
||||
value: 'report',
|
||||
},
|
||||
],
|
||||
default: 'company',
|
||||
},
|
||||
// Company
|
||||
...companyOperations,
|
||||
...companyFields,
|
||||
// Industry
|
||||
...industryOperations,
|
||||
...industryFields,
|
||||
// Invite
|
||||
...inviteOperations,
|
||||
...inviteFields,
|
||||
// Portfolio
|
||||
...portfolioOperations,
|
||||
...portfolioFields,
|
||||
// Portfolio Company
|
||||
...portfolioCompanyOperations,
|
||||
...portfolioCompanyFields,
|
||||
// Report
|
||||
...reportOperations,
|
||||
...reportFields,
|
||||
],
|
||||
};
|
||||
|
||||
async execute(this: IExecuteFunctions): Promise<INodeExecutionData[][]> {
|
||||
const items = this.getInputData();
|
||||
const returnData: IDataObject[] = [];
|
||||
let responseData;
|
||||
const length = (items.length as unknown) as number;
|
||||
|
||||
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 === 'portfolio') {
|
||||
|
||||
if (operation === 'create') {
|
||||
const name = this.getNodeParameter('name', i) as string;
|
||||
const description = this.getNodeParameter('description', i) as string;
|
||||
const privacy = this.getNodeParameter('privacy', i) as string;
|
||||
|
||||
const body: IDataObject = {
|
||||
name,
|
||||
description,
|
||||
privacy,
|
||||
};
|
||||
|
||||
responseData = await scorecardApiRequest.call(
|
||||
this,
|
||||
'POST',
|
||||
'portfolios',
|
||||
body,
|
||||
);
|
||||
returnData.push(responseData as IDataObject);
|
||||
}
|
||||
|
||||
if (operation === 'delete') {
|
||||
const portfolioId = this.getNodeParameter('portfolioId', i) as string;
|
||||
responseData = await scorecardApiRequest.call(
|
||||
this,
|
||||
'DELETE',
|
||||
`portfolios/${portfolioId}`,
|
||||
);
|
||||
returnData.push({ success: true });
|
||||
}
|
||||
|
||||
if (operation === 'update') {
|
||||
const portfolioId = this.getNodeParameter('portfolioId', i) as string;
|
||||
const name = this.getNodeParameter('name', i) as string;
|
||||
const description = this.getNodeParameter('description', i) as string;
|
||||
const privacy = this.getNodeParameter('privacy', i) as string;
|
||||
|
||||
const body: IDataObject = {
|
||||
name,
|
||||
description,
|
||||
privacy,
|
||||
};
|
||||
|
||||
responseData = await scorecardApiRequest.call(
|
||||
this,
|
||||
'PUT',
|
||||
`portfolios/${portfolioId}`,
|
||||
body,
|
||||
);
|
||||
returnData.push(responseData as IDataObject);
|
||||
}
|
||||
|
||||
if (operation === 'getAll') {
|
||||
const returnAll = this.getNodeParameter('returnAll', 0) as boolean;
|
||||
responseData = await scorecardApiRequest.call(
|
||||
this,
|
||||
'GET',
|
||||
'portfolios',
|
||||
);
|
||||
responseData = responseData.entries;
|
||||
|
||||
if (returnAll === false) {
|
||||
const limit = this.getNodeParameter('limit', 0) as number;
|
||||
responseData = responseData.splice(0, limit);
|
||||
}
|
||||
|
||||
returnData.push.apply(returnData, responseData as IDataObject[]);
|
||||
}
|
||||
}
|
||||
|
||||
if (resource === 'portfolioCompany') {
|
||||
if (operation === 'add') {
|
||||
const portfolioId = this.getNodeParameter('portfolioId', i) as string;
|
||||
const domain = this.getNodeParameter('domain', i);
|
||||
responseData = await scorecardApiRequest.call(
|
||||
this,
|
||||
'PUT',
|
||||
`portfolios/${portfolioId}/companies/${domain}`,
|
||||
);
|
||||
returnData.push(responseData as IDataObject);
|
||||
}
|
||||
|
||||
if (operation === 'remove') {
|
||||
const portfolioId = this.getNodeParameter('portfolioId', i) as string;
|
||||
const domain = this.getNodeParameter('domain', i);
|
||||
responseData = await scorecardApiRequest.call(
|
||||
this,
|
||||
'DELETE',
|
||||
`portfolios/${portfolioId}/companies/${domain}`,
|
||||
);
|
||||
returnData.push({ success: true });
|
||||
}
|
||||
|
||||
if (operation === 'getAll') {
|
||||
const returnAll = this.getNodeParameter('returnAll', 0) as boolean;
|
||||
const portfolioId = this.getNodeParameter('portfolioId', i) as string;
|
||||
const filterParams = this.getNodeParameter('filters', i) as IDataObject;
|
||||
responseData = await scorecardApiRequest.call(
|
||||
this,
|
||||
'GET',
|
||||
`portfolios/${portfolioId}/companies`,
|
||||
{},
|
||||
filterParams,
|
||||
);
|
||||
|
||||
responseData = responseData.entries;
|
||||
|
||||
if (returnAll === false) {
|
||||
const limit = this.getNodeParameter('limit', 0) as boolean;
|
||||
responseData = responseData.splice(0, limit);
|
||||
}
|
||||
|
||||
returnData.push.apply(returnData, responseData as IDataObject[]);
|
||||
}
|
||||
}
|
||||
|
||||
if (resource === 'report') {
|
||||
if (operation === 'download') {
|
||||
const reportUrl = this.getNodeParameter('url', i) as string;
|
||||
|
||||
const response = await scorecardApiRequest.call(
|
||||
this,
|
||||
'GET',
|
||||
'',
|
||||
{},
|
||||
{},
|
||||
reportUrl,
|
||||
{ encoding: null, resolveWithFullResponse: true });
|
||||
|
||||
let mimeType: string | undefined;
|
||||
if (response.headers['content-type']) {
|
||||
mimeType = response.headers['content-type'];
|
||||
}
|
||||
|
||||
const newItem: INodeExecutionData = {
|
||||
json: items[i].json,
|
||||
binary: {},
|
||||
};
|
||||
|
||||
if (items[i].binary !== undefined) {
|
||||
// Create a shallow copy of the binary data so that the old
|
||||
// data references which do not get changed still stay behind
|
||||
// but the incoming data does not get changed.
|
||||
Object.assign(newItem.binary, items[i].binary);
|
||||
}
|
||||
|
||||
items[i] = newItem;
|
||||
|
||||
const dataPropertyNameDownload = this.getNodeParameter('binaryPropertyName', i) as string;
|
||||
|
||||
const fileName = reportUrl.split('/').pop();
|
||||
|
||||
const data = Buffer.from(response.body as string, 'utf8');
|
||||
items[i].binary![dataPropertyNameDownload] = await this.helpers.prepareBinaryData(data as unknown as Buffer, fileName, mimeType);
|
||||
}
|
||||
|
||||
if (operation === 'generate') {
|
||||
const reportType = this.getNodeParameter('report', i) as string;
|
||||
let body: IDataObject = {};
|
||||
|
||||
if (reportType !== 'portfolio') {
|
||||
body['scorecard_identifier'] = this.getNodeParameter('scorecardIdentifier', i);
|
||||
} else {
|
||||
body['portfolio_id'] = this.getNodeParameter('portfolioId', i);
|
||||
}
|
||||
if (reportType === 'events-json') {
|
||||
body['date'] = this.getNodeParameter('date', i);
|
||||
}
|
||||
if (['issues', 'portfolio'].indexOf(reportType) > -1) {
|
||||
body['format'] = this.getNodeParameter('options.format', i) || 'pdf';
|
||||
}
|
||||
if (['detailed', 'summary'].indexOf(reportType) > -1) {
|
||||
body['branding'] = this.getNodeParameter('branding', i);
|
||||
}
|
||||
// json reports want the params differently
|
||||
if (['events-json', 'full-scorecard-json'].indexOf(reportType) > -1) {
|
||||
body = { params: body };
|
||||
}
|
||||
if (reportType === 'scorecard-footprint') {
|
||||
const options = this.getNodeParameter('options', i) as IDataObject;
|
||||
Object.assign(body, options);
|
||||
}
|
||||
|
||||
responseData = await scorecardApiRequest.call(
|
||||
this,
|
||||
'POST',
|
||||
`reports/${reportType}`,
|
||||
body,
|
||||
);
|
||||
|
||||
returnData.push(responseData as IDataObject);
|
||||
}
|
||||
|
||||
if (operation === 'getAll') {
|
||||
const returnAll = this.getNodeParameter('returnAll', 0) as boolean;
|
||||
responseData = await scorecardApiRequest.call(
|
||||
this,
|
||||
'GET',
|
||||
'reports/recent',
|
||||
);
|
||||
responseData = responseData.entries;
|
||||
|
||||
if (returnAll === false) {
|
||||
const limit = this.getNodeParameter('limit', i) as number;
|
||||
responseData = responseData.splice(0, limit);
|
||||
}
|
||||
returnData.push.apply(returnData, responseData as IDataObject[]);
|
||||
}
|
||||
}
|
||||
|
||||
if (resource === 'invite') {
|
||||
if (operation === 'create') {
|
||||
const body: IDataObject = {
|
||||
email: this.getNodeParameter('email', i),
|
||||
first_name: this.getNodeParameter('firstName', i),
|
||||
last_name: this.getNodeParameter('lastName', i),
|
||||
message: this.getNodeParameter('message', i),
|
||||
};
|
||||
const additionalFields = this.getNodeParameter('additionalFields', i);
|
||||
Object.assign(body, additionalFields);
|
||||
|
||||
responseData = await scorecardApiRequest.call(
|
||||
this,
|
||||
'POST',
|
||||
`invitations`,
|
||||
body,
|
||||
);
|
||||
returnData.push(responseData as IDataObject);
|
||||
}
|
||||
}
|
||||
|
||||
if (resource === 'industry') {
|
||||
if (operation === 'getScore') {
|
||||
const industry = this.getNodeParameter('industry', i);
|
||||
responseData = await scorecardApiRequest.call(
|
||||
this,
|
||||
'GET',
|
||||
`industries/${industry}/score`,
|
||||
);
|
||||
returnData.push(responseData as IDataObject);
|
||||
}
|
||||
|
||||
if (operation === 'getFactor') {
|
||||
const simple = this.getNodeParameter('simple', 0) as boolean;
|
||||
const returnAll = this.getNodeParameter('returnAll', 0) as boolean;
|
||||
const industry = this.getNodeParameter('industry', i);
|
||||
responseData = await scorecardApiRequest.call(
|
||||
this,
|
||||
'GET',
|
||||
`industries/${industry}/history/factors`,
|
||||
);
|
||||
responseData = responseData.entries;
|
||||
|
||||
if (returnAll === false) {
|
||||
const limit = this.getNodeParameter('limit', i) as number;
|
||||
responseData = responseData.splice(0, limit);
|
||||
}
|
||||
|
||||
if (simple === true) {
|
||||
responseData = simplify(responseData);
|
||||
}
|
||||
|
||||
returnData.push.apply(returnData, responseData as IDataObject[]);
|
||||
}
|
||||
|
||||
if (operation === 'getFactorHistorical') {
|
||||
const simple = this.getNodeParameter('simple', 0) as boolean;
|
||||
const returnAll = this.getNodeParameter('returnAll', i) as boolean;
|
||||
const industry = this.getNodeParameter('industry', i);
|
||||
const options = this.getNodeParameter('options', i) as IDataObject;
|
||||
// Convert to YYYY-MM-DD
|
||||
if (options['from']) {
|
||||
options['from'] = moment(options['from'] as Date).format('YYYY-MM-DD');
|
||||
}
|
||||
|
||||
if (options['to']) {
|
||||
options['to'] = moment(options['to'] as Date).format('YYYY-MM-DD');
|
||||
}
|
||||
responseData = await scorecardApiRequest.call(
|
||||
this,
|
||||
'GET',
|
||||
`industries/${industry}/history/factors`,
|
||||
{},
|
||||
options,
|
||||
);
|
||||
responseData = responseData.entries;
|
||||
|
||||
if (returnAll === false) {
|
||||
const limit = this.getNodeParameter('limit', i) as number;
|
||||
responseData = responseData.splice(0, limit);
|
||||
}
|
||||
|
||||
if (simple === true) {
|
||||
responseData = simplify(responseData);
|
||||
}
|
||||
|
||||
returnData.push.apply(returnData, responseData as IDataObject[]);
|
||||
}
|
||||
}
|
||||
|
||||
if (resource === 'company') {
|
||||
if (operation === 'getScorecard') {
|
||||
const scorecardIdentifier = this.getNodeParameter('scorecardIdentifier', i) as string;
|
||||
responseData = await scorecardApiRequest.call(
|
||||
this,
|
||||
'GET',
|
||||
`companies/${scorecardIdentifier}`,
|
||||
);
|
||||
returnData.push(responseData as IDataObject);
|
||||
}
|
||||
|
||||
if (operation === 'getFactor') {
|
||||
const returnAll = this.getNodeParameter('returnAll', i) as boolean;
|
||||
const scorecardIdentifier = this.getNodeParameter('scorecardIdentifier', i);
|
||||
const filterParams = this.getNodeParameter('filters', i) as IDataObject;
|
||||
responseData = await scorecardApiRequest.call(
|
||||
this,
|
||||
'GET',
|
||||
`companies/${scorecardIdentifier}/factors`,
|
||||
{},
|
||||
filterParams,
|
||||
);
|
||||
|
||||
responseData = responseData.entries;
|
||||
|
||||
if (returnAll === false) {
|
||||
const limit = this.getNodeParameter('limit', i) as number;
|
||||
responseData = responseData.splice(0, limit);
|
||||
}
|
||||
|
||||
returnData.push.apply(returnData, responseData as IDataObject[]);
|
||||
}
|
||||
|
||||
if (operation === 'getFactorHistorical') {
|
||||
const simple = this.getNodeParameter('simple', 0) as boolean;
|
||||
const returnAll = this.getNodeParameter('returnAll', i) as boolean;
|
||||
const scorecardIdentifier = this.getNodeParameter('scorecardIdentifier', i) as string;
|
||||
const options = this.getNodeParameter('options', i) as IDataObject;
|
||||
// Convert to YYYY-MM-DD
|
||||
if (options['date_from']) {
|
||||
options['date_from'] = moment(options['date_from'] as Date).format('YYYY-MM-DD');
|
||||
}
|
||||
|
||||
if (options['date_to']) {
|
||||
options['date_to'] = moment(options['date_to'] as Date).format('YYYY-MM-DD');
|
||||
}
|
||||
responseData = await scorecardApiRequest.call(
|
||||
this,
|
||||
'GET',
|
||||
`companies/${scorecardIdentifier}/history/factors/score`,
|
||||
{},
|
||||
options,
|
||||
);
|
||||
|
||||
responseData = responseData.entries;
|
||||
|
||||
if (returnAll === false) {
|
||||
const limit = this.getNodeParameter('limit', i) as number;
|
||||
responseData = responseData.splice(0, limit);
|
||||
}
|
||||
|
||||
if (simple === true) {
|
||||
responseData = simplify(responseData);
|
||||
}
|
||||
|
||||
returnData.push.apply(returnData, responseData as IDataObject[]);
|
||||
}
|
||||
|
||||
if (operation === 'getHistoricalScore') {
|
||||
const simple = this.getNodeParameter('simple', 0) as boolean;
|
||||
const returnAll = this.getNodeParameter('returnAll', i) as boolean;
|
||||
const scorecardIdentifier = this.getNodeParameter('scorecardIdentifier', i);
|
||||
const options = this.getNodeParameter('options', i) as IDataObject;
|
||||
|
||||
// for some reason the params are different between these two APis :/
|
||||
if (options['date_from']) {
|
||||
options['from'] = moment(options['date_from'] as Date).format('YYYY-MM-DD');
|
||||
delete options['date_from'];
|
||||
}
|
||||
if (options['date_to']) {
|
||||
options['to'] = moment(options['date_to'] as Date).format('YYYY-MM-DD');
|
||||
delete options['date_to'];
|
||||
}
|
||||
responseData = await scorecardApiRequest.call(
|
||||
this,
|
||||
'GET',
|
||||
`companies/${scorecardIdentifier}/history/factors/score`,
|
||||
{},
|
||||
options,
|
||||
);
|
||||
responseData = responseData.entries;
|
||||
|
||||
if (returnAll === false) {
|
||||
const limit = this.getNodeParameter('limit', i) as number;
|
||||
responseData = responseData.splice(0, limit);
|
||||
}
|
||||
|
||||
if (simple === true) {
|
||||
responseData = simplify(responseData);
|
||||
}
|
||||
|
||||
returnData.push.apply(returnData, responseData as IDataObject[]);
|
||||
}
|
||||
|
||||
if (operation === 'getScorePlan') {
|
||||
const returnAll = this.getNodeParameter('returnAll', i) as boolean;
|
||||
const scorecardIdentifier = this.getNodeParameter('scorecardIdentifier', i) as string;
|
||||
const targetScore = this.getNodeParameter('score', i);
|
||||
responseData = await scorecardApiRequest.call(
|
||||
this,
|
||||
'GET',
|
||||
`companies/${scorecardIdentifier}/score-plans/by-target/${targetScore}`,
|
||||
);
|
||||
|
||||
responseData = responseData.entries;
|
||||
|
||||
if (returnAll === false) {
|
||||
const limit = this.getNodeParameter('limit', i) as number;
|
||||
responseData = responseData.splice(0, limit);
|
||||
}
|
||||
|
||||
returnData.push.apply(returnData, responseData as IDataObject[]);
|
||||
}
|
||||
}
|
||||
}
|
||||
// Handle file download output data differently
|
||||
if (resource === 'report' && operation === 'download') {
|
||||
return this.prepareOutputData(items);
|
||||
}
|
||||
return [this.helpers.returnJsonArray(returnData)];
|
||||
}
|
||||
}
|
||||
Reference in New Issue
Block a user