ToolJet/plugins/packages/dynamodb/lib/index.ts
Ganesh Kumar b26ecbee8b
Feature/dynamodb plugin update item (#6875)
* harperdb plugin initial commit

* added required fields in manifest and provided default values in operation json

* removed not required query options from harperdb

* modified nosql code

* review changes inprogress

* added new fields in manifest file to customize port and protocol

* protocol drop down in harperdb is changed to ssl toggle

* search-by-condition query in NoSQL

* harperdb NoSQL version tested along with search with condition query

* Dynamodb new operations added UpdateItem - DescribeTable - CreateTable

* dynamodb plugin new operation putItem added
2023-07-13 15:54:20 +05:30

120 lines
4.1 KiB
TypeScript

import { ConnectionTestResult, QueryService, QueryResult, QueryError } from '@tooljet-plugins/common';
import {
deleteItem,
getItem,
listTables,
queryTable,
scanTable,
describeTable,
updateItem,
createTable,
putItem,
} from './operations';
const AWS = require('aws-sdk');
import { AssumeRoleCredentials, SourceOptions, QueryOptions } from './types';
export default class DynamodbQueryService implements QueryService {
async run(sourceOptions: SourceOptions, queryOptions: QueryOptions): Promise<QueryResult> {
const operation = queryOptions.operation;
const client = await this.getConnection(sourceOptions, { operation });
let result = {};
try {
switch (operation) {
case 'list_tables':
result = await listTables(client);
break;
case 'get_item':
result = await getItem(client, queryOptions.table, JSON.parse(queryOptions.key));
break;
case 'delete_item':
result = await deleteItem(client, queryOptions.table, JSON.parse(queryOptions.key));
break;
case 'query_table':
result = await queryTable(client, JSON.parse(queryOptions.query_condition));
break;
case 'scan_table':
result = await scanTable(client, JSON.parse(queryOptions.scan_condition));
break;
case 'update_item':
result = await updateItem(client, JSON.parse(queryOptions.update_condition));
break;
case 'create_table':
result = await createTable(client, JSON.parse(queryOptions.table_parameters));
break;
case 'describe_table':
result = await describeTable(client, queryOptions.table);
break;
case 'put_item':
result = await putItem(client, JSON.parse(queryOptions.new_item_details));
break;
}
} catch (err) {
throw new QueryError('Query could not be completed', err.message, {});
}
return {
status: 'ok',
data: result,
};
}
async testConnection(sourceOptions: SourceOptions): Promise<ConnectionTestResult> {
const client = await this.getConnection(sourceOptions, { operation: 'list_tables' });
await listTables(client);
return {
status: 'ok',
};
}
async getAssumeRoleCredentials(roleArn: string): Promise<AssumeRoleCredentials> {
const sts = new AWS.STS();
return new Promise((resolve, reject) => {
const timestamp = new Date().getTime();
const roleName = roleArn.split('/')[1];
const params = {
RoleArn: roleArn,
RoleSessionName: `dynamodb-${roleName}-${timestamp}`,
};
sts.assumeRole(params, (err, data) => {
if (err) {
reject(err);
} else {
resolve({
accessKeyId: data.Credentials.AccessKeyId,
secretAccessKey: data.Credentials.SecretAccessKey,
sessionToken: data.Credentials.SessionToken,
});
}
});
});
}
async getConnection(sourceOptions: SourceOptions, options?: object): Promise<any> {
const useAWSInstanceProfile = sourceOptions['instance_metadata_credentials'] === 'aws_instance_credentials';
const region = sourceOptions['region'];
const useRoleArn = sourceOptions['instance_metadata_credentials'] === 'aws_arn_role';
let credentials = null;
if (useAWSInstanceProfile) {
credentials = new AWS.EC2MetadataCredentials({ httpOptions: { timeout: 5000 } });
} else if (useRoleArn) {
const assumeRoleCredentials = await this.getAssumeRoleCredentials(sourceOptions['role_arn']);
credentials = new AWS.Credentials(
assumeRoleCredentials.accessKeyId,
assumeRoleCredentials.secretAccessKey,
assumeRoleCredentials.sessionToken
);
} else {
credentials = new AWS.Credentials(sourceOptions['access_key'], sourceOptions['secret_key']);
}
if (['create_table', 'list_tables', 'describe_table'].includes(options['operation'])) {
return new AWS.DynamoDB({ region, credentials });
} else {
return new AWS.DynamoDB.DocumentClient({ region, credentials });
}
}
}