UNPKG

27.6 kBJavaScriptView Raw
1"use strict";
2
3Object.defineProperty(exports, "__esModule", {
4 value: true
5});
6exports.default = void 0;
7
8var _awsSdk = _interopRequireDefault(require("aws-sdk"));
9
10var _System = _interopRequireDefault(require("./System"));
11
12var _Module = _interopRequireDefault(require("./Module"));
13
14var _errorUtils = require("./errorUtils");
15
16function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
17
18function _objectSpread(target) { for (var i = 1; i < arguments.length; i++) { var source = arguments[i] != null ? arguments[i] : {}; var ownKeys = Object.keys(source); if (typeof Object.getOwnPropertySymbols === 'function') { ownKeys = ownKeys.concat(Object.getOwnPropertySymbols(source).filter(function (sym) { return Object.getOwnPropertyDescriptor(source, sym).enumerable; })); } ownKeys.forEach(function (key) { _defineProperty(target, key, source[key]); }); } return target; }
19
20function _defineProperty(obj, key, value) { if (key in obj) { Object.defineProperty(obj, key, { value: value, enumerable: true, configurable: true, writable: true }); } else { obj[key] = value; } return obj; }
21
22const {
23 MODULE_NAME,
24 debug,
25 log,
26 warn,
27 error,
28 noteGauge,
29 noteCount,
30 noteTimer,
31 trackOp
32} = new _Module.default(__filename); // eslint-disable-line no-unused-vars
33
34class AWSError extends Error {
35 constructor(err, opName, data) {
36 // $FlowIgnore
37 const message = `AWS Op '${opName}' threw '${err.code}': ${err.message}`;
38 super(message);
39
40 _defineProperty(this, "message", void 0);
41
42 _defineProperty(this, "code", void 0);
43
44 _defineProperty(this, "time", void 0);
45
46 _defineProperty(this, "statusCode", void 0);
47
48 _defineProperty(this, "retryable", void 0);
49
50 _defineProperty(this, "data", void 0);
51
52 this.message = message; // $FlowIgnore
53
54 this.code = err.code; // $FlowIgnore
55
56 this.time = err.time; // $FlowIgnore
57
58 this.statusCode = err.statusCode; // $FlowIgnore
59
60 this.retryable = err.retryable;
61 this.data = data;
62 }
63
64}
65
66class AWSClient {
67 constructor() {
68 const awsConfig = _System.default.getConfig().aws;
69
70 if (!awsConfig) throw new Error('aws system config is unset');
71 _awsSdk.default.config.region = awsConfig.defaultRegion; // support local execution
72
73 if (awsConfig.profile) {
74 log('Using profile', {
75 profile: awsConfig.profile
76 });
77 _awsSdk.default.config.credentials = new _awsSdk.default.SharedIniFileCredentials({
78 profile: awsConfig.profile
79 });
80 }
81 }
82
83 async runSES(func, options) {
84 return await this._runOp('SES', clientConfig => new _awsSdk.default.SES(clientConfig), func, options);
85 }
86
87 async runS3(func, options) {
88 return await this._runOp('S3', clientConfig => new _awsSdk.default.S3(clientConfig), func, options);
89 }
90
91 async runDocumentClient(func, options) {
92 return await this._runOp('DocumentClient', clientConfig => new _awsSdk.default.DynamoDB.DocumentClient(_objectSpread({
93 convertEmptyValues: true
94 }, clientConfig)), func, options);
95 }
96
97 async runDynamoDB(func, options) {
98 return await this._runOp('DynamoDB', clientConfig => new _awsSdk.default.DynamoDB(clientConfig), func, options);
99 }
100
101 async runApplicationAutoScaling(func, options) {
102 return await this._runOp('ApplicationAutoScaling', clientConfig => new _awsSdk.default.ApplicationAutoScaling(clientConfig), func, options);
103 }
104
105 async runCloudFront(func, options) {
106 return await this._runOp('CloudFront', clientConfig => new _awsSdk.default.CloudFront(clientConfig), func, options);
107 }
108
109 async runAthena(func, options) {
110 return await this._runOp('Athena', clientConfig => new _awsSdk.default.Athena(clientConfig), func, options);
111 }
112
113 async runSQS(func, options) {
114 return await this._runOp('SQS', clientConfig => new _awsSdk.default.SQS(clientConfig), func, options);
115 }
116
117 async runCloudWatch(func, options) {
118 return await this._runOp('CloudWatch', clientConfig => new _awsSdk.default.CloudWatch(clientConfig), func, options);
119 }
120
121 async runFirehose(func, options) {
122 return await this._runOp('Firehose', clientConfig => new _awsSdk.default.Firehose(clientConfig), func, options);
123 }
124
125 async runES(func, options) {
126 return await this._runOp('Firehose', clientConfig => new _awsSdk.default.ES(clientConfig), func, options);
127 }
128
129 async _runOp(serviceName, serviceGenerator, func, options) {
130 const awsConfig = _System.default.getConfig().aws;
131
132 if (!awsConfig) throw new Error('aws config not set in system config');
133 const region = awsConfig.defaultRegion || (options || {}).region || (0, _errorUtils.throwUnsetArg)('aws region');
134 options = options || {};
135 options.errorHandlers = options.errorHandlers || {};
136 const credentials = _awsSdk.default.config.credentials;
137 let start = 0;
138 let opName = 'unknown-operation';
139
140 try {
141 // TODO: recreate only when creds expire?
142 const service = serviceGenerator({
143 credentials: credentials,
144 region: region
145 });
146 const req = func(service);
147 if (req === undefined) throw new Error(`AWS runXXX method call on service '${serviceName}' is missing, did you user 'return'?`); // $FlowIgnore
148
149 opName = options.opName || req['operation'];
150 if (!options.noLog) debug(`AWS operation started...`, {
151 serviceName,
152 opName
153 });
154 let result;
155 let success = false;
156 let code;
157
158 do {
159 try {
160 start = Date.now();
161 result = await req.promise();
162 success = true;
163 } catch (err) {
164 // $FlowIgnore
165 const errorHandler = options.errorHandlers[err.code];
166 if (!errorHandler) throw err;
167
168 switch (errorHandler.type) {
169 case 'retry':
170 try {
171 errorHandler.tries = errorHandler.tries || 1;
172
173 if (errorHandler.tries-- > 0 && errorHandler.action) {
174 await errorHandler.action(err);
175 }
176 } catch (err2) {
177 err2.leadingError = err;
178 throw err2;
179 }
180
181 break;
182
183 case 'ignore':
184 code = err.code;
185 success = true;
186 log(`${serviceName} op '${opName}' completed with an ignorable error code`, {
187 code: code,
188 ms: Date.now() - start
189 });
190 break;
191
192 default:
193 throw new Error(`Unexpected errorHandler type: '${errorHandler.type}'`);
194 }
195 }
196 } while (!success);
197
198 if (!options.noLog) debug(`AWS operation completed`, {
199 serviceName,
200 opName,
201 ms: Date.now() - start
202 });
203 noteCount(`${MODULE_NAME}.${serviceName}.${opName}.success`, 1); // $FlowIgnore
204
205 return {
206 result: result,
207 code: code
208 };
209 } catch (err) {
210 log(`${serviceName} op '${opName}' failed`, {
211 err,
212 args: options.args
213 });
214 noteCount(`${MODULE_NAME}.${serviceName}.${opName}.fail`, 1);
215 throw new AWSError(err, `${serviceName}.${opName}`, {
216 args: options.args
217 });
218 } finally {
219 noteTimer(`${MODULE_NAME}.${serviceName}.${opName}`, Date.now() - start);
220 }
221 }
222
223}
224
225exports.default = AWSClient;
226//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../src/AWSClient.js"],"names":["MODULE_NAME","debug","log","warn","error","noteGauge","noteCount","noteTimer","trackOp","Module","__filename","AWSError","Error","constructor","err","opName","data","message","code","time","statusCode","retryable","AWSClient","awsConfig","System","getConfig","aws","AWS","config","region","defaultRegion","profile","credentials","SharedIniFileCredentials","runSES","func","options","_runOp","clientConfig","SES","runS3","S3","runDocumentClient","DynamoDB","DocumentClient","convertEmptyValues","runDynamoDB","runApplicationAutoScaling","ApplicationAutoScaling","runCloudFront","CloudFront","runAthena","Athena","runSQS","SQS","runCloudWatch","CloudWatch","runFirehose","Firehose","runES","ES","serviceName","serviceGenerator","errorHandlers","start","service","req","undefined","noLog","result","success","Date","now","promise","errorHandler","type","tries","action","err2","leadingError","ms","args"],"mappings":"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","sourcesContent":["//@flow\n\nimport AWS from 'aws-sdk'\n\nimport System from './System'\nimport Module from './Module'\nimport { throwUnsetArg } from './errorUtils'\n\nconst { MODULE_NAME, debug, log, warn, error, noteGauge, noteCount, noteTimer, trackOp } = new Module(__filename) // eslint-disable-line no-unused-vars\n\nclass AWSError extends Error {\n\n  message: string\n  code: string\n  time: string\n  statusCode: number\n  retryable: boolean\n  data: ?any\n\n  \n  constructor(err: Error, opName: string, data: ?any) {\n    // $FlowIgnore    \n    const message = `AWS Op '${opName}' threw '${err.code}': ${err.message}`\n\n    super(message)\n\n    this.message = message\n    // $FlowIgnore\n    this.code = err.code\n    // $FlowIgnore\n    this.time = err.time\n    // $FlowIgnore\n    this.statusCode = err.statusCode\n    // $FlowIgnore\n    this.retryable = err.retryable\n\n    this.data = data\n  }\n}\n\ntype ClientConfig = {\n  region: string,\n}\n\ntype AWSOpOptions = {\n  args?: ?Array<any>,\n  opName?: ?string,\n  region?: ?string,\n  errorHandlers?: ?{ \n    [code: string]: {\n      type: 'ignore' | 'retry',\n      tries?: ?number,\n      action?: ?(err: AWSError) => Promise<void>,\n    }\n  },\n  noLog?: ?boolean,\n}\n\nexport default class AWSClient {\n\n  constructor() {\n    const awsConfig = System.getConfig().aws\n    if (!awsConfig)\n      throw new Error('aws system config is unset')\n\n    AWS.config.region = awsConfig.defaultRegion\n    \n    // support local execution\n    if (awsConfig.profile) {\n      log('Using profile', { profile: awsConfig.profile })\n      AWS.config.credentials = new AWS.SharedIniFileCredentials({ profile: awsConfig.profile })\n    }\n  }\n\n  async runSES<D, E> (\n    func: (service: AWS.SES) => AWS.Request<D, E>,\n    options: ?AWSOpOptions,\n  ): Promise<{ result: D, code?: ?string }> {\n    return await this._runOp(\n      'SES',\n      (clientConfig: ClientConfig) => new AWS.SES(clientConfig),\n      func,\n      options)\n  }\n\n  async runS3<D, E> (\n    func: (service: AWS.S3) => AWS.Request<D, E>,\n    options: ?AWSOpOptions,\n  ): Promise<{ result: D, code?: ?string }> {\n    return await this._runOp(\n      'S3',\n      (clientConfig: ClientConfig) => new AWS.S3(clientConfig),\n      func,\n      options)\n  }\n\n  async runDocumentClient<D, E> (\n    func: (service: AWS.DynamoDB.DocumentClient) => AWS.Request<D, E>,\n    options: ?AWSOpOptions,\n  ): Promise<{ result: D, code?: ?string }> {\n    return await this._runOp(\n      'DocumentClient',\n      (clientConfig: ClientConfig) => new AWS.DynamoDB.DocumentClient({ convertEmptyValues: true, ...clientConfig }),\n      func,\n      options)\n  }\n\n  async runDynamoDB<D, E> (\n    func: (service: AWS.DynamoDB) => AWS.Request<D, E>,\n    options: ?AWSOpOptions,\n  ): Promise<{ result: D, code?: ?string }> {\n    return await this._runOp(\n      'DynamoDB',\n      (clientConfig: ClientConfig) => new AWS.DynamoDB(clientConfig),\n      func,\n      options)\n  }\n\n  async runApplicationAutoScaling<D, E> (\n    func: (service: AWS.ApplicationAutoScaling) => AWS.Request<D, E>,\n    options: ?AWSOpOptions,\n  ): Promise<{ result: D, code?: ?string }> {\n    return await this._runOp(\n      'ApplicationAutoScaling',\n      (clientConfig: ClientConfig) => new AWS.ApplicationAutoScaling(clientConfig),\n      func,\n      options)\n  }\n\n  async runCloudFront<D, E> (\n    func: (service: AWS.CloudFront) => AWS.Request<D, E>,\n    options: ?AWSOpOptions,\n  ): Promise<{ result: D, code?: ?string }> {\n    return await this._runOp(\n      'CloudFront',\n      (clientConfig: ClientConfig) => new AWS.CloudFront(clientConfig),\n      func,\n      options)\n  }\n\n  async runAthena<D, E> (\n    func: (service: AWS.Athena) => AWS.Request<D, E>,\n    options: ?AWSOpOptions,\n  ): Promise<{ result: D, code?: ?string }> {\n    return await this._runOp(\n      'Athena',\n      (clientConfig: ClientConfig) => new AWS.Athena(clientConfig),\n      func,\n      options)\n  }\n\n  async runSQS<D, E> (\n    func: (service: AWS.SQS) => AWS.Request<D, E>,\n    options: ?AWSOpOptions,\n  ): Promise<{ result: D, code?: ?string }> {\n    return await this._runOp(\n      'SQS',\n      (clientConfig: ClientConfig) => new AWS.SQS(clientConfig),\n      func,\n      options)\n  }\n\n  async runCloudWatch<D, E> (\n    func: (service: AWS.SQS) => AWS.Request<D, E>,\n    options: ?AWSOpOptions,\n  ): Promise<{ result: D, code?: ?string }> {\n    return await this._runOp(\n      'CloudWatch',\n      (clientConfig: ClientConfig) => new AWS.CloudWatch(clientConfig),\n      func,\n      options)\n  }\n\n  async runFirehose<D, E> (\n    func: (service: AWS.SQS) => AWS.Request<D, E>,\n    options: ?AWSOpOptions,\n  ): Promise<{ result: D, code?: ?string }> {\n    return await this._runOp(\n      'Firehose',\n      (clientConfig: ClientConfig) => new AWS.Firehose(clientConfig),\n      func,\n      options)\n  }\n\n  async runES<D, E>(\n    func: (service: AWS.ES) => AWS.Request<D, E>,\n    options: ?AWSOpOptions,\n  ): Promise<{ result: D, code?: ?string }> {\n    return await this._runOp(\n      'Firehose',\n      (clientConfig: ClientConfig) => new AWS.ES(clientConfig),\n      func,\n      options)\n  }\n\n  async _runOp<D, E> (\n    serviceName: string,\n    serviceGenerator: (clientConfig: ClientConfig) => Object,\n    func: (service: Object) => AWS.Request<D, E>,\n    options: ?AWSOpOptions,\n  ): Promise<{ result: D, code?: ?string }> {\n    const awsConfig = System.getConfig().aws\n    if (!awsConfig)\n      throw new Error('aws config not set in system config')\n    const region = awsConfig.defaultRegion || (options || {}).region || throwUnsetArg('aws region')\n\n    options = options || {}\n    options.errorHandlers = options.errorHandlers || {}\n\n    const credentials = AWS.config.credentials\n    let start = 0\n    let opName = 'unknown-operation'\n    try {\n      // TODO: recreate only when creds expire?\n      const service = serviceGenerator({ credentials: credentials, region: region })\n      const req = func(service)\n      if (req === undefined)\n        throw new Error(`AWS runXXX method call on service '${serviceName}' is missing, did you user 'return'?`)\n      // $FlowIgnore\n      opName = options.opName || req['operation']\n\n      if (!options.noLog)\n        debug(`AWS operation started...`, {serviceName, opName} )\n\n      let result: D\n      let success = false\n      let code: ?string\n      do {\n        try {\n          start = Date.now()\n          result = await req.promise()\n          success = true\n        } catch (err) {\n          // $FlowIgnore\n          const errorHandler = options.errorHandlers[err.code]\n          if (!errorHandler) \n            throw err\n          switch (errorHandler.type) {\n          case 'retry':\n            try {\n              errorHandler.tries = errorHandler.tries || 1\n              if (errorHandler.tries-- > 0 && errorHandler.action) {\n                await errorHandler.action(err)\n              }\n            } catch (err2) {\n              err2.leadingError = err\n              throw err2\n            }\n            break\n          case 'ignore':\n            code = err.code\n            success = true\n            log(`${serviceName} op '${opName}' completed with an ignorable error code`, { code: code, ms: Date.now() - start })\n            break\n          default:\n            throw new Error(`Unexpected errorHandler type: '${errorHandler.type}'`)\n          }\n        }\n      } while (!success)\n\n      if (!options.noLog)\n        debug(`AWS operation completed`, { serviceName, opName, ms: Date.now() - start })\n      noteCount(`${MODULE_NAME}.${serviceName}.${opName}.success`, 1)\n      // $FlowIgnore   \n      return {\n        result: result,\n        code: code,\n      }\n    } catch (err) {\n      log(`${serviceName} op '${opName}' failed`, { err, args: options.args })\n      noteCount(`${MODULE_NAME}.${serviceName}.${opName}.fail`, 1)\n      throw new AWSError(err, `${serviceName}.${opName}`, { args: options.args })\n    } finally {\n      noteTimer(`${MODULE_NAME}.${serviceName}.${opName}`, Date.now() - start)\n    }\n  }\n}"]}
\No newline at end of file