Skip to content

Commit

Permalink
Fix: 'fromGlobalId' trowing on invalide UTF code points
Browse files Browse the repository at this point in the history
Fixes graphql#358
Note: it's temprorary workaround until we design a better API
  • Loading branch information
IvanGoncharov committed Dec 19, 2021
1 parent d4c533f commit dc5e1dd
Show file tree
Hide file tree
Showing 2 changed files with 29 additions and 8 deletions.
16 changes: 12 additions & 4 deletions src/connection/__tests__/arrayConnection-test.ts
Original file line number Diff line number Diff line change
Expand Up @@ -274,11 +274,19 @@ describe('connectionFromArray()', () => {
});

it('returns all elements if cursors are invalid', () => {
const c = connectionFromArray(arrayABCDE, {
before: 'invalid',
after: 'invalid',
const c1 = connectionFromArray(arrayABCDE, {
before: 'InvalidBase64',
after: 'InvalidBase64',
});
expect(c).to.deep.equal({

const invalidUnicodeInBase64 = '9JCAgA=='; // U+110000
const c2 = connectionFromArray(arrayABCDE, {
before: invalidUnicodeInBase64,
after: invalidUnicodeInBase64,
});

expect(c1).to.deep.equal(c2);
expect(c1).to.deep.equal({
edges: [edgeA, edgeB, edgeC, edgeD, edgeE],
pageInfo: {
startCursor: cursorA,
Expand Down
21 changes: 17 additions & 4 deletions src/utils/base64.ts
Original file line number Diff line number Diff line change
Expand Up @@ -126,29 +126,42 @@ function utf8ArrayToString(input: Array<number>) {
for (let i = 0; i < input.length; ) {
const a = input[i++];
if ((a & 0x80) === 0) {
result += String.fromCodePoint(a);
result += fromCodePoint(a);
continue;
}

const b = input[i++];
if ((a & 0xe0) === 0xc0) {
result += String.fromCodePoint(((a & 0x1f) << 6) | (b & 0x3f));
result += fromCodePoint(((a & 0x1f) << 6) | (b & 0x3f));
continue;
}

const c = input[i++];
if ((a & 0xf0) === 0xe0) {
result += String.fromCodePoint(
result += fromCodePoint(
((a & 0x0f) << 12) | ((b & 0x3f) << 6) | (c & 0x3f),
);
continue;
}

const d = input[i++];
result += String.fromCodePoint(
result += fromCodePoint(
((a & 0x07) << 18) | ((b & 0x3f) << 12) | ((c & 0x3f) << 6) | (d & 0x3f),
);
}

return result;
}

function fromCodePoint(code: number): string {
if (code > 0x10ffff) {
/*
* Previously we used Node's API for parsing Base64 and following code
* Buffer.from(i, 'base64').toString('utf8')
* That silently ignored incorrect input and returned empty string instead
* Let's keep this behaviour for a time being and hopefully fix it in the future.
*/
return '';
}
return String.fromCodePoint(code);
}

0 comments on commit dc5e1dd

Please sign in to comment.