1 | ;
|
2 |
|
3 | Object.defineProperty(exports, "__esModule", {
|
4 | value: true
|
5 | });
|
6 | exports.default = void 0;
|
7 |
|
8 | var _awsSdk = _interopRequireDefault(require("aws-sdk"));
|
9 |
|
10 | var _System = _interopRequireDefault(require("./System"));
|
11 |
|
12 | var _Module = _interopRequireDefault(require("./Module"));
|
13 |
|
14 | var _errorUtils = require("./errorUtils");
|
15 |
|
16 | function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
|
17 |
|
18 | function _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 |
|
20 | function _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 |
|
22 | const {
|
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 |
|
34 | class 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 |
|
66 | class 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 |
|
225 | exports.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 |