@learnpack/learnpack 4.0.10 → 4.0.13

Sign up to get free protection for your applications and to get access to all the features.
Files changed (126) hide show
  1. package/README.md +16 -20
  2. package/lib/commands/audit.d.ts +6 -6
  3. package/lib/commands/audit.js +327 -327
  4. package/lib/commands/clean.d.ts +8 -8
  5. package/lib/commands/clean.js +22 -22
  6. package/lib/commands/download.d.ts +13 -13
  7. package/lib/commands/download.js +52 -52
  8. package/lib/commands/init.d.ts +9 -9
  9. package/lib/commands/init.js +127 -127
  10. package/lib/commands/login.d.ts +14 -14
  11. package/lib/commands/login.js +34 -34
  12. package/lib/commands/logout.d.ts +14 -14
  13. package/lib/commands/logout.js +34 -34
  14. package/lib/commands/publish.d.ts +11 -14
  15. package/lib/commands/publish.js +160 -82
  16. package/lib/commands/start.d.ts +7 -7
  17. package/lib/commands/start.js +252 -250
  18. package/lib/commands/test.d.ts +6 -6
  19. package/lib/commands/test.js +62 -62
  20. package/lib/index.d.ts +1 -1
  21. package/lib/index.js +4 -4
  22. package/lib/managers/config/allowed_files.d.ts +5 -5
  23. package/lib/managers/config/allowed_files.js +30 -30
  24. package/lib/managers/config/defaults.d.ts +47 -48
  25. package/lib/managers/config/defaults.js +51 -51
  26. package/lib/managers/config/exercise.d.ts +36 -36
  27. package/lib/managers/config/exercise.js +243 -236
  28. package/lib/managers/config/index.d.ts +3 -3
  29. package/lib/managers/config/index.js +464 -459
  30. package/lib/managers/file.d.ts +14 -14
  31. package/lib/managers/file.js +190 -184
  32. package/lib/managers/gitpod.d.ts +3 -3
  33. package/lib/managers/gitpod.js +67 -67
  34. package/lib/managers/server/index.d.ts +5 -6
  35. package/lib/managers/server/index.js +58 -58
  36. package/lib/managers/server/routes.d.ts +4 -4
  37. package/lib/managers/server/routes.js +228 -220
  38. package/lib/managers/session.d.ts +3 -3
  39. package/lib/managers/session.js +125 -125
  40. package/lib/managers/socket.d.ts +3 -3
  41. package/lib/managers/socket.js +188 -186
  42. package/lib/managers/telemetry.d.ts +74 -74
  43. package/lib/managers/telemetry.js +215 -214
  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 +86 -86
  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 +29 -30
  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 +31 -31
  76. package/lib/models/session.js +2 -2
  77. package/lib/models/socket.d.ts +37 -37
  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 +62 -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 +43 -43
  99. package/lib/utils/api.d.ts +14 -14
  100. package/lib/utils/api.js +255 -255
  101. package/lib/utils/audit.d.ts +16 -16
  102. package/lib/utils/audit.js +303 -303
  103. package/lib/utils/checkNotInstalled.d.ts +8 -8
  104. package/lib/utils/checkNotInstalled.js +185 -181
  105. package/lib/utils/console.d.ts +12 -12
  106. package/lib/utils/console.js +19 -19
  107. package/lib/utils/errors.d.ts +17 -17
  108. package/lib/utils/errors.js +107 -100
  109. package/lib/utils/exercisesQueue.d.ts +9 -9
  110. package/lib/utils/exercisesQueue.js +38 -38
  111. package/lib/utils/fileQueue.d.ts +43 -43
  112. package/lib/utils/fileQueue.js +169 -169
  113. package/lib/utils/misc.d.ts +1 -1
  114. package/lib/utils/misc.js +24 -23
  115. package/lib/utils/osOperations.d.ts +5 -5
  116. package/lib/utils/osOperations.js +72 -72
  117. package/lib/utils/validators.d.ts +5 -5
  118. package/lib/utils/validators.js +16 -17
  119. package/lib/utils/watcher.d.ts +2 -2
  120. package/lib/utils/watcher.js +25 -25
  121. package/oclif.manifest.json +1 -1
  122. package/package.json +6 -4
  123. package/src/commands/publish.ts +181 -107
  124. package/src/managers/config/index.ts +5 -0
  125. package/src/managers/server/routes.ts +10 -0
  126. package/src/managers/session.ts +145 -145
@@ -1,236 +1,243 @@
1
- "use strict";
2
- Object.defineProperty(exports, "__esModule", { value: true });
3
- exports.filterFiles = exports.detect = exports.isDirectory = exports.shouldBeVisible = exports.isCodable = exports.validateExerciseDirectoryName = exports.exercise = void 0;
4
- const p = require("path");
5
- // import frontMatter from 'front-matter'
6
- const fs = require("fs");
7
- const console_1 = require("../../utils/console");
8
- const allowed_files_1 = require("./allowed_files");
9
- // eslint-disable-next-line
10
- const frontMatter = require("front-matter");
11
- exports.exercise = (path, position, configObject) => {
12
- const { config, exercises } = configObject;
13
- let slug = p.basename(path);
14
- if (!exports.validateExerciseDirectoryName(slug)) {
15
- console_1.default.error(`Exercise directory ${slug} has an invalid name, it has to start with two or three digits followed by words separated by underscors or hyphen (no white spaces). e.g: 01.12-hello-world`);
16
- }
17
- // get all the files
18
- const files = fs.readdirSync(path);
19
- /**
20
- * build the translation array like:
21
- {
22
- "us": "path/to/Readme.md",
23
- "es": "path/to/Readme.es.md"
24
- }
25
- */
26
- const translations = {};
27
- for (const file of files.filter(file => file.toLowerCase().includes("readme"))) {
28
- const parts = file.split(".");
29
- if (parts.length === 3)
30
- translations[parts[1]] = file;
31
- else
32
- translations.us = file;
33
- }
34
- // if the slug is a dot, it means there is not "exercises" folder, and its just a single README.md
35
- if (slug === ".")
36
- slug = "default-index";
37
- const detected = exports.detect(configObject, files);
38
- const exerciseObj = {
39
- position,
40
- path,
41
- slug,
42
- translations,
43
- language: detected === null || detected === void 0 ? void 0 : detected.language,
44
- entry: (detected === null || detected === void 0 ? void 0 : detected.entry) ? path + "/" + detected.entry : null,
45
- title: slug || "Exercise",
46
- graded: files.some((file) => file.toLowerCase().startsWith("test.") ||
47
- file.toLowerCase().startsWith("tests.")),
48
- files: exports.filterFiles(files, path),
49
- // if the exercises was on the config before I may keep the status done
50
- done: Array.isArray(exercises) &&
51
- typeof exercises[position] !== "undefined" &&
52
- path.slice(Math.max(0, path.indexOf("exercises/") + 10)) ===
53
- exercises[position].slug ?
54
- exercises[position].done :
55
- false,
56
- getReadme: function (lang = null) {
57
- if (lang === "us")
58
- lang = null; // <-- english is default, no need to append it to the file name
59
- if (!fs.existsSync(`${this.path}/README${lang ? "." + lang : ""}.md`)) {
60
- console_1.default.error(`Language ${lang} not found for exercise ${slug}, switching to default language`);
61
- if (lang)
62
- lang = null;
63
- if (!fs.existsSync(`${this.path}/README${lang ? "." + lang : ""}.md`))
64
- throw new Error("Readme file not found for exercise: " + this.path + "/README.md");
65
- }
66
- const content = fs.readFileSync(`${this.path}/README${lang ? "." + lang : ""}.md`, "utf8");
67
- const attr = frontMatter(content);
68
- return attr;
69
- },
70
- getFile: function (name) {
71
- const file = this.files.find((f) => f.name === name);
72
- if (!file || !fs.existsSync(file.path)) {
73
- throw new Error(`File not found: + ${file === null || file === void 0 ? void 0 : file.path}`);
74
- }
75
- else if (fs.lstatSync(file.path).isDirectory()) {
76
- return ("Error: This is not a file to be read, but a directory: " + file.path);
77
- }
78
- // get file content
79
- const content = fs.readFileSync(file.path);
80
- // create reset folder
81
- if (!fs.existsSync(`${config === null || config === void 0 ? void 0 : config.dirPath}/resets`))
82
- fs.mkdirSync(`${config === null || config === void 0 ? void 0 : config.dirPath}/resets`);
83
- if (!fs.existsSync(`${config === null || config === void 0 ? void 0 : config.dirPath}/resets/` + this.slug)) {
84
- fs.mkdirSync(`${config === null || config === void 0 ? void 0 : config.dirPath}/resets/` + this.slug);
85
- }
86
- for (const _file of this.files) {
87
- const stats = fs.statSync(_file.path);
88
- if (stats.isDirectory() || _file.hidden)
89
- continue;
90
- const fileContent = fs.readFileSync(_file.path);
91
- if (!fs.existsSync(`${config === null || config === void 0 ? void 0 : config.dirPath}/resets/${this.slug}/${_file.name}`)) {
92
- fs.writeFileSync(`${config === null || config === void 0 ? void 0 : config.dirPath}/resets/${this.slug}/${_file.name}`, fileContent);
93
- }
94
- }
95
- return content;
96
- },
97
- saveFile: function (name, content) {
98
- const file = this.files.find((f) => f.name === name);
99
- if (file) {
100
- if (!fs.existsSync(file.path)) {
101
- throw new Error("File not found: " + file.path);
102
- }
103
- return fs.writeFileSync(file.path, content, "utf8");
104
- }
105
- },
106
- getTestReport: function () {
107
- var _a;
108
- const _path = `${(_a = configObject === null || configObject === void 0 ? void 0 : configObject.confPath) === null || _a === void 0 ? void 0 : _a.base}/reports/${this.slug}.json`;
109
- if (!fs.existsSync(_path))
110
- return {};
111
- const content = fs.readFileSync(_path);
112
- const data = JSON.parse(`${content}`);
113
- return data;
114
- },
115
- };
116
- return exerciseObj;
117
- };
118
- exports.validateExerciseDirectoryName = (str) => {
119
- if (str === "./")
120
- return true;
121
- // TODO: Add nameValidationREgex from the config
122
- const regex = /^(\d{2,3}(\.\d{1,2})?-([\dA-Za-z]+(-|_)?)+)$/;
123
- return regex.test(str);
124
- };
125
- exports.isCodable = (str) => {
126
- const extension = p.extname(str);
127
- return allowed_files_1.default.extensions.includes(extension.slice(1).toLowerCase());
128
- };
129
- const isNotConfiguration = (str) => {
130
- return !allowed_files_1.default.names.includes(str);
131
- };
132
- exports.shouldBeVisible = function (file) {
133
- return (
134
- // doest not have "test." on their name
135
- !file.name.toLocaleLowerCase().includes("test.") &&
136
- !file.name.toLocaleLowerCase().includes("tests.") &&
137
- !file.name.toLocaleLowerCase().includes(".hide.") &&
138
- // ignore hidden files
139
- file.name.charAt(0) !== "." &&
140
- // ignore learn.json and bc.json
141
- !file.name.toLocaleLowerCase().includes("learn.json") &&
142
- !file.name.toLocaleLowerCase().includes("bc.json") &&
143
- // ignore images, videos, vectors, etc.
144
- exports.isCodable(file.name) &&
145
- isNotConfiguration(file.name) &&
146
- // readme's and directories
147
- !file.name.toLowerCase().includes("readme.") &&
148
- !exports.isDirectory(file.path) &&
149
- file.name.charAt(0) !== "_");
150
- };
151
- exports.isDirectory = (source) => {
152
- // if(path.basename(source) === path.basename(config.dirPath)) return false
153
- return fs.lstatSync(source).isDirectory();
154
- };
155
- exports.detect = (configObject, files) => {
156
- if (!configObject) {
157
- return;
158
- }
159
- const { config } = configObject;
160
- if (!config)
161
- throw new Error("No configuration found during the engine detection");
162
- if (!config.entries)
163
- throw new Error("No configuration found for entries, please add a 'entries' object with the default file name for your exercise entry file that is going to be used while compiling, for example: index.html for html, app.py for python3, etc.");
164
- // A language was found on the config object, but this language will only be used as last resort, learnpack will try to guess each exercise language independently based on file extension (js, jsx, html, etc.)
165
- let hasFiles = files.filter(f => f.includes(".py"));
166
- if (hasFiles.length > 0)
167
- return {
168
- language: "python3",
169
- entry: hasFiles.find(f => config.entries.python3 === f),
170
- };
171
- hasFiles = files.filter(f => f.includes(".java"));
172
- if (hasFiles.length > 0)
173
- return {
174
- language: "java",
175
- entry: hasFiles.find(f => config.entries.java === f),
176
- };
177
- hasFiles = files.filter(f => f.includes(".jsx"));
178
- if (hasFiles.length > 0)
179
- return {
180
- language: "react",
181
- entry: hasFiles.find(f => config.entries.react === f),
182
- };
183
- const hasHTML = files.filter(f => f.includes("index.html"));
184
- const hasIndexJS = files.find(f => f.includes("index.js"));
185
- const hasJS = files.filter(f => f.includes(".js"));
186
- // angular, vue, vanillajs needs to have at least 2 files (html,css,js),
187
- // the test.js and the entry file in js
188
- // if not its just another HTML
189
- if (hasIndexJS && hasHTML.length > 0)
190
- return {
191
- language: "vanillajs",
192
- entry: hasIndexJS,
193
- };
194
- if (hasHTML.length > 0)
195
- return {
196
- language: "html",
197
- entry: hasHTML.find(f => config.entries.html === f),
198
- };
199
- if (hasJS.length > 0)
200
- return {
201
- language: "node",
202
- entry: hasJS.find(f => config.entries.node === f),
203
- };
204
- return {
205
- language: null,
206
- entry: null,
207
- };
208
- };
209
- exports.filterFiles = (files, basePath = ".") => files
210
- .map((ex) => ({
211
- path: basePath + "/" + ex,
212
- name: ex,
213
- hidden: !exports.shouldBeVisible({
214
- name: ex,
215
- path: basePath + "/" + ex,
216
- }),
217
- }))
218
- .sort((f1, f2) => {
219
- const score = {
220
- // sorting priority
221
- "index.html": 1,
222
- "styles.css": 2,
223
- "styles.scss": 2,
224
- "style.css": 2,
225
- "style.scss": 2,
226
- "index.css": 2,
227
- "index.scss": 2,
228
- "index.js": 3,
229
- };
230
- return score[f1.name] < score[f2.name] ? -1 : 1;
231
- });
232
- exports.default = {
233
- exercise: exports.exercise,
234
- detect: exports.detect,
235
- filterFiles: exports.filterFiles,
236
- };
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.filterFiles = exports.detect = exports.isDirectory = exports.shouldBeVisible = exports.isCodable = exports.validateExerciseDirectoryName = exports.exercise = void 0;
4
+ const p = require("path");
5
+ // import frontMatter from 'front-matter'
6
+ const fs = require("fs");
7
+ const console_1 = require("../../utils/console");
8
+ const allowed_files_1 = require("./allowed_files");
9
+ // eslint-disable-next-line
10
+ const frontMatter = require("front-matter");
11
+ const exercise = (path, position, configObject) => {
12
+ const { config, exercises } = configObject;
13
+ let slug = p.basename(path);
14
+ if (!(0, exports.validateExerciseDirectoryName)(slug)) {
15
+ console_1.default.error(`Exercise directory ${slug} has an invalid name, it has to start with two or three digits followed by words separated by underscors or hyphen (no white spaces). e.g: 01.12-hello-world`);
16
+ }
17
+ // get all the files
18
+ const files = fs.readdirSync(path);
19
+ /**
20
+ * build the translation array like:
21
+ {
22
+ "us": "path/to/Readme.md",
23
+ "es": "path/to/Readme.es.md"
24
+ }
25
+ */
26
+ const translations = {};
27
+ for (const file of files.filter(file => file.toLowerCase().includes("readme"))) {
28
+ const parts = file.split(".");
29
+ if (parts.length === 3)
30
+ translations[parts[1]] = file;
31
+ else
32
+ translations.us = file;
33
+ }
34
+ // if the slug is a dot, it means there is not "exercises" folder, and its just a single README.md
35
+ if (slug === ".")
36
+ slug = "default-index";
37
+ const detected = (0, exports.detect)(configObject, files);
38
+ const exerciseObj = {
39
+ position,
40
+ path,
41
+ slug,
42
+ translations,
43
+ language: detected === null || detected === void 0 ? void 0 : detected.language,
44
+ entry: (detected === null || detected === void 0 ? void 0 : detected.entry) ? path + "/" + detected.entry : null, // full path to the exercise entry
45
+ title: slug || "Exercise",
46
+ graded: files.some((file) => file.toLowerCase().startsWith("test.") ||
47
+ file.toLowerCase().startsWith("tests.")),
48
+ files: (0, exports.filterFiles)(files, path),
49
+ // if the exercises was on the config before I may keep the status done
50
+ done: Array.isArray(exercises) &&
51
+ typeof exercises[position] !== "undefined" &&
52
+ path.slice(Math.max(0, path.indexOf("exercises/") + 10)) ===
53
+ exercises[position].slug ?
54
+ exercises[position].done :
55
+ false,
56
+ getReadme: function (lang = null) {
57
+ if (lang === "us")
58
+ lang = null; // <-- english is default, no need to append it to the file name
59
+ if (!fs.existsSync(`${this.path}/README${lang ? "." + lang : ""}.md`)) {
60
+ console_1.default.error(`Language ${lang} not found for exercise ${slug}, switching to default language`);
61
+ if (lang)
62
+ lang = null;
63
+ if (!fs.existsSync(`${this.path}/README${lang ? "." + lang : ""}.md`))
64
+ throw new Error("Readme file not found for exercise: " + this.path + "/README.md");
65
+ }
66
+ const content = fs.readFileSync(`${this.path}/README${lang ? "." + lang : ""}.md`, "utf8");
67
+ const attr = frontMatter(content);
68
+ return attr;
69
+ },
70
+ getFile: function (name) {
71
+ const file = this.files.find((f) => f.name === name);
72
+ if (!file || !fs.existsSync(file.path)) {
73
+ throw new Error(`File not found: + ${file === null || file === void 0 ? void 0 : file.path}`);
74
+ }
75
+ else if (fs.lstatSync(file.path).isDirectory()) {
76
+ return ("Error: This is not a file to be read, but a directory: " + file.path);
77
+ }
78
+ // get file content
79
+ const content = fs.readFileSync(file.path);
80
+ // create reset folder
81
+ if (!fs.existsSync(`${config === null || config === void 0 ? void 0 : config.dirPath}/resets`))
82
+ fs.mkdirSync(`${config === null || config === void 0 ? void 0 : config.dirPath}/resets`);
83
+ if (!fs.existsSync(`${config === null || config === void 0 ? void 0 : config.dirPath}/resets/` + this.slug)) {
84
+ fs.mkdirSync(`${config === null || config === void 0 ? void 0 : config.dirPath}/resets/` + this.slug);
85
+ }
86
+ for (const _file of this.files) {
87
+ const stats = fs.statSync(_file.path);
88
+ if (stats.isDirectory() || _file.hidden)
89
+ continue;
90
+ const fileContent = fs.readFileSync(_file.path);
91
+ if (!fs.existsSync(`${config === null || config === void 0 ? void 0 : config.dirPath}/resets/${this.slug}/${_file.name}`)) {
92
+ fs.writeFileSync(`${config === null || config === void 0 ? void 0 : config.dirPath}/resets/${this.slug}/${_file.name}`, fileContent);
93
+ }
94
+ }
95
+ return content;
96
+ },
97
+ saveFile: function (name, content) {
98
+ const file = this.files.find((f) => f.name === name);
99
+ if (file) {
100
+ if (!fs.existsSync(file.path)) {
101
+ throw new Error("File not found: " + file.path);
102
+ }
103
+ return fs.writeFileSync(file.path, content, "utf8");
104
+ }
105
+ },
106
+ getTestReport: function () {
107
+ var _a;
108
+ const _path = `${(_a = configObject === null || configObject === void 0 ? void 0 : configObject.confPath) === null || _a === void 0 ? void 0 : _a.base}/reports/${this.slug}.json`;
109
+ if (!fs.existsSync(_path))
110
+ return {};
111
+ const content = fs.readFileSync(_path);
112
+ const data = JSON.parse(`${content}`);
113
+ return data;
114
+ },
115
+ };
116
+ return exerciseObj;
117
+ };
118
+ exports.exercise = exercise;
119
+ const validateExerciseDirectoryName = (str) => {
120
+ if (str === "./")
121
+ return true;
122
+ // TODO: Add nameValidationREgex from the config
123
+ const regex = /^(\d{2,3}(\.\d{1,2})?-([\dA-Za-z]+(-|_)?)+)$/;
124
+ return regex.test(str);
125
+ };
126
+ exports.validateExerciseDirectoryName = validateExerciseDirectoryName;
127
+ const isCodable = (str) => {
128
+ const extension = p.extname(str);
129
+ return allowed_files_1.default.extensions.includes(extension.slice(1).toLowerCase());
130
+ };
131
+ exports.isCodable = isCodable;
132
+ const isNotConfiguration = (str) => {
133
+ return !allowed_files_1.default.names.includes(str);
134
+ };
135
+ const shouldBeVisible = function (file) {
136
+ return (
137
+ // doest not have "test." on their name
138
+ !file.name.toLocaleLowerCase().includes("test.") &&
139
+ !file.name.toLocaleLowerCase().includes("tests.") &&
140
+ !file.name.toLocaleLowerCase().includes(".hide.") &&
141
+ // ignore hidden files
142
+ file.name.charAt(0) !== "." &&
143
+ // ignore learn.json and bc.json
144
+ !file.name.toLocaleLowerCase().includes("learn.json") &&
145
+ !file.name.toLocaleLowerCase().includes("bc.json") &&
146
+ // ignore images, videos, vectors, etc.
147
+ (0, exports.isCodable)(file.name) &&
148
+ isNotConfiguration(file.name) &&
149
+ // readme's and directories
150
+ !file.name.toLowerCase().includes("readme.") &&
151
+ !(0, exports.isDirectory)(file.path) &&
152
+ file.name.charAt(0) !== "_");
153
+ };
154
+ exports.shouldBeVisible = shouldBeVisible;
155
+ const isDirectory = (source) => {
156
+ // if(path.basename(source) === path.basename(config.dirPath)) return false
157
+ return fs.lstatSync(source).isDirectory();
158
+ };
159
+ exports.isDirectory = isDirectory;
160
+ const detect = (configObject, files) => {
161
+ if (!configObject) {
162
+ return;
163
+ }
164
+ const { config } = configObject;
165
+ if (!config)
166
+ throw new Error("No configuration found during the engine detection");
167
+ if (!config.entries)
168
+ throw new Error("No configuration found for entries, please add a 'entries' object with the default file name for your exercise entry file that is going to be used while compiling, for example: index.html for html, app.py for python3, etc.");
169
+ // A language was found on the config object, but this language will only be used as last resort, learnpack will try to guess each exercise language independently based on file extension (js, jsx, html, etc.)
170
+ let hasFiles = files.filter(f => f.includes(".py"));
171
+ if (hasFiles.length > 0)
172
+ return {
173
+ language: "python3",
174
+ entry: hasFiles.find(f => config.entries.python3 === f),
175
+ };
176
+ hasFiles = files.filter(f => f.includes(".java"));
177
+ if (hasFiles.length > 0)
178
+ return {
179
+ language: "java",
180
+ entry: hasFiles.find(f => config.entries.java === f),
181
+ };
182
+ hasFiles = files.filter(f => f.includes(".jsx"));
183
+ if (hasFiles.length > 0)
184
+ return {
185
+ language: "react",
186
+ entry: hasFiles.find(f => config.entries.react === f),
187
+ };
188
+ const hasHTML = files.filter(f => f.includes("index.html"));
189
+ const hasIndexJS = files.find(f => f.includes("index.js"));
190
+ const hasJS = files.filter(f => f.includes(".js"));
191
+ // angular, vue, vanillajs needs to have at least 2 files (html,css,js),
192
+ // the test.js and the entry file in js
193
+ // if not its just another HTML
194
+ if (hasIndexJS && hasHTML.length > 0)
195
+ return {
196
+ language: "vanillajs",
197
+ entry: hasIndexJS,
198
+ };
199
+ if (hasHTML.length > 0)
200
+ return {
201
+ language: "html",
202
+ entry: hasHTML.find(f => config.entries.html === f),
203
+ };
204
+ if (hasJS.length > 0)
205
+ return {
206
+ language: "node",
207
+ entry: hasJS.find(f => config.entries.node === f),
208
+ };
209
+ return {
210
+ language: null,
211
+ entry: null,
212
+ };
213
+ };
214
+ exports.detect = detect;
215
+ const filterFiles = (files, basePath = ".") => files
216
+ .map((ex) => ({
217
+ path: basePath + "/" + ex,
218
+ name: ex,
219
+ hidden: !(0, exports.shouldBeVisible)({
220
+ name: ex,
221
+ path: basePath + "/" + ex,
222
+ }),
223
+ }))
224
+ .sort((f1, f2) => {
225
+ const score = {
226
+ // sorting priority
227
+ "index.html": 1,
228
+ "styles.css": 2,
229
+ "styles.scss": 2,
230
+ "style.css": 2,
231
+ "style.scss": 2,
232
+ "index.css": 2,
233
+ "index.scss": 2,
234
+ "index.js": 3,
235
+ };
236
+ return score[f1.name] < score[f2.name] ? -1 : 1;
237
+ });
238
+ exports.filterFiles = filterFiles;
239
+ exports.default = {
240
+ exercise: exports.exercise,
241
+ detect: exports.detect,
242
+ filterFiles: exports.filterFiles,
243
+ };
@@ -1,3 +1,3 @@
1
- import { IConfigManagerAttributes, IConfigManager } from "../../models/config-manager";
2
- declare const _default: ({ grading, mode, disableGrading, version, }: IConfigManagerAttributes) => Promise<IConfigManager>;
3
- export default _default;
1
+ import { IConfigManagerAttributes, IConfigManager } from "../../models/config-manager";
2
+ declare const _default: ({ grading, mode, disableGrading, version, }: IConfigManagerAttributes) => Promise<IConfigManager>;
3
+ export default _default;