@learnpack/learnpack 2.1.25 → 2.1.27

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
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
+ }