Skip to content

Commit

Permalink
lib: implement webidl dictionary converter and use it in structuredClone
Browse files Browse the repository at this point in the history
This commit provides a factory to generate `dictionaryConverter`
compliant with the spec. The implemented factory function is used for
the `structuredClone` algorithm with updated test cases.
  • Loading branch information
jazelly committed Oct 22, 2024
1 parent 7ae193d commit d2a6a48
Show file tree
Hide file tree
Showing 3 changed files with 108 additions and 23 deletions.
71 changes: 71 additions & 0 deletions lib/internal/webidl.js
Original file line number Diff line number Diff line change
Expand Up @@ -2,6 +2,7 @@

const {
ArrayPrototypePush,
ArrayPrototypeSort,
MathAbs,
MathMax,
MathMin,
Expand All @@ -12,6 +13,7 @@ const {
NumberMAX_SAFE_INTEGER,
NumberMIN_SAFE_INTEGER,
ObjectAssign,
ObjectPrototypeHasOwnProperty,
ObjectPrototypeIsPrototypeOf,
SafeSet,
String,
Expand Down Expand Up @@ -263,6 +265,74 @@ function type(V) {
}
}

// https://webidl.spec.whatwg.org/#js-dictionary
function createDictionaryConverter(members) {
// The spec requires us to operate members of each dictionary in
// lexicographical order. We are doing all of these in the outer scope to
// reduce the overhead that could happen in the returned function.
const sortedMembers = [];
for (let i = 0; i < members.length; i++) {
const member = members[i];
ArrayPrototypePush(sortedMembers, member);
}

ArrayPrototypeSort(sortedMembers, (a, b) => {
if (a.key === b.key) {
return 0;
}
return a.key < b.key ? -1 : 1;
});

return function(
V,
opts = kEmptyObject,
) {
const typeV = type(V);
if (typeV !== 'Undefined' && typeV !== 'Null' && typeV !== 'Object') {
throw makeException(
'can not be converted to a dictionary',
opts,
);
}

const idlDict = { __proto__: null };
for (let i = 0; i < sortedMembers.length; i++) {
const member = sortedMembers[i];
const key = member.key;
let jsMemberValue;
if (typeV === 'Undefined' || typeV === 'Null') {
jsMemberValue = undefined;
} else {
jsMemberValue = V[key];
}

if (jsMemberValue !== undefined) {
const memberContext = opts.context ? `${key} in ${opts.context}` : `${key}`;
const converter = member.converter;
const idlMemberValue = converter(
jsMemberValue,
{
__proto__: null,
prefix: opts.prefix,
context: memberContext,
},
);
idlDict[key] = idlMemberValue;
} else if (ObjectPrototypeHasOwnProperty(member, 'defaultValue')) {
const idlMemberValue = member.defaultValue;
idlDict[key] = idlMemberValue;
} else if (member.required) {
throw makeException(
`can not be converted because of the missing '${key}'`,
opts,
);
}
}

return idlDict;
};
}

// https://webidl.spec.whatwg.org/#es-sequence
function createSequenceConverter(converter) {
return function(V, opts = kEmptyObject) {
Expand Down Expand Up @@ -318,6 +388,7 @@ module.exports = {
createEnumConverter,
createInterfaceConverter,
createSequenceConverter,
createDictionaryConverter,
evenRound,
makeException,
};
39 changes: 21 additions & 18 deletions lib/internal/worker/js_transferable.js
Original file line number Diff line number Diff line change
Expand Up @@ -5,7 +5,6 @@ const {
} = primordials;
const {
codes: {
ERR_INVALID_ARG_TYPE,
ERR_MISSING_ARGS,
},
} = require('internal/errors');
Expand Down Expand Up @@ -98,29 +97,33 @@ function markTransferMode(obj, cloneable = false, transferable = false) {
obj[transfer_mode_private_symbol] = mode;
}


webidl.converters.StructuredCloneOptions = webidl
.createDictionaryConverter(
[
{
key: 'transfer',
converter: webidl.converters['sequence<object>'],
get defaultValue() {
return [];
},
},
],
);

function structuredClone(value, options) {
if (arguments.length === 0) {
throw new ERR_MISSING_ARGS('The value argument must be specified');
}

// TODO(jazelly): implement generic webidl dictionary converter
const prefix = 'Options';
const optionsType = webidl.type(options);
if (optionsType !== 'Undefined' && optionsType !== 'Null' && optionsType !== 'Object') {
throw new ERR_INVALID_ARG_TYPE(
prefix,
['object', 'null', 'undefined'],
options,
);
}
const key = 'transfer';
const idlOptions = { __proto__: null, [key]: [] };
if (options != null && key in options && options[key] !== undefined) {
idlOptions[key] = webidl.converters['sequence<object>'](options[key], {
const idlOptions = webidl.converters.StructuredCloneOptions(
options,
{
__proto__: null,
context: 'Transfer',
});
}
prefix: "Failed to execute 'structuredClone'",
context: 'Options',
},
);

const serializedData = nativeStructuredClone(value, idlOptions);
return serializedData;
Expand Down
21 changes: 16 additions & 5 deletions test/parallel/test-structuredClone-global.js
Original file line number Diff line number Diff line change
Expand Up @@ -3,12 +3,23 @@
require('../common');
const assert = require('assert');

const prefix = "Failed to execute 'structuredClone'";
const key = 'transfer';
const context = 'Options';
const memberConverterError = `${prefix}: ${key} in ${context} can not be converted to sequence.`;
const dictionaryConverterError = `${prefix}: ${context} can not be converted to a dictionary`;

assert.throws(() => structuredClone(), { code: 'ERR_MISSING_ARGS' });
assert.throws(() => structuredClone(undefined, ''), { code: 'ERR_INVALID_ARG_TYPE' });
assert.throws(() => structuredClone(undefined, 1), { code: 'ERR_INVALID_ARG_TYPE' });
assert.throws(() => structuredClone(undefined, { transfer: 1 }), { code: 'ERR_INVALID_ARG_TYPE' });
assert.throws(() => structuredClone(undefined, { transfer: '' }), { code: 'ERR_INVALID_ARG_TYPE' });
assert.throws(() => structuredClone(undefined, { transfer: null }), { code: 'ERR_INVALID_ARG_TYPE' });
assert.throws(() => structuredClone(undefined, ''),
{ code: 'ERR_INVALID_ARG_TYPE', message: dictionaryConverterError });
assert.throws(() => structuredClone(undefined, 1),
{ code: 'ERR_INVALID_ARG_TYPE', message: dictionaryConverterError });
assert.throws(() => structuredClone(undefined, { transfer: 1 }),
{ code: 'ERR_INVALID_ARG_TYPE', message: memberConverterError });
assert.throws(() => structuredClone(undefined, { transfer: '' }),
{ code: 'ERR_INVALID_ARG_TYPE', message: memberConverterError });
assert.throws(() => structuredClone(undefined, { transfer: null }),
{ code: 'ERR_INVALID_ARG_TYPE', message: memberConverterError });

// Options can be null or undefined.
assert.strictEqual(structuredClone(undefined), undefined);
Expand Down

0 comments on commit d2a6a48

Please sign in to comment.