@learnpack/learnpack 2.1.25 → 2.1.27

Sign up to get free protection for your applications and to get access to all the features.
Files changed (187) hide show
  1. package/README.md +16 -16
  2. package/bin/run +17 -17
  3. package/bin/run.cmd +3 -3
  4. package/oclif.manifest.json +1 -1
  5. package/package.json +139 -138
  6. package/src/commands/audit.ts +134 -109
  7. package/src/commands/clean.ts +29 -29
  8. package/src/commands/download.ts +62 -62
  9. package/src/commands/init.ts +39 -39
  10. package/src/commands/login.ts +42 -42
  11. package/src/commands/logout.ts +43 -43
  12. package/src/commands/publish.ts +107 -107
  13. package/src/commands/start.ts +264 -234
  14. package/src/commands/test.ts +85 -85
  15. package/src/index.ts +1 -1
  16. package/src/managers/config/allowed_files.ts +29 -29
  17. package/src/managers/config/defaults.ts +4 -2
  18. package/src/managers/config/exercise.ts +309 -302
  19. package/src/managers/config/index.ts +159 -138
  20. package/src/managers/file.ts +178 -169
  21. package/src/managers/gitpod.ts +84 -84
  22. package/src/managers/server/index.ts +78 -69
  23. package/src/managers/server/routes.ts +139 -90
  24. package/src/managers/session.ts +147 -118
  25. package/src/managers/socket.ts +252 -239
  26. package/src/managers/test.ts +83 -83
  27. package/src/models/action.ts +10 -3
  28. package/src/models/config-manager.ts +23 -23
  29. package/src/models/config.ts +9 -2
  30. package/src/models/counter.ts +11 -11
  31. package/src/models/errors.ts +22 -22
  32. package/src/models/exercise-obj.ts +6 -3
  33. package/src/models/file.ts +5 -5
  34. package/src/models/findings.ts +18 -18
  35. package/src/models/flags.ts +10 -10
  36. package/src/models/front-matter.ts +11 -11
  37. package/src/models/gitpod-data.ts +19 -19
  38. package/src/models/language.ts +4 -4
  39. package/src/models/package.ts +7 -7
  40. package/src/models/plugin-config.ts +17 -17
  41. package/src/models/session.ts +29 -26
  42. package/src/models/socket.ts +54 -48
  43. package/src/models/status.ts +16 -15
  44. package/src/models/success-types.ts +1 -1
  45. package/src/plugin/command/compile.ts +17 -17
  46. package/src/plugin/command/test.ts +30 -30
  47. package/src/plugin/index.ts +6 -6
  48. package/src/plugin/plugin.ts +94 -94
  49. package/src/plugin/utils.ts +87 -87
  50. package/src/types/node-fetch.d.ts +1 -1
  51. package/src/ui/download.ts +71 -71
  52. package/src/utils/BaseCommand.ts +48 -48
  53. package/src/utils/SessionCommand.ts +48 -48
  54. package/src/utils/api.ts +246 -194
  55. package/src/utils/audit.ts +392 -395
  56. package/src/utils/console.ts +24 -24
  57. package/src/utils/errors.ts +117 -117
  58. package/src/utils/exercisesQueue.ts +51 -51
  59. package/src/utils/fileQueue.ts +198 -198
  60. package/src/utils/misc.ts +23 -23
  61. package/src/utils/templates/gitignore.txt +19 -19
  62. package/src/utils/templates/incremental/.learn/exercises/01-hello-world/README.es.md +24 -24
  63. package/src/utils/templates/incremental/.learn/exercises/01-hello-world/README.md +24 -24
  64. package/src/utils/templates/incremental/README.ejs +4 -4
  65. package/src/utils/templates/incremental/README.es.ejs +4 -4
  66. package/src/utils/templates/isolated/01-hello-world/README.es.md +26 -26
  67. package/src/utils/templates/isolated/01-hello-world/README.md +26 -26
  68. package/src/utils/templates/isolated/README.ejs +4 -4
  69. package/src/utils/templates/isolated/README.es.ejs +4 -4
  70. package/src/utils/templates/no-grading/README.ejs +4 -4
  71. package/src/utils/templates/no-grading/README.es.ejs +4 -4
  72. package/src/utils/validators.ts +18 -18
  73. package/src/utils/watcher.ts +14 -14
  74. package/lib/commands/audit.d.ts +0 -6
  75. package/lib/commands/audit.js +0 -317
  76. package/lib/commands/clean.d.ts +0 -8
  77. package/lib/commands/clean.js +0 -25
  78. package/lib/commands/download.d.ts +0 -13
  79. package/lib/commands/download.js +0 -55
  80. package/lib/commands/init.d.ts +0 -9
  81. package/lib/commands/init.js +0 -123
  82. package/lib/commands/login.d.ts +0 -14
  83. package/lib/commands/login.js +0 -37
  84. package/lib/commands/logout.d.ts +0 -14
  85. package/lib/commands/logout.js +0 -37
  86. package/lib/commands/publish.d.ts +0 -14
  87. package/lib/commands/publish.js +0 -82
  88. package/lib/commands/start.d.ts +0 -7
  89. package/lib/commands/start.js +0 -165
  90. package/lib/commands/test.d.ts +0 -6
  91. package/lib/commands/test.js +0 -62
  92. package/lib/index.d.ts +0 -1
  93. package/lib/index.js +0 -4
  94. package/lib/managers/config/allowed_files.d.ts +0 -5
  95. package/lib/managers/config/allowed_files.js +0 -30
  96. package/lib/managers/config/defaults.d.ts +0 -37
  97. package/lib/managers/config/defaults.js +0 -38
  98. package/lib/managers/config/exercise.d.ts +0 -36
  99. package/lib/managers/config/exercise.js +0 -230
  100. package/lib/managers/config/index.d.ts +0 -3
  101. package/lib/managers/config/index.js +0 -302
  102. package/lib/managers/file.d.ts +0 -13
  103. package/lib/managers/file.js +0 -134
  104. package/lib/managers/gitpod.d.ts +0 -3
  105. package/lib/managers/gitpod.js +0 -67
  106. package/lib/managers/server/index.d.ts +0 -6
  107. package/lib/managers/server/index.js +0 -51
  108. package/lib/managers/server/routes.d.ts +0 -4
  109. package/lib/managers/server/routes.js +0 -167
  110. package/lib/managers/session.d.ts +0 -3
  111. package/lib/managers/session.js +0 -104
  112. package/lib/managers/socket.d.ts +0 -3
  113. package/lib/managers/socket.js +0 -164
  114. package/lib/managers/test.d.ts +0 -0
  115. package/lib/managers/test.js +0 -84
  116. package/lib/models/action.d.ts +0 -2
  117. package/lib/models/action.js +0 -2
  118. package/lib/models/audit.d.ts +0 -15
  119. package/lib/models/audit.js +0 -2
  120. package/lib/models/config-manager.d.ts +0 -21
  121. package/lib/models/config-manager.js +0 -2
  122. package/lib/models/config.d.ts +0 -60
  123. package/lib/models/config.js +0 -2
  124. package/lib/models/counter.d.ts +0 -11
  125. package/lib/models/counter.js +0 -2
  126. package/lib/models/errors.d.ts +0 -15
  127. package/lib/models/errors.js +0 -2
  128. package/lib/models/exercise-obj.d.ts +0 -27
  129. package/lib/models/exercise-obj.js +0 -2
  130. package/lib/models/file.d.ts +0 -5
  131. package/lib/models/file.js +0 -2
  132. package/lib/models/findings.d.ts +0 -17
  133. package/lib/models/findings.js +0 -2
  134. package/lib/models/flags.d.ts +0 -10
  135. package/lib/models/flags.js +0 -2
  136. package/lib/models/front-matter.d.ts +0 -11
  137. package/lib/models/front-matter.js +0 -2
  138. package/lib/models/gitpod-data.d.ts +0 -16
  139. package/lib/models/gitpod-data.js +0 -2
  140. package/lib/models/language.d.ts +0 -4
  141. package/lib/models/language.js +0 -2
  142. package/lib/models/package.d.ts +0 -7
  143. package/lib/models/package.js +0 -2
  144. package/lib/models/plugin-config.d.ts +0 -16
  145. package/lib/models/plugin-config.js +0 -2
  146. package/lib/models/session.d.ts +0 -23
  147. package/lib/models/session.js +0 -2
  148. package/lib/models/socket.d.ts +0 -31
  149. package/lib/models/socket.js +0 -2
  150. package/lib/models/status.d.ts +0 -1
  151. package/lib/models/status.js +0 -2
  152. package/lib/models/success-types.d.ts +0 -1
  153. package/lib/models/success-types.js +0 -2
  154. package/lib/plugin/command/compile.d.ts +0 -6
  155. package/lib/plugin/command/compile.js +0 -18
  156. package/lib/plugin/command/test.d.ts +0 -6
  157. package/lib/plugin/command/test.js +0 -25
  158. package/lib/plugin/index.d.ts +0 -27
  159. package/lib/plugin/index.js +0 -7
  160. package/lib/plugin/plugin.d.ts +0 -8
  161. package/lib/plugin/plugin.js +0 -68
  162. package/lib/plugin/utils.d.ts +0 -16
  163. package/lib/plugin/utils.js +0 -58
  164. package/lib/ui/download.d.ts +0 -5
  165. package/lib/ui/download.js +0 -61
  166. package/lib/utils/BaseCommand.d.ts +0 -8
  167. package/lib/utils/BaseCommand.js +0 -41
  168. package/lib/utils/SessionCommand.d.ts +0 -10
  169. package/lib/utils/SessionCommand.js +0 -47
  170. package/lib/utils/api.d.ts +0 -12
  171. package/lib/utils/api.js +0 -173
  172. package/lib/utils/audit.d.ts +0 -16
  173. package/lib/utils/audit.js +0 -302
  174. package/lib/utils/console.d.ts +0 -12
  175. package/lib/utils/console.js +0 -19
  176. package/lib/utils/errors.d.ts +0 -17
  177. package/lib/utils/errors.js +0 -100
  178. package/lib/utils/exercisesQueue.d.ts +0 -9
  179. package/lib/utils/exercisesQueue.js +0 -38
  180. package/lib/utils/fileQueue.d.ts +0 -40
  181. package/lib/utils/fileQueue.js +0 -168
  182. package/lib/utils/misc.d.ts +0 -1
  183. package/lib/utils/misc.js +0 -23
  184. package/lib/utils/validators.d.ts +0 -5
  185. package/lib/utils/validators.js +0 -17
  186. package/lib/utils/watcher.d.ts +0 -2
  187. package/lib/utils/watcher.js +0 -23
@@ -1,395 +1,392 @@
1
- import { IAuditErrors } from "../models/audit";
2
- import { IConfigObj } from "../models/config";
3
- import { ICounter } from "../models/counter";
4
- import { IFindings } from "../models/findings";
5
- import { IExercise } from "../models/exercise-obj";
6
- import { IFrontmatter } from "../models/front-matter";
7
- import Console from "./console";
8
- import * as fs from "fs";
9
- import * as path from "path";
10
-
11
- // eslint-disable-next-line
12
- const fetch = require("node-fetch");
13
- // eslint-disable-next-line
14
- const fm = require("front-matter");
15
-
16
- // This function checks if a url is valid.
17
- const isUrl = async (
18
- url: string,
19
- errors: IAuditErrors[],
20
- counter: ICounter
21
- ) => {
22
- const regexUrl = /(https?:\/\/[\w./-]+)/gm;
23
- counter.links.total++;
24
- if (!regexUrl.test(url)) {
25
- counter.links.error++;
26
- errors.push({
27
- exercise: undefined,
28
- msg: `The repository value of the configuration file is not a link: ${url}`,
29
- });
30
- return false;
31
- }
32
-
33
- const res = await fetch(url, { method: "HEAD" });
34
- if (!res.ok) {
35
- counter.links.error++;
36
- errors.push({
37
- exercise: undefined,
38
- msg: `The link of the repository is broken: ${url}`,
39
- });
40
- }
41
-
42
- return true;
43
- };
44
-
45
- const checkForEmptySpaces = (str: string) => {
46
- const isEmpty = true;
47
- for (const letter of str) {
48
- if (letter !== " ") {
49
- return false;
50
- }
51
- }
52
-
53
- return isEmpty;
54
- };
55
-
56
- const checkLearnpackClean = (configObj: IConfigObj, errors: IAuditErrors[]) => {
57
- if (
58
- (configObj.config?.outputPath &&
59
- fs.existsSync(configObj.config?.outputPath)) ||
60
- fs.existsSync(`${configObj.config?.dirPath}/_app`) ||
61
- fs.existsSync(`${configObj.config?.dirPath}/reports`) ||
62
- fs.existsSync(`${configObj.config?.dirPath}/resets`) ||
63
- fs.existsSync(`${configObj.config?.dirPath}/app.tar.gz`) ||
64
- fs.existsSync(`${configObj.config?.dirPath}/config.json`) ||
65
- fs.existsSync(`${configObj.config?.dirPath}/vscode_queue.json`)
66
- ) {
67
- errors.push({
68
- exercise: undefined,
69
- msg: "You have to run learnpack clean command",
70
- });
71
- }
72
- };
73
-
74
- const findInFile = (types: string[], content: string) => {
75
- const regex: any = {
76
- relativeImages:
77
- /!\[.*]\s*\((((\.\/)?(\.{2}\/){1,5})(.*\/)*(.[^\s/]*\.[A-Za-z]{2,4})\S*)\)/gm,
78
- externalImages: /!\[.*]\((https?:\/(\/[^)/]+)+\/?)\)/gm,
79
- markdownLinks: /(\s)+\[.*]\((https?:\/(\/[^)/]+)+\/?)\)/gm,
80
- url: /(https?:\/\/[\w./-]+)/gm,
81
- uploadcare: /https:\/\/ucarecdn.com\/(?:.*\/)*([\w./-]+)/gm,
82
- };
83
-
84
- const validTypes = Object.keys(regex);
85
- if (!Array.isArray(types))
86
- types = [types];
87
-
88
- const findings: IFindings = {};
89
- type findingsType =
90
- | "relativeImages"
91
- | "externalImages"
92
- | "markdownLinks"
93
- | "url"
94
- | "uploadcare";
95
-
96
- for (const type of types) {
97
- if (!validTypes.includes(type))
98
- throw new Error("Invalid type: " + type);
99
- else
100
- findings[type as findingsType] = {};
101
- }
102
-
103
- for (const type of types) {
104
- let m: RegExpExecArray;
105
- while ((m = regex[type].exec(content)) !== null) {
106
- // This is necessary to avoid infinite loops with zero-width matches
107
- if (m.index === regex.lastIndex) {
108
- regex.lastIndex++;
109
- }
110
-
111
- // The result can be accessed through the `m`-variable.
112
- // m.forEach((match, groupIndex) => values.push(match));
113
-
114
- findings[type as findingsType]![m[0]] = {
115
- content: m[0],
116
- absUrl: m[1],
117
- mdUrl: m[2],
118
- relUrl: m[6],
119
- };
120
- }
121
- }
122
-
123
- return findings;
124
- };
125
-
126
- // This function checks that each of the url's are working.
127
- const checkUrl = async (
128
- config: IConfigObj,
129
- filePath: string,
130
- fileName: string,
131
- exercise: IExercise | undefined,
132
- errors: IAuditErrors[],
133
- warnings: IAuditErrors[],
134
- counter: ICounter | undefined
135
- ) => {
136
- if (!fs.existsSync(filePath))
137
- return false;
138
- const content: string = fs.readFileSync(filePath).toString();
139
- const isEmpty = checkForEmptySpaces(content);
140
- if (isEmpty || !content)
141
- errors.push({
142
- exercise: exercise?.title!,
143
- msg: `This file (${fileName}) doesn't have any content inside.`,
144
- });
145
-
146
- const frontmatter: IFrontmatter = fm(content);
147
- for (const attribute in frontmatter.attributes) {
148
- if (
149
- Object.prototype.hasOwnProperty.call(frontmatter.attributes, attribute) &&
150
- (attribute === "intro" || attribute === "tutorial")
151
- ) {
152
- counter && counter.links.total++;
153
- try {
154
- // eslint-disable-next-line
155
- let res = await fetch(frontmatter.attributes[attribute], {
156
- method: "HEAD",
157
- });
158
- if (!res.ok) {
159
- counter && counter.links.error++;
160
- errors.push({
161
- exercise: exercise?.title!,
162
- msg: `This link is broken (${res.ok}): ${frontmatter.attributes[attribute]}`,
163
- });
164
- }
165
- } catch {
166
- counter && counter.links.error++;
167
- errors.push({
168
- exercise: exercise?.title,
169
- msg: `This link is broken: ${frontmatter.attributes[attribute]}`,
170
- });
171
- }
172
- }
173
- }
174
-
175
- // Check url's of each README file.
176
- const findings: IFindings = findInFile(
177
- ["relativeImages", "externalImages", "markdownLinks"],
178
- content
179
- );
180
- type findingsType =
181
- | "relativeImages"
182
- | "externalImages"
183
- | "markdownLinks"
184
- | "url"
185
- | "uploadcare";
186
- for (const finding in findings) {
187
- if (Object.prototype.hasOwnProperty.call(findings, finding)) {
188
- const obj = findings[finding as findingsType];
189
- // Valdites all the relative path images.
190
- if (finding === "relativeImages" && Object.keys(obj!).length > 0) {
191
- for (const img in obj) {
192
- if (Object.prototype.hasOwnProperty.call(obj, img)) {
193
- // Validates if the image is in the assets folder.
194
- counter && counter.images.total++;
195
- const relativePath = path
196
- .relative(
197
- exercise ? exercise.path.replace(/\\/gm, "/") : "./",
198
- `${config!.config?.dirPath}/assets/${obj[img].relUrl}`
199
- )
200
- .replace(/\\/gm, "/");
201
- if (relativePath !== obj[img].absUrl.split("?").shift()) {
202
- counter && counter.images.error++;
203
- errors.push({
204
- exercise: exercise?.title,
205
- msg: `This relative path (${obj[img].relUrl}) is not pointing to the assets folder.`,
206
- });
207
- }
208
-
209
- if (
210
- !fs.existsSync(
211
- `${config!.config?.dirPath}/assets/${obj[img].relUrl}`
212
- )
213
- ) {
214
- counter && counter.images.error++;
215
- errors.push({
216
- exercise: exercise?.title,
217
- msg: `The file ${obj[img].relUrl} doesn't exist in the assets folder.`,
218
- });
219
- }
220
- }
221
- }
222
- } else if (finding === "externalImages" && Object.keys(obj!).length > 0) {
223
- // Valdites all the aboslute path images.
224
- for (const img in obj) {
225
- if (Object.prototype.hasOwnProperty.call(obj, img)) {
226
- counter && counter.images.total++;
227
- if (
228
- fs.existsSync(
229
- `${config!.config?.dirPath}/assets${obj[img].mdUrl
230
- .split("?")
231
- .shift()}`
232
- )
233
- ) {
234
- const relativePath = path
235
- .relative(
236
- exercise ? exercise.path.replace(/\\/gm, "/") : "./",
237
- `${config!.config?.dirPath}/assets/${obj[img].mdUrl}`
238
- )
239
- .replace(/\\/gm, "/");
240
- warnings.push({
241
- exercise: exercise?.title,
242
- msg: `On this exercise you have an image with an absolute path "${obj[img].absUrl}". We recommend you to replace it by the relative path: "${relativePath}".`,
243
- });
244
- }
245
-
246
- try {
247
- // eslint-disable-next-line
248
- let res = await fetch(obj[img].absUrl, {
249
- method: "HEAD",
250
- });
251
- if (!res.ok) {
252
- counter && counter.images.error++;
253
- errors.push({
254
- exercise: exercise?.title,
255
- msg: `This link is broken: ${obj[img].absUrl}`,
256
- });
257
- }
258
- } catch {
259
- counter && counter.images.error++;
260
- errors.push({
261
- exercise: exercise?.title,
262
- msg: `This link is broken: ${obj[img].absUrl}`,
263
- });
264
- }
265
- }
266
- }
267
- } else if (finding === "markdownLinks" && Object.keys(obj!).length > 0) {
268
- for (const link in obj) {
269
- if (Object.prototype.hasOwnProperty.call(obj, link)) {
270
- counter && counter.links.total++;
271
- if (!obj[link].mdUrl.includes("twitter")) {
272
- try {
273
- // eslint-disable-next-line
274
- let res = await fetch(obj[link].mdUrl, {
275
- method: "HEAD",
276
- });
277
- if (res.status > 399 && res.status < 500) {
278
- counter && counter.links.error++;
279
- errors.push({
280
- exercise: exercise?.title,
281
- msg: `This link is broken: ${obj[link].mdUrl}`,
282
- });
283
- }
284
- } catch {
285
- counter && counter.links.error++;
286
- errors.push({
287
- exercise: exercise?.title,
288
- msg: `This link is broken: ${obj[link].mdUrl}`,
289
- });
290
- }
291
- }
292
- }
293
- }
294
- }
295
- }
296
- }
297
-
298
- return true;
299
- };
300
-
301
- // This function writes a given file with the given content.
302
- const writeFile = async (content: string, filePath: string) => {
303
- try {
304
- await fs.promises.writeFile(filePath, content);
305
- } catch (error) {
306
- if (error)
307
- Console.error(
308
- `We weren't able to write the file in this path "${filePath}".`,
309
- error
310
- );
311
- }
312
- };
313
-
314
- // This function checks if there are errors, and show them in the console at the end.
315
- const showErrors = (
316
- errors: IAuditErrors[],
317
- counter: ICounter | undefined
318
- ) => {
319
- return new Promise((resolve, reject) => {
320
- if (errors) {
321
- if (errors.length > 0) {
322
- Console.log("Checking for errors...");
323
- for (const [i, error] of errors.entries())
324
- Console.error(
325
- `${i + 1}) ${error.msg} ${
326
- error.exercise ? `(Exercise: ${error.exercise})` : ""
327
- }`
328
- );
329
- if (counter) {
330
- Console.error(
331
- ` We found ${errors.length} error${
332
- errors.length > 1 ? "s" : ""
333
- } among ${counter.images.total} images, ${
334
- counter.links.total
335
- } link, ${counter.readmeFiles} README files and ${
336
- counter.exercises
337
- } exercises.`
338
- );
339
- } else {
340
- Console.error(
341
- ` We found ${errors.length} error${
342
- errors.length > 1 ? "s" : ""
343
- } related with the project integrity.`
344
- );
345
- }
346
-
347
- process.exit(1);
348
- } else {
349
- if (counter) {
350
- Console.success(
351
- `We didn't find any errors in this repository among ${counter.images.total} images, ${counter.links.total} link, ${counter.readmeFiles} README files and ${counter.exercises} exercises.`
352
- );
353
- } else {
354
- Console.success(`We didn't find any errors in this repository.`);
355
- }
356
-
357
- process.exit(0);
358
- }
359
- } else {
360
- reject("Failed");
361
- }
362
- });
363
- };
364
-
365
- // This function checks if there are warnings, and show them in the console at the end.
366
- const showWarnings = (warnings: IAuditErrors[]) => {
367
- return new Promise((resolve, reject) => {
368
- if (warnings) {
369
- if (warnings.length > 0) {
370
- Console.log("Checking for warnings...");
371
- for (const [i, warning] of warnings.entries())
372
- Console.warning(
373
- `${i + 1}) ${warning.msg} ${
374
- warning.exercise ? `File: ${warning.exercise}` : ""
375
- }`
376
- );
377
- }
378
-
379
- resolve("SUCCESS");
380
- } else {
381
- reject("Failed");
382
- }
383
- });
384
- };
385
-
386
- export default {
387
- isUrl,
388
- checkForEmptySpaces,
389
- checkLearnpackClean,
390
- findInFile,
391
- checkUrl,
392
- writeFile,
393
- showErrors,
394
- showWarnings,
395
- };
1
+ import { IAuditErrors } from "../models/audit"
2
+ import { IConfigObj } from "../models/config"
3
+ import { ICounter } from "../models/counter"
4
+ import { IFindings } from "../models/findings"
5
+ import { IExercise } from "../models/exercise-obj"
6
+ import { IFrontmatter } from "../models/front-matter"
7
+ import Console from "./console"
8
+ import * as fs from "fs"
9
+ import * as path from "path"
10
+
11
+ // eslint-disable-next-line
12
+ const fetch = require("node-fetch");
13
+ // eslint-disable-next-line
14
+ const fm = require("front-matter");
15
+
16
+ // This function checks if a url is valid.
17
+ const isUrl = async (
18
+ url: string,
19
+ errors: IAuditErrors[],
20
+ counter: ICounter
21
+ ) => {
22
+ const regexUrl = /(https?:\/\/[\w./-]+)/gm
23
+ counter.links.total++
24
+ if (!regexUrl.test(url)) {
25
+ counter.links.error++
26
+ errors.push({
27
+ exercise: undefined,
28
+ msg: `The repository value of the configuration file is not a link: ${url}`,
29
+ })
30
+ return false
31
+ }
32
+
33
+ const res = await fetch(url, { method: "HEAD" })
34
+ if (!res.ok) {
35
+ counter.links.error++
36
+ errors.push({
37
+ exercise: undefined,
38
+ msg: `The link of the repository is broken: ${url}`,
39
+ })
40
+ }
41
+
42
+ return true
43
+ }
44
+
45
+ const checkForEmptySpaces = (str: string) => {
46
+ const isEmpty = true
47
+ for (const letter of str) {
48
+ if (letter !== " ") {
49
+ return false
50
+ }
51
+ }
52
+
53
+ return isEmpty
54
+ }
55
+
56
+ const checkLearnpackClean = (configObj: IConfigObj, errors: IAuditErrors[]) => {
57
+ if (
58
+ (configObj.config?.outputPath &&
59
+ fs.existsSync(configObj.config?.outputPath)) ||
60
+ fs.existsSync(`${configObj.config?.dirPath}/_app`) ||
61
+ fs.existsSync(`${configObj.config?.dirPath}/reports`) ||
62
+ fs.existsSync(`${configObj.config?.dirPath}/resets`) ||
63
+ fs.existsSync(`${configObj.config?.dirPath}/app.tar.gz`) ||
64
+ fs.existsSync(`${configObj.config?.dirPath}/config.json`) ||
65
+ fs.existsSync(`${configObj.config?.dirPath}/vscode_queue.json`)
66
+ ) {
67
+ errors.push({
68
+ exercise: undefined,
69
+ msg: "You have to run learnpack clean command",
70
+ })
71
+ }
72
+ }
73
+
74
+ const findInFile = (types: string[], content: string) => {
75
+ const regex: any = {
76
+ relativeImages:
77
+ /!\[.*]\s*\((((\.\/)?(\.{2}\/){1,5})(.*\/)*(.[^\s/]*\.[A-Za-z]{2,4})\S*)\)/gm,
78
+ externalImages: /!\[.*]\((https?:\/(\/[^)/]+)+\/?)\)/gm,
79
+ markdownLinks: /(\s)+\[.*]\((https?:\/(\/[^)/]+)+\/?)\)/gm,
80
+ url: /(https?:\/\/[\w./-]+)/gm,
81
+ uploadcare: /https:\/\/ucarecdn.com\/(?:.*\/)*([\w./-]+)/gm,
82
+ }
83
+
84
+ const validTypes = Object.keys(regex)
85
+ if (!Array.isArray(types))
86
+ types = [types]
87
+
88
+ const findings: IFindings = {}
89
+ type findingsType =
90
+ | "relativeImages"
91
+ | "externalImages"
92
+ | "markdownLinks"
93
+ | "url"
94
+ | "uploadcare";
95
+
96
+ for (const type of types) {
97
+ if (!validTypes.includes(type))
98
+ throw new Error("Invalid type: " + type)
99
+ else
100
+ findings[type as findingsType] = {}
101
+ }
102
+
103
+ for (const type of types) {
104
+ let m: RegExpExecArray
105
+ while ((m = regex[type].exec(content)) !== null) {
106
+ // This is necessary to avoid infinite loops with zero-width matches
107
+ if (m.index === regex.lastIndex) {
108
+ regex.lastIndex++
109
+ }
110
+
111
+ // The result can be accessed through the `m`-variable.
112
+ // m.forEach((match, groupIndex) => values.push(match));
113
+
114
+ findings[type as findingsType]![m[0]] = {
115
+ content: m[0],
116
+ absUrl: m[1],
117
+ mdUrl: m[2],
118
+ relUrl: m[6],
119
+ }
120
+ }
121
+ }
122
+
123
+ return findings
124
+ }
125
+
126
+ // This function checks that each of the url's are working.
127
+ const checkUrl = async (
128
+ config: IConfigObj,
129
+ filePath: string,
130
+ fileName: string,
131
+ exercise: IExercise | undefined,
132
+ errors: IAuditErrors[],
133
+ warnings: IAuditErrors[],
134
+ counter: ICounter | undefined
135
+ ) => {
136
+ if (!fs.existsSync(filePath))
137
+ return false
138
+ const content: string = fs.readFileSync(filePath).toString()
139
+ const isEmpty = checkForEmptySpaces(content)
140
+ if (isEmpty || !content)
141
+ errors.push({
142
+ exercise: exercise?.title!,
143
+ msg: `This file (${fileName}) doesn't have any content inside.`,
144
+ })
145
+
146
+ const frontmatter: IFrontmatter = fm(content)
147
+ for (const attribute in frontmatter.attributes) {
148
+ if (
149
+ Object.prototype.hasOwnProperty.call(frontmatter.attributes, attribute) &&
150
+ (attribute === "intro" || attribute === "tutorial")
151
+ ) {
152
+ counter && counter.links.total++
153
+ try {
154
+ // eslint-disable-next-line
155
+ let res = await fetch(frontmatter.attributes[attribute], {
156
+ method: "HEAD",
157
+ })
158
+ if (!res.ok) {
159
+ counter && counter.links.error++
160
+ errors.push({
161
+ exercise: exercise?.title!,
162
+ msg: `This link is broken (${res.ok}): ${frontmatter.attributes[attribute]}`,
163
+ })
164
+ }
165
+ } catch {
166
+ counter && counter.links.error++
167
+ errors.push({
168
+ exercise: exercise?.title,
169
+ msg: `This link is broken: ${frontmatter.attributes[attribute]}`,
170
+ })
171
+ }
172
+ }
173
+ }
174
+
175
+ // Check url's of each README file.
176
+ const findings: IFindings = findInFile(
177
+ ["relativeImages", "externalImages", "markdownLinks"],
178
+ content
179
+ )
180
+ type findingsType =
181
+ | "relativeImages"
182
+ | "externalImages"
183
+ | "markdownLinks"
184
+ | "url"
185
+ | "uploadcare";
186
+ for (const finding in findings) {
187
+ if (Object.prototype.hasOwnProperty.call(findings, finding)) {
188
+ const obj = findings[finding as findingsType]
189
+ // Valdites all the relative path images.
190
+ if (finding === "relativeImages" && Object.keys(obj!).length > 0) {
191
+ for (const img in obj) {
192
+ if (Object.prototype.hasOwnProperty.call(obj, img)) {
193
+ // Validates if the image is in the assets folder.
194
+ counter && counter.images.total++
195
+ const relativePath = path
196
+ .relative(
197
+ exercise ? exercise.path.replace(/\\/gm, "/") : "./",
198
+ `${config!.config?.dirPath}/assets/${obj[img].relUrl}`
199
+ )
200
+ .replace(/\\/gm, "/")
201
+ if (relativePath !== obj[img].absUrl.split("?").shift()) {
202
+ counter && counter.images.error++
203
+ errors.push({
204
+ exercise: exercise?.title,
205
+ msg: `This relative path (${obj[img].relUrl}) is not pointing to the assets folder.`,
206
+ })
207
+ }
208
+
209
+ if (
210
+ !fs.existsSync(
211
+ `${config!.config?.dirPath}/assets/${obj[img].relUrl}`
212
+ )
213
+ ) {
214
+ counter && counter.images.error++
215
+ errors.push({
216
+ exercise: exercise?.title,
217
+ msg: `The file ${obj[img].relUrl} doesn't exist in the assets folder.`,
218
+ })
219
+ }
220
+ }
221
+ }
222
+ } else if (finding === "externalImages" && Object.keys(obj!).length > 0) {
223
+ // Valdites all the aboslute path images.
224
+ for (const img in obj) {
225
+ if (Object.prototype.hasOwnProperty.call(obj, img)) {
226
+ counter && counter.images.total++
227
+ if (
228
+ fs.existsSync(
229
+ `${config!.config?.dirPath}/assets${obj[img].mdUrl
230
+ .split("?")
231
+ .shift()}`
232
+ )
233
+ ) {
234
+ const relativePath = path
235
+ .relative(
236
+ exercise ? exercise.path.replace(/\\/gm, "/") : "./",
237
+ `${config!.config?.dirPath}/assets/${obj[img].mdUrl}`
238
+ )
239
+ .replace(/\\/gm, "/")
240
+ warnings.push({
241
+ exercise: exercise?.title,
242
+ msg: `On this exercise you have an image with an absolute path "${obj[img].absUrl}". We recommend you to replace it by the relative path: "${relativePath}".`,
243
+ })
244
+ }
245
+
246
+ try {
247
+ // eslint-disable-next-line
248
+ let res = await fetch(obj[img].absUrl, {
249
+ method: "HEAD",
250
+ })
251
+ if (!res.ok) {
252
+ counter && counter.images.error++
253
+ errors.push({
254
+ exercise: exercise?.title,
255
+ msg: `This link is broken: ${obj[img].absUrl}`,
256
+ })
257
+ }
258
+ } catch {
259
+ counter && counter.images.error++
260
+ errors.push({
261
+ exercise: exercise?.title,
262
+ msg: `This link is broken: ${obj[img].absUrl}`,
263
+ })
264
+ }
265
+ }
266
+ }
267
+ } else if (finding === "markdownLinks" && Object.keys(obj!).length > 0) {
268
+ for (const link in obj) {
269
+ if (Object.prototype.hasOwnProperty.call(obj, link)) {
270
+ counter && counter.links.total++
271
+ if (!obj[link].mdUrl.includes("twitter")) {
272
+ try {
273
+ // eslint-disable-next-line
274
+ let res = await fetch(obj[link].mdUrl, {
275
+ method: "HEAD",
276
+ })
277
+ if (res.status > 399 && res.status < 500) {
278
+ counter && counter.links.error++
279
+ errors.push({
280
+ exercise: exercise?.title,
281
+ msg: `This link is broken: ${obj[link].mdUrl}`,
282
+ })
283
+ }
284
+ } catch {
285
+ counter && counter.links.error++
286
+ errors.push({
287
+ exercise: exercise?.title,
288
+ msg: `This link is broken: ${obj[link].mdUrl}`,
289
+ })
290
+ }
291
+ }
292
+ }
293
+ }
294
+ }
295
+ }
296
+ }
297
+
298
+ return true
299
+ }
300
+
301
+ // This function writes a given file with the given content.
302
+ const writeFile = async (content: string, filePath: string) => {
303
+ try {
304
+ await fs.promises.writeFile(filePath, content)
305
+ } catch (error) {
306
+ if (error)
307
+ Console.error(
308
+ `We weren't able to write the file in this path "${filePath}".`,
309
+ error
310
+ )
311
+ }
312
+ }
313
+
314
+ // This function checks if there are errors, and show them in the console at the end.
315
+ const showErrors = (errors: IAuditErrors[], counter: ICounter | undefined) => {
316
+ return new Promise((resolve, reject) => {
317
+ if (errors) {
318
+ if (errors.length > 0) {
319
+ Console.log("Checking for errors...")
320
+ for (const [i, error] of errors.entries())
321
+ Console.error(
322
+ `${i + 1}) ${error.msg} ${
323
+ error.exercise ? `(Exercise: ${error.exercise})` : ""
324
+ }`
325
+ )
326
+ if (counter) {
327
+ Console.error(
328
+ ` We found ${errors.length} error${
329
+ errors.length > 1 ? "s" : ""
330
+ } among ${counter.images.total} images, ${
331
+ counter.links.total
332
+ } link, ${counter.readmeFiles} README files and ${
333
+ counter.exercises
334
+ } exercises.`
335
+ )
336
+ } else {
337
+ Console.error(
338
+ ` We found ${errors.length} error${
339
+ errors.length > 1 ? "s" : ""
340
+ } related with the project integrity.`
341
+ )
342
+ }
343
+
344
+ process.exit(1)
345
+ } else {
346
+ if (counter) {
347
+ Console.success(
348
+ `We didn't find any errors in this repository among ${counter.images.total} images, ${counter.links.total} link, ${counter.readmeFiles} README files and ${counter.exercises} exercises.`
349
+ )
350
+ } else {
351
+ Console.success(`We didn't find any errors in this repository.`)
352
+ }
353
+
354
+ process.exit(0)
355
+ }
356
+ } else {
357
+ reject("Failed")
358
+ }
359
+ })
360
+ }
361
+
362
+ // This function checks if there are warnings, and show them in the console at the end.
363
+ const showWarnings = (warnings: IAuditErrors[]) => {
364
+ return new Promise((resolve, reject) => {
365
+ if (warnings) {
366
+ if (warnings.length > 0) {
367
+ Console.log("Checking for warnings...")
368
+ for (const [i, warning] of warnings.entries())
369
+ Console.warning(
370
+ `${i + 1}) ${warning.msg} ${
371
+ warning.exercise ? `File: ${warning.exercise}` : ""
372
+ }`
373
+ )
374
+ }
375
+
376
+ resolve("SUCCESS")
377
+ } else {
378
+ reject("Failed")
379
+ }
380
+ })
381
+ }
382
+
383
+ export default {
384
+ isUrl,
385
+ checkForEmptySpaces,
386
+ checkLearnpackClean,
387
+ findInFile,
388
+ checkUrl,
389
+ writeFile,
390
+ showErrors,
391
+ showWarnings,
392
+ }