Skip to content

Commit

Permalink
feat(pass-style): Far GET_METHOD_NAMES meta method
Browse files Browse the repository at this point in the history
  • Loading branch information
erights committed Nov 8, 2023
1 parent ad86b0c commit f47e3a5
Show file tree
Hide file tree
Showing 10 changed files with 106 additions and 25 deletions.
1 change: 1 addition & 0 deletions packages/eventual-send/utils.js
Original file line number Diff line number Diff line change
@@ -0,0 +1 @@
export { getMethodNames } from './src/local.js';
3 changes: 2 additions & 1 deletion packages/exo/src/exo-tools.js
Original file line number Diff line number Diff line change
@@ -1,4 +1,5 @@
import { getMethodNames } from '@endo/eventual-send/src/local.js';
import { getMethodNames } from '@endo/eventual-send/utils.js';
import { hasOwnPropertyOf } from '@endo/pass-style';
import { E, Far } from '@endo/far';
import {
listDifference,
Expand Down
2 changes: 2 additions & 0 deletions packages/exo/src/get-interface.js
Original file line number Diff line number Diff line change
Expand Up @@ -4,6 +4,8 @@
* The name of the automatically added default meta-method for
* obtaining an exo's interface, if it has one.
*
* Intended to be similar to `GET_METHOD_NAMES` from `@endo/pass-style`.
*
* TODO Name to be bikeshed. Perhaps even whether it is a
* string or symbol to be bikeshed.
*
Expand Down
16 changes: 11 additions & 5 deletions packages/exo/test/test-exo-wobbly-point.js
Original file line number Diff line number Diff line change
Expand Up @@ -10,10 +10,9 @@
/* eslint-disable-next-line import/order */
import { test } from './prepare-test-env-ava.js';

// TODO enable import of getMethodNames without deep import
// eslint-disable-next-line import/order
import { getMethodNames } from '@endo/eventual-send/src/local.js';
import { passStyleOf, Far } from '@endo/pass-style';
import { getMethodNames } from '@endo/eventual-send/utils.js';
import { passStyleOf, Far, GET_METHOD_NAMES } from '@endo/pass-style';
import { M } from '@endo/patterns';
import { defineExoClass } from '../src/exo-makers.js';
import { GET_INTERFACE_GUARD } from '../src/get-interface.js';
Expand Down Expand Up @@ -96,12 +95,18 @@ harden(ExoPoint);

const makeExoPoint = defineExoClassFromJSClass(ExoPoint);

const assertMethodNames = (t, obj, names) => {
t.deepEqual(getMethodNames(obj), names);
t.deepEqual(obj[GET_METHOD_NAMES](), names);
};

test('ExoPoint instances', t => {
const pt = makeExoPoint(3, 5);
t.is(passStyleOf(pt), 'remotable');
t.false(pt instanceof ExoPoint);
t.deepEqual(getMethodNames(pt), [
assertMethodNames(t, pt, [
GET_INTERFACE_GUARD,
GET_METHOD_NAMES,
'getX',
'getY',
'setY',
Expand Down Expand Up @@ -152,8 +157,9 @@ test('FarWobblyPoint inheritance', t => {
t.false(wpt instanceof ExoWobblyPoint);
t.false(wpt instanceof ExoPoint);
t.is(passStyleOf(wpt), 'remotable');
t.deepEqual(getMethodNames(wpt), [
assertMethodNames(t, wpt, [
GET_INTERFACE_GUARD,
GET_METHOD_NAMES,
'getX',
'getY',
'setY',
Expand Down
12 changes: 8 additions & 4 deletions packages/marshal/test/test-marshal-testing.js
Original file line number Diff line number Diff line change
@@ -1,14 +1,18 @@
import { test } from './prepare-test-env-ava.js';

// eslint-disable-next-line import/order
import { Far, passStyleOf, Remotable } from '@endo/pass-style';
import { passStyleOf, Remotable } from '@endo/pass-style';
import { makeMarshal } from '../src/marshal.js';

const { create } = Object;

const alice = Far('alice');
const bob1 = Far('bob');
const bob2 = Far('bob');
// Use the lower level `Remotable` rather than `Far` to make an empty
// far object, i.e., one without even the miranda meta methods like
// `GET_METHOD_NAMES`. Such an empty far object should be `t.deepEqual`
// to its remote presences.
const alice = Remotable('Alleged: alice');
const bob1 = Remotable('Alleged: bob');
const bob2 = Remotable('Alleged: bob');

const convertValToSlot = val =>
passStyleOf(val) === 'remotable' ? 'far' : val;
Expand Down
7 changes: 6 additions & 1 deletion packages/pass-style/index.js
Original file line number Diff line number Diff line change
Expand Up @@ -24,7 +24,12 @@ export {
export { passStyleOf, assertPassable } from './src/passStyleOf.js';

export { makeTagged } from './src/makeTagged.js';
export { Remotable, Far, ToFarFunction } from './src/make-far.js';
export {
Remotable,
Far,
ToFarFunction,
GET_METHOD_NAMES,
} from './src/make-far.js';

export {
assertRecord,
Expand Down
2 changes: 1 addition & 1 deletion packages/pass-style/package.json
Original file line number Diff line number Diff line change
Expand Up @@ -33,11 +33,11 @@
"test": "ava"
},
"dependencies": {
"@endo/eventual-send": "^0.17.6",
"@endo/promise-kit": "^0.2.60",
"@fast-check/ava": "^1.1.5"
},
"devDependencies": {
"@endo/eventual-send": "^0.17.6",
"@endo/init": "^0.5.60",
"@endo/ses-ava": "^0.2.44",
"ava": "^5.3.0",
Expand Down
42 changes: 42 additions & 0 deletions packages/pass-style/src/make-far.js
Original file line number Diff line number Diff line change
@@ -1,5 +1,6 @@
/// <reference types="ses"/>

import { getMethodNames } from '@endo/eventual-send/utils.js';
import { assertChecker, PASS_STYLE } from './passStyle-helpers.js';
import { assertIface, getInterfaceOf, RemotableHelper } from './remotable.js';

Expand Down Expand Up @@ -128,16 +129,57 @@ export const Remotable = (
};
harden(Remotable);

/**
* The name of the automatically added default meta-method for obtaining a
* list of all methods of an object declared with `Far`, or an object that
* inherits from an object declared with `Far`.
*
* Modeled on `GET_INTERFACE_GUARD` from `@endo/exo`.
*
* TODO Name to be bikeshed. Perhaps even whether it is a
* string or symbol to be bikeshed.
*
* TODO Beware that an exo's interface can change across an upgrade,
* so remotes that cache it can become stale.
*/
export const GET_METHOD_NAMES = Symbol.for('getMethodNames');

/**
* Note that `getMethodNamesMethod` is a thisful method! It must be so that
* it works as expected with far-object inheritance.
*
* @returns {(string|symbol)[]}
*/
const getMethodNamesMethod = harden({
[GET_METHOD_NAMES]() {
return getMethodNames(this);
},
})[GET_METHOD_NAMES];

/**
* A concise convenience for the most common `Remotable` use.
*
* For far objects (as opposed to far functions), also adds a miranda
* `GET_METHOD_NAMES` method that returns an array of all the method names,
* if there is not yet any method named `GET_METHOD_NAMES`. (Hence "miranda")
*
* @template {{}} T
* @param {string} farName This name will be prepended with `Alleged: `
* for now to form the `Remotable` `iface` argument.
* @param {T} [remotable] The object used as the remotable
*/
export const Far = (farName, remotable = undefined) => {
const r = remotable === undefined ? /** @type {T} */ ({}) : remotable;
if (typeof r === 'object' && !(GET_METHOD_NAMES in r)) {
// This test excludes far functions, since we currently consider them
// to only have a call-behavior, with no callable methods.
// Beware: Mutates the input argument! But `Remotable`
// * requires the object to be mutable
// * does further mutations,
// * hardens the mutated object before returning it.
// so this mutation is not unprecedented. But it is surprising!
r[GET_METHOD_NAMES] = getMethodNamesMethod;
}
return Remotable(`Alleged: ${farName}`, undefined, r);
};
harden(Far);
Expand Down
30 changes: 22 additions & 8 deletions packages/pass-style/test/test-far-class-instances.js
Original file line number Diff line number Diff line change
Expand Up @@ -2,11 +2,10 @@
/* eslint-disable max-classes-per-file */
import { test } from './prepare-test-env-ava.js';

// TODO enable import of getMethodNames without deep import
// eslint-disable-next-line import/order
import { getMethodNames } from '@endo/eventual-send/src/local.js';
import { getMethodNames } from '@endo/eventual-send/utils.js';
import { passStyleOf } from '../src/passStyleOf.js';
import { Far } from '../src/make-far.js';
import { Far, GET_METHOD_NAMES } from '../src/make-far.js';

/**
* Classes whose instances should be Far objects may find it convenient to
Expand All @@ -17,7 +16,7 @@ import { Far } from '../src/make-far.js';
* this object's identity. However, we discourage (but cannot prevent) such
* use of private fields, as they cannot easily be refactored into Exo state.
*/
export const FarBaseClass = class FarBaseClass {
const FarBaseClass = class FarBaseClass {
constructor() {
harden(this);
}
Expand Down Expand Up @@ -45,10 +44,15 @@ class FarSubclass2 extends FarSubclass1 {
}
}

const assertMethodNames = (t, obj, names) => {
t.deepEqual(getMethodNames(obj), names);
t.deepEqual(obj[GET_METHOD_NAMES](), names);
};

test('far class instances', t => {
const fb = new FarBaseClass();
t.is(passStyleOf(fb), 'remotable');
t.deepEqual(getMethodNames(fb), ['constructor']);
assertMethodNames(t, fb, [GET_METHOD_NAMES, 'constructor']);

t.assert(new fb.constructor() instanceof FarBaseClass);
t.throws(() => fb.constructor(), {
Expand All @@ -60,13 +64,18 @@ test('far class instances', t => {
t.is(passStyleOf(fs1), 'remotable');
t.is(fs1.double(4), 8);
t.assert(new fs1.constructor() instanceof FarSubclass1);
t.deepEqual(getMethodNames(fs1), ['constructor', 'double']);
assertMethodNames(t, fs1, [GET_METHOD_NAMES, 'constructor', 'double']);

const fs2 = new FarSubclass2(3);
t.is(passStyleOf(fs2), 'remotable');
t.is(fs2.double(4), 8);
t.is(fs2.doubleAdd(4), 11);
t.deepEqual(getMethodNames(fs2), ['constructor', 'double', 'doubleAdd']);
assertMethodNames(t, fs2, [
GET_METHOD_NAMES,
'constructor',
'double',
'doubleAdd',
]);

const yField = new WeakMap();
class FarSubclass3 extends FarSubclass1 {
Expand All @@ -84,7 +93,12 @@ test('far class instances', t => {
t.is(passStyleOf(fs3), 'remotable');
t.is(fs3.double(4), 8);
t.is(fs3.doubleAdd(4), 11);
t.deepEqual(getMethodNames(fs3), ['constructor', 'double', 'doubleAdd']);
assertMethodNames(t, fs3, [
GET_METHOD_NAMES,
'constructor',
'double',
'doubleAdd',
]);
});

test('far class instance hardened empty', t => {
Expand Down
16 changes: 11 additions & 5 deletions packages/pass-style/test/test-far-wobbly-point.js
Original file line number Diff line number Diff line change
Expand Up @@ -7,11 +7,10 @@
/* eslint-disable max-classes-per-file */
import { test } from './prepare-test-env-ava.js';

// TODO enable import of getMethodNames without deep import
// eslint-disable-next-line import/order
import { getMethodNames } from '@endo/eventual-send/src/local.js';
import { getMethodNames } from '@endo/eventual-send/utils.js';
import { passStyleOf } from '../src/passStyleOf.js';
import { Far } from '../src/make-far.js';
import { Far, GET_METHOD_NAMES } from '../src/make-far.js';

const { apply } = Reflect;

Expand Down Expand Up @@ -61,11 +60,17 @@ class FarPoint extends FarBaseClass {
}
harden(FarPoint);

const assertMethodNames = (t, obj, names) => {
t.deepEqual(getMethodNames(obj), names);
t.deepEqual(obj[GET_METHOD_NAMES](), names);
};

test('FarPoint instances', t => {
const pt = new FarPoint(3, 5);
t.is(passStyleOf(pt), 'remotable');
t.assert(pt instanceof FarPoint);
t.deepEqual(getMethodNames(pt), [
assertMethodNames(t, pt, [
GET_METHOD_NAMES,
'constructor',
'getX',
'getY',
Expand Down Expand Up @@ -103,7 +108,8 @@ test('FarWobblyPoint inheritance', t => {
t.assert(wpt instanceof FarWobblyPoint);
t.assert(wpt instanceof FarPoint);
t.is(passStyleOf(wpt), 'remotable');
t.deepEqual(getMethodNames(wpt), [
assertMethodNames(t, wpt, [
GET_METHOD_NAMES,
'constructor',
'getX',
'getY',
Expand Down

0 comments on commit f47e3a5

Please sign in to comment.