Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

fix spec char issue #235

Open
wants to merge 7 commits into
base: main
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
55 changes: 44 additions & 11 deletions lib/tdf3/src/binary.ts
Original file line number Diff line number Diff line change
Expand Up @@ -44,6 +44,10 @@ export abstract class Binary {

abstract asArrayBuffer(): ArrayBuffer;

abstract asB64(): string;

abstract asHex(): string;

abstract asByteArray(): number[];

abstract asString(): string;
Expand All @@ -60,7 +64,12 @@ function adjustSliceParams(length: number, start: number, end?: number): [number
if (end && end < 0) {
end = length + end;
}
return [start, end];
const result = [start];
if (end) {
result.push(end);
}
Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Because [number, number | undefined] is not the same with [number, number?]


return result as [number, number?];
}

class ArrayBufferBinary extends Binary {
Expand All @@ -87,11 +96,18 @@ class ArrayBufferBinary extends Binary {

override asString(): string {
const uint8Array = new Uint8Array(this.value);
let str = '';
for (let i = 0; i < uint8Array.length; i++) {
str = str + String.fromCharCode(uint8Array[i]);
}
return str;
return new TextDecoder('utf-8').decode(uint8Array);
}

override asB64(): string {
const uint8Array = new Uint8Array(this.value);
return window.btoa([...uint8Array].map((byte) => String.fromCharCode(byte)).join(''));
}

override asHex(): string {
return Array.from(new Uint8Array(this.value))
.map((byte) => byte.toString(16).padStart(2, '0'))
.join('');
}

override isArrayBuffer(): boolean {
Expand Down Expand Up @@ -127,11 +143,16 @@ class ByteArrayBinary extends Binary {

override asString(): string {
const uint8Array = new Uint8Array(this.value);
let str = '';
for (let i = 0; i < uint8Array.length; i++) {
str = str + String.fromCharCode(uint8Array[i]);
}
return str;
return new TextDecoder('utf-8').decode(uint8Array);
}

override asB64(): string {
const uint8Array = new Uint8Array(this.value);
return window.btoa([...uint8Array].map((byte) => String.fromCharCode(byte)).join(''));
}

override asHex(): string {
return this.value.map((byte) => byte.toString(16).padStart(2, '0')).join('');
}

override isByteArray(): boolean {
Expand Down Expand Up @@ -178,6 +199,18 @@ class StringBinary extends Binary {
return this.value;
}

override asB64(): string {
const uint8Array = new TextEncoder().encode(this.asString());
const charArray = Array.from(uint8Array, (byte) => String.fromCharCode(byte));
return btoa(charArray.join(''));
}

override asHex(): string {
return Array.from(new TextEncoder().encode(this.value))
.map((byte) => byte.toString(16).padStart(2, '0'))
.join('');
}

override isString(): boolean {
return true;
}
Expand Down
6 changes: 3 additions & 3 deletions lib/tdf3/src/models/encryption-information.ts
Original file line number Diff line number Diff line change
Expand Up @@ -106,8 +106,8 @@ export class SplitKey {
);

const encryptedMetadataOb = {
ciphertext: base64.encode(encryptedMetadataResult.payload.asString()),
iv: base64.encode(keyInfo.unwrappedKeyIvBinary.asString()),
ciphertext: encryptedMetadataResult.payload.asB64(),
iv: keyInfo.unwrappedKeyIvBinary.asB64(),
};

const encryptedMetadataStr = JSON.stringify(encryptedMetadataOb);
Expand Down Expand Up @@ -143,7 +143,7 @@ export class SplitKey {
method: {
algorithm,
isStreamable: false,
iv: base64.encode(keyInfo.unwrappedKeyIvBinary.asString()),
iv: keyInfo.unwrappedKeyIvBinary.asB64(),
},
integrityInformation: {
rootSignature: {
Expand Down
5 changes: 2 additions & 3 deletions lib/tdf3/src/models/key-access.ts
Original file line number Diff line number Diff line change
Expand Up @@ -40,7 +40,7 @@ export class Wrapped {
type: 'wrapped',
url: this.url,
protocol: 'kas',
wrappedKey: base64.encode(wrappedKeyBinary.asString()),
wrappedKey: wrappedKeyBinary.asB64(),
encryptedMetadata: base64.encode(encryptedMetadataStr),
policyBinding: base64.encode(policyBinding),
};
Expand Down Expand Up @@ -77,8 +77,7 @@ export class Remote {
this.publicKey
);

// this.wrappedKey = wrappedKeyBinary.asBuffer().toString('hex');
this.wrappedKey = base64.encode(wrappedKeyBinary.asString());
this.wrappedKey = wrappedKeyBinary.asB64();

this.keyAccessObject = {
type: 'remote',
Expand Down
10 changes: 4 additions & 6 deletions lib/tdf3/src/tdf.ts
Original file line number Diff line number Diff line change
Expand Up @@ -25,6 +25,7 @@ import {
ZipReader,
ZipWriter,
base64ToBuffer,
bufferToBase64,
fromUrl,
isAppIdProviderCheck,
keyMerge,
Expand Down Expand Up @@ -193,7 +194,7 @@ export class TDF extends EventEmitter {
transferUrl,
transferBaseUrl: origin,
manifest: base64.encode(exportManifest),
payload: base64.encodeArrayBuffer(payload.buffer),
payload: bufferToBase64(payload),
});

return new TextEncoder().encode(fullHtmlString);
Expand Down Expand Up @@ -475,10 +476,7 @@ export class TDF extends EventEmitter {
return hex.encodeArrayBuffer(new Uint8Array(payloadBinary.asByteArray()).slice(-16).buffer);
case 'hs256':
// simple hmac is the default
return await this.cryptoService.hmac(
hex.encodeArrayBuffer(new Uint8Array(unwrappedKeyBinary.asByteArray()).buffer),
new TextDecoder().decode(new Uint8Array(payloadBinary.asByteArray()).buffer)
);
return await this.cryptoService.hmac(unwrappedKeyBinary.asHex(), payloadBinary.asString());
default:
throw new IllegalArgumentError(`Unsupported signature alg [${algorithmType}]`);
}
Expand Down Expand Up @@ -781,7 +779,7 @@ export class TDF extends EventEmitter {
if (upsertResponse) {
plaintextStream.upsertResponse = upsertResponse;
plaintextStream.tdfSize = totalByteCount;
plaintextStream.KEK = payloadKey ? null : btoa(kek.payload.asString());
plaintextStream.KEK = payloadKey ? null : kek.payload.asB64();
plaintextStream.algorithm = manifest.encryptionInformation.method.algorithm;
}

Expand Down
8 changes: 8 additions & 0 deletions lib/tdf3/src/utils/index.ts
Original file line number Diff line number Diff line change
Expand Up @@ -10,6 +10,14 @@ export function base64ToBuffer(b64: string): Uint8Array {
return Uint8Array.from(atob(b64).split(''), (c) => c.charCodeAt(0));
}

export function bufferToBase64(slice: Uint8Array) {
return btoa(
Array.from(slice)
.map((byte) => String.fromCharCode(byte))
.join('')
);
}

export function isAppIdProviderCheck(
provider: AuthProvider | AppIdAuthProvider
): provider is AppIdAuthProvider {
Expand Down
2 changes: 1 addition & 1 deletion lib/tests/mocha/unit/tdf.spec.ts
Original file line number Diff line number Diff line change
Expand Up @@ -74,7 +74,7 @@ describe('TDF', () => {
const wrapped = TDF.wrapHtml(cipherText, JSON.stringify({ thisIs: 'metadata' }), transferUrl);
expect(TDF.unwrapHtml(wrapped)).to.eql(cipherText);
expect(TDF.unwrapHtml(wrapped.buffer)).to.eql(cipherText);
expect(TDF.unwrapHtml(new TextDecoder().decode(wrapped))).to.eql(cipherText);
expect(TDF.unwrapHtml(new TextDecoder('utf-8').decode(wrapped))).to.eql(cipherText);
});

it('should fail on invalid cypher param', () => {
Expand Down