@learnpack/learnpack 2.1.25 → 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 +327 -302
  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 +25 -0
  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,94 +1,94 @@
1
- import * as shell from "shelljs";
2
- import { IPluginConfig } from "../models/plugin-config";
3
- /**
4
- * Main Plugin Runner, it defines the behavior of a learnpack plugin
5
- * dividing it in "actions" like: Compile, test, etc.
6
- * @param {object} pluginConfig Configuration object that must defined language and each possible action.
7
- */
8
- export default (pluginConfig: IPluginConfig) => {
9
- return async (args: any) => {
10
- const { action, exercise, socket, configuration } = args;
11
-
12
- if (pluginConfig.language === undefined)
13
- throw new Error(`Missing language on the plugin configuration object`);
14
-
15
- if (typeof action !== "string") {
16
- throw new TypeError("Missing action property on hook details");
17
- }
18
-
19
- if (!exercise || exercise === undefined) {
20
- throw new Error("Missing exercise information");
21
- }
22
-
23
- type actionType = "compile" | "test";
24
-
25
- // if the action does not exist I don't do anything
26
- if (pluginConfig[action as actionType] === undefined) {
27
- console.log(`Ignoring ${action}`);
28
- return () => null;
29
- }
30
-
31
- // ignore if the plugin language its not the same as the exercise language
32
- if (exercise.language !== pluginConfig.language) {
33
- return () => null;
34
- }
35
-
36
- if (!exercise.files || exercise.files.length === 0) {
37
- throw new Error(`No files to process`);
38
- }
39
-
40
- try {
41
- const _action = pluginConfig[action as actionType];
42
-
43
- if (_action === null || typeof _action !== "object")
44
- throw new Error(
45
- `The ${pluginConfig.language} ${action} module must export an object configuration`
46
- );
47
- if (_action.validate === undefined)
48
- throw new Error(
49
- `Missing validate method for ${pluginConfig.language} ${action}`
50
- );
51
- if (_action.run === undefined)
52
- throw new Error(
53
- `Missing run method for ${pluginConfig.language} ${action}`
54
- );
55
- if (_action.dependencies !== undefined) {
56
- if (!Array.isArray(_action.dependencies))
57
- throw new Error(
58
- `${action}.dependencies must be an array of package names`
59
- );
60
-
61
- for (const packageName of _action.dependencies) {
62
- if (!shell.which(packageName)) {
63
- throw new Error(
64
- `🚫 You need to have ${packageName} installed to run test the exercises`
65
- );
66
- }
67
- }
68
- }
69
-
70
- const valid = await _action.validate({ exercise, configuration });
71
- if (valid) {
72
- // look for the command standard implementation and execute it
73
- const execute = require("./command/" + action + ".js").default;
74
- // no matter the command, the response must always be a stdout
75
- const stdout = await execute({
76
- ...args,
77
- action: _action,
78
- configuration,
79
- });
80
-
81
- // Map the action names to socket messaging standards
82
- const actionToSuccessMapper = { compile: "compiler", test: "testing" };
83
-
84
- socket.success(actionToSuccessMapper[action as actionType], stdout);
85
- return stdout;
86
- }
87
- } catch (error: any) {
88
- if (error.type === undefined)
89
- socket.fatal(error);
90
- else
91
- socket.error(error.type, error.stdout);
92
- }
93
- };
94
- };
1
+ import * as shell from "shelljs";
2
+ import { IPluginConfig } from "../models/plugin-config";
3
+ /**
4
+ * Main Plugin Runner, it defines the behavior of a learnpack plugin
5
+ * dividing it in "actions" like: Compile, test, etc.
6
+ * @param {object} pluginConfig Configuration object that must defined language and each possible action.
7
+ */
8
+ export default (pluginConfig: IPluginConfig) => {
9
+ return async (args: any) => {
10
+ const { action, exercise, socket, configuration } = args;
11
+
12
+ if (pluginConfig.language === undefined)
13
+ throw new Error(`Missing language on the plugin configuration object`);
14
+
15
+ if (typeof action !== "string") {
16
+ throw new TypeError("Missing action property on hook details");
17
+ }
18
+
19
+ if (!exercise || exercise === undefined) {
20
+ throw new Error("Missing exercise information");
21
+ }
22
+
23
+ type actionType = "compile" | "test";
24
+
25
+ // if the action does not exist I don't do anything
26
+ if (pluginConfig[action as actionType] === undefined) {
27
+ console.log(`Ignoring ${action}`);
28
+ return () => null;
29
+ }
30
+
31
+ // ignore if the plugin language its not the same as the exercise language
32
+ if (exercise.language !== pluginConfig.language) {
33
+ return () => null;
34
+ }
35
+
36
+ if (!exercise.files || exercise.files.length === 0) {
37
+ throw new Error(`No files to process`);
38
+ }
39
+
40
+ try {
41
+ const _action = pluginConfig[action as actionType];
42
+
43
+ if (_action === null || typeof _action !== "object")
44
+ throw new Error(
45
+ `The ${pluginConfig.language} ${action} module must export an object configuration`
46
+ );
47
+ if (_action.validate === undefined)
48
+ throw new Error(
49
+ `Missing validate method for ${pluginConfig.language} ${action}`
50
+ );
51
+ if (_action.run === undefined)
52
+ throw new Error(
53
+ `Missing run method for ${pluginConfig.language} ${action}`
54
+ );
55
+ if (_action.dependencies !== undefined) {
56
+ if (!Array.isArray(_action.dependencies))
57
+ throw new Error(
58
+ `${action}.dependencies must be an array of package names`
59
+ );
60
+
61
+ for (const packageName of _action.dependencies) {
62
+ if (!shell.which(packageName)) {
63
+ throw new Error(
64
+ `🚫 You need to have ${packageName} installed to run test the exercises`
65
+ );
66
+ }
67
+ }
68
+ }
69
+
70
+ const valid = await _action.validate({ exercise, configuration });
71
+ if (valid) {
72
+ // look for the command standard implementation and execute it
73
+ const execute = require("./command/" + action + ".js").default;
74
+ // no matter the command, the response must always be a stdout
75
+ const stdout = await execute({
76
+ ...args,
77
+ action: _action,
78
+ configuration,
79
+ });
80
+
81
+ // Map the action names to socket messaging standards
82
+ const actionToSuccessMapper = { compile: "compiler", test: "testing" };
83
+
84
+ socket.success(actionToSuccessMapper[action as actionType], stdout);
85
+ return stdout;
86
+ }
87
+ } catch (error: any) {
88
+ if (error.type === undefined)
89
+ socket.fatal(error);
90
+ else
91
+ socket.error(error.type, error.stdout);
92
+ }
93
+ };
94
+ };
@@ -1,87 +1,87 @@
1
- import * as chalk from "chalk";
2
-
3
- const getMatches = (reg: RegExp, content: string) => {
4
- const inputs = [];
5
- let m;
6
- while ((m = reg.exec(content)) !== null) {
7
- // This is necessary to avoid infinite loops with zero-width matches
8
- if (m.index === reg.lastIndex)
9
- reg.lastIndex++;
10
-
11
- // The result can be accessed through the `m`-variable.
12
- inputs.push(m[1] || null);
13
- }
14
-
15
- return inputs;
16
- };
17
-
18
- const cleanStdout = (buffer: string, inputs: string[]) => {
19
- if (Array.isArray(inputs))
20
- for (let i = 0; i < inputs.length; i++)
21
- if (inputs[i])
22
- buffer = buffer.replace(inputs[i], "");
23
-
24
- return buffer;
25
- };
26
-
27
- const indent = (string: string, options: any, count = 1) => {
28
- options = {
29
- indent: " ",
30
- includeEmptyLines: false,
31
- ...options,
32
- };
33
-
34
- if (typeof string !== "string") {
35
- throw new TypeError(
36
- `Expected \`input\` to be a \`string\`, got \`${typeof string}\``
37
- );
38
- }
39
-
40
- if (typeof count !== "number") {
41
- throw new TypeError(
42
- `Expected \`count\` to be a \`number\`, got \`${typeof count}\``
43
- );
44
- }
45
-
46
- if (count < 0) {
47
- throw new RangeError(
48
- `Expected \`count\` to be at least 0, got \`${count}\``
49
- );
50
- }
51
-
52
- if (typeof options.indent !== "string") {
53
- throw new TypeError(
54
- `Expected \`options.indent\` to be a \`string\`, got \`${typeof options.indent}\``
55
- );
56
- }
57
-
58
- if (count === 0) {
59
- return string;
60
- }
61
-
62
- const regex = options.includeEmptyLines ? /^/gm : /^(?!\s*$)/gm;
63
-
64
- return string.replace(regex, options.indent.repeat(count));
65
- };
66
-
67
- const Console = {
68
- // _debug: true,
69
- _debug: process.env.DEBUG === "true",
70
- startDebug: function () {
71
- this._debug = true;
72
- },
73
- log: (msg: string, ...args: any[]) => console.log(chalk.gray(msg), ...args),
74
- error: (msg: string, ...args: any[]) =>
75
- console.log(chalk.red("⨉ " + msg), ...args),
76
- success: (msg: string, ...args: any[]) =>
77
- console.log(chalk.green("✓ " + msg), ...args),
78
- info: (msg: string, ...args: any[]) =>
79
- console.log(chalk.blue("ⓘ " + msg), ...args),
80
- help: (msg: string) =>
81
- console.log(`${chalk.white.bold("⚠ help:")} ${chalk.white(msg)}`),
82
- debug(...args: any[]) {
83
- this._debug && console.log(chalk.magentaBright(`⚠ debug: `), args);
84
- },
85
- };
86
-
87
- export default { getMatches, cleanStdout, indent, Console };
1
+ import * as chalk from "chalk";
2
+
3
+ const getMatches = (reg: RegExp, content: string) => {
4
+ const inputs = [];
5
+ let m;
6
+ while ((m = reg.exec(content)) !== null) {
7
+ // This is necessary to avoid infinite loops with zero-width matches
8
+ if (m.index === reg.lastIndex)
9
+ reg.lastIndex++;
10
+
11
+ // The result can be accessed through the `m`-variable.
12
+ inputs.push(m[1] || null);
13
+ }
14
+
15
+ return inputs;
16
+ };
17
+
18
+ const cleanStdout = (buffer: string, inputs: string[]) => {
19
+ if (Array.isArray(inputs))
20
+ for (let i = 0; i < inputs.length; i++)
21
+ if (inputs[i])
22
+ buffer = buffer.replace(inputs[i], "");
23
+
24
+ return buffer;
25
+ };
26
+
27
+ const indent = (string: string, options: any, count = 1) => {
28
+ options = {
29
+ indent: " ",
30
+ includeEmptyLines: false,
31
+ ...options,
32
+ };
33
+
34
+ if (typeof string !== "string") {
35
+ throw new TypeError(
36
+ `Expected \`input\` to be a \`string\`, got \`${typeof string}\``
37
+ );
38
+ }
39
+
40
+ if (typeof count !== "number") {
41
+ throw new TypeError(
42
+ `Expected \`count\` to be a \`number\`, got \`${typeof count}\``
43
+ );
44
+ }
45
+
46
+ if (count < 0) {
47
+ throw new RangeError(
48
+ `Expected \`count\` to be at least 0, got \`${count}\``
49
+ );
50
+ }
51
+
52
+ if (typeof options.indent !== "string") {
53
+ throw new TypeError(
54
+ `Expected \`options.indent\` to be a \`string\`, got \`${typeof options.indent}\``
55
+ );
56
+ }
57
+
58
+ if (count === 0) {
59
+ return string;
60
+ }
61
+
62
+ const regex = options.includeEmptyLines ? /^/gm : /^(?!\s*$)/gm;
63
+
64
+ return string.replace(regex, options.indent.repeat(count));
65
+ };
66
+
67
+ const Console = {
68
+ // _debug: true,
69
+ _debug: process.env.DEBUG === "true",
70
+ startDebug: function () {
71
+ this._debug = true;
72
+ },
73
+ log: (msg: string, ...args: any[]) => console.log(chalk.gray(msg), ...args),
74
+ error: (msg: string, ...args: any[]) =>
75
+ console.log(chalk.red("⨉ " + msg), ...args),
76
+ success: (msg: string, ...args: any[]) =>
77
+ console.log(chalk.green("✓ " + msg), ...args),
78
+ info: (msg: string, ...args: any[]) =>
79
+ console.log(chalk.blue("ⓘ " + msg), ...args),
80
+ help: (msg: string) =>
81
+ console.log(`${chalk.white.bold("⚠ help:")} ${chalk.white(msg)}`),
82
+ debug(...args: any[]) {
83
+ this._debug && console.log(chalk.magentaBright(`⚠ debug: `), args);
84
+ },
85
+ };
86
+
87
+ export default { getMatches, cleanStdout, indent, Console };
@@ -1 +1 @@
1
- declare module '*';
1
+ declare module '*';
@@ -1,71 +1,71 @@
1
- import {prompt} from 'enquirer'
2
- import Console from '../utils/console'
3
- import api from '../utils/api'
4
- // import fetch from 'node-fetch'
5
-
6
- import {ILanguage} from '../models/language'
7
- import {IPackage} from '../models/package'
8
-
9
- export const askPackage = async () => {
10
- Console.info('No package was specified')
11
- const languages = await api.getLangs()
12
-
13
- return new Promise((resolve, reject) => {
14
- if (languages.length === 0) {
15
- // reject(new Error('No categories available'))
16
- reject('No categories available')
17
- // return null;
18
- }
19
-
20
- // let packages = []
21
- prompt([
22
- {
23
- type: 'select',
24
- name: 'lang',
25
- message: 'What language do you want to practice?',
26
- choices: languages.map((l: ILanguage) => ({
27
- message: l.title,
28
- name: l.slug,
29
- })),
30
- },
31
- ])
32
- .then(({lang}: any) => {
33
- return (async () => {
34
- const response = await api.getAllPackages({lang})
35
- const packages = response.results
36
- if (packages.length === 0) {
37
- const error = new Error(`No packages found for language ${lang}`)
38
- Console.error(error.message) // TODO: Look this
39
- return error
40
- }
41
-
42
- return prompt([
43
- {
44
- type: 'select',
45
- name: 'pack',
46
- message: 'Choose one of the packages available',
47
- choices: packages.map((l: IPackage) => ({
48
- message: `${l.title}, difficulty: ${l.difficulty}, downloads: ${
49
- l.downloads
50
- } ${
51
- l.skills.length > 0 ? `(Skills: ${l.skills.join(',')})` : ''
52
- }`,
53
- name: l.slug,
54
- })),
55
- },
56
- ])
57
- })()
58
- })
59
- .then((resp: any) => {
60
- if (!resp)
61
- reject(resp.message || resp)
62
- else
63
- resolve(resp.pack)
64
- })
65
- .catch(error => {
66
- Console.error(error.message || error)
67
- })
68
- })
69
- }
70
-
71
- export default {askPackage}
1
+ import {prompt} from 'enquirer'
2
+ import Console from '../utils/console'
3
+ import api from '../utils/api'
4
+ // import fetch from 'node-fetch'
5
+
6
+ import {ILanguage} from '../models/language'
7
+ import {IPackage} from '../models/package'
8
+
9
+ export const askPackage = async () => {
10
+ Console.info('No package was specified')
11
+ const languages = await api.getLangs()
12
+
13
+ return new Promise((resolve, reject) => {
14
+ if (languages.length === 0) {
15
+ // reject(new Error('No categories available'))
16
+ reject('No categories available')
17
+ // return null;
18
+ }
19
+
20
+ // let packages = []
21
+ prompt([
22
+ {
23
+ type: 'select',
24
+ name: 'lang',
25
+ message: 'What language do you want to practice?',
26
+ choices: languages.map((l: ILanguage) => ({
27
+ message: l.title,
28
+ name: l.slug,
29
+ })),
30
+ },
31
+ ])
32
+ .then(({lang}: any) => {
33
+ return (async () => {
34
+ const response = await api.getAllPackages({lang})
35
+ const packages = response.results
36
+ if (packages.length === 0) {
37
+ const error = new Error(`No packages found for language ${lang}`)
38
+ Console.error(error.message) // TODO: Look this
39
+ return error
40
+ }
41
+
42
+ return prompt([
43
+ {
44
+ type: 'select',
45
+ name: 'pack',
46
+ message: 'Choose one of the packages available',
47
+ choices: packages.map((l: IPackage) => ({
48
+ message: `${l.title}, difficulty: ${l.difficulty}, downloads: ${
49
+ l.downloads
50
+ } ${
51
+ l.skills.length > 0 ? `(Skills: ${l.skills.join(',')})` : ''
52
+ }`,
53
+ name: l.slug,
54
+ })),
55
+ },
56
+ ])
57
+ })()
58
+ })
59
+ .then((resp: any) => {
60
+ if (!resp)
61
+ reject(resp.message || resp)
62
+ else
63
+ resolve(resp.pack)
64
+ })
65
+ .catch(error => {
66
+ Console.error(error.message || error)
67
+ })
68
+ })
69
+ }
70
+
71
+ export default {askPackage}
@@ -1,48 +1,48 @@
1
- import { Command } from "@oclif/command";
2
- import Console from "./console";
3
- import { createInterface } from "readline";
4
- // import SessionManager from '../managers/session'
5
-
6
- class BaseCommand extends Command {
7
- async catch(err: any) {
8
- Console.debug("COMMAND CATCH", err);
9
-
10
- throw err;
11
- }
12
-
13
- async init() {
14
- const { flags, args } = this.parse(BaseCommand);
15
- Console.debug("COMMAND INIT");
16
- Console.debug("These are your flags: ", flags);
17
- Console.debug("These are your args: ", args);
18
-
19
- // quick fix for listening to the process termination on windows
20
- if (process.platform === "win32") {
21
- const rl = createInterface({
22
- input: process.stdin,
23
- output: process.stdout,
24
- });
25
-
26
- rl.on("SIGINT", function () {
27
- // process.emit('SIGINT')
28
- // process.emit('SIGINT')
29
- });
30
- }
31
-
32
- process.on("SIGINT", function () {
33
- Console.debug("Terminated (SIGINT)");
34
- process.exit();
35
- });
36
- }
37
-
38
- async finally() {
39
- Console.debug("COMMAND FINALLY");
40
- // called after run and catch regardless of whether or not the command errored
41
- }
42
-
43
- async run() {
44
- // console.log('running my command')
45
- }
46
- }
47
-
48
- export default BaseCommand;
1
+ import { Command } from "@oclif/command";
2
+ import Console from "./console";
3
+ import { createInterface } from "readline";
4
+ // import SessionManager from '../managers/session'
5
+
6
+ class BaseCommand extends Command {
7
+ async catch(err: any) {
8
+ Console.debug("COMMAND CATCH", err);
9
+
10
+ throw err;
11
+ }
12
+
13
+ async init() {
14
+ const { flags, args } = this.parse(BaseCommand);
15
+ Console.debug("COMMAND INIT");
16
+ Console.debug("These are your flags: ", flags);
17
+ Console.debug("These are your args: ", args);
18
+
19
+ // quick fix for listening to the process termination on windows
20
+ if (process.platform === "win32") {
21
+ const rl = createInterface({
22
+ input: process.stdin,
23
+ output: process.stdout,
24
+ });
25
+
26
+ rl.on("SIGINT", function () {
27
+ // process.emit('SIGINT')
28
+ // process.emit('SIGINT')
29
+ });
30
+ }
31
+
32
+ process.on("SIGINT", function () {
33
+ Console.debug("Terminated (SIGINT)");
34
+ process.exit();
35
+ });
36
+ }
37
+
38
+ async finally() {
39
+ Console.debug("COMMAND FINALLY");
40
+ // called after run and catch regardless of whether or not the command errored
41
+ }
42
+
43
+ async run() {
44
+ // console.log('running my command')
45
+ }
46
+ }
47
+
48
+ export default BaseCommand;