Skip to content

Commit ba25e9d

Browse files
committed
An atom based identity.
A major question lingering over the atomic microstates refactor is how it would work with identity (the mechanism for maintaing a stable graph of references). This change contains an implementation of identity using the atomic microstates architecture. The following new constructs were required: ** Polymorphic `atomOf()` ** With an atomic microstate architecture, microstates no longer hold a reference directly to their values. Instead, they hold a reference directly to their paths, and in order to retrieve their values, they look up that their path inside of an atom. This poses a problem for stable identities because the atom needs to change, but the JavaScript reference must remain the same. To solve this, we make `atomOf()` a polymorhpic function of the `AtomOf` typeclass. The default implementation of `atomOf` returns the atom to which the current microstate holds a reference. The `Id<Type>` implementation however, returns the top level value of the identity which changes over time with each transition. ** Extended MicrostateType() ** There is a lot that is the same between the base microstate type instantiated with `create`, and the the Identity microstate type. Things like maintaining the meta information, the `set` transition, caching instance properties, etc... However, there are some important places where we want to diverge, specifically in how transitions are handled, and properties are resolved. This introduces new parameters to the `transitionFn` and `propertyFn`, specifically, the `Type`, and `path` parameters where the transition is happening or the property is being resolved are both now passed. is called, or a property is resolved. This allows us to have a single. ** Tree ** In order to reason about the entire tree as a whole, we introduce a lightweight `Tree` class not dissimilar to the `Profuctor` class in the current implementation. It is expected that this will only be needed for the identity map, and not for implementing the rest of the architecture. ** Higher Order Store ** Identity was previously interwined with the concept of `Store`. This meant that integrating with other stores like Redux and Obvservable, required a little bit of massaging or integrating with the observation mechanism. The Identity() function implemented here takes a microstate and a transition event callback, returns two things: a function `get` which returns the current reference to the root id microstate, and a `transition` function which actually causes a transition to happen. So, when a caller invokes a transition from the microstate graph, it invokes the callback with the `Type`, `path`, `name`, `args` of where the transition happened. This is the same signature as the transition callback, and so the idea is that at some point, you invoke it with those arguments. So why would you want to split a transition into two phases? The answer is that by makeing it So for example, a fully synchronous Store would immediately delegate from the callback to the transition function: ```js let { get, transition } = Identity(create(Number, 0), (...args) => transition(...args)); id.get().increment(); id.get() //> Id<Number> { 1 } ``` Redux, which is also a synchronous store can be implementing in a straightforward fashion ```js let { get, transition } = Identity(Create(Number, 0), (Type, path, name, args) => { store.dispatch({ isMicrostateTransition: true type: `${Type.name}.#{name}${path.join('/')}`, Type, path, name, args }) return get() }) // and in reducer: function microstateRducer(state, action) { if (action.isMicrostateTransition) { let { Type, path, name, args } = action; transition(Type, path, name, args); return id.get(); } } ``` Obvioeusly, the current store is implemented easily too: ```js export default function Store(Type, value, observe = x => x) { let { get, transition } = Identity(Type, value, (...args) => observe(transition(...args))); return get(); } ``` Follow on work: - Make sure that microstate types are stable: same constructor for given `Type` - make sure all function parameters make sense and are aligned - a uniform linking function (link / link2 is a no-go)
1 parent ff8e3b7 commit ba25e9d

9 files changed

+229
-32
lines changed

examples/table.js

+3-3
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,6 @@
11
import { create } from '../index';
22
import { query } from '../src/query';
3-
import { link, ownerOf, pathOf, valueOf, metaOf, atomOf, location } from '../src/meta';
3+
import { link, ownerOf, pathOf, valueOf, atomOf, location } from '../src/meta';
44

55
export default function Table(T) {
66
class Table {
@@ -59,7 +59,7 @@ export default function Table(T) {
5959
*[Symbol.iterator]() {
6060
let i = -1;
6161
let { table, index } = this;
62-
for (let _ of valueOf(table)[index]) {
62+
for (let _ of valueOf(table)[index]) { //eslint-disable-line
6363
i++;
6464
yield link(create(T), location(T, pathOf(table).concat([index, i])), atomOf(table), ownerOf(table));
6565
}
@@ -75,7 +75,7 @@ export default function Table(T) {
7575
*[Symbol.iterator]() {
7676
let i = -1;
7777
let { table, index } = this;
78-
for (let _ of valueOf(table)) {
78+
for (let _ of valueOf(table)) { //eslint-disable-line
7979
i++;
8080
yield link(create(T), location(T, pathOf(table).concat([i, index])), atomOf(table), ownerOf(table));
8181
}

index.js

+2-1
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,5 @@
11
import create from './src/create';
2+
import Identity from './src/identity';
23

3-
export { create };
4+
export { create, Identity };
45
export { valueOf, metaOf, atomOf } from './src/meta';

src/create.js

+11-8
Original file line numberDiff line numberDiff line change
@@ -1,20 +1,16 @@
11
import { set } from './lens';
2-
import { link, locationOf, metaOf, atomOf, ownerOf } from './meta';
2+
import { link, mount, locationOf, metaOf, atomOf, ownerOf, valueOf } from './meta';
33
import MicrostateType from './microstate-type';
44

55
export default function create(Type, value) {
6-
let Microstate = MicrostateType(Type, transition);
6+
let Microstate = MicrostateType(Type, transition, property);
7+
// return root(Type, value, () => new Microstate(value))
78
return new Microstate(value);
89
}
910

10-
function transition(microstate, name, method, ...args) {
11-
12-
//location of the owner of this transition.
11+
function transition(microstate, Type, path, name, method, ...args) {
1312
let owner = ownerOf(microstate);
14-
15-
//in the context of the transition, the owner will be the same as the microstate.
1613
let context = link(microstate, locationOf(microstate), atomOf(microstate));
17-
1814
let result = method.apply(context, args);
1915

2016
function patch() {
@@ -28,3 +24,10 @@ function transition(microstate, name, method, ...args) {
2824

2925
return link(create(owner.Type), owner, patch());
3026
}
27+
28+
export function property(microstate, slot, key) {
29+
let value = valueOf(microstate);
30+
let expanded = typeof slot === 'function' ? create(slot, value) : slot;
31+
let substate = value != null && value[key] != null ? expanded.set(value[key]) : expanded;
32+
return mount(microstate, substate, key);
33+
}

src/identity.js

+75
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,75 @@
1+
import MicrostateType from './microstate-type';
2+
import { At, view, compose, Path } from './lens';
3+
import { typeOf, valueOf, pathOf, link2, AtomOf } from './meta';
4+
import { treemap } from './tree';
5+
import create from './create';
6+
7+
class Id {
8+
static symbol = Symbol('@id');
9+
static data = At(Id.symbol);
10+
static ref = compose(Id.data, At('ref'));
11+
static value = compose(Id.data, At('value'));
12+
static Type = compose(Id.data, At('Type'));
13+
14+
constructor(Type, path, value, ref) {
15+
this.Type = Type;
16+
this.path = path;
17+
this.value = value;
18+
this.ref = ref;
19+
}
20+
}
21+
22+
export default function Identity(microstate, fn) {
23+
24+
let paths = identify(microstate, {});
25+
26+
return { get, transition };
27+
28+
function get() {
29+
return view(Id.ref, paths);
30+
}
31+
32+
function transition(Type, name, path, args) {
33+
let atom = view(Id.value, paths);
34+
let Root = view(Id.Type, paths);
35+
let local = link2(create(Type), Type, path, atom, Root, []);
36+
let next = local[name](...args);
37+
return paths = identify(next, paths);
38+
}
39+
40+
function identify(microstate, pathmap) {
41+
return treemap(node => {
42+
let path = pathOf(node);
43+
let id = view(compose(Path(path), Id.data), pathmap);
44+
if (id != null && id.Type === typeOf(node) && id.value === valueOf(node)) {
45+
return id;
46+
} else {
47+
return proxy(node, path);
48+
}
49+
}, microstate);
50+
51+
function proxy(microstate, path) {
52+
let Type = typeOf(microstate);
53+
let Proxy = MicrostateType(Type, transitionFn, propertyFn);
54+
Proxy.name = `Id<${Type.name}>`;
55+
AtomOf.instance(Proxy, { atomOf: () => view(Id.value, paths) });
56+
57+
let value = valueOf(microstate);
58+
59+
let ref = link2(new Proxy(value), Type, path, 'polymorphic', view(Id.Type, paths), []);
60+
61+
return {
62+
[Id.symbol]: new Id(Type, path, value, ref)
63+
};
64+
}
65+
}
66+
67+
function transitionFn(object, Type, path, name, method, ...args) {
68+
return fn(Type, name, path, args);
69+
}
70+
71+
function propertyFn(self, slot, key, path) {
72+
let location = compose(Path(path), Id.ref);
73+
return view(location, paths);
74+
}
75+
}

src/meta.js

+24-8
Original file line numberDiff line numberDiff line change
@@ -1,13 +1,19 @@
1+
import { type } from 'funcadelic';
2+
13
import { At, view, set, over, compose, Path } from './lens';
24

35
export function root(microstate, Type, value) {
4-
return link(microstate, new Location(Type, []), value);
6+
return set(Meta.data, new Meta(new Location(Type, []), value), microstate);
57
}
68

79
export function link(microstate, location, atom, owner = location) {
810
return set(Meta.data, new Meta(location, atom, owner), microstate);
911
}
1012

13+
export function link2(object, Type, path, atom, Owner, ownerPath) {
14+
return set(Meta.data, new Meta(new Location(Type, path), atom, new Location(Owner, ownerPath)), object);
15+
}
16+
1117
export function mount(microstate, substate, key) {
1218
return over(Meta.data, meta => meta.mount(microstate, key), substate);
1319
}
@@ -29,10 +35,6 @@ export function locationOf(microstate) {
2935
return view(Meta.location, microstate);
3036
}
3137

32-
export function atomOf(microstate) {
33-
return view(Meta.atom, microstate);
34-
}
35-
3638
export function ownerOf(microstate) {
3739
return view(Meta.owner, microstate);
3840
}
@@ -42,7 +44,7 @@ export function typeOf(microstate) {
4244
}
4345

4446
export function pathOf(microstate) {
45-
return view(Meta.path, microstate);
47+
return view(Meta.path, microstate) || [];
4648
}
4749

4850
export class Meta {
@@ -54,9 +56,9 @@ export class Meta {
5456
static Type = compose(Meta.location, At("Type"));
5557
static path = compose(Meta.location, At("path"));
5658

57-
constructor(location, atom, owner) {
58-
this.location = location;
59+
constructor(location, atom, owner = location) {
5960
this.atom = atom;
61+
this.location = location;
6062
this.owner = owner;
6163
}
6264

@@ -80,3 +82,17 @@ class Location {
8082
return Path(this.path);
8183
}
8284
}
85+
86+
export const AtomOf = type(class AtomOf {
87+
atomOf(object) {
88+
return this(object).atomOf(object);
89+
}
90+
});
91+
92+
AtomOf.instance(Object, {
93+
atomOf(object) {
94+
return view(Meta.atom, object);
95+
}
96+
});
97+
98+
export const { atomOf } = AtomOf.prototype;

src/microstate-type.js

+4-12
Original file line numberDiff line numberDiff line change
@@ -1,37 +1,29 @@
11
import { append, map } from 'funcadelic';
22

33
import CachedProperty from './cached-property';
4-
import { mount, valueOf, root } from './meta';
4+
import { root, pathOf } from './meta';
55
import { methodsOf } from './reflection';
6-
import create from './create';
76

8-
export default function MicrostateType(Type, transition) {
7+
export default function MicrostateType(Type, transitionFn, propertyFn) {
98
let Microstate = class extends Type {
109
static Type = Type;
1110
static name = `Microstate<${Type.name}>`;
1211

1312
constructor(value) {
1413
super(value);
1514
Object.defineProperties(this, map((slot, key) => {
16-
return CachedProperty(key, self => {
17-
let value = valueOf(self);
18-
let expanded = typeof slot === 'function' ? create(slot, value) : slot;
19-
let substate = value != null && value[key] != null ? expanded.set(value[key]) : expanded;
20-
return mount(self, substate, key);
21-
});
15+
return CachedProperty(key, self => propertyFn(self, slot, key, pathOf(this).concat(key)));
2216
}, this));
23-
2417
return root(this, Type, value);
2518
}
2619
};
2720

2821
Object.defineProperties(Microstate.prototype, map((descriptor, name) => {
2922
return {
3023
value(...args) {
31-
return transition(this, name, descriptor.value, ...args);
24+
return transitionFn(this, Type, pathOf(this), name, descriptor.value, ...args);
3225
}
3326
};
3427
}, append({ set: { value: x => x} }, methodsOf(Type))));
35-
3628
return Microstate;
3729
}

src/store.js

+6
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,6 @@
1+
import Identity from './identity';
2+
3+
export default function Store(Type, value, observe = x => x) {
4+
let { get, transition } = Identity(Type, value, (...args) => observe(transition(...args)));
5+
return get();
6+
}

src/tree.js

+37
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,37 @@
1+
import { append, type } from 'funcadelic';
2+
import { metaOf } from './meta';
3+
4+
export const Tree = type(class Tree {
5+
visit(object) {
6+
let visit = this(object).visit || (node => metaOf(node) != null);
7+
return visit(object);
8+
}
9+
10+
treemap(fn, object) {
11+
if (visit(object)) {
12+
return this(object).treemap(fn, object);
13+
} else {
14+
return object;
15+
}
16+
}
17+
});
18+
19+
export const { visit, treemap } = Tree.prototype;
20+
21+
Tree.instance(Object, {
22+
treemap(fn, object) {
23+
let next = fn(object);
24+
let keys = Object.keys(object);
25+
if (next === object || keys.length === 0) {
26+
return next;
27+
} else {
28+
return append(next, keys.reduce((properties, key) => {
29+
return append(properties, {
30+
get [key]() {
31+
return treemap(fn, object[key]);
32+
}
33+
});
34+
}, {}));
35+
}
36+
}
37+
});

tests/identity.test.js

+67
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,67 @@
1+
import expect from 'expect';
2+
3+
import { create, Identity, valueOf } from '../index';
4+
5+
6+
class Str {
7+
get state() {
8+
return valueOf(this);
9+
}
10+
}
11+
12+
class Person {
13+
firstName = Str;
14+
lastName = Str;
15+
}
16+
17+
describe('identity', function() {
18+
let t;
19+
let id;
20+
let initial;
21+
let initialFirstName;
22+
let initialLastName;
23+
let value = { firstName: "Charles", lastName: "Lowell" };
24+
beforeEach(function() {
25+
id = Identity(create(Person, value), (...args) => t = args);
26+
initial = id.get();
27+
initialFirstName = initial.firstName;
28+
initialLastName = initial.lastName;
29+
});
30+
31+
it('is an instance of Person', () => {
32+
expect(initial).toBeInstanceOf(Person);
33+
expect(initial.firstName).toBeInstanceOf(Str);
34+
});
35+
36+
it('has the right properties', function() {
37+
expect(valueOf(initial)).toBe(value);
38+
expect(initial.firstName.state).toEqual('Charles');
39+
expect(initial.lastName.state).toEqual('Lowell');
40+
});
41+
42+
describe('invoking a transition', function() {
43+
beforeEach(()=> {
44+
id.get().firstName.set('Carlos');
45+
});
46+
it('invokes the transition callback to capture the location of the the transition', ()=> {
47+
expect(t).toEqual([Str, 'set', ['firstName'], ['Carlos']]);
48+
});
49+
it('does not actually perform the transition yet', ()=> {
50+
expect(id.get()).toBe(initial);
51+
});
52+
53+
describe('calling the id.transition function with the supplied arguments ', ()=> {
54+
beforeEach(()=> {
55+
id.transition(...t);
56+
});
57+
it('updates the root reference to a new instance of the same type', ()=> {
58+
expect(initial).toBeInstanceOf(Person);
59+
expect(id.get()).not.toBe(initial);
60+
});
61+
it('changes the child node that changed, but not the child node that did not', () => {
62+
expect(id.get().firstName).not.toBe(initialFirstName);
63+
expect(id.get().lastName).not.toBe(initialLastName);
64+
});
65+
});
66+
});
67+
});

0 commit comments

Comments
 (0)