Skip to content

Commit

Permalink
Merge pull request #42 from ZitaneLabs/task/base64-nopad
Browse files Browse the repository at this point in the history
Update base64 to ignore padding
  • Loading branch information
SplittyDev authored Aug 14, 2023
2 parents da05d3a + 0d4df14 commit bca66fb
Show file tree
Hide file tree
Showing 2 changed files with 32 additions and 19 deletions.
15 changes: 12 additions & 3 deletions hdrop-web-next/src/crypto/Base64.test.ts
Original file line number Diff line number Diff line change
Expand Up @@ -8,7 +8,7 @@ describe('Base64', () => {
// when
const result = Base64.encode(bytes)
// then
expect(result).toBe('AAECAwQFBgc=')
expect(result).toBe('AAECAwQFBgc')
})

test('encode zero bytes', () => {
Expand All @@ -26,10 +26,19 @@ describe('Base64', () => {
// when
const result = Base64.encode(bytes)
// then
expect(result).toBe('SGVsbG8gdGhpcyBpcyBhIHRlc3Q=')
expect(result).toBe('SGVsbG8gdGhpcyBpcyBhIHRlc3Q')
})

test('decode', () => {
// given
const str = 'AAECAwQFBgc'
// when
const result = Base64.decode(str)
// then
expect(result).toEqual(new Uint8Array([0, 1, 2, 3, 4, 5, 6, 7]))
})

test('decode with padding', () => {
// given
const str = 'AAECAwQFBgc='
// when
Expand All @@ -49,7 +58,7 @@ describe('Base64', () => {

test('decode long string', () => {
// given
const str = 'SGVsbG8gdGhpcyBpcyBhIHRlc3Q='
const str = 'SGVsbG8gdGhpcyBpcyBhIHRlc3Q'
const expected = Array.from(new TextEncoder().encode("Hello this is a test"))
// when
const result = Array.from(Base64.decode(str))
Expand Down
36 changes: 20 additions & 16 deletions hdrop-web-next/src/crypto/Base64.ts
Original file line number Diff line number Diff line change
Expand Up @@ -29,28 +29,34 @@ export default class Base64 {
return byte;
}

private static determineMissingOctetCount(len: number): number {
const remainder = len % 4
if (remainder === 1) {
throw new Error("Unable to parse base64 string.")
}
const lut = {
0: 0,
2: 2,
3: 1,
} as Record<number, number>
return lut[remainder]
}

/** Decode a base64 string into a byte array. */
static decode(str: string): Uint8Array {
str = Base64.fromDisplay(str)
if (str.length % 4 !== 0) {
throw new Error("Unable to parse base64 string.")
}
const index = str.indexOf("=")
if (index !== -1 && index < str.length - 2) {
throw new Error("Unable to parse base64 string.")
}
const missingOctets = str.endsWith("==") ? 2 : str.endsWith("=") ? 1 : 0
const n = str.length
const missingOctets = this.determineMissingOctetCount(str.length)
const n = str.length + missingOctets
const result = new Uint8Array(3 * (n / 4))
for (let i = 0, j = 0; i < n; i += 4, j += 3) {
const buffer =
(Base64.decodeChar(str.charCodeAt(i)) << 18) |
(Base64.decodeChar(str.charCodeAt(i + 1)) << 12) |
(Base64.decodeChar(str.charCodeAt(i + 2)) << 6) |
(Base64.decodeChar(str.charCodeAt(i + 3)))
(i + 2 < n ? Base64.decodeChar(str.charCodeAt(i + 2)) << 6 : 0) |
(i + 3 < n ? Base64.decodeChar(str.charCodeAt(i + 3)) : 0)
result[j] = buffer >> 16
result[j + 1] = (buffer >> 8) & 0xFF
result[j + 2] = buffer & 0xFF
if (i + 2 < n) result[j + 1] = (buffer >> 8) & 0xFF
if (i + 3 < n) result[j + 2] = buffer & 0xFF
}
return result.subarray(0, result.length - missingOctets)
}
Expand All @@ -67,19 +73,17 @@ export default class Base64 {
if (i === l + 1) { // 1 octet yet to write
result += ALPHABET[bytes[i - 2] >> 2]
result += ALPHABET[(bytes[i - 2] & 0x03) << 4]
result += "=="
}
if (i === l) { // 2 octets yet to write
result += ALPHABET[bytes[i - 2] >> 2]
result += ALPHABET[((bytes[i - 2] & 0x03) << 4) | (bytes[i - 1] >> 4)]
result += ALPHABET[(bytes[i - 1] & 0x0F) << 2]
result += "="
}
return Base64.toDisplay(result)
}

private static fromDisplay(text: string): string {
return text.replaceAll('-', '+').replaceAll('_', '/')
return text.replaceAll('-', '+').replaceAll('_', '/').replaceAll('=', '')
}

private static toDisplay(base64: string): string {
Expand Down

0 comments on commit bca66fb

Please sign in to comment.