@learnpack/learnpack 2.1.24 → 2.1.26

Sign up to get free protection for your applications and to get access to all the features.
Files changed (182) hide show
  1. package/README.md +16 -16
  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 -317
  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 +165 -165
  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 +39 -37
  27. package/lib/managers/config/defaults.js +40 -38
  28. package/lib/managers/config/exercise.d.ts +36 -36
  29. package/lib/managers/config/exercise.js +233 -230
  30. package/lib/managers/config/index.d.ts +3 -3
  31. package/lib/managers/config/index.js +320 -302
  32. package/lib/managers/file.d.ts +13 -13
  33. package/lib/managers/file.js +134 -134
  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 -51
  38. package/lib/managers/server/routes.d.ts +4 -4
  39. package/lib/managers/server/routes.js +167 -167
  40. package/lib/managers/session.d.ts +3 -3
  41. package/lib/managers/session.js +104 -104
  42. package/lib/managers/socket.d.ts +3 -3
  43. package/lib/managers/socket.js +164 -164
  44. package/lib/managers/test.js +84 -84
  45. package/lib/models/action.d.ts +2 -2
  46. package/lib/models/action.js +2 -2
  47. package/lib/models/audit.d.ts +15 -15
  48. package/lib/models/audit.js +2 -2
  49. package/lib/models/config-manager.d.ts +21 -21
  50. package/lib/models/config-manager.js +2 -2
  51. package/lib/models/config.d.ts +62 -60
  52. package/lib/models/config.js +2 -2
  53. package/lib/models/counter.d.ts +11 -11
  54. package/lib/models/counter.js +2 -2
  55. package/lib/models/errors.d.ts +15 -15
  56. package/lib/models/errors.js +2 -2
  57. package/lib/models/exercise-obj.d.ts +27 -27
  58. package/lib/models/exercise-obj.js +2 -2
  59. package/lib/models/file.d.ts +5 -5
  60. package/lib/models/file.js +2 -2
  61. package/lib/models/findings.d.ts +17 -17
  62. package/lib/models/findings.js +2 -2
  63. package/lib/models/flags.d.ts +10 -10
  64. package/lib/models/flags.js +2 -2
  65. package/lib/models/front-matter.d.ts +11 -11
  66. package/lib/models/front-matter.js +2 -2
  67. package/lib/models/gitpod-data.d.ts +16 -16
  68. package/lib/models/gitpod-data.js +2 -2
  69. package/lib/models/language.d.ts +4 -4
  70. package/lib/models/language.js +2 -2
  71. package/lib/models/package.d.ts +7 -7
  72. package/lib/models/package.js +2 -2
  73. package/lib/models/plugin-config.d.ts +16 -16
  74. package/lib/models/plugin-config.js +2 -2
  75. package/lib/models/session.d.ts +23 -23
  76. package/lib/models/session.js +2 -2
  77. package/lib/models/socket.d.ts +31 -31
  78. package/lib/models/socket.js +2 -2
  79. package/lib/models/status.d.ts +1 -1
  80. package/lib/models/status.js +2 -2
  81. package/lib/models/success-types.d.ts +1 -1
  82. package/lib/models/success-types.js +2 -2
  83. package/lib/plugin/command/compile.d.ts +6 -6
  84. package/lib/plugin/command/compile.js +18 -18
  85. package/lib/plugin/command/test.d.ts +6 -6
  86. package/lib/plugin/command/test.js +25 -25
  87. package/lib/plugin/index.d.ts +27 -27
  88. package/lib/plugin/index.js +7 -7
  89. package/lib/plugin/plugin.d.ts +8 -8
  90. package/lib/plugin/plugin.js +68 -68
  91. package/lib/plugin/utils.d.ts +16 -16
  92. package/lib/plugin/utils.js +58 -58
  93. package/lib/ui/download.d.ts +5 -5
  94. package/lib/ui/download.js +61 -61
  95. package/lib/utils/BaseCommand.d.ts +8 -8
  96. package/lib/utils/BaseCommand.js +41 -41
  97. package/lib/utils/SessionCommand.d.ts +10 -10
  98. package/lib/utils/SessionCommand.js +47 -47
  99. package/lib/utils/api.d.ts +12 -12
  100. package/lib/utils/api.js +173 -173
  101. package/lib/utils/audit.d.ts +16 -16
  102. package/lib/utils/audit.js +302 -302
  103. package/lib/utils/console.d.ts +12 -12
  104. package/lib/utils/console.js +19 -19
  105. package/lib/utils/errors.d.ts +17 -17
  106. package/lib/utils/errors.js +100 -100
  107. package/lib/utils/exercisesQueue.d.ts +9 -9
  108. package/lib/utils/exercisesQueue.js +38 -38
  109. package/lib/utils/fileQueue.d.ts +40 -40
  110. package/lib/utils/fileQueue.js +168 -168
  111. package/lib/utils/misc.d.ts +1 -1
  112. package/lib/utils/misc.js +23 -23
  113. package/lib/utils/validators.d.ts +5 -5
  114. package/lib/utils/validators.js +17 -17
  115. package/lib/utils/watcher.d.ts +2 -2
  116. package/lib/utils/watcher.js +23 -23
  117. package/oclif.manifest.json +1 -1
  118. package/package.json +138 -138
  119. package/src/commands/audit.ts +443 -418
  120. package/src/commands/clean.ts +29 -29
  121. package/src/commands/download.ts +62 -62
  122. package/src/commands/login.ts +42 -42
  123. package/src/commands/logout.ts +43 -43
  124. package/src/commands/publish.ts +107 -107
  125. package/src/commands/start.ts +238 -234
  126. package/src/commands/test.ts +85 -85
  127. package/src/index.ts +1 -1
  128. package/src/managers/config/allowed_files.ts +29 -29
  129. package/src/managers/config/defaults.ts +2 -0
  130. package/src/managers/config/exercise.ts +309 -302
  131. package/src/managers/config/index.ts +22 -1
  132. package/src/managers/file.ts +169 -169
  133. package/src/managers/gitpod.ts +84 -84
  134. package/src/managers/server/index.ts +77 -69
  135. package/src/managers/session.ts +118 -118
  136. package/src/managers/socket.ts +239 -239
  137. package/src/managers/test.ts +83 -83
  138. package/src/models/action.ts +3 -3
  139. package/src/models/config-manager.ts +23 -23
  140. package/src/models/config.ts +2 -0
  141. package/src/models/counter.ts +11 -11
  142. package/src/models/errors.ts +22 -22
  143. package/src/models/file.ts +5 -5
  144. package/src/models/findings.ts +18 -18
  145. package/src/models/flags.ts +10 -10
  146. package/src/models/front-matter.ts +11 -11
  147. package/src/models/gitpod-data.ts +19 -19
  148. package/src/models/language.ts +4 -4
  149. package/src/models/package.ts +7 -7
  150. package/src/models/plugin-config.ts +17 -17
  151. package/src/models/session.ts +26 -26
  152. package/src/models/socket.ts +48 -48
  153. package/src/models/status.ts +15 -15
  154. package/src/models/success-types.ts +1 -1
  155. package/src/plugin/command/compile.ts +17 -17
  156. package/src/plugin/command/test.ts +30 -30
  157. package/src/plugin/index.ts +6 -6
  158. package/src/plugin/plugin.ts +94 -94
  159. package/src/plugin/utils.ts +87 -87
  160. package/src/types/node-fetch.d.ts +1 -1
  161. package/src/ui/download.ts +71 -71
  162. package/src/utils/BaseCommand.ts +48 -48
  163. package/src/utils/SessionCommand.ts +48 -48
  164. package/src/utils/api.ts +194 -194
  165. package/src/utils/audit.ts +395 -395
  166. package/src/utils/console.ts +24 -24
  167. package/src/utils/errors.ts +117 -117
  168. package/src/utils/exercisesQueue.ts +51 -51
  169. package/src/utils/fileQueue.ts +198 -198
  170. package/src/utils/misc.ts +23 -23
  171. package/src/utils/templates/gitignore.txt +19 -19
  172. package/src/utils/templates/incremental/.learn/exercises/01-hello-world/README.es.md +24 -24
  173. package/src/utils/templates/incremental/.learn/exercises/01-hello-world/README.md +24 -24
  174. package/src/utils/templates/incremental/README.ejs +4 -4
  175. package/src/utils/templates/incremental/README.es.ejs +4 -4
  176. package/src/utils/templates/isolated/01-hello-world/README.es.md +26 -26
  177. package/src/utils/templates/isolated/01-hello-world/README.md +26 -26
  178. package/src/utils/templates/isolated/README.ejs +4 -4
  179. package/src/utils/templates/isolated/README.es.ejs +4 -4
  180. package/src/utils/templates/no-grading/README.ejs +4 -4
  181. package/src/utils/templates/no-grading/README.es.ejs +4 -4
  182. package/src/utils/validators.ts +18 -18
@@ -1,395 +1,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 = (
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 = (
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
+ };