2025-04-02 06:50:39 -04:00

204 lines
4.5 KiB
JavaScript

'use strict';
var conversions = {};
module.exports = conversions;
function sign(x) {
return x < 0 ? -1 : 1;
}
function evenRound(x) {
// Round x to the nearest integer, choosing the even integer if it lies halfway between two.
if (x % 1 === 0.5 && (x & 1) === 0) {
// [even number].5; round down (i.e. floor)
return Math.floor(x);
} else {
return Math.round(x);
}
}
function createNumberConversion(bitLength, typeOpts) {
if (!typeOpts.unsigned) {
--bitLength;
}
const lowerBound = typeOpts.unsigned ? 0 : -Math.pow(2, bitLength);
const upperBound = Math.pow(2, bitLength) - 1;
const moduloVal =
typeOpts.moduloBitLength ?
Math.pow(2, typeOpts.moduloBitLength)
: Math.pow(2, bitLength);
const moduloBound =
typeOpts.moduloBitLength ?
Math.pow(2, typeOpts.moduloBitLength - 1)
: Math.pow(2, bitLength - 1);
return function (V, opts) {
if (!opts) opts = {};
let x = +V;
if (opts.enforceRange) {
if (!Number.isFinite(x)) {
throw new TypeError('Argument is not a finite number');
}
x = sign(x) * Math.floor(Math.abs(x));
if (x < lowerBound || x > upperBound) {
throw new TypeError('Argument is not in byte range');
}
return x;
}
if (!isNaN(x) && opts.clamp) {
x = evenRound(x);
if (x < lowerBound) x = lowerBound;
if (x > upperBound) x = upperBound;
return x;
}
if (!Number.isFinite(x) || x === 0) {
return 0;
}
x = sign(x) * Math.floor(Math.abs(x));
x = x % moduloVal;
if (!typeOpts.unsigned && x >= moduloBound) {
return x - moduloVal;
} else if (typeOpts.unsigned) {
if (x < 0) {
x += moduloVal;
} else if (x === -0) {
// don't return negative zero
return 0;
}
}
return x;
};
}
conversions['void'] = function () {
return undefined;
};
conversions['boolean'] = function (val) {
return !!val;
};
conversions['byte'] = createNumberConversion(8, { unsigned: false });
conversions['octet'] = createNumberConversion(8, { unsigned: true });
conversions['short'] = createNumberConversion(16, { unsigned: false });
conversions['unsigned short'] = createNumberConversion(16, { unsigned: true });
conversions['long'] = createNumberConversion(32, { unsigned: false });
conversions['unsigned long'] = createNumberConversion(32, { unsigned: true });
conversions['long long'] = createNumberConversion(32, {
unsigned: false,
moduloBitLength: 64,
});
conversions['unsigned long long'] = createNumberConversion(32, {
unsigned: true,
moduloBitLength: 64,
});
conversions['double'] = function (V) {
const x = +V;
if (!Number.isFinite(x)) {
throw new TypeError('Argument is not a finite floating-point value');
}
return x;
};
conversions['unrestricted double'] = function (V) {
const x = +V;
if (isNaN(x)) {
throw new TypeError('Argument is NaN');
}
return x;
};
// not quite valid, but good enough for JS
conversions['float'] = conversions['double'];
conversions['unrestricted float'] = conversions['unrestricted double'];
conversions['DOMString'] = function (V, opts) {
if (!opts) opts = {};
if (opts.treatNullAsEmptyString && V === null) {
return '';
}
return String(V);
};
conversions['ByteString'] = function (V, opts) {
const x = String(V);
let c = undefined;
for (let i = 0; (c = x.codePointAt(i)) !== undefined; ++i) {
if (c > 255) {
throw new TypeError('Argument is not a valid bytestring');
}
}
return x;
};
conversions['USVString'] = function (V) {
const S = String(V);
const n = S.length;
const U = [];
for (let i = 0; i < n; ++i) {
const c = S.charCodeAt(i);
if (c < 0xd800 || c > 0xdfff) {
U.push(String.fromCodePoint(c));
} else if (0xdc00 <= c && c <= 0xdfff) {
U.push(String.fromCodePoint(0xfffd));
} else {
if (i === n - 1) {
U.push(String.fromCodePoint(0xfffd));
} else {
const d = S.charCodeAt(i + 1);
if (0xdc00 <= d && d <= 0xdfff) {
const a = c & 0x3ff;
const b = d & 0x3ff;
U.push(String.fromCodePoint((2 << 15) + (2 << 9) * a + b));
++i;
} else {
U.push(String.fromCodePoint(0xfffd));
}
}
}
}
return U.join('');
};
conversions['Date'] = function (V, opts) {
if (!(V instanceof Date)) {
throw new TypeError('Argument is not a Date object');
}
if (isNaN(V)) {
return undefined;
}
return V;
};
conversions['RegExp'] = function (V, opts) {
if (!(V instanceof RegExp)) {
V = new RegExp(V);
}
return V;
};