mirror of
https://github.com/n8n-io/n8n.git
synced 2025-01-14 14:28:14 -08:00
140 lines
3.4 KiB
TypeScript
140 lines
3.4 KiB
TypeScript
|
import type {
|
||
|
IDataObject,
|
||
|
IExecuteFunctions,
|
||
|
INodeExecutionData,
|
||
|
INodeProperties,
|
||
|
} from 'n8n-workflow';
|
||
|
import { NodeOperationError } from 'n8n-workflow';
|
||
|
import set from 'lodash/set';
|
||
|
import { theHiveApiRequest } from '../../transport';
|
||
|
import { fixFieldType, prepareInputItem } from '../../helpers/utils';
|
||
|
import { updateDisplayOptions, wrapData } from '@utils/utilities';
|
||
|
|
||
|
const properties: INodeProperties[] = [
|
||
|
{
|
||
|
displayName: 'Fields',
|
||
|
name: 'observableUpdateFields',
|
||
|
type: 'resourceMapper',
|
||
|
default: {
|
||
|
mappingMode: 'defineBelow',
|
||
|
value: null,
|
||
|
},
|
||
|
noDataExpression: true,
|
||
|
required: true,
|
||
|
typeOptions: {
|
||
|
resourceMapper: {
|
||
|
resourceMapperMethod: 'getObservableUpdateFields',
|
||
|
mode: 'update',
|
||
|
valuesLabel: 'Fields',
|
||
|
addAllFields: true,
|
||
|
multiKeyMatch: true,
|
||
|
},
|
||
|
},
|
||
|
},
|
||
|
];
|
||
|
|
||
|
const displayOptions = {
|
||
|
show: {
|
||
|
resource: ['observable'],
|
||
|
operation: ['update'],
|
||
|
},
|
||
|
};
|
||
|
|
||
|
export const description = updateDisplayOptions(displayOptions, properties);
|
||
|
|
||
|
export async function execute(
|
||
|
this: IExecuteFunctions,
|
||
|
i: number,
|
||
|
item: INodeExecutionData,
|
||
|
): Promise<INodeExecutionData[]> {
|
||
|
let body: IDataObject = {};
|
||
|
let updated = 1;
|
||
|
|
||
|
const dataMode = this.getNodeParameter('observableUpdateFields.mappingMode', i) as string;
|
||
|
|
||
|
if (dataMode === 'autoMapInputData') {
|
||
|
const schema = this.getNodeParameter('observableUpdateFields.schema', i) as IDataObject[];
|
||
|
body = prepareInputItem(item.json, schema, i);
|
||
|
}
|
||
|
|
||
|
if (dataMode === 'defineBelow') {
|
||
|
const observableUpdateFields = this.getNodeParameter(
|
||
|
'observableUpdateFields.value',
|
||
|
i,
|
||
|
[],
|
||
|
) as IDataObject;
|
||
|
body = observableUpdateFields;
|
||
|
}
|
||
|
|
||
|
body = fixFieldType(body);
|
||
|
|
||
|
const fieldsToMatchOn = this.getNodeParameter(
|
||
|
'observableUpdateFields.matchingColumns',
|
||
|
i,
|
||
|
) as string[];
|
||
|
|
||
|
const updateBody: IDataObject = {};
|
||
|
const matchFields: IDataObject = {};
|
||
|
const { id } = body; // id would be used if matching on id, also we need to remove it from the body
|
||
|
|
||
|
for (const field of Object.keys(body)) {
|
||
|
if (fieldsToMatchOn.includes(field)) {
|
||
|
// if field is in fieldsToMatchOn, we need to exclude it from the updateBody, as values used for matching should not be updated
|
||
|
matchFields[field] = body[field];
|
||
|
} else {
|
||
|
// use set to construct the updateBody, as it allows to process customFields.fieldName
|
||
|
// if customFields provided under customFields property, it will be send as is
|
||
|
set(updateBody, field, body[field]);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if (fieldsToMatchOn.includes('id')) {
|
||
|
await theHiveApiRequest.call(this, 'PATCH', `/v1/observable/${id}`, body);
|
||
|
} else {
|
||
|
const filter = {
|
||
|
_name: 'filter',
|
||
|
_and: fieldsToMatchOn.map((field) => ({
|
||
|
_eq: {
|
||
|
_field: field,
|
||
|
_value: matchFields[field],
|
||
|
},
|
||
|
})),
|
||
|
};
|
||
|
|
||
|
const queryBody = {
|
||
|
query: [
|
||
|
{
|
||
|
_name: 'listObservable',
|
||
|
},
|
||
|
filter,
|
||
|
],
|
||
|
};
|
||
|
|
||
|
const matches = (await theHiveApiRequest.call(
|
||
|
this,
|
||
|
'POST',
|
||
|
'/v1/query',
|
||
|
queryBody,
|
||
|
)) as IDataObject[];
|
||
|
|
||
|
if (!matches.length) {
|
||
|
throw new NodeOperationError(this.getNode(), 'No matching alerts found');
|
||
|
}
|
||
|
const ids = matches.map((match) => match._id);
|
||
|
updated = ids.length;
|
||
|
|
||
|
updateBody.ids = ids;
|
||
|
|
||
|
await theHiveApiRequest.call(this, 'PATCH', '/v1/observable/_bulk', updateBody);
|
||
|
}
|
||
|
|
||
|
const executionData = this.helpers.constructExecutionMetaData(
|
||
|
wrapData({ success: true, updated }),
|
||
|
{
|
||
|
itemData: { item: i },
|
||
|
},
|
||
|
);
|
||
|
|
||
|
return executionData;
|
||
|
}
|