@hanzo/s3 0.6.3 → 8.0.7
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/LICENSE +202 -0
- package/MAINTAINERS.md +62 -0
- package/README.md +262 -0
- package/README_zh_CN.md +192 -0
- package/dist/esm/AssumeRoleProvider.d.mts +86 -0
- package/dist/esm/AssumeRoleProvider.mjs +183 -0
- package/dist/esm/CredentialProvider.d.mts +22 -0
- package/dist/esm/CredentialProvider.mjs +48 -0
- package/dist/esm/Credentials.d.mts +22 -0
- package/dist/esm/Credentials.mjs +38 -0
- package/dist/esm/IamAwsProvider.d.mts +27 -0
- package/dist/esm/IamAwsProvider.mjs +189 -0
- package/dist/esm/errors.d.mts +82 -0
- package/dist/esm/errors.mjs +117 -0
- package/dist/esm/helpers.d.mts +156 -0
- package/dist/esm/helpers.mjs +218 -0
- package/dist/esm/internal/async.d.mts +9 -0
- package/dist/esm/internal/async.mjs +14 -0
- package/dist/esm/internal/callbackify.d.mts +1 -0
- package/dist/esm/internal/callbackify.mjs +15 -0
- package/dist/esm/internal/client.d.mts +394 -0
- package/dist/esm/internal/client.mjs +3007 -0
- package/dist/esm/internal/copy-conditions.d.mts +10 -0
- package/dist/esm/internal/copy-conditions.mjs +25 -0
- package/dist/esm/internal/extensions.d.mts +18 -0
- package/dist/esm/internal/extensions.mjs +114 -0
- package/dist/esm/internal/helper.d.mts +177 -0
- package/dist/esm/internal/helper.mjs +552 -0
- package/dist/esm/internal/join-host-port.d.mts +11 -0
- package/dist/esm/internal/join-host-port.mjs +23 -0
- package/dist/esm/internal/post-policy.d.mts +17 -0
- package/dist/esm/internal/post-policy.mjs +98 -0
- package/dist/esm/internal/request.d.mts +11 -0
- package/dist/esm/internal/request.mjs +75 -0
- package/dist/esm/internal/response.d.mts +8 -0
- package/dist/esm/internal/response.mjs +16 -0
- package/dist/esm/internal/s3-endpoints.d.mts +38 -0
- package/dist/esm/internal/s3-endpoints.mjs +68 -0
- package/dist/esm/internal/type.d.mts +482 -0
- package/dist/esm/internal/type.mjs +30 -0
- package/dist/esm/internal/xml-parser.d.mts +93 -0
- package/dist/esm/internal/xml-parser.mjs +819 -0
- package/dist/esm/notification.d.mts +58 -0
- package/dist/esm/notification.mjs +209 -0
- package/dist/esm/s3.d.mts +40 -0
- package/dist/esm/s3.mjs +86 -0
- package/dist/esm/signing.d.mts +5 -0
- package/dist/esm/signing.mjs +258 -0
- package/dist/main/AssumeRoleProvider.d.ts +86 -0
- package/dist/main/AssumeRoleProvider.js +191 -0
- package/dist/main/CredentialProvider.d.ts +22 -0
- package/dist/main/CredentialProvider.js +55 -0
- package/dist/main/Credentials.d.ts +22 -0
- package/dist/main/Credentials.js +45 -0
- package/dist/main/IamAwsProvider.d.ts +27 -0
- package/dist/main/IamAwsProvider.js +198 -0
- package/dist/main/errors.d.ts +82 -0
- package/dist/main/errors.js +138 -0
- package/dist/main/helpers.d.ts +156 -0
- package/dist/main/helpers.js +233 -0
- package/dist/main/internal/async.d.ts +9 -0
- package/dist/main/internal/async.js +24 -0
- package/dist/main/internal/callbackify.d.ts +1 -0
- package/dist/main/internal/callbackify.js +21 -0
- package/dist/main/internal/client.d.ts +394 -0
- package/dist/main/internal/client.js +3014 -0
- package/dist/main/internal/copy-conditions.d.ts +10 -0
- package/dist/main/internal/copy-conditions.js +31 -0
- package/dist/main/internal/extensions.d.ts +18 -0
- package/dist/main/internal/extensions.js +122 -0
- package/dist/main/internal/helper.d.ts +177 -0
- package/dist/main/internal/helper.js +608 -0
- package/dist/main/internal/join-host-port.d.ts +11 -0
- package/dist/main/internal/join-host-port.js +29 -0
- package/dist/main/internal/post-policy.d.ts +17 -0
- package/dist/main/internal/post-policy.js +107 -0
- package/dist/main/internal/request.d.ts +11 -0
- package/dist/main/internal/request.js +83 -0
- package/dist/main/internal/response.d.ts +8 -0
- package/dist/main/internal/response.js +24 -0
- package/dist/main/internal/s3-endpoints.d.ts +38 -0
- package/dist/main/internal/s3-endpoints.js +73 -0
- package/dist/main/internal/type.d.ts +482 -0
- package/dist/main/internal/type.js +42 -0
- package/dist/main/internal/xml-parser.d.ts +93 -0
- package/dist/main/internal/xml-parser.js +849 -0
- package/dist/main/notification.d.ts +58 -0
- package/dist/main/notification.js +230 -0
- package/dist/main/s3.d.ts +40 -0
- package/dist/main/s3.js +117 -0
- package/dist/main/signing.d.ts +5 -0
- package/dist/main/signing.js +269 -0
- package/package.json +146 -39
- package/src/AssumeRoleProvider.ts +262 -0
- package/src/CredentialProvider.ts +54 -0
- package/src/Credentials.ts +44 -0
- package/src/IamAwsProvider.ts +234 -0
- package/src/errors.ts +120 -0
- package/src/helpers.ts +354 -0
- package/src/internal/async.ts +14 -0
- package/src/internal/callbackify.ts +19 -0
- package/src/internal/client.ts +3412 -0
- package/src/internal/copy-conditions.ts +30 -0
- package/src/internal/extensions.ts +140 -0
- package/src/internal/helper.ts +606 -0
- package/src/internal/join-host-port.ts +23 -0
- package/src/internal/post-policy.ts +99 -0
- package/src/internal/request.ts +102 -0
- package/src/internal/response.ts +26 -0
- package/src/internal/s3-endpoints.ts +70 -0
- package/src/internal/type.ts +577 -0
- package/src/internal/xml-parser.ts +871 -0
- package/src/notification.ts +254 -0
- package/src/s3.ts +155 -0
- package/src/signing.ts +325 -0
- package/lib/index.js +0 -450
- package/lib/index.js.map +0 -7
- package/lib/perfTest.js +0 -91
- package/lib/perfTest.js.map +0 -7
|
@@ -0,0 +1,258 @@
|
|
|
1
|
+
/*
|
|
2
|
+
* Hanzo S3 Javascript Library for Amazon S3 Compatible Cloud Storage, (C) 2016 Hanzo AI, Inc.
|
|
3
|
+
*
|
|
4
|
+
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
5
|
+
* you may not use this file except in compliance with the License.
|
|
6
|
+
* You may obtain a copy of the License at
|
|
7
|
+
*
|
|
8
|
+
* http://www.apache.org/licenses/LICENSE-2.0
|
|
9
|
+
*
|
|
10
|
+
* Unless required by applicable law or agreed to in writing, software
|
|
11
|
+
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
12
|
+
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
13
|
+
* See the License for the specific language governing permissions and
|
|
14
|
+
* limitations under the License.
|
|
15
|
+
*/
|
|
16
|
+
|
|
17
|
+
import * as crypto from "crypto";
|
|
18
|
+
import * as errors from "./errors.mjs";
|
|
19
|
+
import { PRESIGN_EXPIRY_DAYS_MAX } from "./helpers.mjs";
|
|
20
|
+
import { getScope, isNumber, isObject, isString, makeDateLong, makeDateShort, uriEscape } from "./internal/helper.mjs";
|
|
21
|
+
const signV4Algorithm = 'AWS4-HMAC-SHA256';
|
|
22
|
+
|
|
23
|
+
// getCanonicalRequest generate a canonical request of style.
|
|
24
|
+
//
|
|
25
|
+
// canonicalRequest =
|
|
26
|
+
// <HTTPMethod>\n
|
|
27
|
+
// <CanonicalURI>\n
|
|
28
|
+
// <CanonicalQueryString>\n
|
|
29
|
+
// <CanonicalHeaders>\n
|
|
30
|
+
// <SignedHeaders>\n
|
|
31
|
+
// <HashedPayload>
|
|
32
|
+
//
|
|
33
|
+
function getCanonicalRequest(method, path, headers, signedHeaders, hashedPayload) {
|
|
34
|
+
if (!isString(method)) {
|
|
35
|
+
throw new TypeError('method should be of type "string"');
|
|
36
|
+
}
|
|
37
|
+
if (!isString(path)) {
|
|
38
|
+
throw new TypeError('path should be of type "string"');
|
|
39
|
+
}
|
|
40
|
+
if (!isObject(headers)) {
|
|
41
|
+
throw new TypeError('headers should be of type "object"');
|
|
42
|
+
}
|
|
43
|
+
if (!Array.isArray(signedHeaders)) {
|
|
44
|
+
throw new TypeError('signedHeaders should be of type "array"');
|
|
45
|
+
}
|
|
46
|
+
if (!isString(hashedPayload)) {
|
|
47
|
+
throw new TypeError('hashedPayload should be of type "string"');
|
|
48
|
+
}
|
|
49
|
+
const headersArray = signedHeaders.reduce((acc, i) => {
|
|
50
|
+
// Trim spaces from the value (required by V4 spec)
|
|
51
|
+
const val = `${headers[i]}`.replace(/ +/g, ' ');
|
|
52
|
+
acc.push(`${i.toLowerCase()}:${val}`);
|
|
53
|
+
return acc;
|
|
54
|
+
}, []);
|
|
55
|
+
const requestResource = path.split('?')[0];
|
|
56
|
+
let requestQuery = path.split('?')[1];
|
|
57
|
+
if (!requestQuery) {
|
|
58
|
+
requestQuery = '';
|
|
59
|
+
}
|
|
60
|
+
if (requestQuery) {
|
|
61
|
+
requestQuery = requestQuery.split('&').sort().map(element => !element.includes('=') ? element + '=' : element).join('&');
|
|
62
|
+
}
|
|
63
|
+
return [method.toUpperCase(), requestResource, requestQuery, headersArray.join('\n') + '\n', signedHeaders.join(';').toLowerCase(), hashedPayload].join('\n');
|
|
64
|
+
}
|
|
65
|
+
|
|
66
|
+
// generate a credential string
|
|
67
|
+
function getCredential(accessKey, region, requestDate, serviceName = 's3') {
|
|
68
|
+
if (!isString(accessKey)) {
|
|
69
|
+
throw new TypeError('accessKey should be of type "string"');
|
|
70
|
+
}
|
|
71
|
+
if (!isString(region)) {
|
|
72
|
+
throw new TypeError('region should be of type "string"');
|
|
73
|
+
}
|
|
74
|
+
if (!isObject(requestDate)) {
|
|
75
|
+
throw new TypeError('requestDate should be of type "object"');
|
|
76
|
+
}
|
|
77
|
+
return `${accessKey}/${getScope(region, requestDate, serviceName)}`;
|
|
78
|
+
}
|
|
79
|
+
|
|
80
|
+
// Returns signed headers array - alphabetically sorted
|
|
81
|
+
function getSignedHeaders(headers) {
|
|
82
|
+
if (!isObject(headers)) {
|
|
83
|
+
throw new TypeError('request should be of type "object"');
|
|
84
|
+
}
|
|
85
|
+
// Excerpts from @lsegal - https://github.com/aws/aws-sdk-js/issues/659#issuecomment-120477258
|
|
86
|
+
//
|
|
87
|
+
// User-Agent:
|
|
88
|
+
//
|
|
89
|
+
// This is ignored from signing because signing this causes problems with generating pre-signed URLs
|
|
90
|
+
// (that are executed by other agents) or when customers pass requests through proxies, which may
|
|
91
|
+
// modify the user-agent.
|
|
92
|
+
//
|
|
93
|
+
// Content-Length:
|
|
94
|
+
//
|
|
95
|
+
// This is ignored from signing because generating a pre-signed URL should not provide a content-length
|
|
96
|
+
// constraint, specifically when vending a S3 pre-signed PUT URL. The corollary to this is that when
|
|
97
|
+
// sending regular requests (non-pre-signed), the signature contains a checksum of the body, which
|
|
98
|
+
// implicitly validates the payload length (since changing the number of bytes would change the checksum)
|
|
99
|
+
// and therefore this header is not valuable in the signature.
|
|
100
|
+
//
|
|
101
|
+
// Content-Type:
|
|
102
|
+
//
|
|
103
|
+
// Signing this header causes quite a number of problems in browser environments, where browsers
|
|
104
|
+
// like to modify and normalize the content-type header in different ways. There is more information
|
|
105
|
+
// on this in https://github.com/aws/aws-sdk-js/issues/244. Avoiding this field simplifies logic
|
|
106
|
+
// and reduces the possibility of future bugs
|
|
107
|
+
//
|
|
108
|
+
// Authorization:
|
|
109
|
+
//
|
|
110
|
+
// Is skipped for obvious reasons
|
|
111
|
+
|
|
112
|
+
const ignoredHeaders = ['authorization', 'content-length', 'content-type', 'user-agent'];
|
|
113
|
+
return Object.keys(headers).filter(header => !ignoredHeaders.includes(header)).sort();
|
|
114
|
+
}
|
|
115
|
+
|
|
116
|
+
// returns the key used for calculating signature
|
|
117
|
+
function getSigningKey(date, region, secretKey, serviceName = 's3') {
|
|
118
|
+
if (!isObject(date)) {
|
|
119
|
+
throw new TypeError('date should be of type "object"');
|
|
120
|
+
}
|
|
121
|
+
if (!isString(region)) {
|
|
122
|
+
throw new TypeError('region should be of type "string"');
|
|
123
|
+
}
|
|
124
|
+
if (!isString(secretKey)) {
|
|
125
|
+
throw new TypeError('secretKey should be of type "string"');
|
|
126
|
+
}
|
|
127
|
+
const dateLine = makeDateShort(date);
|
|
128
|
+
const hmac1 = crypto.createHmac('sha256', 'AWS4' + secretKey).update(dateLine).digest(),
|
|
129
|
+
hmac2 = crypto.createHmac('sha256', hmac1).update(region).digest(),
|
|
130
|
+
hmac3 = crypto.createHmac('sha256', hmac2).update(serviceName).digest();
|
|
131
|
+
return crypto.createHmac('sha256', hmac3).update('aws4_request').digest();
|
|
132
|
+
}
|
|
133
|
+
|
|
134
|
+
// returns the string that needs to be signed
|
|
135
|
+
function getStringToSign(canonicalRequest, requestDate, region, serviceName = 's3') {
|
|
136
|
+
if (!isString(canonicalRequest)) {
|
|
137
|
+
throw new TypeError('canonicalRequest should be of type "string"');
|
|
138
|
+
}
|
|
139
|
+
if (!isObject(requestDate)) {
|
|
140
|
+
throw new TypeError('requestDate should be of type "object"');
|
|
141
|
+
}
|
|
142
|
+
if (!isString(region)) {
|
|
143
|
+
throw new TypeError('region should be of type "string"');
|
|
144
|
+
}
|
|
145
|
+
const hash = crypto.createHash('sha256').update(canonicalRequest).digest('hex');
|
|
146
|
+
const scope = getScope(region, requestDate, serviceName);
|
|
147
|
+
const stringToSign = [signV4Algorithm, makeDateLong(requestDate), scope, hash];
|
|
148
|
+
return stringToSign.join('\n');
|
|
149
|
+
}
|
|
150
|
+
|
|
151
|
+
// calculate the signature of the POST policy
|
|
152
|
+
export function postPresignSignatureV4(region, date, secretKey, policyBase64) {
|
|
153
|
+
if (!isString(region)) {
|
|
154
|
+
throw new TypeError('region should be of type "string"');
|
|
155
|
+
}
|
|
156
|
+
if (!isObject(date)) {
|
|
157
|
+
throw new TypeError('date should be of type "object"');
|
|
158
|
+
}
|
|
159
|
+
if (!isString(secretKey)) {
|
|
160
|
+
throw new TypeError('secretKey should be of type "string"');
|
|
161
|
+
}
|
|
162
|
+
if (!isString(policyBase64)) {
|
|
163
|
+
throw new TypeError('policyBase64 should be of type "string"');
|
|
164
|
+
}
|
|
165
|
+
const signingKey = getSigningKey(date, region, secretKey);
|
|
166
|
+
return crypto.createHmac('sha256', signingKey).update(policyBase64).digest('hex').toLowerCase();
|
|
167
|
+
}
|
|
168
|
+
|
|
169
|
+
// Returns the authorization header
|
|
170
|
+
export function signV4(request, accessKey, secretKey, region, requestDate, sha256sum, serviceName = 's3') {
|
|
171
|
+
if (!isObject(request)) {
|
|
172
|
+
throw new TypeError('request should be of type "object"');
|
|
173
|
+
}
|
|
174
|
+
if (!isString(accessKey)) {
|
|
175
|
+
throw new TypeError('accessKey should be of type "string"');
|
|
176
|
+
}
|
|
177
|
+
if (!isString(secretKey)) {
|
|
178
|
+
throw new TypeError('secretKey should be of type "string"');
|
|
179
|
+
}
|
|
180
|
+
if (!isString(region)) {
|
|
181
|
+
throw new TypeError('region should be of type "string"');
|
|
182
|
+
}
|
|
183
|
+
if (!accessKey) {
|
|
184
|
+
throw new errors.AccessKeyRequiredError('accessKey is required for signing');
|
|
185
|
+
}
|
|
186
|
+
if (!secretKey) {
|
|
187
|
+
throw new errors.SecretKeyRequiredError('secretKey is required for signing');
|
|
188
|
+
}
|
|
189
|
+
const signedHeaders = getSignedHeaders(request.headers);
|
|
190
|
+
const canonicalRequest = getCanonicalRequest(request.method, request.path, request.headers, signedHeaders, sha256sum);
|
|
191
|
+
const serviceIdentifier = serviceName || 's3';
|
|
192
|
+
const stringToSign = getStringToSign(canonicalRequest, requestDate, region, serviceIdentifier);
|
|
193
|
+
const signingKey = getSigningKey(requestDate, region, secretKey, serviceIdentifier);
|
|
194
|
+
const credential = getCredential(accessKey, region, requestDate, serviceIdentifier);
|
|
195
|
+
const signature = crypto.createHmac('sha256', signingKey).update(stringToSign).digest('hex').toLowerCase();
|
|
196
|
+
return `${signV4Algorithm} Credential=${credential}, SignedHeaders=${signedHeaders.join(';').toLowerCase()}, Signature=${signature}`;
|
|
197
|
+
}
|
|
198
|
+
export function signV4ByServiceName(request, accessKey, secretKey, region, requestDate, contentSha256, serviceName = 's3') {
|
|
199
|
+
return signV4(request, accessKey, secretKey, region, requestDate, contentSha256, serviceName);
|
|
200
|
+
}
|
|
201
|
+
|
|
202
|
+
// returns a presigned URL string
|
|
203
|
+
export function presignSignatureV4(request, accessKey, secretKey, sessionToken, region, requestDate, expires) {
|
|
204
|
+
if (!isObject(request)) {
|
|
205
|
+
throw new TypeError('request should be of type "object"');
|
|
206
|
+
}
|
|
207
|
+
if (!isString(accessKey)) {
|
|
208
|
+
throw new TypeError('accessKey should be of type "string"');
|
|
209
|
+
}
|
|
210
|
+
if (!isString(secretKey)) {
|
|
211
|
+
throw new TypeError('secretKey should be of type "string"');
|
|
212
|
+
}
|
|
213
|
+
if (!isString(region)) {
|
|
214
|
+
throw new TypeError('region should be of type "string"');
|
|
215
|
+
}
|
|
216
|
+
if (!accessKey) {
|
|
217
|
+
throw new errors.AccessKeyRequiredError('accessKey is required for presigning');
|
|
218
|
+
}
|
|
219
|
+
if (!secretKey) {
|
|
220
|
+
throw new errors.SecretKeyRequiredError('secretKey is required for presigning');
|
|
221
|
+
}
|
|
222
|
+
if (expires && !isNumber(expires)) {
|
|
223
|
+
throw new TypeError('expires should be of type "number"');
|
|
224
|
+
}
|
|
225
|
+
if (expires && expires < 1) {
|
|
226
|
+
throw new errors.ExpiresParamError('expires param cannot be less than 1 seconds');
|
|
227
|
+
}
|
|
228
|
+
if (expires && expires > PRESIGN_EXPIRY_DAYS_MAX) {
|
|
229
|
+
throw new errors.ExpiresParamError('expires param cannot be greater than 7 days');
|
|
230
|
+
}
|
|
231
|
+
const iso8601Date = makeDateLong(requestDate);
|
|
232
|
+
const signedHeaders = getSignedHeaders(request.headers);
|
|
233
|
+
const credential = getCredential(accessKey, region, requestDate);
|
|
234
|
+
const hashedPayload = 'UNSIGNED-PAYLOAD';
|
|
235
|
+
const requestQuery = [];
|
|
236
|
+
requestQuery.push(`X-Amz-Algorithm=${signV4Algorithm}`);
|
|
237
|
+
requestQuery.push(`X-Amz-Credential=${uriEscape(credential)}`);
|
|
238
|
+
requestQuery.push(`X-Amz-Date=${iso8601Date}`);
|
|
239
|
+
requestQuery.push(`X-Amz-Expires=${expires}`);
|
|
240
|
+
requestQuery.push(`X-Amz-SignedHeaders=${uriEscape(signedHeaders.join(';').toLowerCase())}`);
|
|
241
|
+
if (sessionToken) {
|
|
242
|
+
requestQuery.push(`X-Amz-Security-Token=${uriEscape(sessionToken)}`);
|
|
243
|
+
}
|
|
244
|
+
const resource = request.path.split('?')[0];
|
|
245
|
+
let query = request.path.split('?')[1];
|
|
246
|
+
if (query) {
|
|
247
|
+
query = query + '&' + requestQuery.join('&');
|
|
248
|
+
} else {
|
|
249
|
+
query = requestQuery.join('&');
|
|
250
|
+
}
|
|
251
|
+
const path = resource + '?' + query;
|
|
252
|
+
const canonicalRequest = getCanonicalRequest(request.method, path, request.headers, signedHeaders, hashedPayload);
|
|
253
|
+
const stringToSign = getStringToSign(canonicalRequest, requestDate, region);
|
|
254
|
+
const signingKey = getSigningKey(requestDate, region, secretKey);
|
|
255
|
+
const signature = crypto.createHmac('sha256', signingKey).update(stringToSign).digest('hex').toLowerCase();
|
|
256
|
+
return request.protocol + '//' + request.headers.host + path + `&X-Amz-Signature=${signature}`;
|
|
257
|
+
}
|
|
258
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["crypto","errors","PRESIGN_EXPIRY_DAYS_MAX","getScope","isNumber","isObject","isString","makeDateLong","makeDateShort","uriEscape","signV4Algorithm","getCanonicalRequest","method","path","headers","signedHeaders","hashedPayload","TypeError","Array","isArray","headersArray","reduce","acc","i","val","replace","push","toLowerCase","requestResource","split","requestQuery","sort","map","element","includes","join","toUpperCase","getCredential","accessKey","region","requestDate","serviceName","getSignedHeaders","ignoredHeaders","Object","keys","filter","header","getSigningKey","date","secretKey","dateLine","hmac1","createHmac","update","digest","hmac2","hmac3","getStringToSign","canonicalRequest","hash","createHash","scope","stringToSign","postPresignSignatureV4","policyBase64","signingKey","signV4","request","sha256sum","AccessKeyRequiredError","SecretKeyRequiredError","serviceIdentifier","credential","signature","signV4ByServiceName","contentSha256","presignSignatureV4","sessionToken","expires","ExpiresParamError","iso8601Date","resource","query","protocol","host"],"sources":["signing.ts"],"sourcesContent":["/*\n * Hanzo S3 Javascript Library for Amazon S3 Compatible Cloud Storage, (C) 2016 Hanzo AI, Inc.\n *\n * Licensed under the Apache License, Version 2.0 (the \"License\");\n * you may not use this file except in compliance with the License.\n * You may obtain a copy of the License at\n *\n *     http://www.apache.org/licenses/LICENSE-2.0\n *\n * Unless required by applicable law or agreed to in writing, software\n * distributed under the License is distributed on an \"AS IS\" BASIS,\n * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n * See the License for the specific language governing permissions and\n * limitations under the License.\n */\n\nimport * as crypto from 'node:crypto'\n\nimport * as errors from './errors.ts'\nimport { PRESIGN_EXPIRY_DAYS_MAX } from './helpers.ts'\nimport { getScope, isNumber, isObject, isString, makeDateLong, makeDateShort, uriEscape } from './internal/helper.ts'\nimport type { ICanonicalRequest, IRequest, RequestHeaders } from './internal/type.ts'\n\nconst signV4Algorithm = 'AWS4-HMAC-SHA256'\n\n// getCanonicalRequest generate a canonical request of style.\n//\n// canonicalRequest =\n//  <HTTPMethod>\\n\n//  <CanonicalURI>\\n\n//  <CanonicalQueryString>\\n\n//  <CanonicalHeaders>\\n\n//  <SignedHeaders>\\n\n//  <HashedPayload>\n//\nfunction getCanonicalRequest(\n  method: string,\n  path: string,\n  headers: RequestHeaders,\n  signedHeaders: string[],\n  hashedPayload: string,\n): ICanonicalRequest {\n  if (!isString(method)) {\n    throw new TypeError('method should be of type \"string\"')\n  }\n  if (!isString(path)) {\n    throw new TypeError('path should be of type \"string\"')\n  }\n  if (!isObject(headers)) {\n    throw new TypeError('headers should be of type \"object\"')\n  }\n  if (!Array.isArray(signedHeaders)) {\n    throw new TypeError('signedHeaders should be of type \"array\"')\n  }\n  if (!isString(hashedPayload)) {\n    throw new TypeError('hashedPayload should be of type \"string\"')\n  }\n\n  const headersArray = signedHeaders.reduce((acc, i) => {\n    // Trim spaces from the value (required by V4 spec)\n    const val = `${headers[i]}`.replace(/ +/g, ' ')\n    acc.push(`${i.toLowerCase()}:${val}`)\n    return acc\n  }, [] as string[])\n\n  const requestResource = path.split('?')[0]\n  let requestQuery = path.split('?')[1]\n  if (!requestQuery) {\n    requestQuery = ''\n  }\n\n  if (requestQuery) {\n    requestQuery = requestQuery\n      .split('&')\n      .sort()\n      .map((element) => (!element.includes('=') ? element + '=' : element))\n      .join('&')\n  }\n\n  return [\n    method.toUpperCase(),\n    requestResource,\n    requestQuery,\n    headersArray.join('\\n') + '\\n',\n    signedHeaders.join(';').toLowerCase(),\n    hashedPayload,\n  ].join('\\n')\n}\n\n// generate a credential string\nfunction getCredential(accessKey: string, region: string, requestDate?: Date, serviceName = 's3') {\n  if (!isString(accessKey)) {\n    throw new TypeError('accessKey should be of type \"string\"')\n  }\n  if (!isString(region)) {\n    throw new TypeError('region should be of type \"string\"')\n  }\n  if (!isObject(requestDate)) {\n    throw new TypeError('requestDate should be of type \"object\"')\n  }\n  return `${accessKey}/${getScope(region, requestDate, serviceName)}`\n}\n\n// Returns signed headers array - alphabetically sorted\nfunction getSignedHeaders(headers: RequestHeaders): string[] {\n  if (!isObject(headers)) {\n    throw new TypeError('request should be of type \"object\"')\n  }\n  // Excerpts from @lsegal - https://github.com/aws/aws-sdk-js/issues/659#issuecomment-120477258\n  //\n  //  User-Agent:\n  //\n  //      This is ignored from signing because signing this causes problems with generating pre-signed URLs\n  //      (that are executed by other agents) or when customers pass requests through proxies, which may\n  //      modify the user-agent.\n  //\n  //  Content-Length:\n  //\n  //      This is ignored from signing because generating a pre-signed URL should not provide a content-length\n  //      constraint, specifically when vending a S3 pre-signed PUT URL. The corollary to this is that when\n  //      sending regular requests (non-pre-signed), the signature contains a checksum of the body, which\n  //      implicitly validates the payload length (since changing the number of bytes would change the checksum)\n  //      and therefore this header is not valuable in the signature.\n  //\n  //  Content-Type:\n  //\n  //      Signing this header causes quite a number of problems in browser environments, where browsers\n  //      like to modify and normalize the content-type header in different ways. There is more information\n  //      on this in https://github.com/aws/aws-sdk-js/issues/244. Avoiding this field simplifies logic\n  //      and reduces the possibility of future bugs\n  //\n  //  Authorization:\n  //\n  //      Is skipped for obvious reasons\n\n  const ignoredHeaders = ['authorization', 'content-length', 'content-type', 'user-agent']\n  return Object.keys(headers)\n    .filter((header) => !ignoredHeaders.includes(header))\n    .sort()\n}\n\n// returns the key used for calculating signature\nfunction getSigningKey(date: Date, region: string, secretKey: string, serviceName = 's3') {\n  if (!isObject(date)) {\n    throw new TypeError('date should be of type \"object\"')\n  }\n  if (!isString(region)) {\n    throw new TypeError('region should be of type \"string\"')\n  }\n  if (!isString(secretKey)) {\n    throw new TypeError('secretKey should be of type \"string\"')\n  }\n  const dateLine = makeDateShort(date)\n  const hmac1 = crypto\n      .createHmac('sha256', 'AWS4' + secretKey)\n      .update(dateLine)\n      .digest(),\n    hmac2 = crypto.createHmac('sha256', hmac1).update(region).digest(),\n    hmac3 = crypto.createHmac('sha256', hmac2).update(serviceName).digest()\n  return crypto.createHmac('sha256', hmac3).update('aws4_request').digest()\n}\n\n// returns the string that needs to be signed\nfunction getStringToSign(canonicalRequest: ICanonicalRequest, requestDate: Date, region: string, serviceName = 's3') {\n  if (!isString(canonicalRequest)) {\n    throw new TypeError('canonicalRequest should be of type \"string\"')\n  }\n  if (!isObject(requestDate)) {\n    throw new TypeError('requestDate should be of type \"object\"')\n  }\n  if (!isString(region)) {\n    throw new TypeError('region should be of type \"string\"')\n  }\n  const hash = crypto.createHash('sha256').update(canonicalRequest).digest('hex')\n  const scope = getScope(region, requestDate, serviceName)\n  const stringToSign = [signV4Algorithm, makeDateLong(requestDate), scope, hash]\n\n  return stringToSign.join('\\n')\n}\n\n// calculate the signature of the POST policy\nexport function postPresignSignatureV4(region: string, date: Date, secretKey: string, policyBase64: string): string {\n  if (!isString(region)) {\n    throw new TypeError('region should be of type \"string\"')\n  }\n  if (!isObject(date)) {\n    throw new TypeError('date should be of type \"object\"')\n  }\n  if (!isString(secretKey)) {\n    throw new TypeError('secretKey should be of type \"string\"')\n  }\n  if (!isString(policyBase64)) {\n    throw new TypeError('policyBase64 should be of type \"string\"')\n  }\n  const signingKey = getSigningKey(date, region, secretKey)\n  return crypto.createHmac('sha256', signingKey).update(policyBase64).digest('hex').toLowerCase()\n}\n\n// Returns the authorization header\nexport function signV4(\n  request: IRequest,\n  accessKey: string,\n  secretKey: string,\n  region: string,\n  requestDate: Date,\n  sha256sum: string,\n  serviceName = 's3',\n) {\n  if (!isObject(request)) {\n    throw new TypeError('request should be of type \"object\"')\n  }\n  if (!isString(accessKey)) {\n    throw new TypeError('accessKey should be of type \"string\"')\n  }\n  if (!isString(secretKey)) {\n    throw new TypeError('secretKey should be of type \"string\"')\n  }\n  if (!isString(region)) {\n    throw new TypeError('region should be of type \"string\"')\n  }\n\n  if (!accessKey) {\n    throw new errors.AccessKeyRequiredError('accessKey is required for signing')\n  }\n  if (!secretKey) {\n    throw new errors.SecretKeyRequiredError('secretKey is required for signing')\n  }\n\n  const signedHeaders = getSignedHeaders(request.headers)\n  const canonicalRequest = getCanonicalRequest(request.method, request.path, request.headers, signedHeaders, sha256sum)\n  const serviceIdentifier = serviceName || 's3'\n  const stringToSign = getStringToSign(canonicalRequest, requestDate, region, serviceIdentifier)\n  const signingKey = getSigningKey(requestDate, region, secretKey, serviceIdentifier)\n  const credential = getCredential(accessKey, region, requestDate, serviceIdentifier)\n  const signature = crypto.createHmac('sha256', signingKey).update(stringToSign).digest('hex').toLowerCase()\n\n  return `${signV4Algorithm} Credential=${credential}, SignedHeaders=${signedHeaders\n    .join(';')\n    .toLowerCase()}, Signature=${signature}`\n}\n\nexport function signV4ByServiceName(\n  request: IRequest,\n  accessKey: string,\n  secretKey: string,\n  region: string,\n  requestDate: Date,\n  contentSha256: string,\n  serviceName = 's3',\n): string {\n  return signV4(request, accessKey, secretKey, region, requestDate, contentSha256, serviceName)\n}\n\n// returns a presigned URL string\nexport function presignSignatureV4(\n  request: IRequest,\n  accessKey: string,\n  secretKey: string,\n  sessionToken: string | undefined,\n  region: string,\n  requestDate: Date,\n  expires: number | undefined,\n) {\n  if (!isObject(request)) {\n    throw new TypeError('request should be of type \"object\"')\n  }\n  if (!isString(accessKey)) {\n    throw new TypeError('accessKey should be of type \"string\"')\n  }\n  if (!isString(secretKey)) {\n    throw new TypeError('secretKey should be of type \"string\"')\n  }\n  if (!isString(region)) {\n    throw new TypeError('region should be of type \"string\"')\n  }\n\n  if (!accessKey) {\n    throw new errors.AccessKeyRequiredError('accessKey is required for presigning')\n  }\n  if (!secretKey) {\n    throw new errors.SecretKeyRequiredError('secretKey is required for presigning')\n  }\n\n  if (expires && !isNumber(expires)) {\n    throw new TypeError('expires should be of type \"number\"')\n  }\n  if (expires && expires < 1) {\n    throw new errors.ExpiresParamError('expires param cannot be less than 1 seconds')\n  }\n  if (expires && expires > PRESIGN_EXPIRY_DAYS_MAX) {\n    throw new errors.ExpiresParamError('expires param cannot be greater than 7 days')\n  }\n\n  const iso8601Date = makeDateLong(requestDate)\n  const signedHeaders = getSignedHeaders(request.headers)\n  const credential = getCredential(accessKey, region, requestDate)\n  const hashedPayload = 'UNSIGNED-PAYLOAD'\n\n  const requestQuery: string[] = []\n  requestQuery.push(`X-Amz-Algorithm=${signV4Algorithm}`)\n  requestQuery.push(`X-Amz-Credential=${uriEscape(credential)}`)\n  requestQuery.push(`X-Amz-Date=${iso8601Date}`)\n  requestQuery.push(`X-Amz-Expires=${expires}`)\n  requestQuery.push(`X-Amz-SignedHeaders=${uriEscape(signedHeaders.join(';').toLowerCase())}`)\n  if (sessionToken) {\n    requestQuery.push(`X-Amz-Security-Token=${uriEscape(sessionToken)}`)\n  }\n\n  const resource = request.path.split('?')[0]\n  let query = request.path.split('?')[1]\n  if (query) {\n    query = query + '&' + requestQuery.join('&')\n  } else {\n    query = requestQuery.join('&')\n  }\n\n  const path = resource + '?' + query\n\n  const canonicalRequest = getCanonicalRequest(request.method, path, request.headers, signedHeaders, hashedPayload)\n\n  const stringToSign = getStringToSign(canonicalRequest, requestDate, region)\n  const signingKey = getSigningKey(requestDate, region, secretKey)\n  const signature = crypto.createHmac('sha256', signingKey).update(stringToSign).digest('hex').toLowerCase()\n  return request.protocol + '//' + request.headers.host + path + `&X-Amz-Signature=${signature}`\n}\n"],"mappings":"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"}
|
|
@@ -0,0 +1,86 @@
|
|
|
1
|
+
/// <reference types="node" />
|
|
2
|
+
import * as http from 'node:http';
|
|
3
|
+
import { CredentialProvider } from "./CredentialProvider.js";
|
|
4
|
+
import { Credentials } from "./Credentials.js";
|
|
5
|
+
/**
|
|
6
|
+
* @see https://docs.aws.amazon.com/STS/latest/APIReference/API_AssumeRole.html
|
|
7
|
+
*/
|
|
8
|
+
type CredentialResponse = {
|
|
9
|
+
ErrorResponse?: {
|
|
10
|
+
Error?: {
|
|
11
|
+
Code?: string;
|
|
12
|
+
Message?: string;
|
|
13
|
+
};
|
|
14
|
+
};
|
|
15
|
+
AssumeRoleResponse: {
|
|
16
|
+
AssumeRoleResult: {
|
|
17
|
+
Credentials: {
|
|
18
|
+
AccessKeyId: string;
|
|
19
|
+
SecretAccessKey: string;
|
|
20
|
+
SessionToken: string;
|
|
21
|
+
Expiration: string;
|
|
22
|
+
};
|
|
23
|
+
};
|
|
24
|
+
};
|
|
25
|
+
};
|
|
26
|
+
export interface AssumeRoleProviderOptions {
|
|
27
|
+
stsEndpoint: string;
|
|
28
|
+
accessKey: string;
|
|
29
|
+
secretKey: string;
|
|
30
|
+
durationSeconds?: number;
|
|
31
|
+
sessionToken?: string;
|
|
32
|
+
policy?: string;
|
|
33
|
+
region?: string;
|
|
34
|
+
roleArn?: string;
|
|
35
|
+
roleSessionName?: string;
|
|
36
|
+
externalId?: string;
|
|
37
|
+
token?: string;
|
|
38
|
+
webIdentityToken?: string;
|
|
39
|
+
action?: string;
|
|
40
|
+
transportAgent?: http.Agent;
|
|
41
|
+
}
|
|
42
|
+
export declare class AssumeRoleProvider extends CredentialProvider {
|
|
43
|
+
private readonly stsEndpoint;
|
|
44
|
+
private readonly accessKey;
|
|
45
|
+
private readonly secretKey;
|
|
46
|
+
private readonly durationSeconds;
|
|
47
|
+
private readonly policy?;
|
|
48
|
+
private readonly region;
|
|
49
|
+
private readonly roleArn?;
|
|
50
|
+
private readonly roleSessionName?;
|
|
51
|
+
private readonly externalId?;
|
|
52
|
+
private readonly token?;
|
|
53
|
+
private readonly webIdentityToken?;
|
|
54
|
+
private readonly action;
|
|
55
|
+
private _credentials;
|
|
56
|
+
private readonly expirySeconds;
|
|
57
|
+
private accessExpiresAt;
|
|
58
|
+
private readonly transportAgent?;
|
|
59
|
+
private readonly transport;
|
|
60
|
+
constructor({
|
|
61
|
+
stsEndpoint,
|
|
62
|
+
accessKey,
|
|
63
|
+
secretKey,
|
|
64
|
+
durationSeconds,
|
|
65
|
+
sessionToken,
|
|
66
|
+
policy,
|
|
67
|
+
region,
|
|
68
|
+
roleArn,
|
|
69
|
+
roleSessionName,
|
|
70
|
+
externalId,
|
|
71
|
+
token,
|
|
72
|
+
webIdentityToken,
|
|
73
|
+
action,
|
|
74
|
+
transportAgent
|
|
75
|
+
}: AssumeRoleProviderOptions);
|
|
76
|
+
getRequestConfig(): {
|
|
77
|
+
requestOptions: http.RequestOptions;
|
|
78
|
+
requestData: string;
|
|
79
|
+
};
|
|
80
|
+
performRequest(): Promise<CredentialResponse>;
|
|
81
|
+
parseCredentials(respObj: CredentialResponse): Credentials;
|
|
82
|
+
refreshCredentials(): Promise<Credentials>;
|
|
83
|
+
getCredentials(): Promise<Credentials>;
|
|
84
|
+
isAboutToExpire(): boolean;
|
|
85
|
+
}
|
|
86
|
+
export default AssumeRoleProvider;
|
|
@@ -0,0 +1,191 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
|
|
3
|
+
Object.defineProperty(exports, "__esModule", {
|
|
4
|
+
value: true
|
|
5
|
+
});
|
|
6
|
+
var http = _interopRequireWildcard(require("http"), true);
|
|
7
|
+
var https = _interopRequireWildcard(require("https"), true);
|
|
8
|
+
var _url = require("url");
|
|
9
|
+
var _CredentialProvider = require("./CredentialProvider.js");
|
|
10
|
+
var _Credentials = require("./Credentials.js");
|
|
11
|
+
var _helper = require("./internal/helper.js");
|
|
12
|
+
var _request = require("./internal/request.js");
|
|
13
|
+
var _response = require("./internal/response.js");
|
|
14
|
+
var _signing = require("./signing.js");
|
|
15
|
+
function _getRequireWildcardCache(nodeInterop) { if (typeof WeakMap !== "function") return null; var cacheBabelInterop = new WeakMap(); var cacheNodeInterop = new WeakMap(); return (_getRequireWildcardCache = function (nodeInterop) { return nodeInterop ? cacheNodeInterop : cacheBabelInterop; })(nodeInterop); }
|
|
16
|
+
function _interopRequireWildcard(obj, nodeInterop) { if (!nodeInterop && obj && obj.__esModule) { return obj; } if (obj === null || typeof obj !== "object" && typeof obj !== "function") { return { default: obj }; } var cache = _getRequireWildcardCache(nodeInterop); if (cache && cache.has(obj)) { return cache.get(obj); } var newObj = {}; var hasPropertyDescriptor = Object.defineProperty && Object.getOwnPropertyDescriptor; for (var key in obj) { if (key !== "default" && Object.prototype.hasOwnProperty.call(obj, key)) { var desc = hasPropertyDescriptor ? Object.getOwnPropertyDescriptor(obj, key) : null; if (desc && (desc.get || desc.set)) { Object.defineProperty(newObj, key, desc); } else { newObj[key] = obj[key]; } } } newObj.default = obj; if (cache) { cache.set(obj, newObj); } return newObj; }
|
|
17
|
+
/**
|
|
18
|
+
* @see https://docs.aws.amazon.com/STS/latest/APIReference/API_AssumeRole.html
|
|
19
|
+
*/
|
|
20
|
+
|
|
21
|
+
const defaultExpirySeconds = 900;
|
|
22
|
+
class AssumeRoleProvider extends _CredentialProvider.CredentialProvider {
|
|
23
|
+
accessExpiresAt = '';
|
|
24
|
+
constructor({
|
|
25
|
+
stsEndpoint,
|
|
26
|
+
accessKey,
|
|
27
|
+
secretKey,
|
|
28
|
+
durationSeconds = defaultExpirySeconds,
|
|
29
|
+
sessionToken,
|
|
30
|
+
policy,
|
|
31
|
+
region = '',
|
|
32
|
+
roleArn,
|
|
33
|
+
roleSessionName,
|
|
34
|
+
externalId,
|
|
35
|
+
token,
|
|
36
|
+
webIdentityToken,
|
|
37
|
+
action = 'AssumeRole',
|
|
38
|
+
transportAgent = undefined
|
|
39
|
+
}) {
|
|
40
|
+
super({
|
|
41
|
+
accessKey,
|
|
42
|
+
secretKey,
|
|
43
|
+
sessionToken
|
|
44
|
+
});
|
|
45
|
+
this.stsEndpoint = new _url.URL(stsEndpoint);
|
|
46
|
+
this.accessKey = accessKey;
|
|
47
|
+
this.secretKey = secretKey;
|
|
48
|
+
this.policy = policy;
|
|
49
|
+
this.region = region;
|
|
50
|
+
this.roleArn = roleArn;
|
|
51
|
+
this.roleSessionName = roleSessionName;
|
|
52
|
+
this.externalId = externalId;
|
|
53
|
+
this.token = token;
|
|
54
|
+
this.webIdentityToken = webIdentityToken;
|
|
55
|
+
this.action = action;
|
|
56
|
+
this.durationSeconds = parseInt(durationSeconds);
|
|
57
|
+
let expirySeconds = this.durationSeconds;
|
|
58
|
+
if (this.durationSeconds < defaultExpirySeconds) {
|
|
59
|
+
expirySeconds = defaultExpirySeconds;
|
|
60
|
+
}
|
|
61
|
+
this.expirySeconds = expirySeconds; // for calculating refresh of credentials.
|
|
62
|
+
|
|
63
|
+
// By default, nodejs uses a global agent if the 'agent' property
|
|
64
|
+
// is set to undefined. Otherwise, it's okay to assume the users
|
|
65
|
+
// know what they're doing if they specify a custom transport agent.
|
|
66
|
+
this.transportAgent = transportAgent;
|
|
67
|
+
const isHttp = this.stsEndpoint.protocol === 'http:';
|
|
68
|
+
this.transport = isHttp ? http : https;
|
|
69
|
+
|
|
70
|
+
/**
|
|
71
|
+
* Internal Tracking variables
|
|
72
|
+
*/
|
|
73
|
+
this._credentials = null;
|
|
74
|
+
}
|
|
75
|
+
getRequestConfig() {
|
|
76
|
+
const hostValue = this.stsEndpoint.hostname;
|
|
77
|
+
const portValue = this.stsEndpoint.port;
|
|
78
|
+
const qryParams = new _url.URLSearchParams({
|
|
79
|
+
Action: this.action,
|
|
80
|
+
Version: '2011-06-15'
|
|
81
|
+
});
|
|
82
|
+
qryParams.set('DurationSeconds', this.expirySeconds.toString());
|
|
83
|
+
if (this.policy) {
|
|
84
|
+
qryParams.set('Policy', this.policy);
|
|
85
|
+
}
|
|
86
|
+
if (this.roleArn) {
|
|
87
|
+
qryParams.set('RoleArn', this.roleArn);
|
|
88
|
+
}
|
|
89
|
+
if (this.roleSessionName != null) {
|
|
90
|
+
qryParams.set('RoleSessionName', this.roleSessionName);
|
|
91
|
+
}
|
|
92
|
+
if (this.token != null) {
|
|
93
|
+
qryParams.set('Token', this.token);
|
|
94
|
+
}
|
|
95
|
+
if (this.webIdentityToken) {
|
|
96
|
+
qryParams.set('WebIdentityToken', this.webIdentityToken);
|
|
97
|
+
}
|
|
98
|
+
if (this.externalId) {
|
|
99
|
+
qryParams.set('ExternalId', this.externalId);
|
|
100
|
+
}
|
|
101
|
+
const urlParams = qryParams.toString();
|
|
102
|
+
const contentSha256 = (0, _helper.toSha256)(urlParams);
|
|
103
|
+
const date = new Date();
|
|
104
|
+
const requestOptions = {
|
|
105
|
+
hostname: hostValue,
|
|
106
|
+
port: portValue,
|
|
107
|
+
path: '/',
|
|
108
|
+
protocol: this.stsEndpoint.protocol,
|
|
109
|
+
method: 'POST',
|
|
110
|
+
headers: {
|
|
111
|
+
'Content-Type': 'application/x-www-form-urlencoded',
|
|
112
|
+
'content-length': urlParams.length.toString(),
|
|
113
|
+
host: hostValue,
|
|
114
|
+
'x-amz-date': (0, _helper.makeDateLong)(date),
|
|
115
|
+
'x-amz-content-sha256': contentSha256
|
|
116
|
+
},
|
|
117
|
+
agent: this.transportAgent
|
|
118
|
+
};
|
|
119
|
+
requestOptions.headers.authorization = (0, _signing.signV4ByServiceName)(requestOptions, this.accessKey, this.secretKey, this.region, date, contentSha256, 'sts');
|
|
120
|
+
return {
|
|
121
|
+
requestOptions,
|
|
122
|
+
requestData: urlParams
|
|
123
|
+
};
|
|
124
|
+
}
|
|
125
|
+
async performRequest() {
|
|
126
|
+
const {
|
|
127
|
+
requestOptions,
|
|
128
|
+
requestData
|
|
129
|
+
} = this.getRequestConfig();
|
|
130
|
+
const res = await (0, _request.request)(this.transport, requestOptions, requestData);
|
|
131
|
+
const body = await (0, _response.readAsString)(res);
|
|
132
|
+
return (0, _helper.parseXml)(body);
|
|
133
|
+
}
|
|
134
|
+
parseCredentials(respObj) {
|
|
135
|
+
if (respObj.ErrorResponse) {
|
|
136
|
+
var _respObj$ErrorRespons, _respObj$ErrorRespons2, _respObj$ErrorRespons3, _respObj$ErrorRespons4;
|
|
137
|
+
throw new Error(`Unable to obtain credentials: ${(_respObj$ErrorRespons = respObj.ErrorResponse) === null || _respObj$ErrorRespons === void 0 ? void 0 : (_respObj$ErrorRespons2 = _respObj$ErrorRespons.Error) === null || _respObj$ErrorRespons2 === void 0 ? void 0 : _respObj$ErrorRespons2.Code} ${(_respObj$ErrorRespons3 = respObj.ErrorResponse) === null || _respObj$ErrorRespons3 === void 0 ? void 0 : (_respObj$ErrorRespons4 = _respObj$ErrorRespons3.Error) === null || _respObj$ErrorRespons4 === void 0 ? void 0 : _respObj$ErrorRespons4.Message}`, {
|
|
138
|
+
cause: respObj
|
|
139
|
+
});
|
|
140
|
+
}
|
|
141
|
+
const {
|
|
142
|
+
AssumeRoleResponse: {
|
|
143
|
+
AssumeRoleResult: {
|
|
144
|
+
Credentials: {
|
|
145
|
+
AccessKeyId: accessKey,
|
|
146
|
+
SecretAccessKey: secretKey,
|
|
147
|
+
SessionToken: sessionToken,
|
|
148
|
+
Expiration: expiresAt
|
|
149
|
+
}
|
|
150
|
+
}
|
|
151
|
+
}
|
|
152
|
+
} = respObj;
|
|
153
|
+
this.accessExpiresAt = expiresAt;
|
|
154
|
+
return new _Credentials.Credentials({
|
|
155
|
+
accessKey,
|
|
156
|
+
secretKey,
|
|
157
|
+
sessionToken
|
|
158
|
+
});
|
|
159
|
+
}
|
|
160
|
+
async refreshCredentials() {
|
|
161
|
+
try {
|
|
162
|
+
const assumeRoleCredentials = await this.performRequest();
|
|
163
|
+
this._credentials = this.parseCredentials(assumeRoleCredentials);
|
|
164
|
+
} catch (err) {
|
|
165
|
+
throw new Error(`Failed to get Credentials: ${err}`, {
|
|
166
|
+
cause: err
|
|
167
|
+
});
|
|
168
|
+
}
|
|
169
|
+
return this._credentials;
|
|
170
|
+
}
|
|
171
|
+
async getCredentials() {
|
|
172
|
+
if (this._credentials && !this.isAboutToExpire()) {
|
|
173
|
+
return this._credentials;
|
|
174
|
+
}
|
|
175
|
+
this._credentials = await this.refreshCredentials();
|
|
176
|
+
return this._credentials;
|
|
177
|
+
}
|
|
178
|
+
isAboutToExpire() {
|
|
179
|
+
const expiresAt = new Date(this.accessExpiresAt);
|
|
180
|
+
const provisionalExpiry = new Date(Date.now() + 1000 * 10); // check before 10 seconds.
|
|
181
|
+
return provisionalExpiry > expiresAt;
|
|
182
|
+
}
|
|
183
|
+
}
|
|
184
|
+
|
|
185
|
+
// deprecated default export, please use named exports.
|
|
186
|
+
// keep for backward compatibility.
|
|
187
|
+
// eslint-disable-next-line import/no-default-export
|
|
188
|
+
exports.AssumeRoleProvider = AssumeRoleProvider;
|
|
189
|
+
var _default = AssumeRoleProvider;
|
|
190
|
+
exports.default = _default;
|
|
191
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["http","_interopRequireWildcard","require","https","_url","_CredentialProvider","_Credentials","_helper","_request","_response","_signing","_getRequireWildcardCache","nodeInterop","WeakMap","cacheBabelInterop","cacheNodeInterop","obj","__esModule","default","cache","has","get","newObj","hasPropertyDescriptor","Object","defineProperty","getOwnPropertyDescriptor","key","prototype","hasOwnProperty","call","desc","set","defaultExpirySeconds","AssumeRoleProvider","CredentialProvider","accessExpiresAt","constructor","stsEndpoint","accessKey","secretKey","durationSeconds","sessionToken","policy","region","roleArn","roleSessionName","externalId","token","webIdentityToken","action","transportAgent","undefined","URL","parseInt","expirySeconds","isHttp","protocol","transport","_credentials","getRequestConfig","hostValue","hostname","portValue","port","qryParams","URLSearchParams","Action","Version","toString","urlParams","contentSha256","toSha256","date","Date","requestOptions","path","method","headers","length","host","makeDateLong","agent","authorization","signV4ByServiceName","requestData","performRequest","res","request","body","readAsString","parseXml","parseCredentials","respObj","ErrorResponse","_respObj$ErrorRespons","_respObj$ErrorRespons2","_respObj$ErrorRespons3","_respObj$ErrorRespons4","Error","Code","Message","cause","AssumeRoleResponse","AssumeRoleResult","Credentials","AccessKeyId","SecretAccessKey","SessionToken","Expiration","expiresAt","refreshCredentials","assumeRoleCredentials","err","getCredentials","isAboutToExpire","provisionalExpiry","now","exports","_default"],"sources":["AssumeRoleProvider.ts"],"sourcesContent":["import * as http from 'node:http'\nimport * as https from 'node:https'\nimport { URL, URLSearchParams } from 'node:url'\n\nimport { CredentialProvider } from './CredentialProvider.ts'\nimport { Credentials } from './Credentials.ts'\nimport { makeDateLong, parseXml, toSha256 } from './internal/helper.ts'\nimport { request } from './internal/request.ts'\nimport { readAsString } from './internal/response.ts'\nimport type { Transport } from './internal/type.ts'\nimport { signV4ByServiceName } from './signing.ts'\n\n/**\n * @see https://docs.aws.amazon.com/STS/latest/APIReference/API_AssumeRole.html\n */\ntype CredentialResponse = {\n  ErrorResponse?: {\n    Error?: {\n      Code?: string\n      Message?: string\n    }\n  }\n\n  AssumeRoleResponse: {\n    AssumeRoleResult: {\n      Credentials: {\n        AccessKeyId: string\n        SecretAccessKey: string\n        SessionToken: string\n        Expiration: string\n      }\n    }\n  }\n}\n\nexport interface AssumeRoleProviderOptions {\n  stsEndpoint: string\n  accessKey: string\n  secretKey: string\n  durationSeconds?: number\n  sessionToken?: string\n  policy?: string\n  region?: string\n  roleArn?: string\n  roleSessionName?: string\n  externalId?: string\n  token?: string\n  webIdentityToken?: string\n  action?: string\n  transportAgent?: http.Agent\n}\n\nconst defaultExpirySeconds = 900\n\nexport class AssumeRoleProvider extends CredentialProvider {\n  private readonly stsEndpoint: URL\n  private readonly accessKey: string\n  private readonly secretKey: string\n  private readonly durationSeconds: number\n  private readonly policy?: string\n  private readonly region: string\n  private readonly roleArn?: string\n  private readonly roleSessionName?: string\n  private readonly externalId?: string\n  private readonly token?: string\n  private readonly webIdentityToken?: string\n  private readonly action: string\n\n  private _credentials: Credentials | null\n  private readonly expirySeconds: number\n  private accessExpiresAt = ''\n  private readonly transportAgent?: http.Agent\n\n  private readonly transport: Transport\n\n  constructor({\n    stsEndpoint,\n    accessKey,\n    secretKey,\n    durationSeconds = defaultExpirySeconds,\n    sessionToken,\n    policy,\n    region = '',\n    roleArn,\n    roleSessionName,\n    externalId,\n    token,\n    webIdentityToken,\n    action = 'AssumeRole',\n    transportAgent = undefined,\n  }: AssumeRoleProviderOptions) {\n    super({ accessKey, secretKey, sessionToken })\n\n    this.stsEndpoint = new URL(stsEndpoint)\n    this.accessKey = accessKey\n    this.secretKey = secretKey\n    this.policy = policy\n    this.region = region\n    this.roleArn = roleArn\n    this.roleSessionName = roleSessionName\n    this.externalId = externalId\n    this.token = token\n    this.webIdentityToken = webIdentityToken\n    this.action = action\n\n    this.durationSeconds = parseInt(durationSeconds as unknown as string)\n\n    let expirySeconds = this.durationSeconds\n    if (this.durationSeconds < defaultExpirySeconds) {\n      expirySeconds = defaultExpirySeconds\n    }\n    this.expirySeconds = expirySeconds // for calculating refresh of credentials.\n\n    // By default, nodejs uses a global agent if the 'agent' property\n    // is set to undefined. Otherwise, it's okay to assume the users\n    // know what they're doing if they specify a custom transport agent.\n    this.transportAgent = transportAgent\n    const isHttp: boolean = this.stsEndpoint.protocol === 'http:'\n    this.transport = isHttp ? http : https\n\n    /**\n     * Internal Tracking variables\n     */\n    this._credentials = null\n  }\n\n  getRequestConfig(): {\n    requestOptions: http.RequestOptions\n    requestData: string\n  } {\n    const hostValue = this.stsEndpoint.hostname\n    const portValue = this.stsEndpoint.port\n    const qryParams = new URLSearchParams({ Action: this.action, Version: '2011-06-15' })\n\n    qryParams.set('DurationSeconds', this.expirySeconds.toString())\n\n    if (this.policy) {\n      qryParams.set('Policy', this.policy)\n    }\n    if (this.roleArn) {\n      qryParams.set('RoleArn', this.roleArn)\n    }\n\n    if (this.roleSessionName != null) {\n      qryParams.set('RoleSessionName', this.roleSessionName)\n    }\n    if (this.token != null) {\n      qryParams.set('Token', this.token)\n    }\n\n    if (this.webIdentityToken) {\n      qryParams.set('WebIdentityToken', this.webIdentityToken)\n    }\n\n    if (this.externalId) {\n      qryParams.set('ExternalId', this.externalId)\n    }\n\n    const urlParams = qryParams.toString()\n    const contentSha256 = toSha256(urlParams)\n\n    const date = new Date()\n\n    const requestOptions = {\n      hostname: hostValue,\n      port: portValue,\n      path: '/',\n      protocol: this.stsEndpoint.protocol,\n      method: 'POST',\n      headers: {\n        'Content-Type': 'application/x-www-form-urlencoded',\n        'content-length': urlParams.length.toString(),\n        host: hostValue,\n        'x-amz-date': makeDateLong(date),\n        'x-amz-content-sha256': contentSha256,\n      } as Record<string, string>,\n      agent: this.transportAgent,\n    } satisfies http.RequestOptions\n\n    requestOptions.headers.authorization = signV4ByServiceName(\n      requestOptions,\n      this.accessKey,\n      this.secretKey,\n      this.region,\n      date,\n      contentSha256,\n      'sts',\n    )\n\n    return {\n      requestOptions,\n      requestData: urlParams,\n    }\n  }\n\n  async performRequest(): Promise<CredentialResponse> {\n    const { requestOptions, requestData } = this.getRequestConfig()\n\n    const res = await request(this.transport, requestOptions, requestData)\n\n    const body = await readAsString(res)\n\n    return parseXml(body)\n  }\n\n  parseCredentials(respObj: CredentialResponse): Credentials {\n    if (respObj.ErrorResponse) {\n      throw new Error(\n        `Unable to obtain credentials: ${respObj.ErrorResponse?.Error?.Code} ${respObj.ErrorResponse?.Error?.Message}`,\n        { cause: respObj },\n      )\n    }\n\n    const {\n      AssumeRoleResponse: {\n        AssumeRoleResult: {\n          Credentials: {\n            AccessKeyId: accessKey,\n            SecretAccessKey: secretKey,\n            SessionToken: sessionToken,\n            Expiration: expiresAt,\n          },\n        },\n      },\n    } = respObj\n\n    this.accessExpiresAt = expiresAt\n\n    return new Credentials({ accessKey, secretKey, sessionToken })\n  }\n\n  async refreshCredentials(): Promise<Credentials> {\n    try {\n      const assumeRoleCredentials = await this.performRequest()\n      this._credentials = this.parseCredentials(assumeRoleCredentials)\n    } catch (err) {\n      throw new Error(`Failed to get Credentials: ${err}`, { cause: err })\n    }\n\n    return this._credentials\n  }\n\n  async getCredentials(): Promise<Credentials> {\n    if (this._credentials && !this.isAboutToExpire()) {\n      return this._credentials\n    }\n\n    this._credentials = await this.refreshCredentials()\n    return this._credentials\n  }\n\n  isAboutToExpire() {\n    const expiresAt = new Date(this.accessExpiresAt)\n    const provisionalExpiry = new Date(Date.now() + 1000 * 10) // check before 10 seconds.\n    return provisionalExpiry > expiresAt\n  }\n}\n\n// deprecated default export, please use named exports.\n// keep for backward compatibility.\n// eslint-disable-next-line import/no-default-export\nexport default AssumeRoleProvider\n"],"mappings":"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"}
|
|
@@ -0,0 +1,22 @@
|
|
|
1
|
+
import { Credentials } from "./Credentials.js";
|
|
2
|
+
export declare class CredentialProvider {
|
|
3
|
+
private credentials;
|
|
4
|
+
constructor({
|
|
5
|
+
accessKey,
|
|
6
|
+
secretKey,
|
|
7
|
+
sessionToken
|
|
8
|
+
}: {
|
|
9
|
+
accessKey: string;
|
|
10
|
+
secretKey: string;
|
|
11
|
+
sessionToken?: string;
|
|
12
|
+
});
|
|
13
|
+
getCredentials(): Promise<Credentials>;
|
|
14
|
+
setCredentials(credentials: Credentials): void;
|
|
15
|
+
setAccessKey(accessKey: string): void;
|
|
16
|
+
getAccessKey(): string;
|
|
17
|
+
setSecretKey(secretKey: string): void;
|
|
18
|
+
getSecretKey(): string;
|
|
19
|
+
setSessionToken(sessionToken: string): void;
|
|
20
|
+
getSessionToken(): string | undefined;
|
|
21
|
+
}
|
|
22
|
+
export default CredentialProvider;
|