-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathwatcher.js
450 lines (348 loc) · 13.7 KB
/
watcher.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
class Watcher {
constructor() {
}
static sendOperationMessage(operation) {
Watcher.sendMessage('op|' + JSON.stringify(operation));
}
watchify(object) {
var proxy = this.getWatchifyProxy(object);
return proxy;
}
//Returns a Proxy object for the given object which will send Lucidity data structure operation
//info when certain properties are modified or functions are called on the given object.
getWatchifyProxy(object) {
//Augment object with ds type-specific logic for handling property changes and method calls
object.watcher_dslogic = this.newDSLogicForObject(object);
var onChange = (obj, prop, oldVal, newVal) => {
var operation = obj.watcher_dslogic.operationFromChangeData(obj, prop, oldVal, newVal);
if (!Util.isFalsey(operation))
Watcher.sendOperationMessage(operation);
};
var handler = {
set (obj, prop, value) {
const oldVal = obj[prop];
//'value' is potentially already an object, in which case
//getBoxedValue(...) will just return it unchanged.
var boxedVal = Watcher.getBoxedValue(value);
Reflect.set(obj, prop, boxedVal);
if (obj.watcher_dslogic.isTrackedProperty(obj, prop)) {
onChange(obj, prop, oldVal, boxedVal);
}
return true;
},
}
var proxy = new Proxy(object, handler);
return proxy;
}
newDSLogicForObject(obj) {
if (Array.isArray(obj)) {
return new ListLogic(obj);
} else {
throw "We can only watch arrays right now!";
}
}
//Traverses the object tree and watchifies every sub object too
watchifyRecursive(object) {
}
//Replaces object property with a watched version of the given property
watch(object, property) {
object[property] = this.watchify(object[property]);
}
watchScope(scopeString) {
//parse scopeString
//find variable declarations
//watch all watchable declared objects/arrays
}
//Walk all properties and box primitive values that need unique identifiers
//associated with them.
static objectifyDS(obj, dsLogic, visited) {
if (visited === undefined)
visited = [];
for (var property in obj) {
if (obj.hasOwnProperty(property)) {
if (Util.isPrimitive(obj[property])) {
if (dsLogic.isTrackedProperty(obj, property)) {
Watcher.objectifyProperty(obj, property);
}
} else if (!visited.includes(obj[property])) {
visited.push(obj[property]);
Watcher.objectifyDS(obj[property], dsLogic, visited);
}
}
}
}
static objectifyProperty(obj, property) {
obj[property] = Watcher.getBoxedValue(obj[property]);
}
static getBoxedValue(val) {
if (Util.isFalsey(val) || !Util.isPrimitive(val))
return val;
if (typeof val === 'string') {
return new String(val);
} else if (typeof val === 'number') {
return new Number(val);
} else { //boolean
return new Boolean(val);
}
}
//If we'd like Lucidiy to produce correct 'move' animations when an element moves from
//one position to another (or from one DS to another), we need to uniquely identify elements.
static elementId(element) {
if (Util.isFalsey(element)) {
return -1;
}
if (element.watcher_element_id === undefined) {
Object.defineProperty(element, "watcher_element_id", {
value: Watcher.nextElementId(),
enumerable: false,
writable: false
});
}
return element.watcher_element_id;
}
static nextElementId() {
if (Watcher.next_element_id === undefined)
Watcher.next_element_id = 0;
return Watcher.next_element_id++;
}
//Unique identifiers for data structures are sent over in each Lucidity operation.
//These are generated once, each time a new data structure is created, but then
//every operation meant to be applied to that data structure includes the DS's ID.
static nextDSId() {
if (Watcher.next_ds_id === undefined)
Watcher.next_ds_id = 0;
return Watcher.next_ds_id++;
}
}
const OpType = {
ADD: 'add',
REMOVE: 'remove',
CREATE: 'create',
COMPOUND: 'compound'
}
//Does all the data structure type-specific work to generate Lucidity operations when
//an associated object (named 'ds') is modified in certain ways.
class DSLogic {
constructor(ds, dsType) {
this.ds = ds;
this.dsType = dsType;
this.trackedFunctionCallStack = [];
//Need to box all primitive properties since there is no way of associating
//unique identifiers with primitive values (that I can think of).
//Order matters when calling this (i.e. it probably shouldn't be called earlier
//nor later than on the following line).
Watcher.objectifyDS(ds, this);
this.proxifyTrackedFunctions();
//Initialize data structure in Lucidity
this.ds_id = this.sendCreateOp();
var initializeOps = this.getInitializeOps();
if (initializeOps.length !== 0) {
Watcher.sendOperationMessage(this.compoundOp(...initializeOps));
}
}
sendCreateOp() {
var operation = {
dataStructureType: this.dsType,
targetID: Watcher.nextDSId(),
type: OpType.CREATE,
location: [-1],
timestamp: 0
};
Watcher.sendOperationMessage(operation);
return operation.targetID;
}
//If the data structure should be initialize with some elements, sublcasses should
//return the operations that will add those elements here. Return an empty array
//if there are no initial values.
getInitializeOps() {
throw "Subclasses must override getInitializeOps!";
}
//Should be implemented by subclasses. Called whenever a property on 'ds' is modified.
//On some of these modifications nothing will happen, on others we'll generate appropriate
//Lucidity operations.
operationFromChangeData(obj, prop, oldVal, newVal) {
throw "Subclasses must override operationFromChangeData!";
}
trackedFunctionStarted(funcName, args) {
this.trackedFunctionCallStack.push(funcName);
//If present, call a subclasses method which will generate a ds operation
//corresponding to the tracked function's behavior.
if (this[funcName + 'Op'] !== undefined) {
this[funcName + 'Op'](...args);
}
// console.log("starting: " + funcName + "; args: ", args, "; stack: ", this.trackedFunctionCallStack);
}
trackedFunctionEnded(funcName) {
this.trackedFunctionCallStack.pop();
// console.log("finished: " + funcName + "; stack: ", this.trackedFunctionCallStack);
}
currentlyExecutingTrackedFunction() {
return this.trackedFunctionCallStack[this.trackedFunctionCallStack.length - 1];
}
//Replace function named 'funcName' on 'obj' with a function Proxy used to notify handlers before
//the function starts and after it ends.
proxifyFunction(obj, funcName) {
var self = this;
obj[funcName] = new Proxy(obj[funcName], {apply: function(target, thisArg, argumentsList) {
//IMPORTANT: this is happening.
//Just box any primitive that's potentially getting added to our DS
for (var i = 0; i < argumentsList.length; i++) {
argumentsList[i] = Watcher.getBoxedValue(argumentsList[i]);
}
self.trackedFunctionStarted(funcName, argumentsList);
Reflect.apply(target, thisArg, argumentsList)
self.trackedFunctionEnded(funcName);
}});
}
//Should be implemented by subclasses. Subclasses should use proxifyFunction(...) to
//set up tracking on all functions which should be tracked (i.e. those for which we generate
//Lucidity operations on their invokation).
proxifyTrackedFunctions() {
throw "Subclasses must override proxifyTrackedFunctions!";
}
isTrackedProperty(object, propertyName) {
throw "Subclasses must override isTrackedProperty!";
}
//Returns a 'modification' operation, e.g. ADD/REMOVE
modificationOp(location, elementValue, opType) {
if (!Array.isArray(location))
throw "'location' argument in modificationOp(...) must be an array.";
return {
targetID: this.ds_id,
elementID: Watcher.elementId(elementValue),
type: opType,
location: location,
untypedArgument: DSLogic.operationArgumentToString(elementValue),
timestamp: 0
};
}
//Returns a compound operation comprised of the given operations
compoundOp(...subOps) {
if (subOps.length >= 1 && Array.isArray(subOps[0]))
throw "compoundOp() uses the spread operator; don't pass it an array of things."
return {
targetID: this.ds_id,
elementID: "null",
type: OpType.COMPOUND,
subOperations: subOps,
timestamp: 0
}
}
static operationArgumentToString(arg) {
if (arg === null) {
return 'null';
} else if (arg === undefined) {
return 'undefined';
} else if (typeof arg !== 'string') {
return new String(arg);
} else {
return arg;
}
}
}
//Needs 'logics' for these types:
// []
// Map
// custom treenode class
// other js object
class ListLogic extends DSLogic {
constructor(ds) {
super(ds, 'list');
this.old_ds = this.ds.slice();
}
getInitializeOps() {
var addOps = [];
this.ds.forEach((value, i) => {
addOps.push(this.modificationOp([i], value, OpType.ADD));
});
return addOps;
}
//Overrides parent function
//Generate add/remove operations based on property change data
operationFromChangeData(obj, prop, oldVal, newVal) {
// console.log("direct array set; list[" + prop + "] = ", newVal, "; was ", oldVal);
var operation;
if (prop == 'length' && newVal < oldVal) { //length was reduced; truncate list
operation = this.lengthReducedOp(oldVal, newVal);
} else if (Util.isNumber(prop)) { //setting some array val, e.g.
var index = Number(prop);
operation = this.arraySetOp(index, oldVal, newVal);
}
//Copy whole array after every operation so we can access the old state of the array on the next
//property change event.
this.old_ds = this.ds.slice();
return operation;
}
lengthReducedOp(oldLength, newLength) {
var removeOps = [];
for (var i = newLength; i < oldLength; i++) {
removeOps.push(this.modificationOp([newLength], this.old_ds[i], OpType.REMOVE));
}
var compound = this.compoundOp(...removeOps);
return compound;
}
//Returns the operation for setting the value of some array element
arraySetOp(index, oldVal, newVal) {
var addOp = this.modificationOp([index], newVal, OpType.ADD);
if (index >= this.old_ds.length) {
return addOp;
}
else { //Remove the old value first
var removeOp = this.modificationOp([index], oldVal, OpType.REMOVE);
var setOp = this.compoundOp(removeOp, addOp);
return setOp;
}
}
//Overrides parent function
proxifyTrackedFunctions() {
//We don't need any function-specific operation generation for arrays
}
isTrackedProperty(obj, propertyName) {
//If it is a number, it's an array element
//we also check that obj is this.ds since we might
//be getting asked about a 'nested' property on the ds
//and we don't want to track any of those.
return obj === this.ds && (Util.isNumber(propertyName) || propertyName === 'length');
}
}
class Util {
static isPrimitive(val) {
return val !== Object(val);
}
static isNumber(val) {
return !isNaN(val);
}
static isFalsey(val) {
return val === null || val === undefined;
}
}
exports.Watcher = Watcher;
// var w = new Watcher();
// Watcher.sendMessage = function(message) {
// //send network message to Lucidity
// console.log("sending: " + message);
// };
// var a = w.watchify([1, 2, 3, 4, 5, 6, 7, 8]);
// a[8] = 1234;
// a.push('xyz');
// a.splice(2, 1, 'instead', 'and_another', 'blah');
//Access global scope anywhere:
// var global = Function('return this')();
// var global = {};
// Object.defineProperty(global, '__stack', {
// get: function(){
// var orig = Error.prepareStackTrace;
// Error.prepareStackTrace = function(_, stack){ return stack; };
// var err = new Error;
// Error.captureStackTrace(err, arguments.callee);
// var stack = err.stack;
// Error.prepareStackTrace = orig;
// return stack;
// }
// });
// Object.defineProperty(global, '__line', {
// get: function(){
// return global.__stack[1].getLineNumber();
// }
// });
// console.log(global.__line);