Skip to content

Commit

Permalink
cleanup
Browse files Browse the repository at this point in the history
  • Loading branch information
samthor committed Jul 1, 2020
1 parent 9788823 commit c8c1d8b
Show file tree
Hide file tree
Showing 4 changed files with 71 additions and 25 deletions.
10 changes: 10 additions & 0 deletions build.js
Original file line number Diff line number Diff line change
Expand Up @@ -14,6 +14,9 @@
* the License.
*/

const fs = require('fs');
const path = require('path');

const ClosureCompiler = require('google-closure-compiler').compiler;

const closureCompiler = new ClosureCompiler({
Expand All @@ -28,6 +31,13 @@ const closureCompiler = new ClosureCompiler({
output_wrapper: '(function(){%output%})();', // don't pollute global scope
});

// FIXME(samthor): There's probably a better way to do this. Avoid Java on macOS.
const checkNativeMac = path.join(__dirname, 'node_modules/google-closure-compiler-osx/compiler');
if (fs.existsSync(checkNativeMac)) {
closureCompiler.JAR_PATH = undefined;
ClosureCompiler.prototype.javaPath = './node_modules/google-closure-compiler-osx/compiler';
}

const compilerProcess = closureCompiler.run((code, stdout, stderr) => {
if (stderr) {
console.error('err!', stderr);
Expand Down
10 changes: 5 additions & 5 deletions proxy.min.js

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

40 changes: 20 additions & 20 deletions src/proxy.js
Original file line number Diff line number Diff line change
Expand Up @@ -35,7 +35,7 @@ module.exports = function proxyPolyfill() {
const $Object = Object;

// Closure assumes that `{__proto__: null} instanceof Object` is always true, hence why we check against a different name.
const canCreateNullProtoObjects = !!$Object.create || !({ __proto__: null } instanceof $Object);
const canCreateNullProtoObjects = Boolean($Object.create) || !({ __proto__: null } instanceof $Object);
const objectCreate =
$Object.create ||
(canCreateNullProtoObjects
Expand All @@ -55,6 +55,8 @@ module.exports = function proxyPolyfill() {
return new T();
});

const noop = function() { return null; };

const getProto =
$Object.getPrototypeOf ||
([].__proto__ === Array.prototype
Expand All @@ -64,19 +66,7 @@ module.exports = function proxyPolyfill() {
const proto = O.__proto__;
return isObject(proto) ? proto : null;
}
: null);

// Some old engines support Object.getPrototypeOf but not Object.setPrototypeOf,
// because Object.setPrototypeOf was standardized later.
const setProto =
$Object.setPrototypeOf ||
([].__proto__ === Array.prototype
? function setPrototypeOf(O, proto) {
validateProto(proto);
O.__proto__ = proto;
return O;
}
: null);
: noop);

/**
* @constructor
Expand Down Expand Up @@ -126,7 +116,7 @@ module.exports = function proxyPolyfill() {

// Define proxy as an object that extends target.[[Prototype]],
// or a Function (if either it's callable, or apply is set).
const proto = getProto ? getProto(target) : null;
const proto = getProto(target); // can return null in old browsers
let proxy;
let isMethod = false;
let isArray = false;
Expand Down Expand Up @@ -158,7 +148,7 @@ module.exports = function proxyPolyfill() {
proxy = [];
isArray = true;
} else {
proxy = canCreateNullProtoObjects || proto !== null ? objectCreate(proto) : {};
proxy = (canCreateNullProtoObjects || proto !== null) ? objectCreate(proto) : {};
}

// Create default getters/setters. Create different code paths as handler.get/handler.set can't
Expand Down Expand Up @@ -192,7 +182,7 @@ module.exports = function proxyPolyfill() {
}
const real = $Object.getOwnPropertyDescriptor(target, prop);
const desc = {
enumerable: !!real.enumerable,
enumerable: Boolean(real.enumerable),
get: getter.bind(target, prop),
set: setter.bind(target, prop),
};
Expand All @@ -205,9 +195,19 @@ module.exports = function proxyPolyfill() {
// An alternative here would be to _just_ clone methods to keep behavior consistent.
let prototypeOk = true;
if (isMethod || isArray) {
if (setProto && proto !== undefined) {
setProto(proxy, proto);
} else {
// Arrays and methods are special: above, we instantiate boring versions of these then swap
// our their prototype later. So we only need to use setPrototypeOf in these cases. Some old
// engines support `Object.getPrototypeOf` but not `Object.setPrototypeOf`.
const setProto =
$Object.setPrototypeOf ||
([].__proto__ === Array.prototype
? function setPrototypeOf(O, proto) {
validateProto(proto);
O.__proto__ = proto;
return O;
}
: noop);
if (!(proto && setProto(proxy, proto))) {
prototypeOk = false;
}
}
Expand Down
36 changes: 36 additions & 0 deletions suite.js
Original file line number Diff line number Diff line change
Expand Up @@ -303,6 +303,32 @@ module.exports = () => function(scope) {
assert.equal(p.y, 1);
}
});

test('subclass of array', function() {
class Foo extends Array {}
var f = new Foo(2);
f[0] = 'first';
f[1] = 'second';

var p = new impl(f, {get: function(obj, prop) {
return obj[prop];
}});

assert.equal(p[0], 'first');
assert.equal(p.length, 2);
assert.isTrue(p instanceof Array, 'should be instanceof Array');

if (impl === scope.NativeProxy) {
// nb. this fails in NativeProxy
assert.throws(function() {
p instanceof impl;
}, 'Function has non-object prototype', 'cannot instanceof native proxy');
} else {
assert.isFalse(p instanceof impl, 'should not be instanceof Proxy');
}

assert.equal(Object.getPrototypeOf(p), Object.getPrototypeOf(f));
});
});
}

Expand All @@ -312,6 +338,16 @@ module.exports = () => function(scope) {
}

suite('general polyfill', function() {
test('prototypal inheritance', function() {
function Foo() {
}
var f = new Foo();
var p = new Proxy(f, {});

assert.isTrue(p instanceof Foo, 'proxy looks like proxied object');
assert.isFalse(p instanceof Proxy, 'proxy does not look like Proxy')
});

test('seals object', function() {
var testObj = buildObject();
assert.isNotSealed(testObj);
Expand Down

0 comments on commit c8c1d8b

Please sign in to comment.