@learnpack/learnpack 2.1.39 → 2.1.40

Sign up to get free protection for your applications and to get access to all the features.
Files changed (196) hide show
  1. package/README.md +370 -35
  2. package/bin/run +17 -17
  3. package/bin/run.cmd +3 -3
  4. package/lib/commands/audit.d.ts +6 -6
  5. package/lib/commands/audit.js +342 -342
  6. package/lib/commands/clean.d.ts +8 -8
  7. package/lib/commands/clean.js +25 -25
  8. package/lib/commands/download.d.ts +13 -13
  9. package/lib/commands/download.js +55 -55
  10. package/lib/commands/init.d.ts +9 -9
  11. package/lib/commands/init.js +123 -123
  12. package/lib/commands/login.d.ts +14 -14
  13. package/lib/commands/login.js +37 -37
  14. package/lib/commands/logout.d.ts +14 -14
  15. package/lib/commands/logout.js +37 -37
  16. package/lib/commands/publish.d.ts +14 -14
  17. package/lib/commands/publish.js +82 -82
  18. package/lib/commands/start.d.ts +7 -7
  19. package/lib/commands/start.js +239 -208
  20. package/lib/commands/test.d.ts +6 -6
  21. package/lib/commands/test.js +62 -62
  22. package/lib/index.d.ts +1 -1
  23. package/lib/index.js +4 -4
  24. package/lib/managers/config/allowed_files.d.ts +5 -5
  25. package/lib/managers/config/allowed_files.js +30 -30
  26. package/lib/managers/config/defaults.d.ts +41 -41
  27. package/lib/managers/config/defaults.js +44 -44
  28. package/lib/managers/config/exercise.d.ts +36 -36
  29. package/lib/managers/config/exercise.js +236 -236
  30. package/lib/managers/config/index.d.ts +3 -3
  31. package/lib/managers/config/index.js +337 -337
  32. package/lib/managers/file.d.ts +14 -14
  33. package/lib/managers/file.js +153 -153
  34. package/lib/managers/gitpod.d.ts +3 -3
  35. package/lib/managers/gitpod.js +67 -67
  36. package/lib/managers/server/index.d.ts +6 -6
  37. package/lib/managers/server/index.js +58 -58
  38. package/lib/managers/server/routes.d.ts +4 -4
  39. package/lib/managers/server/routes.js +219 -219
  40. package/lib/managers/session.d.ts +3 -3
  41. package/lib/managers/session.js +125 -125
  42. package/lib/managers/socket.d.ts +3 -3
  43. package/lib/managers/socket.js +176 -176
  44. package/lib/managers/telemetry.d.ts +74 -74
  45. package/lib/managers/telemetry.js +206 -206
  46. package/lib/managers/test.js +84 -84
  47. package/lib/models/action.d.ts +2 -2
  48. package/lib/models/action.js +2 -2
  49. package/lib/models/audit.d.ts +15 -15
  50. package/lib/models/audit.js +2 -2
  51. package/lib/models/config-manager.d.ts +21 -21
  52. package/lib/models/config-manager.js +2 -2
  53. package/lib/models/config.d.ts +68 -67
  54. package/lib/models/config.js +2 -2
  55. package/lib/models/counter.d.ts +11 -11
  56. package/lib/models/counter.js +2 -2
  57. package/lib/models/errors.d.ts +15 -15
  58. package/lib/models/errors.js +2 -2
  59. package/lib/models/exercise-obj.d.ts +30 -30
  60. package/lib/models/exercise-obj.js +2 -2
  61. package/lib/models/file.d.ts +5 -5
  62. package/lib/models/file.js +2 -2
  63. package/lib/models/findings.d.ts +17 -17
  64. package/lib/models/findings.js +2 -2
  65. package/lib/models/flags.d.ts +10 -10
  66. package/lib/models/flags.js +2 -2
  67. package/lib/models/front-matter.d.ts +11 -11
  68. package/lib/models/front-matter.js +2 -2
  69. package/lib/models/gitpod-data.d.ts +16 -16
  70. package/lib/models/gitpod-data.js +2 -2
  71. package/lib/models/language.d.ts +4 -4
  72. package/lib/models/language.js +2 -2
  73. package/lib/models/package.d.ts +7 -7
  74. package/lib/models/package.js +2 -2
  75. package/lib/models/plugin-config.d.ts +16 -16
  76. package/lib/models/plugin-config.js +2 -2
  77. package/lib/models/session.d.ts +31 -31
  78. package/lib/models/session.js +2 -2
  79. package/lib/models/socket.d.ts +36 -32
  80. package/lib/models/socket.js +2 -2
  81. package/lib/models/status.d.ts +1 -1
  82. package/lib/models/status.js +2 -2
  83. package/lib/models/success-types.d.ts +1 -1
  84. package/lib/models/success-types.js +2 -2
  85. package/lib/plugin/command/compile.d.ts +6 -6
  86. package/lib/plugin/command/compile.js +18 -18
  87. package/lib/plugin/command/test.d.ts +6 -6
  88. package/lib/plugin/command/test.js +25 -25
  89. package/lib/plugin/index.d.ts +27 -27
  90. package/lib/plugin/index.js +7 -7
  91. package/lib/plugin/plugin.d.ts +8 -8
  92. package/lib/plugin/plugin.js +68 -68
  93. package/lib/plugin/utils.d.ts +16 -16
  94. package/lib/plugin/utils.js +58 -58
  95. package/lib/ui/download.d.ts +5 -5
  96. package/lib/ui/download.js +61 -61
  97. package/lib/utils/BaseCommand.d.ts +8 -8
  98. package/lib/utils/BaseCommand.js +41 -41
  99. package/lib/utils/SessionCommand.d.ts +10 -10
  100. package/lib/utils/SessionCommand.js +43 -43
  101. package/lib/utils/api.d.ts +14 -14
  102. package/lib/utils/api.js +255 -255
  103. package/lib/utils/audit.d.ts +16 -16
  104. package/lib/utils/audit.js +303 -303
  105. package/lib/utils/checkNotInstalled.d.ts +2 -0
  106. package/lib/utils/checkNotInstalled.js +36 -0
  107. package/lib/utils/console.d.ts +12 -12
  108. package/lib/utils/console.js +19 -19
  109. package/lib/utils/errors.d.ts +17 -17
  110. package/lib/utils/errors.js +100 -100
  111. package/lib/utils/exercisesQueue.d.ts +9 -9
  112. package/lib/utils/exercisesQueue.js +38 -38
  113. package/lib/utils/fileQueue.d.ts +40 -40
  114. package/lib/utils/fileQueue.js +168 -168
  115. package/lib/utils/misc.d.ts +1 -1
  116. package/lib/utils/misc.js +23 -23
  117. package/lib/utils/osOperations.d.ts +5 -5
  118. package/lib/utils/osOperations.js +72 -72
  119. package/lib/utils/validators.d.ts +5 -5
  120. package/lib/utils/validators.js +17 -17
  121. package/lib/utils/watcher.d.ts +2 -2
  122. package/lib/utils/watcher.js +25 -25
  123. package/oclif.manifest.json +1 -1
  124. package/package.json +139 -139
  125. package/src/commands/audit.ts +443 -443
  126. package/src/commands/clean.ts +29 -29
  127. package/src/commands/download.ts +61 -61
  128. package/src/commands/init.ts +170 -170
  129. package/src/commands/login.ts +42 -42
  130. package/src/commands/logout.ts +43 -43
  131. package/src/commands/publish.ts +107 -107
  132. package/src/commands/start.ts +53 -23
  133. package/src/commands/test.ts +85 -85
  134. package/src/index.ts +1 -1
  135. package/src/managers/config/allowed_files.ts +29 -29
  136. package/src/managers/config/defaults.ts +42 -42
  137. package/src/managers/config/exercise.ts +311 -311
  138. package/src/managers/config/index.ts +455 -455
  139. package/src/managers/file.ts +196 -196
  140. package/src/managers/gitpod.ts +84 -84
  141. package/src/managers/server/index.ts +78 -78
  142. package/src/managers/server/routes.ts +330 -330
  143. package/src/managers/session.ts +145 -145
  144. package/src/managers/socket.ts +250 -250
  145. package/src/managers/telemetry.ts +346 -346
  146. package/src/managers/test.ts +83 -83
  147. package/src/models/action.ts +10 -10
  148. package/src/models/audit.ts +16 -16
  149. package/src/models/config-manager.ts +23 -23
  150. package/src/models/config.ts +5 -3
  151. package/src/models/counter.ts +11 -11
  152. package/src/models/errors.ts +22 -22
  153. package/src/models/exercise-obj.ts +29 -29
  154. package/src/models/file.ts +5 -5
  155. package/src/models/findings.ts +18 -18
  156. package/src/models/flags.ts +10 -10
  157. package/src/models/front-matter.ts +11 -11
  158. package/src/models/gitpod-data.ts +19 -19
  159. package/src/models/language.ts +4 -4
  160. package/src/models/package.ts +7 -7
  161. package/src/models/plugin-config.ts +17 -17
  162. package/src/models/session.ts +34 -34
  163. package/src/models/socket.ts +5 -0
  164. package/src/models/status.ts +16 -16
  165. package/src/models/success-types.ts +1 -1
  166. package/src/plugin/command/compile.ts +17 -17
  167. package/src/plugin/command/test.ts +30 -30
  168. package/src/plugin/index.ts +6 -6
  169. package/src/plugin/plugin.ts +94 -94
  170. package/src/plugin/utils.ts +87 -87
  171. package/src/types/node-fetch.d.ts +1 -1
  172. package/src/ui/download.ts +71 -71
  173. package/src/utils/BaseCommand.ts +48 -48
  174. package/src/utils/SessionCommand.ts +43 -43
  175. package/src/utils/api.ts +303 -303
  176. package/src/utils/audit.ts +393 -393
  177. package/src/utils/checkNotInstalled.ts +46 -0
  178. package/src/utils/console.ts +24 -24
  179. package/src/utils/errors.ts +117 -117
  180. package/src/utils/exercisesQueue.ts +51 -51
  181. package/src/utils/fileQueue.ts +198 -198
  182. package/src/utils/misc.ts +23 -23
  183. package/src/utils/osOperations.ts +79 -79
  184. package/src/utils/templates/gitignore.txt +19 -19
  185. package/src/utils/templates/incremental/.learn/exercises/01-hello-world/README.es.md +24 -24
  186. package/src/utils/templates/incremental/.learn/exercises/01-hello-world/README.md +24 -24
  187. package/src/utils/templates/incremental/README.ejs +4 -4
  188. package/src/utils/templates/incremental/README.es.ejs +4 -4
  189. package/src/utils/templates/isolated/01-hello-world/README.es.md +26 -26
  190. package/src/utils/templates/isolated/01-hello-world/README.md +26 -26
  191. package/src/utils/templates/isolated/README.ejs +4 -4
  192. package/src/utils/templates/isolated/README.es.ejs +4 -4
  193. package/src/utils/templates/no-grading/README.ejs +4 -4
  194. package/src/utils/templates/no-grading/README.es.ejs +4 -4
  195. package/src/utils/validators.ts +18 -18
  196. package/src/utils/watcher.ts +27 -27
@@ -1,219 +1,219 @@
1
- "use strict";
2
- Object.defineProperty(exports, "__esModule", { value: true });
3
- const console_1 = require("../../utils/console");
4
- const express = require("express");
5
- const fs = require("fs");
6
- const bodyParser = require("body-parser");
7
- const socket_1 = require("../socket");
8
- const fileQueue_1 = require("../../utils/fileQueue");
9
- // import gitpod from '../gitpod'
10
- const exercise_1 = require("../config/exercise");
11
- const session_1 = require("../../managers/session");
12
- const telemetry_1 = require("../telemetry");
13
- const osOperations_1 = require("../../utils/osOperations");
14
- const withHandler = (func) => (req, res) => {
15
- try {
16
- func(req, res);
17
- }
18
- catch (error) {
19
- console_1.default.debug(error);
20
- const _err = {
21
- message: error.message || "There has been an error",
22
- status: error.status || 500,
23
- type: error.type || null,
24
- };
25
- console_1.default.error(_err.message);
26
- // send rep to the server
27
- res.status(_err.status);
28
- res.json(_err);
29
- }
30
- };
31
- async function default_1(app, configObject, configManager) {
32
- const { config, exercises } = configObject;
33
- const session = await session_1.default.get(configManager === null || configManager === void 0 ? void 0 : configManager.get());
34
- const dispatcher = fileQueue_1.default.dispatcher({
35
- create: true,
36
- path: `${config === null || config === void 0 ? void 0 : config.dirPath}/vscode_queue.json`,
37
- });
38
- app.get("/config", withHandler((_, res) => {
39
- res.json(configObject);
40
- }));
41
- // Added this line to parse the json body
42
- const jsonBodyParser = bodyParser.json();
43
- // Trying to log in from frontend
44
- app.post("/login", jsonBodyParser, withHandler(async (req, res) => {
45
- const email = req.body.email;
46
- const password = req.body.password;
47
- session_1.default.destroy();
48
- const payload = await session_1.default.loginWeb(email, password);
49
- res.json(payload);
50
- }));
51
- app.post("/set-rigobot-token", jsonBodyParser, withHandler(async (req, res) => {
52
- const token = req.body.token;
53
- // Ensure token is provided in the request body
54
- if (!token) {
55
- return res.status(400).json({ error: "Token is required" });
56
- }
57
- try {
58
- const tokenSaved = await session_1.default.setRigoToken(token);
59
- // Check if the token was saved successfully
60
- if (tokenSaved) {
61
- res.json({ status: "ok" });
62
- }
63
- else {
64
- res.status(500).json({ error: "Failed to save the token" });
65
- }
66
- }
67
- catch (_a) {
68
- // Handle any unexpected errors during the process
69
- res.status(500).json({ error: "Internal server error" });
70
- }
71
- }));
72
- app.get("/check/rigo/status", withHandler(async (_, res) => {
73
- const payload = await session_1.default.getPayload();
74
- if (payload && payload.rigobot && payload.rigobot.key) {
75
- res.json({ rigoToken: payload.rigobot.key });
76
- }
77
- else {
78
- res
79
- .status(400)
80
- .json({ details: `Rigobot token not found, please log in first!` });
81
- }
82
- }));
83
- // symbolic link to maintain path compatiblity
84
- const fetchStaticAsset = withHandler((req, res) => {
85
- const filePath = `${config === null || config === void 0 ? void 0 : config.dirPath}/assets/${req.params.filePath}`;
86
- if (!fs.existsSync(filePath))
87
- throw new Error("File not found: " + filePath);
88
- const content = fs.readFileSync(filePath);
89
- res.write(content);
90
- res.end();
91
- });
92
- app.get(`${(config === null || config === void 0 ? void 0 : config.dirPath.indexOf("./")) === 0 ?
93
- config.dirPath.slice(1) : config === null || config === void 0 ? void 0 : config.dirPath}/assets/:filePath`, fetchStaticAsset);
94
- app.get("/assets/:filePath", fetchStaticAsset);
95
- app.get("/exercise", withHandler((_, res) => {
96
- res.json(exercises);
97
- }));
98
- app.get("/exercise/:slug/readme", withHandler(({ params: { slug }, query: { lang } }, res) => {
99
- const exercise = configManager.getExercise(slug);
100
- if (exercise && exercise.getReadme) {
101
- const readme = exercise.getReadme(lang || null);
102
- res.json(readme);
103
- }
104
- else {
105
- res.status(400);
106
- }
107
- }));
108
- app.get("/exercise/:slug/report", withHandler(({ params: { slug } }, res) => {
109
- const exercise = configManager.getExercise(slug);
110
- if (exercise && exercise.getTestReport) {
111
- const report = exercise.getTestReport();
112
- res.json(JSON.stringify(report));
113
- }
114
- }));
115
- app.get("/exercise/:slug", withHandler((req, res) => {
116
- var _a, _b, _c, _d, _e;
117
- // no need to re-start exercise if it's already started
118
- if (configObject.currentExercise &&
119
- req.params.slug === configObject.currentExercise) {
120
- const exercise = configManager.getExercise(req.params.slug);
121
- res.json(exercise);
122
- if (exercise.position) {
123
- telemetry_1.default.registerStepEvent(exercise.position, "open_step", {});
124
- }
125
- return;
126
- }
127
- const exercise = configManager.startExercise(req.params.slug);
128
- if (exercise.position) {
129
- telemetry_1.default.registerStepEvent(exercise.position, "open_step", {});
130
- }
131
- if (((_a = configObject.config) === null || _a === void 0 ? void 0 : _a.editor.agent) === "os") {
132
- osOperations_1.eventManager.enqueue(dispatcher.events.START_EXERCISE, exercise);
133
- }
134
- else {
135
- dispatcher.enqueue(dispatcher.events.START_EXERCISE, req.params.slug);
136
- }
137
- const entries = new Set(Object.keys(config === null || config === void 0 ? void 0 : config.entries).map(lang => config === null || config === void 0 ? void 0 : config.entries[lang]));
138
- // if we are in incremental grading, the entry file can by dinamically detected
139
- // based on the changes the student is making during the exercise
140
- if ((config === null || config === void 0 ? void 0 : config.grading) === "incremental") {
141
- const scanedFiles = fs.readdirSync("./");
142
- // update the file hierarchy with updates
143
- exercise.files = [
144
- ...exercise.files.filter(f => f.name.includes("test.")),
145
- ...exercise_1.filterFiles(scanedFiles),
146
- ];
147
- console_1.default.debug(`Exercise updated files: `, exercise.files);
148
- }
149
- const detected = exercise_1.detect(configObject, exercise.files
150
- .filter(fileName => entries.has(fileName.name))
151
- .map(f => f.name || f));
152
- // if a new language for the testing engine is detected, we replace it
153
- // if not we leave it as it was before
154
- if ((config === null || config === void 0 ? void 0 : config.language) && !["", "auto"].includes(config === null || config === void 0 ? void 0 : config.language)) {
155
- console_1.default.debug(`Exercise language ignored, instead imported from configuration ${config === null || config === void 0 ? void 0 : config.language}`);
156
- exercise.language = detected === null || detected === void 0 ? void 0 : detected.language;
157
- }
158
- else if ((detected === null || detected === void 0 ? void 0 : detected.language) &&
159
- (!(config === null || config === void 0 ? void 0 : config.language) || (config === null || config === void 0 ? void 0 : config.language) === "auto")) {
160
- console_1.default.debug(`Switching to ${detected.language} engine in this exercise`);
161
- exercise.language = detected.language;
162
- }
163
- // WARNING: has to be the FULL PATH to the entry path
164
- // We need to detect entry in both gradings: Incremental and Isolate
165
- exercise.entry = detected === null || detected === void 0 ? void 0 : detected.entry;
166
- console_1.default.debug(`Exercise detected entry: ${detected === null || detected === void 0 ? void 0 : detected.entry} and language ${exercise.language}`);
167
- // exercises.graded and exercises.disableGrading deprecated.
168
- if (!exercise.graded || (config === null || config === void 0 ? void 0 : config.disableGrading) || ((_b = config === null || config === void 0 ? void 0 : config.disabledActions) === null || _b === void 0 ? void 0 : _b.includes("test"))) {
169
- socket_1.default.removeAllowed("test");
170
- }
171
- else {
172
- socket_1.default.addAllowed("test");
173
- }
174
- if (!exercise.entry || ((_c = config === null || config === void 0 ? void 0 : config.disabledActions) === null || _c === void 0 ? void 0 : _c.includes("build"))) {
175
- socket_1.default.removeAllowed("build");
176
- }
177
- else {
178
- socket_1.default.addAllowed("build");
179
- }
180
- if (exercise.files.filter((f) => !f.name.toLowerCase().includes("readme.") &&
181
- !f.name.toLowerCase().includes("test.")).length === 0 || ((_d = config === null || config === void 0 ? void 0 : config.disabledActions) === null || _d === void 0 ? void 0 : _d.includes("reset"))) {
182
- socket_1.default.removeAllowed("reset");
183
- }
184
- else if (!((_e = config === null || config === void 0 ? void 0 : config.disabledActions) === null || _e === void 0 ? void 0 : _e.includes("reset"))) {
185
- socket_1.default.addAllowed("reset");
186
- }
187
- socket_1.default.log("ready");
188
- res.json(exercise);
189
- }));
190
- app.get("/exercise/:slug/file/:fileName", withHandler((req, res) => {
191
- const exercise = configManager.getExercise(req.params.slug);
192
- if (exercise && exercise.getFile) {
193
- res.write(exercise.getFile(req.params.fileName));
194
- res.end();
195
- }
196
- }));
197
- /*
198
- app.post(
199
- "/exercise/:slug/file/:fileName",
200
- withHandler((req: express.Request, res: express.Response) => {
201
- get tokens but also, add allowed action for 'generate'
202
- use the sessionManager to keep compatibility with the cli login command.
203
- })
204
- );
205
- */
206
- const textBodyParser = bodyParser.text();
207
- app.put("/exercise/:slug/file/:fileName", textBodyParser, withHandler((req, res) => {
208
- const exercise = configManager.getExercise(req.params.slug);
209
- if (exercise && exercise.saveFile) {
210
- exercise.saveFile(req.params.fileName, req.body);
211
- res.end();
212
- }
213
- }));
214
- if (config === null || config === void 0 ? void 0 : config.outputPath) {
215
- app.use("/preview", express.static(config.outputPath));
216
- }
217
- app.use("/", express.static(`${config === null || config === void 0 ? void 0 : config.dirPath}/_app`));
218
- }
219
- exports.default = default_1;
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ const console_1 = require("../../utils/console");
4
+ const express = require("express");
5
+ const fs = require("fs");
6
+ const bodyParser = require("body-parser");
7
+ const socket_1 = require("../socket");
8
+ const fileQueue_1 = require("../../utils/fileQueue");
9
+ // import gitpod from '../gitpod'
10
+ const exercise_1 = require("../config/exercise");
11
+ const session_1 = require("../../managers/session");
12
+ const telemetry_1 = require("../telemetry");
13
+ const osOperations_1 = require("../../utils/osOperations");
14
+ const withHandler = (func) => (req, res) => {
15
+ try {
16
+ func(req, res);
17
+ }
18
+ catch (error) {
19
+ console_1.default.debug(error);
20
+ const _err = {
21
+ message: error.message || "There has been an error",
22
+ status: error.status || 500,
23
+ type: error.type || null,
24
+ };
25
+ console_1.default.error(_err.message);
26
+ // send rep to the server
27
+ res.status(_err.status);
28
+ res.json(_err);
29
+ }
30
+ };
31
+ async function default_1(app, configObject, configManager) {
32
+ const { config, exercises } = configObject;
33
+ const session = await session_1.default.get(configManager === null || configManager === void 0 ? void 0 : configManager.get());
34
+ const dispatcher = fileQueue_1.default.dispatcher({
35
+ create: true,
36
+ path: `${config === null || config === void 0 ? void 0 : config.dirPath}/vscode_queue.json`,
37
+ });
38
+ app.get("/config", withHandler((_, res) => {
39
+ res.json(configObject);
40
+ }));
41
+ // Added this line to parse the json body
42
+ const jsonBodyParser = bodyParser.json();
43
+ // Trying to log in from frontend
44
+ app.post("/login", jsonBodyParser, withHandler(async (req, res) => {
45
+ const email = req.body.email;
46
+ const password = req.body.password;
47
+ session_1.default.destroy();
48
+ const payload = await session_1.default.loginWeb(email, password);
49
+ res.json(payload);
50
+ }));
51
+ app.post("/set-rigobot-token", jsonBodyParser, withHandler(async (req, res) => {
52
+ const token = req.body.token;
53
+ // Ensure token is provided in the request body
54
+ if (!token) {
55
+ return res.status(400).json({ error: "Token is required" });
56
+ }
57
+ try {
58
+ const tokenSaved = await session_1.default.setRigoToken(token);
59
+ // Check if the token was saved successfully
60
+ if (tokenSaved) {
61
+ res.json({ status: "ok" });
62
+ }
63
+ else {
64
+ res.status(500).json({ error: "Failed to save the token" });
65
+ }
66
+ }
67
+ catch (_a) {
68
+ // Handle any unexpected errors during the process
69
+ res.status(500).json({ error: "Internal server error" });
70
+ }
71
+ }));
72
+ app.get("/check/rigo/status", withHandler(async (_, res) => {
73
+ const payload = await session_1.default.getPayload();
74
+ if (payload && payload.rigobot && payload.rigobot.key) {
75
+ res.json({ rigoToken: payload.rigobot.key });
76
+ }
77
+ else {
78
+ res
79
+ .status(400)
80
+ .json({ details: `Rigobot token not found, please log in first!` });
81
+ }
82
+ }));
83
+ // symbolic link to maintain path compatiblity
84
+ const fetchStaticAsset = withHandler((req, res) => {
85
+ const filePath = `${config === null || config === void 0 ? void 0 : config.dirPath}/assets/${req.params.filePath}`;
86
+ if (!fs.existsSync(filePath))
87
+ throw new Error("File not found: " + filePath);
88
+ const content = fs.readFileSync(filePath);
89
+ res.write(content);
90
+ res.end();
91
+ });
92
+ app.get(`${(config === null || config === void 0 ? void 0 : config.dirPath.indexOf("./")) === 0 ?
93
+ config.dirPath.slice(1) : config === null || config === void 0 ? void 0 : config.dirPath}/assets/:filePath`, fetchStaticAsset);
94
+ app.get("/assets/:filePath", fetchStaticAsset);
95
+ app.get("/exercise", withHandler((_, res) => {
96
+ res.json(exercises);
97
+ }));
98
+ app.get("/exercise/:slug/readme", withHandler(({ params: { slug }, query: { lang } }, res) => {
99
+ const exercise = configManager.getExercise(slug);
100
+ if (exercise && exercise.getReadme) {
101
+ const readme = exercise.getReadme(lang || null);
102
+ res.json(readme);
103
+ }
104
+ else {
105
+ res.status(400);
106
+ }
107
+ }));
108
+ app.get("/exercise/:slug/report", withHandler(({ params: { slug } }, res) => {
109
+ const exercise = configManager.getExercise(slug);
110
+ if (exercise && exercise.getTestReport) {
111
+ const report = exercise.getTestReport();
112
+ res.json(JSON.stringify(report));
113
+ }
114
+ }));
115
+ app.get("/exercise/:slug", withHandler((req, res) => {
116
+ var _a, _b, _c, _d, _e;
117
+ // no need to re-start exercise if it's already started
118
+ if (configObject.currentExercise &&
119
+ req.params.slug === configObject.currentExercise) {
120
+ const exercise = configManager.getExercise(req.params.slug);
121
+ res.json(exercise);
122
+ if (exercise.position) {
123
+ telemetry_1.default.registerStepEvent(exercise.position, "open_step", {});
124
+ }
125
+ return;
126
+ }
127
+ const exercise = configManager.startExercise(req.params.slug);
128
+ if (exercise.position) {
129
+ telemetry_1.default.registerStepEvent(exercise.position, "open_step", {});
130
+ }
131
+ if (((_a = configObject.config) === null || _a === void 0 ? void 0 : _a.editor.agent) === "os") {
132
+ osOperations_1.eventManager.enqueue(dispatcher.events.START_EXERCISE, exercise);
133
+ }
134
+ else {
135
+ dispatcher.enqueue(dispatcher.events.START_EXERCISE, req.params.slug);
136
+ }
137
+ const entries = new Set(Object.keys(config === null || config === void 0 ? void 0 : config.entries).map(lang => config === null || config === void 0 ? void 0 : config.entries[lang]));
138
+ // if we are in incremental grading, the entry file can by dinamically detected
139
+ // based on the changes the student is making during the exercise
140
+ if ((config === null || config === void 0 ? void 0 : config.grading) === "incremental") {
141
+ const scanedFiles = fs.readdirSync("./");
142
+ // update the file hierarchy with updates
143
+ exercise.files = [
144
+ ...exercise.files.filter(f => f.name.includes("test.")),
145
+ ...exercise_1.filterFiles(scanedFiles),
146
+ ];
147
+ console_1.default.debug(`Exercise updated files: `, exercise.files);
148
+ }
149
+ const detected = exercise_1.detect(configObject, exercise.files
150
+ .filter(fileName => entries.has(fileName.name))
151
+ .map(f => f.name || f));
152
+ // if a new language for the testing engine is detected, we replace it
153
+ // if not we leave it as it was before
154
+ if ((config === null || config === void 0 ? void 0 : config.language) && !["", "auto"].includes(config === null || config === void 0 ? void 0 : config.language)) {
155
+ console_1.default.debug(`Exercise language ignored, instead imported from configuration ${config === null || config === void 0 ? void 0 : config.language}`);
156
+ exercise.language = detected === null || detected === void 0 ? void 0 : detected.language;
157
+ }
158
+ else if ((detected === null || detected === void 0 ? void 0 : detected.language) &&
159
+ (!(config === null || config === void 0 ? void 0 : config.language) || (config === null || config === void 0 ? void 0 : config.language) === "auto")) {
160
+ console_1.default.debug(`Switching to ${detected.language} engine in this exercise`);
161
+ exercise.language = detected.language;
162
+ }
163
+ // WARNING: has to be the FULL PATH to the entry path
164
+ // We need to detect entry in both gradings: Incremental and Isolate
165
+ exercise.entry = detected === null || detected === void 0 ? void 0 : detected.entry;
166
+ console_1.default.debug(`Exercise detected entry: ${detected === null || detected === void 0 ? void 0 : detected.entry} and language ${exercise.language}`);
167
+ // exercises.graded and exercises.disableGrading deprecated.
168
+ if (!exercise.graded || (config === null || config === void 0 ? void 0 : config.disableGrading) || ((_b = config === null || config === void 0 ? void 0 : config.disabledActions) === null || _b === void 0 ? void 0 : _b.includes("test"))) {
169
+ socket_1.default.removeAllowed("test");
170
+ }
171
+ else {
172
+ socket_1.default.addAllowed("test");
173
+ }
174
+ if (!exercise.entry || ((_c = config === null || config === void 0 ? void 0 : config.disabledActions) === null || _c === void 0 ? void 0 : _c.includes("build"))) {
175
+ socket_1.default.removeAllowed("build");
176
+ }
177
+ else {
178
+ socket_1.default.addAllowed("build");
179
+ }
180
+ if (exercise.files.filter((f) => !f.name.toLowerCase().includes("readme.") &&
181
+ !f.name.toLowerCase().includes("test.")).length === 0 || ((_d = config === null || config === void 0 ? void 0 : config.disabledActions) === null || _d === void 0 ? void 0 : _d.includes("reset"))) {
182
+ socket_1.default.removeAllowed("reset");
183
+ }
184
+ else if (!((_e = config === null || config === void 0 ? void 0 : config.disabledActions) === null || _e === void 0 ? void 0 : _e.includes("reset"))) {
185
+ socket_1.default.addAllowed("reset");
186
+ }
187
+ socket_1.default.log("ready");
188
+ res.json(exercise);
189
+ }));
190
+ app.get("/exercise/:slug/file/:fileName", withHandler((req, res) => {
191
+ const exercise = configManager.getExercise(req.params.slug);
192
+ if (exercise && exercise.getFile) {
193
+ res.write(exercise.getFile(req.params.fileName));
194
+ res.end();
195
+ }
196
+ }));
197
+ /*
198
+ app.post(
199
+ "/exercise/:slug/file/:fileName",
200
+ withHandler((req: express.Request, res: express.Response) => {
201
+ get tokens but also, add allowed action for 'generate'
202
+ use the sessionManager to keep compatibility with the cli login command.
203
+ })
204
+ );
205
+ */
206
+ const textBodyParser = bodyParser.text();
207
+ app.put("/exercise/:slug/file/:fileName", textBodyParser, withHandler((req, res) => {
208
+ const exercise = configManager.getExercise(req.params.slug);
209
+ if (exercise && exercise.saveFile) {
210
+ exercise.saveFile(req.params.fileName, req.body);
211
+ res.end();
212
+ }
213
+ }));
214
+ if (config === null || config === void 0 ? void 0 : config.outputPath) {
215
+ app.use("/preview", express.static(config.outputPath));
216
+ }
217
+ app.use("/", express.static(`${config === null || config === void 0 ? void 0 : config.dirPath}/_app`));
218
+ }
219
+ exports.default = default_1;
@@ -1,3 +1,3 @@
1
- import { ISession } from "../models/session";
2
- declare const Session: ISession;
3
- export default Session;
1
+ import { ISession } from "../models/session";
2
+ declare const Session: ISession;
3
+ export default Session;