@discordeno/utils 22.0.1-next.df8cc46 → 22.0.1-next.e5063e1
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/dist/base64.d.ts +6 -0
- package/dist/base64.d.ts.map +1 -1
- package/dist/base64.js +90 -13
- package/dist/oauth2.d.ts +38 -0
- package/dist/oauth2.d.ts.map +1 -1
- package/dist/oauth2.js +33 -1
- package/package.json +10 -10
package/dist/base64.d.ts
CHANGED
|
@@ -4,6 +4,12 @@
|
|
|
4
4
|
* @param data
|
|
5
5
|
*/
|
|
6
6
|
export declare function encode(data: Uint8Array | ArrayBuffer | string): string;
|
|
7
|
+
/**
|
|
8
|
+
* Encodes a given Uint8Array, ArrayBuffer or string into RFC4648 base64url representation
|
|
9
|
+
* @param data The data to encode
|
|
10
|
+
* @returns The base64url encoded string
|
|
11
|
+
*/
|
|
12
|
+
export declare function encodeBase64Url(data: Uint8Array | ArrayBuffer | string): string;
|
|
7
13
|
/**
|
|
8
14
|
* CREDIT: https://gist.github.com/enepomnyaschih/72c423f727d395eeaa09697058238727
|
|
9
15
|
* Decodes RFC4648 base64 string into an Uint8Array
|
package/dist/base64.d.ts.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"base64.d.ts","sourceRoot":"","sources":["../src/base64.ts"],"names":[],"mappings":"AAAA;;;;GAIG;AACH,wBAAgB,MAAM,CAAC,IAAI,EAAE,UAAU,GAAG,WAAW,GAAG,MAAM,GAAG,MAAM,
|
|
1
|
+
{"version":3,"file":"base64.d.ts","sourceRoot":"","sources":["../src/base64.ts"],"names":[],"mappings":"AAAA;;;;GAIG;AACH,wBAAgB,MAAM,CAAC,IAAI,EAAE,UAAU,GAAG,WAAW,GAAG,MAAM,GAAG,MAAM,CAGtE;AAED;;;;GAIG;AACH,wBAAgB,eAAe,CAAC,IAAI,EAAE,UAAU,GAAG,WAAW,GAAG,MAAM,GAAG,MAAM,CAG/E;AA6BD;;;;GAIG;AACH,wBAAgB,MAAM,CAAC,IAAI,EAAE,MAAM,GAAG,UAAU,CAuB/C"}
|
package/dist/base64.js
CHANGED
|
@@ -4,27 +4,38 @@
|
|
|
4
4
|
* @param data
|
|
5
5
|
*/ export function encode(data) {
|
|
6
6
|
const uint8 = typeof data === 'string' ? new TextEncoder().encode(data) : data instanceof Uint8Array ? data : new Uint8Array(data);
|
|
7
|
+
return _encode(uint8, base64abc, false);
|
|
8
|
+
}
|
|
9
|
+
/**
|
|
10
|
+
* Encodes a given Uint8Array, ArrayBuffer or string into RFC4648 base64url representation
|
|
11
|
+
* @param data The data to encode
|
|
12
|
+
* @returns The base64url encoded string
|
|
13
|
+
*/ export function encodeBase64Url(data) {
|
|
14
|
+
const uint8 = typeof data === 'string' ? new TextEncoder().encode(data) : data instanceof Uint8Array ? data : new Uint8Array(data);
|
|
15
|
+
return _encode(uint8, base64urlAbc, true);
|
|
16
|
+
}
|
|
17
|
+
/** @private */ function _encode(data, alpha, skipPadding) {
|
|
7
18
|
let result = '';
|
|
8
19
|
let i;
|
|
9
|
-
const l =
|
|
20
|
+
const l = data.length;
|
|
10
21
|
for(i = 2; i < l; i += 3){
|
|
11
|
-
result +=
|
|
12
|
-
result +=
|
|
13
|
-
result +=
|
|
14
|
-
result +=
|
|
22
|
+
result += alpha[data[i - 2] >> 2];
|
|
23
|
+
result += alpha[(data[i - 2] & 0x03) << 4 | data[i - 1] >> 4];
|
|
24
|
+
result += alpha[(data[i - 1] & 0x0f) << 2 | data[i] >> 6];
|
|
25
|
+
result += alpha[data[i] & 0x3f];
|
|
15
26
|
}
|
|
16
27
|
if (i === l + 1) {
|
|
17
28
|
// 1 octet yet to write
|
|
18
|
-
result +=
|
|
19
|
-
result +=
|
|
20
|
-
result += '==';
|
|
29
|
+
result += alpha[data[i - 2] >> 2];
|
|
30
|
+
result += alpha[(data[i - 2] & 0x03) << 4];
|
|
31
|
+
if (!skipPadding) result += '==';
|
|
21
32
|
}
|
|
22
33
|
if (i === l) {
|
|
23
34
|
// 2 octets yet to write
|
|
24
|
-
result +=
|
|
25
|
-
result +=
|
|
26
|
-
result +=
|
|
27
|
-
result += '=';
|
|
35
|
+
result += alpha[data[i - 2] >> 2];
|
|
36
|
+
result += alpha[(data[i - 2] & 0x03) << 4 | data[i - 1] >> 4];
|
|
37
|
+
result += alpha[(data[i - 1] & 0x0f) << 2];
|
|
38
|
+
if (!skipPadding) result += '=';
|
|
28
39
|
}
|
|
29
40
|
return result;
|
|
30
41
|
}
|
|
@@ -132,6 +143,72 @@ const base64abc = [
|
|
|
132
143
|
'+',
|
|
133
144
|
'/'
|
|
134
145
|
];
|
|
146
|
+
const base64urlAbc = [
|
|
147
|
+
'A',
|
|
148
|
+
'B',
|
|
149
|
+
'C',
|
|
150
|
+
'D',
|
|
151
|
+
'E',
|
|
152
|
+
'F',
|
|
153
|
+
'G',
|
|
154
|
+
'H',
|
|
155
|
+
'I',
|
|
156
|
+
'J',
|
|
157
|
+
'K',
|
|
158
|
+
'L',
|
|
159
|
+
'M',
|
|
160
|
+
'N',
|
|
161
|
+
'O',
|
|
162
|
+
'P',
|
|
163
|
+
'Q',
|
|
164
|
+
'R',
|
|
165
|
+
'S',
|
|
166
|
+
'T',
|
|
167
|
+
'U',
|
|
168
|
+
'V',
|
|
169
|
+
'W',
|
|
170
|
+
'X',
|
|
171
|
+
'Y',
|
|
172
|
+
'Z',
|
|
173
|
+
'a',
|
|
174
|
+
'b',
|
|
175
|
+
'c',
|
|
176
|
+
'd',
|
|
177
|
+
'e',
|
|
178
|
+
'f',
|
|
179
|
+
'g',
|
|
180
|
+
'h',
|
|
181
|
+
'i',
|
|
182
|
+
'j',
|
|
183
|
+
'k',
|
|
184
|
+
'l',
|
|
185
|
+
'm',
|
|
186
|
+
'n',
|
|
187
|
+
'o',
|
|
188
|
+
'p',
|
|
189
|
+
'q',
|
|
190
|
+
'r',
|
|
191
|
+
's',
|
|
192
|
+
't',
|
|
193
|
+
'u',
|
|
194
|
+
'v',
|
|
195
|
+
'w',
|
|
196
|
+
'x',
|
|
197
|
+
'y',
|
|
198
|
+
'z',
|
|
199
|
+
'0',
|
|
200
|
+
'1',
|
|
201
|
+
'2',
|
|
202
|
+
'3',
|
|
203
|
+
'4',
|
|
204
|
+
'5',
|
|
205
|
+
'6',
|
|
206
|
+
'7',
|
|
207
|
+
'8',
|
|
208
|
+
'9',
|
|
209
|
+
'-',
|
|
210
|
+
'_'
|
|
211
|
+
];
|
|
135
212
|
// CREDIT: https://gist.github.com/enepomnyaschih/72c423f727d395eeaa09697058238727
|
|
136
213
|
const base64codes = [
|
|
137
214
|
255,
|
|
@@ -259,4 +336,4 @@ const base64codes = [
|
|
|
259
336
|
51
|
|
260
337
|
];
|
|
261
338
|
|
|
262
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../src/base64.ts"],"sourcesContent":["/**\n * CREDIT: https://gist.github.com/enepomnyaschih/72c423f727d395eeaa09697058238727\n * Encodes a given Uint8Array, ArrayBuffer or string into RFC4648 base64 representation\n * @param data\n */\nexport function encode(data: Uint8Array | ArrayBuffer | string): string {\n  const uint8 = typeof data === 'string' ? new TextEncoder().encode(data) : data instanceof Uint8Array ? data : new Uint8Array(data)\n  let result = ''\n  let i\n  const l = uint8.length\n  for (i = 2; i < l; i += 3) {\n    result += base64abc[uint8[i - 2] >> 2]\n    result += base64abc[((uint8[i - 2] & 0x03) << 4) | (uint8[i - 1] >> 4)]\n    result += base64abc[((uint8[i - 1] & 0x0f) << 2) | (uint8[i] >> 6)]\n    result += base64abc[uint8[i] & 0x3f]\n  }\n  if (i === l + 1) {\n    // 1 octet yet to write\n    result += base64abc[uint8[i - 2] >> 2]\n    result += base64abc[(uint8[i - 2] & 0x03) << 4]\n    result += '=='\n  }\n  if (i === l) {\n    // 2 octets yet to write\n    result += base64abc[uint8[i - 2] >> 2]\n    result += base64abc[((uint8[i - 2] & 0x03) << 4) | (uint8[i - 1] >> 4)]\n    result += base64abc[(uint8[i - 1] & 0x0f) << 2]\n    result += '='\n  }\n  return result\n}\n\n/**\n * CREDIT: https://gist.github.com/enepomnyaschih/72c423f727d395eeaa09697058238727\n * Decodes RFC4648 base64 string into an Uint8Array\n * @param data\n */\nexport function decode(data: string): Uint8Array {\n  if (data.length % 4 !== 0) {\n    throw new Error('Unable to parse base64 string.')\n  }\n  const index = data.indexOf('=')\n  if (index !== -1 && index < data.length - 2) {\n    throw new Error('Unable to parse base64 string.')\n  }\n  const missingOctets = data.endsWith('==') ? 2 : data.endsWith('=') ? 1 : 0\n  const n = data.length\n  const result = new Uint8Array(3 * (n / 4))\n  let buffer\n  for (let i = 0, j = 0; i < n; i += 4, j += 3) {\n    buffer =\n      (getBase64Code(data.charCodeAt(i)) << 18) |\n      (getBase64Code(data.charCodeAt(i + 1)) << 12) |\n      (getBase64Code(data.charCodeAt(i + 2)) << 6) |\n      getBase64Code(data.charCodeAt(i + 3))\n    result[j] = buffer >> 16\n    result[j + 1] = (buffer >> 8) & 0xff\n    result[j + 2] = buffer & 0xff\n  }\n  return result.subarray(0, result.length - missingOctets)\n}\n\n/**\n * CREDIT: https://gist.github.com/enepomnyaschih/72c423f727d395eeaa09697058238727\n * @param charCode\n */\nfunction getBase64Code(charCode: number): number {\n  if (charCode >= base64codes.length) {\n    throw new Error('Unable to parse base64 string.')\n  }\n  const code = base64codes[charCode]\n  if (code === 255) {\n    throw new Error('Unable to parse base64 string.')\n  }\n  return code\n}\n\n// Copyright 2018-2021 the Deno authors. All rights reserved. MIT license.\nconst base64abc = [\n  'A',\n  'B',\n  'C',\n  'D',\n  'E',\n  'F',\n  'G',\n  'H',\n  'I',\n  'J',\n  'K',\n  'L',\n  'M',\n  'N',\n  'O',\n  'P',\n  'Q',\n  'R',\n  'S',\n  'T',\n  'U',\n  'V',\n  'W',\n  'X',\n  'Y',\n  'Z',\n  'a',\n  'b',\n  'c',\n  'd',\n  'e',\n  'f',\n  'g',\n  'h',\n  'i',\n  'j',\n  'k',\n  'l',\n  'm',\n  'n',\n  'o',\n  'p',\n  'q',\n  'r',\n  's',\n  't',\n  'u',\n  'v',\n  'w',\n  'x',\n  'y',\n  'z',\n  '0',\n  '1',\n  '2',\n  '3',\n  '4',\n  '5',\n  '6',\n  '7',\n  '8',\n  '9',\n  '+',\n  '/',\n]\n\n// CREDIT: https://gist.github.com/enepomnyaschih/72c423f727d395eeaa09697058238727\nconst base64codes = [\n  255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255,\n  255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 62, 255, 255, 255, 63, 52, 53, 54, 55, 56, 57, 58, 59, 60, 61, 255, 255, 255,\n  0, 255, 255, 255, 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16, 17, 18, 19, 20, 21, 22, 23, 24, 25, 255, 255, 255, 255, 255, 255, 26,\n  27, 28, 29, 30, 31, 32, 33, 34, 35, 36, 37, 38, 39, 40, 41, 42, 43, 44, 45, 46, 47, 48, 49, 50, 51,\n]\n"],"names":["encode","data","uint8","TextEncoder","Uint8Array","result","i","l","length","base64abc","decode","Error","index","indexOf","missingOctets","endsWith","n","buffer","j","getBase64Code","charCodeAt","subarray","charCode","base64codes","code"],"mappings":"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"}
|
|
339
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../src/base64.ts"],"sourcesContent":["/**\n * CREDIT: https://gist.github.com/enepomnyaschih/72c423f727d395eeaa09697058238727\n * Encodes a given Uint8Array, ArrayBuffer or string into RFC4648 base64 representation\n * @param data\n */\nexport function encode(data: Uint8Array | ArrayBuffer | string): string {\n  const uint8 = typeof data === 'string' ? new TextEncoder().encode(data) : data instanceof Uint8Array ? data : new Uint8Array(data)\n  return _encode(uint8, base64abc, false)\n}\n\n/**\n * Encodes a given Uint8Array, ArrayBuffer or string into RFC4648 base64url representation\n * @param data The data to encode\n * @returns The base64url encoded string\n */\nexport function encodeBase64Url(data: Uint8Array | ArrayBuffer | string): string {\n  const uint8 = typeof data === 'string' ? new TextEncoder().encode(data) : data instanceof Uint8Array ? data : new Uint8Array(data)\n  return _encode(uint8, base64urlAbc, true)\n}\n\n/** @private */\nfunction _encode(data: Uint8Array, alpha: string[], skipPadding: boolean): string {\n  let result = ''\n  let i\n  const l = data.length\n  for (i = 2; i < l; i += 3) {\n    result += alpha[data[i - 2] >> 2]\n    result += alpha[((data[i - 2] & 0x03) << 4) | (data[i - 1] >> 4)]\n    result += alpha[((data[i - 1] & 0x0f) << 2) | (data[i] >> 6)]\n    result += alpha[data[i] & 0x3f]\n  }\n  if (i === l + 1) {\n    // 1 octet yet to write\n    result += alpha[data[i - 2] >> 2]\n    result += alpha[(data[i - 2] & 0x03) << 4]\n    if (!skipPadding) result += '=='\n  }\n  if (i === l) {\n    // 2 octets yet to write\n    result += alpha[data[i - 2] >> 2]\n    result += alpha[((data[i - 2] & 0x03) << 4) | (data[i - 1] >> 4)]\n    result += alpha[(data[i - 1] & 0x0f) << 2]\n    if (!skipPadding) result += '='\n  }\n  return result\n}\n\n/**\n * CREDIT: https://gist.github.com/enepomnyaschih/72c423f727d395eeaa09697058238727\n * Decodes RFC4648 base64 string into an Uint8Array\n * @param data\n */\nexport function decode(data: string): Uint8Array {\n  if (data.length % 4 !== 0) {\n    throw new Error('Unable to parse base64 string.')\n  }\n  const index = data.indexOf('=')\n  if (index !== -1 && index < data.length - 2) {\n    throw new Error('Unable to parse base64 string.')\n  }\n  const missingOctets = data.endsWith('==') ? 2 : data.endsWith('=') ? 1 : 0\n  const n = data.length\n  const result = new Uint8Array(3 * (n / 4))\n  let buffer\n  for (let i = 0, j = 0; i < n; i += 4, j += 3) {\n    buffer =\n      (getBase64Code(data.charCodeAt(i)) << 18) |\n      (getBase64Code(data.charCodeAt(i + 1)) << 12) |\n      (getBase64Code(data.charCodeAt(i + 2)) << 6) |\n      getBase64Code(data.charCodeAt(i + 3))\n    result[j] = buffer >> 16\n    result[j + 1] = (buffer >> 8) & 0xff\n    result[j + 2] = buffer & 0xff\n  }\n  return result.subarray(0, result.length - missingOctets)\n}\n\n/**\n * CREDIT: https://gist.github.com/enepomnyaschih/72c423f727d395eeaa09697058238727\n * @param charCode\n */\nfunction getBase64Code(charCode: number): number {\n  if (charCode >= base64codes.length) {\n    throw new Error('Unable to parse base64 string.')\n  }\n  const code = base64codes[charCode]\n  if (code === 255) {\n    throw new Error('Unable to parse base64 string.')\n  }\n  return code\n}\n\n// Copyright 2018-2021 the Deno authors. All rights reserved. MIT license.\nconst base64abc = [\n  'A',\n  'B',\n  'C',\n  'D',\n  'E',\n  'F',\n  'G',\n  'H',\n  'I',\n  'J',\n  'K',\n  'L',\n  'M',\n  'N',\n  'O',\n  'P',\n  'Q',\n  'R',\n  'S',\n  'T',\n  'U',\n  'V',\n  'W',\n  'X',\n  'Y',\n  'Z',\n  'a',\n  'b',\n  'c',\n  'd',\n  'e',\n  'f',\n  'g',\n  'h',\n  'i',\n  'j',\n  'k',\n  'l',\n  'm',\n  'n',\n  'o',\n  'p',\n  'q',\n  'r',\n  's',\n  't',\n  'u',\n  'v',\n  'w',\n  'x',\n  'y',\n  'z',\n  '0',\n  '1',\n  '2',\n  '3',\n  '4',\n  '5',\n  '6',\n  '7',\n  '8',\n  '9',\n  '+',\n  '/',\n]\n\nconst base64urlAbc = [\n  'A',\n  'B',\n  'C',\n  'D',\n  'E',\n  'F',\n  'G',\n  'H',\n  'I',\n  'J',\n  'K',\n  'L',\n  'M',\n  'N',\n  'O',\n  'P',\n  'Q',\n  'R',\n  'S',\n  'T',\n  'U',\n  'V',\n  'W',\n  'X',\n  'Y',\n  'Z',\n  'a',\n  'b',\n  'c',\n  'd',\n  'e',\n  'f',\n  'g',\n  'h',\n  'i',\n  'j',\n  'k',\n  'l',\n  'm',\n  'n',\n  'o',\n  'p',\n  'q',\n  'r',\n  's',\n  't',\n  'u',\n  'v',\n  'w',\n  'x',\n  'y',\n  'z',\n  '0',\n  '1',\n  '2',\n  '3',\n  '4',\n  '5',\n  '6',\n  '7',\n  '8',\n  '9',\n  '-',\n  '_',\n]\n\n// CREDIT: https://gist.github.com/enepomnyaschih/72c423f727d395eeaa09697058238727\nconst base64codes = [\n  255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255,\n  255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 62, 255, 255, 255, 63, 52, 53, 54, 55, 56, 57, 58, 59, 60, 61, 255, 255, 255,\n  0, 255, 255, 255, 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16, 17, 18, 19, 20, 21, 22, 23, 24, 25, 255, 255, 255, 255, 255, 255, 26,\n  27, 28, 29, 30, 31, 32, 33, 34, 35, 36, 37, 38, 39, 40, 41, 42, 43, 44, 45, 46, 47, 48, 49, 50, 51,\n]\n"],"names":["encode","data","uint8","TextEncoder","Uint8Array","_encode","base64abc","encodeBase64Url","base64urlAbc","alpha","skipPadding","result","i","l","length","decode","Error","index","indexOf","missingOctets","endsWith","n","buffer","j","getBase64Code","charCodeAt","subarray","charCode","base64codes","code"],"mappings":"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;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;CACD;AAED,MAAME,eAAe;IACnB;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;CACD;AAED,kFAAkF;AAClF,MAAMoB,cAAc;IAClB;IAAK;IAAK;IAAK;IAAK;IAAK;IAAK;IAAK;IAAK;IAAK;IAAK;IAAK;IAAK;IAAK;IAAK;IAAK;IAAK;IAAK;IAAK;IAAK;IAAK;IAAK;IAAK;IAAK;IAAK;IAAK;IAAK;IAAK;IAAK;IAC5I;IAAK;IAAK;IAAK;IAAK;IAAK;IAAK;IAAK;IAAK;IAAK;IAAK;IAAK;IAAK;IAAK;IAAK;IAAI;IAAK;IAAK;IAAK;IAAI;IAAI;IAAI;IAAI;IAAI;IAAI;IAAI;IAAI;IAAI;IAAI;IAAI;IAAK;IAAK;IAC/I;IAAG;IAAK;IAAK;IAAK;IAAG;IAAG;IAAG;IAAG;IAAG;IAAG;IAAG;IAAG;IAAG;IAAG;IAAI;IAAI;IAAI;IAAI;IAAI;IAAI;IAAI;IAAI;IAAI;IAAI;IAAI;IAAI;IAAI;IAAI;IAAI;IAAI;IAAK;IAAK;IAAK;IAAK;IAAK;IAAK;IAC9I;IAAI;IAAI;IAAI;IAAI;IAAI;IAAI;IAAI;IAAI;IAAI;IAAI;IAAI;IAAI;IAAI;IAAI;IAAI;IAAI;IAAI;IAAI;IAAI;IAAI;IAAI;IAAI;IAAI;IAAI;CACjG"}
|
package/dist/oauth2.d.ts
CHANGED
|
@@ -1,5 +1,28 @@
|
|
|
1
1
|
import type { BigString, DiscordApplicationIntegrationType, OAuth2Scope, PermissionStrings } from '@discordeno/types';
|
|
2
2
|
export declare function createOAuth2Link(options: CreateOAuth2LinkOptions): string;
|
|
3
|
+
/**
|
|
4
|
+
* Generates a code verifier for use in the PKCE extension to OAuth2.
|
|
5
|
+
*
|
|
6
|
+
* @param octetLength - The length of the code verifier in octets (default is 32).
|
|
7
|
+
* @return The base64url encoded code verifier
|
|
8
|
+
*
|
|
9
|
+
* @remarks
|
|
10
|
+
* The entropy of the code verifier should be between 256 and 768 bits (32 to 96 octets), as the resulting base64url encoded string has to be between 43 and 128 characters long.
|
|
11
|
+
*
|
|
12
|
+
* @see https://datatracker.ietf.org/doc/html/rfc7636#section-7.1 for the octet length
|
|
13
|
+
* @see https://datatracker.ietf.org/doc/html/rfc7636#section-4.1 for why 32 octets is the default
|
|
14
|
+
*/
|
|
15
|
+
export declare function generateCodeVerifier(octetLength?: number): string;
|
|
16
|
+
/**
|
|
17
|
+
* Creates a code challenge from the code verifier using the specified method.
|
|
18
|
+
*
|
|
19
|
+
* @param verifier - The code verifier to use.
|
|
20
|
+
* @returns The code challenge.
|
|
21
|
+
*
|
|
22
|
+
* @remarks
|
|
23
|
+
* This performs a SHA-256 hash on the verifier and encodes it using base64url encoding. Discord only supports 'S256' as the code challenge method.
|
|
24
|
+
*/
|
|
25
|
+
export declare function createCodeChallenge(verifier: string): Promise<string>;
|
|
3
26
|
export interface CreateOAuth2LinkOptions {
|
|
4
27
|
/**
|
|
5
28
|
* The type of response
|
|
@@ -65,5 +88,20 @@ export interface CreateOAuth2LinkOptions {
|
|
|
65
88
|
* The application must be configured in the Developer Portal to support the provided `integrationType`.
|
|
66
89
|
*/
|
|
67
90
|
integrationType?: DiscordApplicationIntegrationType;
|
|
91
|
+
/**
|
|
92
|
+
* The code challenge used to verify the authorization request
|
|
93
|
+
*
|
|
94
|
+
* @see https://datatracker.ietf.org/doc/html/rfc7636#section-4.1
|
|
95
|
+
*/
|
|
96
|
+
codeChallenge?: string;
|
|
97
|
+
/**
|
|
98
|
+
* The challenge method used to generate the code challenge
|
|
99
|
+
*
|
|
100
|
+
* @remarks
|
|
101
|
+
* While the RFC allows for the 'plain' value to be set, discord does not allow it
|
|
102
|
+
*
|
|
103
|
+
* @see https://datatracker.ietf.org/doc/html/rfc7636#section-4.2
|
|
104
|
+
*/
|
|
105
|
+
codeChallengeMethod?: 'S256';
|
|
68
106
|
}
|
|
69
107
|
//# sourceMappingURL=oauth2.d.ts.map
|
package/dist/oauth2.d.ts.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"oauth2.d.ts","sourceRoot":"","sources":["../src/oauth2.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,SAAS,EAAE,iCAAiC,EAAE,WAAW,EAAE,iBAAiB,EAAE,MAAM,mBAAmB,CAAA;
|
|
1
|
+
{"version":3,"file":"oauth2.d.ts","sourceRoot":"","sources":["../src/oauth2.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,SAAS,EAAE,iCAAiC,EAAE,WAAW,EAAE,iBAAiB,EAAE,MAAM,mBAAmB,CAAA;AAIrH,wBAAgB,gBAAgB,CAAC,OAAO,EAAE,uBAAuB,GAAG,MAAM,CAmBzE;AAED;;;;;;;;;;;GAWG;AACH,wBAAgB,oBAAoB,CAAC,WAAW,GAAE,MAAW,UAI5D;AAED;;;;;;;;GAQG;AACH,wBAAsB,mBAAmB,CAAC,QAAQ,EAAE,MAAM,mBAGzD;AAED,MAAM,WAAW,uBAAuB;IACtC;;;;;OAKG;IACH,YAAY,CAAC,EAAE,MAAM,GAAG,OAAO,CAAA;IAC/B,gCAAgC;IAChC,QAAQ,EAAE,SAAS,CAAA;IACnB,qCAAqC;IACrC,KAAK,EAAE,WAAW,EAAE,CAAA;IACpB;;;;OAIG;IACH,KAAK,CAAC,EAAE,MAAM,CAAA;IACd;;;;;OAKG;IACH,WAAW,CAAC,EAAE,MAAM,CAAA;IACpB;;;;;;OAMG;IACH,MAAM,CAAC,EAAE,SAAS,GAAG,MAAM,CAAA;IAC3B;;;;;OAKG;IACH,WAAW,CAAC,EAAE,SAAS,GAAG,iBAAiB,EAAE,CAAA;IAC7C;;;;;OAKG;IACH,OAAO,CAAC,EAAE,SAAS,CAAA;IACnB;;;;;OAKG;IACH,kBAAkB,CAAC,EAAE,OAAO,CAAA;IAC5B;;;;;;;;;OASG;IACH,eAAe,CAAC,EAAE,iCAAiC,CAAA;IACnD;;;;OAIG;IACH,aAAa,CAAC,EAAE,MAAM,CAAA;IACtB;;;;;;;OAOG;IACH,mBAAmB,CAAC,EAAE,MAAM,CAAA;CAC7B"}
|
package/dist/oauth2.js
CHANGED
|
@@ -1,3 +1,4 @@
|
|
|
1
|
+
import { encodeBase64Url } from './base64.js';
|
|
1
2
|
import { calculateBits } from './permissions.js';
|
|
2
3
|
export function createOAuth2Link(options) {
|
|
3
4
|
const joinedScopeString = options.scope.join('%20');
|
|
@@ -10,7 +11,38 @@ export function createOAuth2Link(options) {
|
|
|
10
11
|
if (options.guildId) url += `&guild_id=${options.guildId}`;
|
|
11
12
|
if (options.disableGuildSelect !== undefined) url += `&disable_guild_select=${options.disableGuildSelect}`;
|
|
12
13
|
if (options.integrationType) url += `&integration_type=${options.integrationType}`;
|
|
14
|
+
// Options defined by RFC 7636 (https://datatracker.ietf.org/doc/html/rfc7636)
|
|
15
|
+
if (options.codeChallenge) url += `&code_challenge=${options.codeChallenge}`;
|
|
16
|
+
if (options.codeChallengeMethod) url += `&code_challenge_method=${options.codeChallengeMethod}`;
|
|
13
17
|
return url;
|
|
14
18
|
}
|
|
19
|
+
/**
|
|
20
|
+
* Generates a code verifier for use in the PKCE extension to OAuth2.
|
|
21
|
+
*
|
|
22
|
+
* @param octetLength - The length of the code verifier in octets (default is 32).
|
|
23
|
+
* @return The base64url encoded code verifier
|
|
24
|
+
*
|
|
25
|
+
* @remarks
|
|
26
|
+
* The entropy of the code verifier should be between 256 and 768 bits (32 to 96 octets), as the resulting base64url encoded string has to be between 43 and 128 characters long.
|
|
27
|
+
*
|
|
28
|
+
* @see https://datatracker.ietf.org/doc/html/rfc7636#section-7.1 for the octet length
|
|
29
|
+
* @see https://datatracker.ietf.org/doc/html/rfc7636#section-4.1 for why 32 octets is the default
|
|
30
|
+
*/ export function generateCodeVerifier(octetLength = 32) {
|
|
31
|
+
const randomBytes = new Uint8Array(octetLength);
|
|
32
|
+
crypto.getRandomValues(randomBytes);
|
|
33
|
+
return encodeBase64Url(randomBytes);
|
|
34
|
+
}
|
|
35
|
+
/**
|
|
36
|
+
* Creates a code challenge from the code verifier using the specified method.
|
|
37
|
+
*
|
|
38
|
+
* @param verifier - The code verifier to use.
|
|
39
|
+
* @returns The code challenge.
|
|
40
|
+
*
|
|
41
|
+
* @remarks
|
|
42
|
+
* This performs a SHA-256 hash on the verifier and encodes it using base64url encoding. Discord only supports 'S256' as the code challenge method.
|
|
43
|
+
*/ export async function createCodeChallenge(verifier) {
|
|
44
|
+
const hashed = await crypto.subtle.digest('SHA-256', new TextEncoder().encode(verifier));
|
|
45
|
+
return encodeBase64Url(hashed);
|
|
46
|
+
}
|
|
15
47
|
|
|
16
|
-
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uL3NyYy9vYXV0aDIudHMiXSwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHR5cGUgeyBCaWdTdHJpbmcsIERpc2NvcmRBcHBsaWNhdGlvbkludGVncmF0aW9uVHlwZSwgT0F1dGgyU2NvcGUsIFBlcm1pc3Npb25TdHJpbmdzIH0gZnJvbSAnQGRpc2NvcmRlbm8vdHlwZXMnXG5pbXBvcnQgeyBjYWxjdWxhdGVCaXRzIH0gZnJvbSAnLi9wZXJtaXNzaW9ucy5qcydcblxuZXhwb3J0IGZ1bmN0aW9uIGNyZWF0ZU9BdXRoMkxpbmsob3B0aW9uczogQ3JlYXRlT0F1dGgyTGlua09wdGlvbnMpOiBzdHJpbmcge1xuICBjb25zdCBqb2luZWRTY29wZVN0cmluZyA9IG9wdGlvbnMuc2NvcGUuam9pbignJTIwJylcblxuICBsZXQgdXJsID0gYGh0dHBzOi8vZGlzY29yZC5jb20vb2F1dGgyL2F1dGhvcml6ZT9jbGllbnRfaWQ9JHtvcHRpb25zLmNsaWVudElkfSZzY29wZT0ke2pvaW5lZFNjb3BlU3RyaW5nfWBcblxuICBpZiAob3B0aW9ucy5yZXNwb25zZVR5cGUpIHVybCArPSBgJnJlc3BvbnNlX3R5cGU9JHtvcHRpb25zLnJlc3BvbnNlVHlwZX1gXG4gIGlmIChvcHRpb25zLnN0YXRlKSB1cmwgKz0gYCZzdGF0ZT0ke2VuY29kZVVSSUNvbXBvbmVudChvcHRpb25zLnN0YXRlKX1gXG4gIGlmIChvcHRpb25zLnJlZGlyZWN0VXJpKSB1cmwgKz0gYCZyZWRpcmVjdF91cmk9JHtlbmNvZGVVUklDb21wb25lbnQob3B0aW9ucy5yZWRpcmVjdFVyaSl9YFxuICBpZiAob3B0aW9ucy5wcm9tcHQpIHVybCArPSBgJnByb21wdD0ke29wdGlvbnMucHJvbXB0fWBcbiAgaWYgKG9wdGlvbnMucGVybWlzc2lvbnMpIHVybCArPSBgJnBlcm1pc3Npb25zPSR7QXJyYXkuaXNBcnJheShvcHRpb25zLnBlcm1pc3Npb25zKSA/IGNhbGN1bGF0ZUJpdHMob3B0aW9ucy5wZXJtaXNzaW9ucykgOiBvcHRpb25zLnBlcm1pc3Npb25zfWBcbiAgaWYgKG9wdGlvbnMuZ3VpbGRJZCkgdXJsICs9IGAmZ3VpbGRfaWQ9JHtvcHRpb25zLmd1aWxkSWR9YFxuICBpZiAob3B0aW9ucy5kaXNhYmxlR3VpbGRTZWxlY3QgIT09IHVuZGVmaW5lZCkgdXJsICs9IGAmZGlzYWJsZV9ndWlsZF9zZWxlY3Q9JHtvcHRpb25zLmRpc2FibGVHdWlsZFNlbGVjdH1gXG4gIGlmIChvcHRpb25zLmludGVncmF0aW9uVHlwZSkgdXJsICs9IGAmaW50ZWdyYXRpb25fdHlwZT0ke29wdGlvbnMuaW50ZWdyYXRpb25UeXBlfWBcblxuICByZXR1cm4gdXJsXG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgQ3JlYXRlT0F1dGgyTGlua09wdGlvbnMge1xuICAvKipcbiAgICogVGhlIHR5cGUgb2YgcmVzcG9uc2VcbiAgICpcbiAgICogQHJlbWFya3NcbiAgICogU2hvdWxkIGJlIGRlZmluZWQgb25seSBpZiB1c2luZyBlaXRoZXIgT0F1dGgyIGF1dGhvcml6YXRpb24sIGltcGxpY2l0IG9yIG5vdCwgb3IgW2FkdmFuY2VkIGJvdCBhdXRob3JpemF0aW9uXShodHRwczovL2Rpc2NvcmQuY29tL2RldmVsb3BlcnMvZG9jcy90b3BpY3Mvb2F1dGgyI2FkdmFuY2VkLWJvdC1hdXRob3JpemF0aW9uKVxuICAgKi9cbiAgcmVzcG9uc2VUeXBlPzogJ2NvZGUnIHwgJ3Rva2VuJ1xuICAvKiogVGhlIGlkIG9mIHRoZSBhcHBsaWNhdGlvbiAqL1xuICBjbGllbnRJZDogQmlnU3RyaW5nXG4gIC8qKiBUaGUgc2NvcGVzIGZvciB0aGUgYXBwbGljYXRpb24gKi9cbiAgc2NvcGU6IE9BdXRoMlNjb3BlW11cbiAgLyoqXG4gICAqIFRoZSBvcHRpb25hbCBzdGF0ZSBmb3Igc2VjdXJpdHlcbiAgICpcbiAgICogQHNlZSBodHRwczovL2Rpc2NvcmQuY29tL2RldmVsb3BlcnMvZG9jcy90b3BpY3Mvb2F1dGgyI3N0YXRlLWFuZC1zZWN1cml0eVxuICAgKi9cbiAgc3RhdGU/OiBzdHJpbmdcbiAgLyoqXG4gICAqIFRoZSByZWRpcmVjdCB1cmkgZm9yIGFmdGVyIHRoZSBhdXRoZW50aWNhdGlvblxuICAgKlxuICAgKiBAcmVtYXJrc1xuICAgKiBTaG91bGQgYmUgZGVmaW5lZCBvbmx5IGlmIHVzaW5nIGVpdGhlciBPQXV0aDIgYXV0aG9yaXphdGlvbiwgaW1wbGljaXQgb3Igbm90LCBvciBbYWR2YW5jZWQgYm90IGF1dGhvcml6YXRpb25dKGh0dHBzOi8vZGlzY29yZC5jb20vZGV2ZWxvcGVycy9kb2NzL3RvcGljcy9vYXV0aDIjYWR2YW5jZWQtYm90LWF1dGhvcml6YXRpb24pXG4gICAqL1xuICByZWRpcmVjdFVyaT86IHN0cmluZ1xuICAvKipcbiAgICogVGhlIHR5cGUgb2YgcHJvbXB0IHRvIGdpdmUgdG8gdGhlIHVzZXJcbiAgICpcbiAgICogQHJlbWFya3NcbiAgICogSWYgc2V0IHRvIGBub25lYCwgaXQgd2lsbCBza2lwIHRoZSBhdXRob3JpemF0aW9uIHNjcmVlbiBhbmQgcmVkaXJlY3QgdGhlbSBiYWNrIHRvIHlvdXIgcmVkaXJlY3QgVVJJIHdpdGhvdXQgcmVxdWVzdGluZyB0aGVpciBhdXRob3JpemF0aW9uLlxuICAgKiBGb3IgcGFzc3Rocm91Z2ggc2NvcGVzLCBsaWtlIGJvdCBhbmQgd2ViaG9vay5pbmNvbWluZywgYXV0aG9yaXphdGlvbiBpcyBhbHdheXMgcmVxdWlyZWQuXG4gICAqL1xuICBwcm9tcHQ/OiAnY29uc2VudCcgfCAnbm9uZSdcbiAgLyoqXG4gICAqIFRoZSBwZXJtaXNzaW9ucyBvZiB0aGUgaW52aXRlZCBib3RcbiAgICpcbiAgICogQHJlbWFya3NcbiAgICogU2hvdWxkIGJlIGRlZmluZWQgb25seSBpbiBhIFtib3QgYXV0aG9yaXphdGlvbiBmbG93XShodHRwczovL2Rpc2NvcmQuY29tL2RldmVsb3BlcnMvZG9jcy90b3BpY3Mvb2F1dGgyI2JvdC1hdXRob3JpemF0aW9uLWZsb3cpIG9yIHdpdGggW2FkdmFuY2VkIGJvdCBhdXRob3JpemF0aW9uXShodHRwczovL2Rpc2NvcmQuY29tL2RldmVsb3BlcnMvZG9jcy90b3BpY3Mvb2F1dGgyI2FkdmFuY2VkLWJvdC1hdXRob3JpemF0aW9uKVxuICAgKi9cbiAgcGVybWlzc2lvbnM/OiBCaWdTdHJpbmcgfCBQZXJtaXNzaW9uU3RyaW5nc1tdXG4gIC8qKlxuICAgKiBQcmUtZmlsbHMgdGhlIGRyb3Bkb3duIHBpY2tlciB3aXRoIGEgZ3VpbGQgZm9yIHRoZSB1c2VyXG4gICAqXG4gICAqIEByZW1hcmtzXG4gICAqIFNob3VsZCBiZSBkZWZpbmVkIG9ubHkgaW4gYSBbYm90IGF1dGhvcml6YXRpb24gZmxvd10oaHR0cHM6Ly9kaXNjb3JkLmNvbS9kZXZlbG9wZXJzL2RvY3MvdG9waWNzL29hdXRoMiNib3QtYXV0aG9yaXphdGlvbi1mbG93KSBvciB3aXRoIFthZHZhbmNlZCBib3QgYXV0aG9yaXphdGlvbl0oaHR0cHM6Ly9kaXNjb3JkLmNvbS9kZXZlbG9wZXJzL2RvY3MvdG9waWNzL29hdXRoMiNhZHZhbmNlZC1ib3QtYXV0aG9yaXphdGlvbikgb3Igd2l0aCB0aGUgYHdlYmhvb2suaW5jb21pbmdgIHNjb3BlXG4gICAqL1xuICBndWlsZElkPzogQmlnU3RyaW5nXG4gIC8qKlxuICAgKiBEaXNhbGxvd3MgdGhlIHVzZXIgZnJvbSBjaGFuZ2luZyB0aGUgZ3VpbGQgZHJvcGRvd24gaWYgc2V0IHRvIHRydWVcbiAgICpcbiAgICogQHJlbWFya3NcbiAgICogU2hvdWxkIGJlIGRlZmluZWQgb25seSBpbiBhIFtib3QgYXV0aG9yaXphdGlvbiBmbG93XShodHRwczovL2Rpc2NvcmQuY29tL2RldmVsb3BlcnMvZG9jcy90b3BpY3Mvb2F1dGgyI2JvdC1hdXRob3JpemF0aW9uLWZsb3cpLCB3aXRoIFthZHZhbmNlZCBib3QgYXV0aG9yaXphdGlvbl0oaHR0cHM6Ly9kaXNjb3JkLmNvbS9kZXZlbG9wZXJzL2RvY3MvdG9waWNzL29hdXRoMiNhZHZhbmNlZC1ib3QtYXV0aG9yaXphdGlvbikgb3Igd2l0aCB0aGUgYHdlYmhvb2suaW5jb21pbmdgIHNjb3BlXG4gICAqL1xuICBkaXNhYmxlR3VpbGRTZWxlY3Q/OiBib29sZWFuXG4gIC8qKlxuICAgKiBTcGVjaWZpZXMgdGhlIGluc3RhbGxhdGlvbiBjb250ZXh0IGZvciB0aGUgYXV0aG9yaXphdGlvblxuICAgKlxuICAgKiBAcmVtYXJrc1xuICAgKiBTaG91bGQgYmUgZGVmaW5lZCBvbmx5IHdoZW4ge0BsaW5rIHNjb3BlfSBpbmNsdWRlcyB7QGxpbmsgT0F1dGgyU2NvcGUuQXBwbGljYXRpb25zQ29tbWFuZHN9LlxuICAgKlxuICAgKiBXaGVuIHNldCB0byBHdWlsZEluc3RhbGwgKDApIHRoZSBhcHBsaWNhdGlvbiB3aWxsIGJlIGF1dGhvcml6ZWQgZm9yIGluc3RhbGxhdGlvbiB0byBhIHNlcnZlciwgYW5kIHdoZW4gc2V0IHRvIFVzZXJJbnN0YWxsICgxKSB0aGUgYXBwbGljYXRpb24gd2lsbCBiZSBhdXRob3JpemVkIGZvciBpbnN0YWxsYXRpb24gdG8gYSB1c2VyLlxuICAgKlxuICAgKiBUaGUgYXBwbGljYXRpb24gbXVzdCBiZSBjb25maWd1cmVkIGluIHRoZSBEZXZlbG9wZXIgUG9ydGFsIHRvIHN1cHBvcnQgdGhlIHByb3ZpZGVkIGBpbnRlZ3JhdGlvblR5cGVgLlxuICAgKi9cbiAgaW50ZWdyYXRpb25UeXBlPzogRGlzY29yZEFwcGxpY2F0aW9uSW50ZWdyYXRpb25UeXBlXG59XG4iXSwibmFtZXMiOlsiY2FsY3VsYXRlQml0cyIsImNyZWF0ZU9BdXRoMkxpbmsiLCJvcHRpb25zIiwiam9pbmVkU2NvcGVTdHJpbmciLCJzY29wZSIsImpvaW4iLCJ1cmwiLCJjbGllbnRJZCIsInJlc3BvbnNlVHlwZSIsInN0YXRlIiwiZW5jb2RlVVJJQ29tcG9uZW50IiwicmVkaXJlY3RVcmkiLCJwcm9tcHQiLCJwZXJtaXNzaW9ucyIsIkFycmF5IiwiaXNBcnJheSIsImd1aWxkSWQiLCJkaXNhYmxlR3VpbGRTZWxlY3QiLCJ1bmRlZmluZWQiLCJpbnRlZ3JhdGlvblR5cGUiXSwibWFwcGluZ3MiOiJBQUNBLFNBQVNBLGFBQWEsUUFBUSxtQkFBa0I7QUFFaEQsT0FBTyxTQUFTQyxpQkFBaUJDLE9BQWdDO0lBQy9ELE1BQU1DLG9CQUFvQkQsUUFBUUUsS0FBSyxDQUFDQyxJQUFJLENBQUM7SUFFN0MsSUFBSUMsTUFBTSxDQUFDLCtDQUErQyxFQUFFSixRQUFRSyxRQUFRLENBQUMsT0FBTyxFQUFFSixtQkFBbUI7SUFFekcsSUFBSUQsUUFBUU0sWUFBWSxFQUFFRixPQUFPLENBQUMsZUFBZSxFQUFFSixRQUFRTSxZQUFZLEVBQUU7SUFDekUsSUFBSU4sUUFBUU8sS0FBSyxFQUFFSCxPQUFPLENBQUMsT0FBTyxFQUFFSSxtQkFBbUJSLFFBQVFPLEtBQUssR0FBRztJQUN2RSxJQUFJUCxRQUFRUyxXQUFXLEVBQUVMLE9BQU8sQ0FBQyxjQUFjLEVBQUVJLG1CQUFtQlIsUUFBUVMsV0FBVyxHQUFHO0lBQzFGLElBQUlULFFBQVFVLE1BQU0sRUFBRU4sT0FBTyxDQUFDLFFBQVEsRUFBRUosUUFBUVUsTUFBTSxFQUFFO0lBQ3RELElBQUlWLFFBQVFXLFdBQVcsRUFBRVAsT0FBTyxDQUFDLGFBQWEsRUFBRVEsTUFBTUMsT0FBTyxDQUFDYixRQUFRVyxXQUFXLElBQUliLGNBQWNFLFFBQVFXLFdBQVcsSUFBSVgsUUFBUVcsV0FBVyxFQUFFO0lBQy9JLElBQUlYLFFBQVFjLE9BQU8sRUFBRVYsT0FBTyxDQUFDLFVBQVUsRUFBRUosUUFBUWMsT0FBTyxFQUFFO0lBQzFELElBQUlkLFFBQVFlLGtCQUFrQixLQUFLQyxXQUFXWixPQUFPLENBQUMsc0JBQXNCLEVBQUVKLFFBQVFlLGtCQUFrQixFQUFFO0lBQzFHLElBQUlmLFFBQVFpQixlQUFlLEVBQUViLE9BQU8sQ0FBQyxrQkFBa0IsRUFBRUosUUFBUWlCLGVBQWUsRUFBRTtJQUVsRixPQUFPYjtBQUNUIn0=
|
|
48
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../src/oauth2.ts"],"sourcesContent":["import type { BigString, DiscordApplicationIntegrationType, OAuth2Scope, PermissionStrings } from '@discordeno/types'\nimport { encodeBase64Url } from './base64.js'\nimport { calculateBits } from './permissions.js'\n\nexport function createOAuth2Link(options: CreateOAuth2LinkOptions): string {\n  const joinedScopeString = options.scope.join('%20')\n\n  let url = `https://discord.com/oauth2/authorize?client_id=${options.clientId}&scope=${joinedScopeString}`\n\n  if (options.responseType) url += `&response_type=${options.responseType}`\n  if (options.state) url += `&state=${encodeURIComponent(options.state)}`\n  if (options.redirectUri) url += `&redirect_uri=${encodeURIComponent(options.redirectUri)}`\n  if (options.prompt) url += `&prompt=${options.prompt}`\n  if (options.permissions) url += `&permissions=${Array.isArray(options.permissions) ? calculateBits(options.permissions) : options.permissions}`\n  if (options.guildId) url += `&guild_id=${options.guildId}`\n  if (options.disableGuildSelect !== undefined) url += `&disable_guild_select=${options.disableGuildSelect}`\n  if (options.integrationType) url += `&integration_type=${options.integrationType}`\n\n  // Options defined by RFC 7636 (https://datatracker.ietf.org/doc/html/rfc7636)\n  if (options.codeChallenge) url += `&code_challenge=${options.codeChallenge}`\n  if (options.codeChallengeMethod) url += `&code_challenge_method=${options.codeChallengeMethod}`\n\n  return url\n}\n\n/**\n * Generates a code verifier for use in the PKCE extension to OAuth2.\n *\n * @param octetLength - The length of the code verifier in octets (default is 32).\n * @return The base64url encoded code verifier\n *\n * @remarks\n * The entropy of the code verifier should be between 256 and 768 bits (32 to 96 octets), as the resulting base64url encoded string has to be between 43 and 128 characters long.\n *\n * @see https://datatracker.ietf.org/doc/html/rfc7636#section-7.1 for the octet length\n * @see https://datatracker.ietf.org/doc/html/rfc7636#section-4.1 for why 32 octets is the default\n */\nexport function generateCodeVerifier(octetLength: number = 32) {\n  const randomBytes = new Uint8Array(octetLength)\n  crypto.getRandomValues(randomBytes)\n  return encodeBase64Url(randomBytes)\n}\n\n/**\n * Creates a code challenge from the code verifier using the specified method.\n *\n * @param verifier - The code verifier to use.\n * @returns The code challenge.\n *\n * @remarks\n * This performs a SHA-256 hash on the verifier and encodes it using base64url encoding. Discord only supports 'S256' as the code challenge method.\n */\nexport async function createCodeChallenge(verifier: string) {\n  const hashed = await crypto.subtle.digest('SHA-256', new TextEncoder().encode(verifier))\n  return encodeBase64Url(hashed)\n}\n\nexport interface CreateOAuth2LinkOptions {\n  /**\n   * The type of response\n   *\n   * @remarks\n   * Should be defined only if using either OAuth2 authorization, implicit or not, or [advanced bot authorization](https://discord.com/developers/docs/topics/oauth2#advanced-bot-authorization)\n   */\n  responseType?: 'code' | 'token'\n  /** The id of the application */\n  clientId: BigString\n  /** The scopes for the application */\n  scope: OAuth2Scope[]\n  /**\n   * The optional state for security\n   *\n   * @see https://discord.com/developers/docs/topics/oauth2#state-and-security\n   */\n  state?: string\n  /**\n   * The redirect uri for after the authentication\n   *\n   * @remarks\n   * Should be defined only if using either OAuth2 authorization, implicit or not, or [advanced bot authorization](https://discord.com/developers/docs/topics/oauth2#advanced-bot-authorization)\n   */\n  redirectUri?: string\n  /**\n   * The type of prompt to give to the user\n   *\n   * @remarks\n   * If set to `none`, it will skip the authorization screen and redirect them back to your redirect URI without requesting their authorization.\n   * For passthrough scopes, like bot and webhook.incoming, authorization is always required.\n   */\n  prompt?: 'consent' | 'none'\n  /**\n   * The permissions of the invited bot\n   *\n   * @remarks\n   * Should be defined only in a [bot authorization flow](https://discord.com/developers/docs/topics/oauth2#bot-authorization-flow) or with [advanced bot authorization](https://discord.com/developers/docs/topics/oauth2#advanced-bot-authorization)\n   */\n  permissions?: BigString | PermissionStrings[]\n  /**\n   * Pre-fills the dropdown picker with a guild for the user\n   *\n   * @remarks\n   * Should be defined only in a [bot authorization flow](https://discord.com/developers/docs/topics/oauth2#bot-authorization-flow) or with [advanced bot authorization](https://discord.com/developers/docs/topics/oauth2#advanced-bot-authorization) or with the `webhook.incoming` scope\n   */\n  guildId?: BigString\n  /**\n   * Disallows the user from changing the guild dropdown if set to true\n   *\n   * @remarks\n   * Should be defined only in a [bot authorization flow](https://discord.com/developers/docs/topics/oauth2#bot-authorization-flow), with [advanced bot authorization](https://discord.com/developers/docs/topics/oauth2#advanced-bot-authorization) or with the `webhook.incoming` scope\n   */\n  disableGuildSelect?: boolean\n  /**\n   * Specifies the installation context for the authorization\n   *\n   * @remarks\n   * Should be defined only when {@link scope} includes {@link OAuth2Scope.ApplicationsCommands}.\n   *\n   * When set to GuildInstall (0) the application will be authorized for installation to a server, and when set to UserInstall (1) the application will be authorized for installation to a user.\n   *\n   * The application must be configured in the Developer Portal to support the provided `integrationType`.\n   */\n  integrationType?: DiscordApplicationIntegrationType\n  /**\n   * The code challenge used to verify the authorization request\n   *\n   * @see https://datatracker.ietf.org/doc/html/rfc7636#section-4.1\n   */\n  codeChallenge?: string\n  /**\n   * The challenge method used to generate the code challenge\n   *\n   * @remarks\n   * While the RFC allows for the 'plain' value to be set, discord does not allow it\n   *\n   * @see https://datatracker.ietf.org/doc/html/rfc7636#section-4.2\n   */\n  codeChallengeMethod?: 'S256'\n}\n"],"names":["encodeBase64Url","calculateBits","createOAuth2Link","options","joinedScopeString","scope","join","url","clientId","responseType","state","encodeURIComponent","redirectUri","prompt","permissions","Array","isArray","guildId","disableGuildSelect","undefined","integrationType","codeChallenge","codeChallengeMethod","generateCodeVerifier","octetLength","randomBytes","Uint8Array","crypto","getRandomValues","createCodeChallenge","verifier","hashed","subtle","digest","TextEncoder","encode"],"mappings":"AACA,SAASA,eAAe,QAAQ,cAAa;AAC7C,SAASC,aAAa,QAAQ,mBAAkB;AAEhD,OAAO,SAASC,iBAAiBC,OAAgC;IAC/D,MAAMC,oBAAoBD,QAAQE,KAAK,CAACC,IAAI,CAAC;IAE7C,IAAIC,MAAM,CAAC,+CAA+C,EAAEJ,QAAQK,QAAQ,CAAC,OAAO,EAAEJ,mBAAmB;IAEzG,IAAID,QAAQM,YAAY,EAAEF,OAAO,CAAC,eAAe,EAAEJ,QAAQM,YAAY,EAAE;IACzE,IAAIN,QAAQO,KAAK,EAAEH,OAAO,CAAC,OAAO,EAAEI,mBAAmBR,QAAQO,KAAK,GAAG;IACvE,IAAIP,QAAQS,WAAW,EAAEL,OAAO,CAAC,cAAc,EAAEI,mBAAmBR,QAAQS,WAAW,GAAG;IAC1F,IAAIT,QAAQU,MAAM,EAAEN,OAAO,CAAC,QAAQ,EAAEJ,QAAQU,MAAM,EAAE;IACtD,IAAIV,QAAQW,WAAW,EAAEP,OAAO,CAAC,aAAa,EAAEQ,MAAMC,OAAO,CAACb,QAAQW,WAAW,IAAIb,cAAcE,QAAQW,WAAW,IAAIX,QAAQW,WAAW,EAAE;IAC/I,IAAIX,QAAQc,OAAO,EAAEV,OAAO,CAAC,UAAU,EAAEJ,QAAQc,OAAO,EAAE;IAC1D,IAAId,QAAQe,kBAAkB,KAAKC,WAAWZ,OAAO,CAAC,sBAAsB,EAAEJ,QAAQe,kBAAkB,EAAE;IAC1G,IAAIf,QAAQiB,eAAe,EAAEb,OAAO,CAAC,kBAAkB,EAAEJ,QAAQiB,eAAe,EAAE;IAElF,8EAA8E;IAC9E,IAAIjB,QAAQkB,aAAa,EAAEd,OAAO,CAAC,gBAAgB,EAAEJ,QAAQkB,aAAa,EAAE;IAC5E,IAAIlB,QAAQmB,mBAAmB,EAAEf,OAAO,CAAC,uBAAuB,EAAEJ,QAAQmB,mBAAmB,EAAE;IAE/F,OAAOf;AACT;AAEA;;;;;;;;;;;CAWC,GACD,OAAO,SAASgB,qBAAqBC,cAAsB,EAAE;IAC3D,MAAMC,cAAc,IAAIC,WAAWF;IACnCG,OAAOC,eAAe,CAACH;IACvB,OAAOzB,gBAAgByB;AACzB;AAEA;;;;;;;;CAQC,GACD,OAAO,eAAeI,oBAAoBC,QAAgB;IACxD,MAAMC,SAAS,MAAMJ,OAAOK,MAAM,CAACC,MAAM,CAAC,WAAW,IAAIC,cAAcC,MAAM,CAACL;IAC9E,OAAO9B,gBAAgB+B;AACzB"}
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@discordeno/utils",
|
|
3
|
-
"version": "22.0.1-next.
|
|
3
|
+
"version": "22.0.1-next.e5063e1",
|
|
4
4
|
"exports": "./dist/index.js",
|
|
5
5
|
"type": "module",
|
|
6
6
|
"license": "Apache-2.0",
|
|
@@ -26,22 +26,22 @@
|
|
|
26
26
|
"test:test-type": "tsc --project tests/tsconfig.json"
|
|
27
27
|
},
|
|
28
28
|
"dependencies": {
|
|
29
|
-
"@discordeno/types": "22.0.1-next.
|
|
29
|
+
"@discordeno/types": "22.0.1-next.e5063e1"
|
|
30
30
|
},
|
|
31
31
|
"devDependencies": {
|
|
32
|
-
"@biomejs/biome": "2.
|
|
33
|
-
"@swc/cli": "^0.7.
|
|
34
|
-
"@swc/core": "^1.
|
|
35
|
-
"@types/chai": "^5.2.
|
|
32
|
+
"@biomejs/biome": "2.3.1",
|
|
33
|
+
"@swc/cli": "^0.7.9",
|
|
34
|
+
"@swc/core": "^1.15.2",
|
|
35
|
+
"@types/chai": "^5.2.3",
|
|
36
36
|
"@types/mocha": "^10.0.10",
|
|
37
|
-
"@types/node": "^24.
|
|
37
|
+
"@types/node": "^24.9.1",
|
|
38
38
|
"@types/sinon": "^17.0.4",
|
|
39
39
|
"c8": "^10.1.3",
|
|
40
|
-
"chai": "^6.0
|
|
41
|
-
"mocha": "^11.7.
|
|
40
|
+
"chai": "^6.2.0",
|
|
41
|
+
"mocha": "^11.7.4",
|
|
42
42
|
"sinon": "^21.0.0",
|
|
43
43
|
"ts-node": "^10.9.2",
|
|
44
44
|
"tsconfig": "*",
|
|
45
|
-
"typescript": "^5.9.
|
|
45
|
+
"typescript": "^5.9.3"
|
|
46
46
|
}
|
|
47
47
|
}
|