forked from MetaMask/utils
-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathbase64.ts
59 lines (53 loc) · 1.55 KB
/
base64.ts
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
26
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
52
53
54
55
56
57
58
59
import type { Struct } from 'superstruct';
import { pattern } from 'superstruct';
import { assert } from './assert';
export type Base64Options = {
/**
* Is the `=` padding at the end required or not.
*
* @default false
*/
// Padding is optional in RFC 4648, that's why the default value is false
paddingRequired?: boolean;
/**
* Which character set should be used.
* The sets are based on {@link https://datatracker.ietf.org/doc/html/rfc4648 RFC 4648}.
*
* @default 'base64'
*/
characterSet?: 'base64' | 'base64url';
};
/**
* Ensure that a provided string-based struct is valid base64.
*
* @param struct - The string based struct.
* @param options - Optional options to specialize base64 validation. See {@link Base64Options} documentation.
* @returns A superstruct validating base64.
*/
export const base64 = <Type extends string, Schema>(
struct: Struct<Type, Schema>,
options: Base64Options = {},
) => {
const paddingRequired = options.paddingRequired ?? false;
const characterSet = options.characterSet ?? 'base64';
let letters: string;
if (characterSet === 'base64') {
letters = String.raw`[A-Za-z0-9+\/]`;
} else {
assert(characterSet === 'base64url');
letters = String.raw`[-_A-Za-z0-9]`;
}
let re: RegExp;
if (paddingRequired) {
re = new RegExp(
`^(?:${letters}{4})*(?:${letters}{3}=|${letters}{2}==)?$`,
'u',
);
} else {
re = new RegExp(
`^(?:${letters}{4})*(?:${letters}{2,3}|${letters}{3}=|${letters}{2}==)?$`,
'u',
);
}
return pattern(struct, re);
};