forked from jscl-project/jscl
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathservice-worker.js
131 lines (116 loc) · 3.99 KB
/
service-worker.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
/* service-worker.js
*
* This file is a service-worker which will be installed to provide
* offline responses to a HTTP service, providing input and some other
* async functionality to jscl. This allows to a jscl installed as a
* web worker to request them synchronously, using the XMLHttpRequest
* with async=false.
*
*/
/* This file, unlike the rest of the project is distributed under a permissive
* license, as it will be included in the generated code. */
/*
* Copyright (C) 2018 David Vazquez Pua
*
* Permission is hereby granted, free of charge, to any person obtaining
* a copy of this software and associated documentation files (the
* "Software"), to deal in the Software without restriction, including
* without limitation the rights to use, copy, modify, merge, publish,
* distribute, sublicense, and/or sell copies of the Software, and to
* permit persons to whom the Software is furnished to do so, subject to
* the following conditions:
*
* The above copyright notice and this permission notice shall be
* included in all copies or substantial portions of the Software.
*
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
* EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
* NON INFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
* LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
* OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
* WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
*
*/
"use strict";
const contexts = {};
// A context represents a client of the service-worker. We want to
// keep separate buffers for different clients for example, as it
// should be possible to use multiple REPLs in multiple tabs.
class Context {
constructor(sessionId) {
this.sessionId = sessionId;
this.resolvePendingRequest = undefined;
}
readStdin() {
return new Promise(async resolve => {
if (this.resolvePendingRequest) {
throw new Error(
`Concurrent requests from the same client are not allowed.`
);
}
const client = await self.clients.get(this.sessionId);
client.postMessage({
command: "prompt"
});
this.resolvePendingRequest = resolve;
});
}
sleep({ seconds }) {
return new Promise(resolve => {
setTimeout(() => resolve(true), seconds * 1000);
});
}
static findOrCreate(clientId) {
if (contexts[clientId]) {
return contexts[clientId];
} else {
contexts[clientId] = new Context(clientId);
return contexts[clientId];
}
}
}
// Message handler
self.addEventListener("message", function(event) {
const { sessionId, input } = event.data;
const context = Context.findOrCreate(sessionId);
context.resolvePendingRequest(input);
context.resolvePendingRequest = undefined;
});
self.addEventListener("fetch", event => {
const { request } = event;
if (request.url.endsWith("/__jscl")) {
event.respondWith(
(async function() {
const cmd = await request.json();
const { command, sessionId, options } = cmd;
let value;
switch (command) {
case "init": {
// Note that the client INIT request is made from the main
// thread.
const context = Context.findOrCreate(event.clientId);
value = context.sessionId;
break;
}
case "readStdin": {
const context = Context.findOrCreate(sessionId);
value = await context.readStdin();
break;
}
case "sleep": {
const context = Context.findOrCreate(sessionId);
value = await context.sleep(options);
break;
}
}
const response = new Response(JSON.stringify({ value: value }), {
headers: { "Content-Type": "application/json" }
});
return response;
})()
);
} else {
return fetch(event.request);
}
});