@learnpack/learnpack 2.1.39 → 2.1.41
Sign up to get free protection for your applications and to get access to all the features.
- package/README.md +370 -35
- package/bin/run +17 -17
- package/bin/run.cmd +3 -3
- package/lib/commands/audit.d.ts +6 -6
- package/lib/commands/audit.js +342 -342
- package/lib/commands/clean.d.ts +8 -8
- package/lib/commands/clean.js +25 -25
- package/lib/commands/download.d.ts +13 -13
- package/lib/commands/download.js +55 -55
- package/lib/commands/init.d.ts +9 -9
- package/lib/commands/init.js +123 -123
- package/lib/commands/login.d.ts +14 -14
- package/lib/commands/login.js +37 -37
- package/lib/commands/logout.d.ts +14 -14
- package/lib/commands/logout.js +37 -37
- package/lib/commands/publish.d.ts +14 -14
- package/lib/commands/publish.js +82 -82
- package/lib/commands/start.d.ts +7 -7
- package/lib/commands/start.js +239 -208
- package/lib/commands/test.d.ts +6 -6
- package/lib/commands/test.js +62 -62
- package/lib/index.d.ts +1 -1
- package/lib/index.js +4 -4
- package/lib/managers/config/allowed_files.d.ts +5 -5
- package/lib/managers/config/allowed_files.js +30 -30
- package/lib/managers/config/defaults.d.ts +41 -41
- package/lib/managers/config/defaults.js +44 -44
- package/lib/managers/config/exercise.d.ts +36 -36
- package/lib/managers/config/exercise.js +236 -236
- package/lib/managers/config/index.d.ts +3 -3
- package/lib/managers/config/index.js +337 -337
- package/lib/managers/file.d.ts +14 -14
- package/lib/managers/file.js +153 -153
- package/lib/managers/gitpod.d.ts +3 -3
- package/lib/managers/gitpod.js +67 -67
- package/lib/managers/server/index.d.ts +6 -6
- package/lib/managers/server/index.js +58 -58
- package/lib/managers/server/routes.d.ts +4 -4
- package/lib/managers/server/routes.js +219 -219
- package/lib/managers/session.d.ts +3 -3
- package/lib/managers/session.js +125 -125
- package/lib/managers/socket.d.ts +3 -3
- package/lib/managers/socket.js +176 -176
- package/lib/managers/telemetry.d.ts +74 -74
- package/lib/managers/telemetry.js +206 -206
- package/lib/managers/test.js +84 -84
- package/lib/models/action.d.ts +2 -2
- package/lib/models/action.js +2 -2
- package/lib/models/audit.d.ts +15 -15
- package/lib/models/audit.js +2 -2
- package/lib/models/config-manager.d.ts +21 -21
- package/lib/models/config-manager.js +2 -2
- package/lib/models/config.d.ts +68 -67
- package/lib/models/config.js +2 -2
- package/lib/models/counter.d.ts +11 -11
- package/lib/models/counter.js +2 -2
- package/lib/models/errors.d.ts +15 -15
- package/lib/models/errors.js +2 -2
- package/lib/models/exercise-obj.d.ts +30 -30
- package/lib/models/exercise-obj.js +2 -2
- package/lib/models/file.d.ts +5 -5
- package/lib/models/file.js +2 -2
- package/lib/models/findings.d.ts +17 -17
- package/lib/models/findings.js +2 -2
- package/lib/models/flags.d.ts +10 -10
- package/lib/models/flags.js +2 -2
- package/lib/models/front-matter.d.ts +11 -11
- package/lib/models/front-matter.js +2 -2
- package/lib/models/gitpod-data.d.ts +16 -16
- package/lib/models/gitpod-data.js +2 -2
- package/lib/models/language.d.ts +4 -4
- package/lib/models/language.js +2 -2
- package/lib/models/package.d.ts +7 -7
- package/lib/models/package.js +2 -2
- package/lib/models/plugin-config.d.ts +16 -16
- package/lib/models/plugin-config.js +2 -2
- package/lib/models/session.d.ts +31 -31
- package/lib/models/session.js +2 -2
- package/lib/models/socket.d.ts +36 -32
- package/lib/models/socket.js +2 -2
- package/lib/models/status.d.ts +1 -1
- package/lib/models/status.js +2 -2
- package/lib/models/success-types.d.ts +1 -1
- package/lib/models/success-types.js +2 -2
- package/lib/plugin/command/compile.d.ts +6 -6
- package/lib/plugin/command/compile.js +18 -18
- package/lib/plugin/command/test.d.ts +6 -6
- package/lib/plugin/command/test.js +25 -25
- package/lib/plugin/index.d.ts +27 -27
- package/lib/plugin/index.js +7 -7
- package/lib/plugin/plugin.d.ts +8 -8
- package/lib/plugin/plugin.js +68 -68
- package/lib/plugin/utils.d.ts +16 -16
- package/lib/plugin/utils.js +58 -58
- package/lib/ui/download.d.ts +5 -5
- package/lib/ui/download.js +61 -61
- package/lib/utils/BaseCommand.d.ts +8 -8
- package/lib/utils/BaseCommand.js +41 -41
- package/lib/utils/SessionCommand.d.ts +10 -10
- package/lib/utils/SessionCommand.js +43 -43
- package/lib/utils/api.d.ts +14 -14
- package/lib/utils/api.js +255 -255
- package/lib/utils/audit.d.ts +16 -16
- package/lib/utils/audit.js +303 -303
- package/lib/utils/checkNotInstalled.d.ts +2 -0
- package/lib/utils/checkNotInstalled.js +36 -0
- package/lib/utils/console.d.ts +12 -12
- package/lib/utils/console.js +19 -19
- package/lib/utils/errors.d.ts +17 -17
- package/lib/utils/errors.js +100 -100
- package/lib/utils/exercisesQueue.d.ts +9 -9
- package/lib/utils/exercisesQueue.js +38 -38
- package/lib/utils/fileQueue.d.ts +40 -40
- package/lib/utils/fileQueue.js +168 -168
- package/lib/utils/misc.d.ts +1 -1
- package/lib/utils/misc.js +23 -23
- package/lib/utils/osOperations.d.ts +5 -5
- package/lib/utils/osOperations.js +72 -72
- package/lib/utils/validators.d.ts +5 -5
- package/lib/utils/validators.js +17 -17
- package/lib/utils/watcher.d.ts +2 -2
- package/lib/utils/watcher.js +25 -25
- package/oclif.manifest.json +1 -1
- package/package.json +139 -139
- package/src/commands/audit.ts +443 -443
- package/src/commands/clean.ts +29 -29
- package/src/commands/download.ts +61 -61
- package/src/commands/init.ts +170 -170
- package/src/commands/login.ts +42 -42
- package/src/commands/logout.ts +43 -43
- package/src/commands/publish.ts +107 -107
- package/src/commands/start.ts +53 -23
- package/src/commands/test.ts +85 -85
- package/src/index.ts +1 -1
- package/src/managers/config/allowed_files.ts +29 -29
- package/src/managers/config/defaults.ts +42 -42
- package/src/managers/config/exercise.ts +311 -311
- package/src/managers/config/index.ts +455 -455
- package/src/managers/file.ts +196 -196
- package/src/managers/gitpod.ts +84 -84
- package/src/managers/server/index.ts +78 -78
- package/src/managers/server/routes.ts +330 -330
- package/src/managers/session.ts +145 -145
- package/src/managers/socket.ts +250 -250
- package/src/managers/telemetry.ts +346 -346
- package/src/managers/test.ts +83 -83
- package/src/models/action.ts +10 -10
- package/src/models/audit.ts +16 -16
- package/src/models/config-manager.ts +23 -23
- package/src/models/config.ts +5 -3
- package/src/models/counter.ts +11 -11
- package/src/models/errors.ts +22 -22
- package/src/models/exercise-obj.ts +29 -29
- package/src/models/file.ts +5 -5
- package/src/models/findings.ts +18 -18
- package/src/models/flags.ts +10 -10
- package/src/models/front-matter.ts +11 -11
- package/src/models/gitpod-data.ts +19 -19
- package/src/models/language.ts +4 -4
- package/src/models/package.ts +7 -7
- package/src/models/plugin-config.ts +17 -17
- package/src/models/session.ts +34 -34
- package/src/models/socket.ts +5 -0
- package/src/models/status.ts +16 -16
- package/src/models/success-types.ts +1 -1
- package/src/plugin/command/compile.ts +17 -17
- package/src/plugin/command/test.ts +30 -30
- package/src/plugin/index.ts +6 -6
- package/src/plugin/plugin.ts +94 -94
- package/src/plugin/utils.ts +87 -87
- package/src/types/node-fetch.d.ts +1 -1
- package/src/ui/download.ts +71 -71
- package/src/utils/BaseCommand.ts +48 -48
- package/src/utils/SessionCommand.ts +43 -43
- package/src/utils/api.ts +303 -303
- package/src/utils/audit.ts +393 -393
- package/src/utils/checkNotInstalled.ts +46 -0
- package/src/utils/console.ts +24 -24
- package/src/utils/errors.ts +117 -117
- package/src/utils/exercisesQueue.ts +51 -51
- package/src/utils/fileQueue.ts +198 -198
- package/src/utils/misc.ts +23 -23
- package/src/utils/osOperations.ts +79 -79
- package/src/utils/templates/gitignore.txt +19 -19
- package/src/utils/templates/incremental/.learn/exercises/01-hello-world/README.es.md +24 -24
- package/src/utils/templates/incremental/.learn/exercises/01-hello-world/README.md +24 -24
- package/src/utils/templates/incremental/README.ejs +4 -4
- package/src/utils/templates/incremental/README.es.ejs +4 -4
- package/src/utils/templates/isolated/01-hello-world/README.es.md +26 -26
- package/src/utils/templates/isolated/01-hello-world/README.md +26 -26
- package/src/utils/templates/isolated/README.ejs +4 -4
- package/src/utils/templates/isolated/README.es.ejs +4 -4
- package/src/utils/templates/no-grading/README.ejs +4 -4
- package/src/utils/templates/no-grading/README.es.ejs +4 -4
- package/src/utils/validators.ts +18 -18
- package/src/utils/watcher.ts +27 -27
package/src/models/session.ts
CHANGED
@@ -1,34 +1,34 @@
|
|
1
|
-
import { IConfig, IConfigObj } from "./config"
|
2
|
-
|
3
|
-
export interface IPayload {
|
4
|
-
email: string;
|
5
|
-
}
|
6
|
-
|
7
|
-
export interface IStartProps {
|
8
|
-
token: string;
|
9
|
-
payload: IPayload | null;
|
10
|
-
}
|
11
|
-
|
12
|
-
type TLoginResponse = {
|
13
|
-
token: string;
|
14
|
-
user_id: string;
|
15
|
-
email: string;
|
16
|
-
};
|
17
|
-
|
18
|
-
export interface ISession {
|
19
|
-
sessionStarted: boolean;
|
20
|
-
token: string | null;
|
21
|
-
config: IConfig | null;
|
22
|
-
currentCohort: null;
|
23
|
-
initialize: () => Promise<boolean>;
|
24
|
-
setRigoToken: (token: string) => Promise<boolean>;
|
25
|
-
setPayload: (value: IPayload) => Promise<boolean>;
|
26
|
-
getPayload: () => Promise<any>;
|
27
|
-
isActive: () => boolean;
|
28
|
-
get: (config?: IConfigObj) => Promise<any>;
|
29
|
-
login: () => Promise<void>;
|
30
|
-
loginWeb: (email: string, password: string) => Promise<TLoginResponse>;
|
31
|
-
sync: () => Promise<void>;
|
32
|
-
start: ({ token, payload }: IStartProps) => Promise<void>;
|
33
|
-
destroy: () => Promise<void>;
|
34
|
-
}
|
1
|
+
import { IConfig, IConfigObj } from "./config"
|
2
|
+
|
3
|
+
export interface IPayload {
|
4
|
+
email: string;
|
5
|
+
}
|
6
|
+
|
7
|
+
export interface IStartProps {
|
8
|
+
token: string;
|
9
|
+
payload: IPayload | null;
|
10
|
+
}
|
11
|
+
|
12
|
+
type TLoginResponse = {
|
13
|
+
token: string;
|
14
|
+
user_id: string;
|
15
|
+
email: string;
|
16
|
+
};
|
17
|
+
|
18
|
+
export interface ISession {
|
19
|
+
sessionStarted: boolean;
|
20
|
+
token: string | null;
|
21
|
+
config: IConfig | null;
|
22
|
+
currentCohort: null;
|
23
|
+
initialize: () => Promise<boolean>;
|
24
|
+
setRigoToken: (token: string) => Promise<boolean>;
|
25
|
+
setPayload: (value: IPayload) => Promise<boolean>;
|
26
|
+
getPayload: () => Promise<any>;
|
27
|
+
isActive: () => boolean;
|
28
|
+
get: (config?: IConfigObj) => Promise<any>;
|
29
|
+
login: () => Promise<void>;
|
30
|
+
loginWeb: (email: string, password: string) => Promise<TLoginResponse>;
|
31
|
+
sync: () => Promise<void>;
|
32
|
+
start: ({ token, payload }: IStartProps) => Promise<void>;
|
33
|
+
destroy: () => Promise<void>;
|
34
|
+
}
|
package/src/models/socket.ts
CHANGED
package/src/models/status.ts
CHANGED
@@ -1,16 +1,16 @@
|
|
1
|
-
export type TStatus =
|
2
|
-
| "ready"
|
3
|
-
| "internal-error"
|
4
|
-
| "compiler-success"
|
5
|
-
| "testing-success"
|
6
|
-
| "compiling"
|
7
|
-
| "testing"
|
8
|
-
| "start_exercise"
|
9
|
-
| "initializing"
|
10
|
-
| "configuration_loaded"
|
11
|
-
| "connection_ended"
|
12
|
-
| "reset_exercise"
|
13
|
-
| "open_files"
|
14
|
-
| "open_window"
|
15
|
-
| "instructions_closed"
|
16
|
-
| "completed";
|
1
|
+
export type TStatus =
|
2
|
+
| "ready"
|
3
|
+
| "internal-error"
|
4
|
+
| "compiler-success"
|
5
|
+
| "testing-success"
|
6
|
+
| "compiling"
|
7
|
+
| "testing"
|
8
|
+
| "start_exercise"
|
9
|
+
| "initializing"
|
10
|
+
| "configuration_loaded"
|
11
|
+
| "connection_ended"
|
12
|
+
| "reset_exercise"
|
13
|
+
| "open_files"
|
14
|
+
| "open_window"
|
15
|
+
| "instructions_closed"
|
16
|
+
| "completed";
|
@@ -1 +1 @@
|
|
1
|
-
export type TSuccessType = 'compiler-success' | 'testing-success';
|
1
|
+
export type TSuccessType = 'compiler-success' | 'testing-success';
|
@@ -1,17 +1,17 @@
|
|
1
|
-
import { IError } from "../../models/errors"
|
2
|
-
|
3
|
-
const CompilationError = (messages: string) => {
|
4
|
-
const _err: IError = new Error(messages)
|
5
|
-
_err.status = 400
|
6
|
-
_err.stdout = messages
|
7
|
-
_err.type = "compiler-error"
|
8
|
-
return _err
|
9
|
-
}
|
10
|
-
|
11
|
-
export default {
|
12
|
-
CompilationError,
|
13
|
-
default: async ({ action, ...rest }: any) => {
|
14
|
-
const stdout = await action.run(rest)
|
15
|
-
return stdout
|
16
|
-
},
|
17
|
-
}
|
1
|
+
import { IError } from "../../models/errors"
|
2
|
+
|
3
|
+
const CompilationError = (messages: string) => {
|
4
|
+
const _err: IError = new Error(messages)
|
5
|
+
_err.status = 400
|
6
|
+
_err.stdout = messages
|
7
|
+
_err.type = "compiler-error"
|
8
|
+
return _err
|
9
|
+
}
|
10
|
+
|
11
|
+
export default {
|
12
|
+
CompilationError,
|
13
|
+
default: async ({ action, ...rest }: any) => {
|
14
|
+
const stdout = await action.run(rest)
|
15
|
+
return stdout
|
16
|
+
},
|
17
|
+
}
|
@@ -1,30 +1,30 @@
|
|
1
|
-
import * as fs from "fs"
|
2
|
-
import { IError } from "../../models/errors"
|
3
|
-
|
4
|
-
const TestingError = (messages: string) => {
|
5
|
-
const _err: IError = new Error(messages)
|
6
|
-
_err.status = 400
|
7
|
-
_err.stdout = messages
|
8
|
-
_err.type = "testing-error"
|
9
|
-
return _err
|
10
|
-
}
|
11
|
-
|
12
|
-
export default {
|
13
|
-
TestingError,
|
14
|
-
default: async function (args: any) {
|
15
|
-
const { action, configuration, socket, exercise } = args
|
16
|
-
|
17
|
-
if (!fs.existsSync(`${configuration.dirPath}/reports`)) {
|
18
|
-
// reports directory
|
19
|
-
fs.mkdirSync(`${configuration.dirPath}/reports`)
|
20
|
-
}
|
21
|
-
|
22
|
-
// compile
|
23
|
-
const stdout = await action.run(args)
|
24
|
-
|
25
|
-
// mark exercise as done
|
26
|
-
exercise.done = true
|
27
|
-
|
28
|
-
return stdout
|
29
|
-
},
|
30
|
-
}
|
1
|
+
import * as fs from "fs"
|
2
|
+
import { IError } from "../../models/errors"
|
3
|
+
|
4
|
+
const TestingError = (messages: string) => {
|
5
|
+
const _err: IError = new Error(messages)
|
6
|
+
_err.status = 400
|
7
|
+
_err.stdout = messages
|
8
|
+
_err.type = "testing-error"
|
9
|
+
return _err
|
10
|
+
}
|
11
|
+
|
12
|
+
export default {
|
13
|
+
TestingError,
|
14
|
+
default: async function (args: any) {
|
15
|
+
const { action, configuration, socket, exercise } = args
|
16
|
+
|
17
|
+
if (!fs.existsSync(`${configuration.dirPath}/reports`)) {
|
18
|
+
// reports directory
|
19
|
+
fs.mkdirSync(`${configuration.dirPath}/reports`)
|
20
|
+
}
|
21
|
+
|
22
|
+
// compile
|
23
|
+
const stdout = await action.run(args)
|
24
|
+
|
25
|
+
// mark exercise as done
|
26
|
+
exercise.done = true
|
27
|
+
|
28
|
+
return stdout
|
29
|
+
},
|
30
|
+
}
|
package/src/plugin/index.ts
CHANGED
@@ -1,6 +1,6 @@
|
|
1
|
-
import CompilationError from "./command/compile"
|
2
|
-
import TestingError from "./command/test"
|
3
|
-
import utils from "./utils"
|
4
|
-
import plugin from "./plugin"
|
5
|
-
|
6
|
-
export default { CompilationError, TestingError, Utils: utils, plugin }
|
1
|
+
import CompilationError from "./command/compile"
|
2
|
+
import TestingError from "./command/test"
|
3
|
+
import utils from "./utils"
|
4
|
+
import plugin from "./plugin"
|
5
|
+
|
6
|
+
export default { CompilationError, TestingError, Utils: utils, plugin }
|
package/src/plugin/plugin.ts
CHANGED
@@ -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
|
+
}
|
package/src/plugin/utils.ts
CHANGED
@@ -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 '*';
|