Skip to content

Commit 8dc5651

Browse files
committed
Suspendable: adds class that provides a "pause" and "resume" method
We can't safely or effectively destroy objects that represent hardware or hardware states, but we need a way to temporarily prevent sensors from emitting data events. This class will move the contents of _events into a side table weakmap when pause() is called and then restore _events when resume() is called.
1 parent 206c080 commit 8dc5651

12 files changed

+201
-10
lines changed

.jshintrc

+1
Original file line numberDiff line numberDiff line change
@@ -14,6 +14,7 @@
1414
"strict": false,
1515
"quotmark": "double",
1616
"unused": true,
17+
"proto": true,
1718
"globals": {
1819
"exports": true,
1920
"document": true,

lib/light.js

+2-2
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,6 @@
11
const Board = require("./board");
22
const EVS = require("./evshield");
3-
const Withinable = require("./mixins/within");
3+
const Suspendable = require("./mixins/suspendable");
44
const { uint16, toFixed, scale } = require("./fn");
55
const priv = new Map();
66

@@ -436,7 +436,7 @@ Controllers.ALSPT19 = Controllers["ALS-PT19"] = Controllers.DEFAULT;
436436
*
437437
*/
438438

439-
class Light extends Withinable {
439+
class Light extends Suspendable {
440440
constructor(options) {
441441
super();
442442

lib/mixins/suspendable.js

+26
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,26 @@
1+
const Withinable = require("./withinable");
2+
const wm = new WeakMap();
3+
4+
class Suspendable extends Withinable {
5+
pause() {
6+
7+
wm.set(this, {
8+
...this._events
9+
});
10+
11+
this._events = { __proto__: null };
12+
}
13+
14+
resume() {
15+
const events = wm.get(this);
16+
if (events) {
17+
this._events = {
18+
__proto__: null,
19+
...events
20+
};
21+
wm.set(this, null);
22+
}
23+
}
24+
}
25+
26+
module.exports = Suspendable;
File renamed without changes.

lib/proximity.js

+2-2
Original file line numberDiff line numberDiff line change
@@ -2,7 +2,7 @@ const Board = require("./board");
22
const Collection = require("./mixins/collection");
33
const EVS = require("./evshield");
44
const Fn = require("./fn");
5-
const Withinable = require("./mixins/within");
5+
const Suspendable = require("./mixins/suspendable");
66
const Pins = Board.Pins;
77

88
const toFixed = Fn.toFixed;
@@ -427,7 +427,7 @@ Controllers.DEFAULT = Controllers["GP2Y0A21YK"];
427427
*
428428
*/
429429

430-
class Proximity extends Withinable {
430+
class Proximity extends Suspendable {
431431
constructor(options) {
432432
super();
433433

lib/sensor.js

+2-2
Original file line numberDiff line numberDiff line change
@@ -1,7 +1,7 @@
11
const Board = require("./board");
22
const Collection = require("./mixins/collection");
33
const Fn = require("./fn");
4-
const Withinable = require("./mixins/within");
4+
const Suspendable = require("./mixins/suspendable");
55

66
// Sensor instance private data
77
const priv = new Map();
@@ -28,7 +28,7 @@ function median(input) {
2828
* @param {Object} options Options: pin, freq, range
2929
*/
3030

31-
class Sensor extends Withinable {
31+
class Sensor extends Suspendable {
3232
constructor(options) {
3333

3434
super();

lib/thermometer.js

+2-2
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,6 @@
11
const Board = require("./board");
22
const Emitter = require("events");
3-
const Withinable = require("./mixins/within");
3+
const Suspendable = require("./mixins/suspendable");
44
const {
55
toFixed,
66
POW_2_16,
@@ -880,7 +880,7 @@ Controllers.DEFAULT = Controllers.ANALOG;
880880

881881
var priv = new Map();
882882

883-
class Thermometer extends Withinable {
883+
class Thermometer extends Suspendable {
884884
constructor(options) {
885885
super();
886886

test/.jshintrc

+1
Original file line numberDiff line numberDiff line change
@@ -90,6 +90,7 @@
9090
"ShiftRegisters": true,
9191
"Sonar": true,
9292
"Stepper": true,
93+
"Suspendable": true,
9394
"Switch": true,
9495
"temporal": true,
9596
"Thermometer": true,

test/common/bootstrap.js

+2-1
Original file line numberDiff line numberDiff line change
@@ -7,7 +7,8 @@ global.Emitter = require("events");
77

88
// Internal
99
global.Collection = require("../../lib/mixins/collection");
10-
global.Withinable = require("../../lib/mixins/within");
10+
global.Suspendable = require("../../lib/mixins/suspendable");
11+
global.Withinable = require("../../lib/mixins/withinable");
1112
global.five = require("../../lib/johnny-five");
1213
global.EVS = require("../../lib/evshield");
1314

test/sensor.js

+1-1
Original file line numberDiff line numberDiff line change
@@ -1171,7 +1171,7 @@ exports["Sensor - Analog"] = {
11711171
this.clock.tick(25);
11721172

11731173
test.done();
1174-
}, // ./within: function(test)
1174+
}, // ./withinable: function(test)
11751175

11761176
booleanAt(test) {
11771177
const callback = this.analogRead.args[0][1];

test/suspendable.js

+162
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,162 @@
1+
require("./common/bootstrap");
2+
3+
const Crypto = require("crypto");
4+
5+
class Component extends Suspendable {
6+
constructor(options) {
7+
super();
8+
9+
var value = null;
10+
11+
this.board = options.board;
12+
this.io = options.board.io;
13+
this.pin = options.pin;
14+
15+
this.io.analogRead(this.pin, (data) => {
16+
value = data;
17+
this.emit("data", value);
18+
});
19+
20+
21+
this.tracking = {
22+
value: {
23+
get: 0,
24+
set: 0,
25+
},
26+
unit: {
27+
get: 0,
28+
},
29+
};
30+
31+
Object.defineProperties(this, {
32+
value: {
33+
get: function() {
34+
this.tracking.value.get++;
35+
return value;
36+
},
37+
set: function(input) {
38+
this.tracking.value.set++;
39+
value = input;
40+
},
41+
},
42+
unit: {
43+
get: function() {
44+
this.tracking.unit.get++;
45+
return value;
46+
},
47+
},
48+
});
49+
}
50+
}
51+
52+
exports["Suspendable"] = {
53+
setUp(done) {
54+
this.sandbox = sinon.sandbox.create();
55+
this.board = newBoard();
56+
this.analogRead = this.sandbox.spy(MockFirmata.prototype, "analogRead");
57+
this.component = new Component({
58+
board: this.board,
59+
pin: "A0",
60+
});
61+
62+
done();
63+
},
64+
tearDown(done) {
65+
Board.purge();
66+
this.sandbox.restore();
67+
done();
68+
},
69+
70+
isInstanceOf(test) {
71+
test.expect(3);
72+
test.equal(new Suspendable() instanceof Withinable, true);
73+
test.equal(this.component instanceof Suspendable, true);
74+
test.equal(this.component instanceof Withinable, true);
75+
test.done();
76+
},
77+
78+
pauseIsAFunction(test) {
79+
test.expect(1);
80+
const suspendable = new Suspendable();
81+
test.equal(typeof suspendable.pause, "function");
82+
test.done();
83+
},
84+
85+
resumeIsAFunction(test) {
86+
test.expect(1);
87+
const suspendable = new Suspendable();
88+
test.equal(typeof suspendable.resume, "function");
89+
test.done();
90+
},
91+
92+
suspendability(test) {
93+
test.expect(5);
94+
const suspendable = new Suspendable();
95+
const expectedBytes = [];
96+
const emittedBytes = [];
97+
98+
suspendable.on("data", byte => emittedBytes.push(byte));
99+
100+
let turns = 0;
101+
let interval = setInterval(() => {
102+
const byte = Crypto.randomBytes(1);
103+
turns++;
104+
suspendable.emit("data", byte);
105+
106+
if (turns === 2) {
107+
suspendable.pause();
108+
}
109+
110+
if (turns === 4) {
111+
suspendable.resume();
112+
}
113+
114+
if (turns <= 2 || turns > 4) {
115+
expectedBytes.push(byte);
116+
}
117+
118+
if (turns === 6) {
119+
test.equal(emittedBytes.length, 4);
120+
emittedBytes.forEach((b, index) => test.equal(b, expectedBytes[index]));
121+
clearInterval(interval);
122+
test.done();
123+
}
124+
}, 5);
125+
},
126+
127+
componentSuspendability(test) {
128+
test.expect(5);
129+
const analogRead = this.analogRead.lastCall.args[1];
130+
const expectedBytes = [];
131+
const emittedBytes = [];
132+
133+
this.component.on("data", byte => emittedBytes.push(byte));
134+
135+
let turns = 0;
136+
let interval = setInterval(() => {
137+
const byte = Crypto.randomBytes(1);
138+
turns++;
139+
analogRead(byte);
140+
141+
if (turns === 2) {
142+
this.component.pause();
143+
}
144+
145+
if (turns === 4) {
146+
this.component.resume();
147+
}
148+
149+
if (turns <= 2 || turns > 4) {
150+
expectedBytes.push(byte);
151+
}
152+
153+
if (turns === 6) {
154+
test.equal(emittedBytes.length, 4);
155+
emittedBytes.forEach((b, index) => test.equal(b, expectedBytes[index]));
156+
clearInterval(interval);
157+
test.done();
158+
}
159+
}, 5);
160+
},
161+
162+
};

test/within.js test/withinable.js

File renamed without changes.

0 commit comments

Comments
 (0)