index.js 1.4 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344
  1. 'use strict';
  2. var GetIntrinsic = require('get-intrinsic');
  3. var define = require('define-data-property');
  4. var hasDescriptors = require('has-property-descriptors')();
  5. var gOPD = require('gopd');
  6. var $TypeError = require('es-errors/type');
  7. var $floor = GetIntrinsic('%Math.floor%');
  8. /** @typedef {(...args: unknown[]) => unknown} Func */
  9. /** @type {<T extends Func = Func>(fn: T, length: number, loose?: boolean) => T} */
  10. module.exports = function setFunctionLength(fn, length) {
  11. if (typeof fn !== 'function') {
  12. throw new $TypeError('`fn` is not a function');
  13. }
  14. if (typeof length !== 'number' || length < 0 || length > 0xFFFFFFFF || $floor(length) !== length) {
  15. throw new $TypeError('`length` must be a positive 32-bit integer');
  16. }
  17. var loose = arguments.length > 2 && !!arguments[2];
  18. var functionLengthIsConfigurable = true;
  19. var functionLengthIsWritable = true;
  20. if ('length' in fn && gOPD) {
  21. var desc = gOPD(fn, 'length');
  22. if (desc && !desc.configurable) {
  23. functionLengthIsConfigurable = false;
  24. }
  25. if (desc && !desc.writable) {
  26. functionLengthIsWritable = false;
  27. }
  28. }
  29. if (functionLengthIsConfigurable || functionLengthIsWritable || !loose) {
  30. if (hasDescriptors) {
  31. define(/** @type {Parameters<define>[0]} */ (fn), 'length', length, true, true);
  32. } else {
  33. define(/** @type {Parameters<define>[0]} */ (fn), 'length', length);
  34. }
  35. }
  36. return fn;
  37. };