mirror of
https://github.com/n8n-io/n8n.git
synced 2025-01-15 06:47:29 -08:00
e4785da2e1
fix(AWS DynamoDB Node): add type assertion
135 lines
3.3 KiB
TypeScript
135 lines
3.3 KiB
TypeScript
import { deepCopy, IDataObject, INodeExecutionData, assert } from 'n8n-workflow';
|
|
|
|
import {
|
|
AdjustedPutItem,
|
|
AttributeValueType,
|
|
EAttributeValueType,
|
|
IAttributeNameUi,
|
|
IAttributeValue,
|
|
IAttributeValueUi,
|
|
IAttributeValueValue,
|
|
PutItemUi,
|
|
} from './types';
|
|
|
|
const addColon = (attribute: string) =>
|
|
(attribute = attribute.charAt(0) === ':' ? attribute : `:${attribute}`);
|
|
|
|
const addPound = (key: string) => (key = key.charAt(0) === '#' ? key : `#${key}`);
|
|
|
|
export function adjustExpressionAttributeValues(eavUi: IAttributeValueUi[]) {
|
|
const eav: IAttributeValue = {};
|
|
|
|
eavUi.forEach(({ attribute, type, value }) => {
|
|
eav[addColon(attribute)] = { [type]: value } as IAttributeValueValue;
|
|
});
|
|
|
|
return eav;
|
|
}
|
|
|
|
export function adjustExpressionAttributeName(eanUi: IAttributeNameUi[]) {
|
|
const ean: { [key: string]: string } = {};
|
|
|
|
eanUi.forEach(({ key, value }) => {
|
|
ean[addPound(key)] = value;
|
|
});
|
|
|
|
return ean;
|
|
}
|
|
|
|
export function adjustPutItem(putItemUi: PutItemUi) {
|
|
const adjustedPutItem: AdjustedPutItem = {};
|
|
|
|
Object.entries(putItemUi).forEach(([attribute, value]) => {
|
|
let type: string;
|
|
|
|
if (typeof value === 'boolean') {
|
|
type = 'BOOL';
|
|
} else if (typeof value === 'object' && !Array.isArray(value) && value !== null) {
|
|
type = 'M';
|
|
// @ts-ignore
|
|
} else if (isNaN(value)) {
|
|
type = 'S';
|
|
} else {
|
|
type = 'N';
|
|
}
|
|
|
|
adjustedPutItem[attribute] = { [type]: value.toString() };
|
|
});
|
|
|
|
return adjustedPutItem;
|
|
}
|
|
|
|
export function simplify(item: IAttributeValue): IDataObject {
|
|
const output: IDataObject = {};
|
|
|
|
for (const [attribute, value] of Object.entries(item)) {
|
|
const [type, content] = Object.entries(value)[0] as [AttributeValueType, string];
|
|
output[attribute] = decodeAttribute(type, content);
|
|
}
|
|
|
|
return output;
|
|
}
|
|
|
|
function decodeAttribute(type: AttributeValueType, attribute: string | IAttributeValue) {
|
|
switch (type) {
|
|
case 'BOOL':
|
|
return Boolean(attribute);
|
|
case 'N':
|
|
return Number(attribute);
|
|
case 'S':
|
|
return String(attribute);
|
|
case 'SS':
|
|
case 'NS':
|
|
return attribute;
|
|
case 'M':
|
|
assert(
|
|
typeof attribute === 'object' && !Array.isArray(attribute) && attribute !== null,
|
|
'Attribute must be an object',
|
|
);
|
|
return simplify(attribute);
|
|
default:
|
|
return null;
|
|
}
|
|
}
|
|
|
|
export function validateJSON(input: any): object {
|
|
try {
|
|
return JSON.parse(input);
|
|
} catch (error) {
|
|
throw new Error('Items must be a valid JSON');
|
|
}
|
|
}
|
|
|
|
export function copyInputItem(item: INodeExecutionData, properties: string[]): IDataObject {
|
|
// Prepare the data to insert and copy it to be returned
|
|
const newItem: IDataObject = {};
|
|
for (const property of properties) {
|
|
if (item.json[property] === undefined) {
|
|
newItem[property] = null;
|
|
} else {
|
|
newItem[property] = deepCopy(item.json[property]);
|
|
}
|
|
}
|
|
return newItem;
|
|
}
|
|
|
|
export function mapToAttributeValues(item: IDataObject): void {
|
|
for (const key of Object.keys(item)) {
|
|
if (!key.startsWith(':')) {
|
|
item[`:${key}`] = item[key];
|
|
delete item[key];
|
|
}
|
|
}
|
|
}
|
|
|
|
export function decodeItem(item: IAttributeValue): IDataObject {
|
|
const _item: IDataObject = {};
|
|
for (const entry of Object.entries(item)) {
|
|
const [attribute, value]: [string, object] = entry;
|
|
const [type, content]: [string, object] = Object.entries(value)[0];
|
|
_item[attribute] = decodeAttribute(type as EAttributeValueType, content as unknown as string);
|
|
}
|
|
|
|
return _item;
|
|
}
|