@learnpack/learnpack 5.0.275 → 5.0.277
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.
- package/README.md +409 -409
- package/lib/commands/audit.js +15 -15
- package/lib/commands/breakToken.js +19 -19
- package/lib/commands/clean.js +3 -3
- package/lib/commands/init.js +41 -41
- package/lib/commands/logout.js +3 -3
- package/lib/commands/publish.js +5 -10
- package/lib/commands/serve.js +55 -2
- package/lib/creatorDist/assets/index-BfLyIQVh.js +10343 -10224
- package/lib/managers/config/index.js +77 -77
- package/lib/utils/api.d.ts +1 -1
- package/lib/utils/api.js +12 -9
- package/lib/utils/creatorUtilities.js +14 -14
- package/lib/utils/export/epub.d.ts +2 -0
- package/lib/utils/export/epub.js +298 -0
- package/lib/utils/export/index.d.ts +3 -0
- package/lib/utils/export/index.js +7 -0
- package/lib/utils/export/scorm.d.ts +2 -0
- package/lib/utils/export/scorm.js +84 -0
- package/lib/utils/export/shared.d.ts +4 -0
- package/lib/utils/export/shared.js +61 -0
- package/lib/utils/export/types.d.ts +15 -0
- package/lib/utils/export/types.js +2 -0
- package/package.json +2 -1
- package/src/commands/audit.ts +487 -487
- package/src/commands/breakToken.ts +67 -67
- package/src/commands/clean.ts +30 -30
- package/src/commands/init.ts +650 -650
- package/src/commands/logout.ts +38 -38
- package/src/commands/publish.ts +20 -25
- package/src/commands/serve.ts +69 -4
- package/src/commands/start.ts +333 -333
- package/src/commands/translate.ts +123 -123
- package/src/creator/README.md +54 -54
- package/src/creator/eslint.config.js +7 -7
- package/src/creator/src/components/syllabus/ContentIndex.tsx +312 -312
- package/src/creator/src/i18n.ts +28 -28
- package/src/creator/src/index.css +217 -217
- package/src/creator/src/locales/en.json +126 -126
- package/src/creator/src/locales/es.json +126 -126
- package/src/creator/src/utils/configTypes.ts +122 -122
- package/src/creator/src/utils/constants.ts +13 -13
- package/src/creator/src/utils/creatorUtils.ts +46 -46
- package/src/creator/src/utils/eventBus.ts +2 -2
- package/src/creator/src/utils/lib.ts +468 -468
- package/src/creator/src/utils/socket.ts +61 -61
- package/src/creator/src/utils/store.ts +222 -222
- package/src/creator/src/vite-env.d.ts +1 -1
- package/src/creator/vite.config.ts +13 -13
- package/src/creatorDist/assets/index-BfLyIQVh.js +10343 -10224
- package/src/managers/config/defaults.ts +49 -49
- package/src/managers/config/exercise.ts +364 -364
- package/src/managers/config/index.ts +775 -775
- package/src/managers/file.ts +236 -236
- package/src/managers/server/routes.ts +554 -554
- package/src/managers/session.ts +182 -182
- package/src/managers/telemetry.ts +188 -188
- package/src/models/action.ts +13 -13
- package/src/models/config-manager.ts +28 -28
- package/src/models/config.ts +106 -106
- package/src/models/creator.ts +47 -47
- package/src/models/exercise-obj.ts +30 -30
- package/src/models/session.ts +39 -39
- package/src/models/socket.ts +61 -61
- package/src/models/status.ts +16 -16
- package/src/ui/_app/app.css +1 -1
- package/src/ui/_app/app.js +400 -397
- package/src/ui/app.tar.gz +0 -0
- package/src/utils/BaseCommand.ts +56 -56
- package/src/utils/api.ts +53 -39
- package/src/utils/audit.ts +392 -392
- package/src/utils/checkNotInstalled.ts +267 -267
- package/src/utils/configBuilder.ts +82 -82
- package/src/utils/convertCreds.js +34 -34
- package/src/utils/creatorUtilities.ts +504 -504
- package/src/utils/export/README.md +178 -0
- package/src/utils/export/epub.ts +400 -0
- package/src/utils/export/index.ts +3 -0
- package/src/utils/export/scorm.ts +121 -0
- package/src/utils/export/shared.ts +61 -0
- package/src/utils/export/types.ts +17 -0
- package/src/utils/incrementVersion.js +74 -74
- package/src/utils/misc.ts +58 -58
- package/src/utils/rigoActions.ts +500 -500
- package/src/utils/sidebarGenerator.ts +195 -195
- package/src/utils/templates/epub/epub.css +133 -0
- package/src/utils/templates/isolated/exercises/01-hello-world/README.es.md +26 -26
- package/src/utils/templates/isolated/exercises/01-hello-world/README.md +26 -26
- package/src/utils/templates/scorm/adlcp_rootv1p2.xsd +110 -0
- package/src/utils/templates/scorm/config/api.js +175 -0
- package/src/utils/templates/scorm/config/index.html +210 -0
- package/src/utils/templates/scorm/ims_xml.xsd +1 -0
- package/src/utils/templates/scorm/imscp_rootv1p1p2.xsd +345 -0
- package/src/utils/templates/scorm/imsmanifest.xml +38 -0
- package/src/utils/templates/scorm/imsmd_rootv1p2p1.xsd +573 -0
@@ -1,82 +1,82 @@
|
|
1
|
-
import { Bucket, File } from "@google-cloud/storage"
|
2
|
-
|
3
|
-
type TFile = {
|
4
|
-
name: string
|
5
|
-
slug: string
|
6
|
-
hidden: boolean
|
7
|
-
}
|
8
|
-
|
9
|
-
export type Exercise = {
|
10
|
-
title: string
|
11
|
-
slug: string
|
12
|
-
graded: boolean
|
13
|
-
files: TFile[]
|
14
|
-
translations: Record<string, string>
|
15
|
-
position: number
|
16
|
-
}
|
17
|
-
|
18
|
-
export type ConfigResponse = {
|
19
|
-
config: any
|
20
|
-
exercises: Exercise[]
|
21
|
-
}
|
22
|
-
|
23
|
-
/**
|
24
|
-
* Crea la configuración y lista de ejercicios para un curso.
|
25
|
-
*
|
26
|
-
* @param bucket - Instancia de GCS Bucket donde está el curso.
|
27
|
-
* @param courseSlug - Slug del curso a procesar.
|
28
|
-
* @returns Promise con objeto { config, exercises } listo para usar.
|
29
|
-
*/
|
30
|
-
export async function buildConfig(
|
31
|
-
bucket: Bucket,
|
32
|
-
courseSlug: string
|
33
|
-
): Promise<ConfigResponse> {
|
34
|
-
const prefix = `courses/${courseSlug}/`
|
35
|
-
const [files] = await bucket.getFiles({ prefix })
|
36
|
-
|
37
|
-
// 1) Leer learn.json
|
38
|
-
const learnFile = files.find(f => f.name.endsWith("learn.json"))!
|
39
|
-
const [learnBuf] = await learnFile.download()
|
40
|
-
const learnJson = JSON.parse(learnBuf.toString())
|
41
|
-
|
42
|
-
// 2) Agrupar ejercicios
|
43
|
-
const map: Record<string, Omit<Exercise, "position">> = {}
|
44
|
-
for (const file of files) {
|
45
|
-
const parts = file.name.split("/")
|
46
|
-
if (!parts.includes("exercises")) continue
|
47
|
-
|
48
|
-
const slug = parts[parts.indexOf("exercises") + 1]
|
49
|
-
if (!map[slug]) {
|
50
|
-
map[slug] = {
|
51
|
-
title: slug,
|
52
|
-
slug,
|
53
|
-
graded: false,
|
54
|
-
files: [],
|
55
|
-
translations: {},
|
56
|
-
}
|
57
|
-
}
|
58
|
-
|
59
|
-
const fname = parts.pop()!
|
60
|
-
const m = fname.match(/^readme(?:\.([a-z]{2}))?\.md$/i)
|
61
|
-
if (m) {
|
62
|
-
const lang = m[1] || "en"
|
63
|
-
map[slug].translations[lang] = fname
|
64
|
-
} else {
|
65
|
-
map[slug].files.push({
|
66
|
-
name: fname,
|
67
|
-
slug: fname,
|
68
|
-
hidden: false,
|
69
|
-
})
|
70
|
-
}
|
71
|
-
}
|
72
|
-
|
73
|
-
const exercises = Object.values(map).map((ex, i) => ({
|
74
|
-
...ex,
|
75
|
-
position: i,
|
76
|
-
}))
|
77
|
-
|
78
|
-
return {
|
79
|
-
config: { ...learnJson },
|
80
|
-
exercises,
|
81
|
-
}
|
82
|
-
}
|
1
|
+
import { Bucket, File } from "@google-cloud/storage"
|
2
|
+
|
3
|
+
type TFile = {
|
4
|
+
name: string
|
5
|
+
slug: string
|
6
|
+
hidden: boolean
|
7
|
+
}
|
8
|
+
|
9
|
+
export type Exercise = {
|
10
|
+
title: string
|
11
|
+
slug: string
|
12
|
+
graded: boolean
|
13
|
+
files: TFile[]
|
14
|
+
translations: Record<string, string>
|
15
|
+
position: number
|
16
|
+
}
|
17
|
+
|
18
|
+
export type ConfigResponse = {
|
19
|
+
config: any
|
20
|
+
exercises: Exercise[]
|
21
|
+
}
|
22
|
+
|
23
|
+
/**
|
24
|
+
* Crea la configuración y lista de ejercicios para un curso.
|
25
|
+
*
|
26
|
+
* @param bucket - Instancia de GCS Bucket donde está el curso.
|
27
|
+
* @param courseSlug - Slug del curso a procesar.
|
28
|
+
* @returns Promise con objeto { config, exercises } listo para usar.
|
29
|
+
*/
|
30
|
+
export async function buildConfig(
|
31
|
+
bucket: Bucket,
|
32
|
+
courseSlug: string
|
33
|
+
): Promise<ConfigResponse> {
|
34
|
+
const prefix = `courses/${courseSlug}/`
|
35
|
+
const [files] = await bucket.getFiles({ prefix })
|
36
|
+
|
37
|
+
// 1) Leer learn.json
|
38
|
+
const learnFile = files.find(f => f.name.endsWith("learn.json"))!
|
39
|
+
const [learnBuf] = await learnFile.download()
|
40
|
+
const learnJson = JSON.parse(learnBuf.toString())
|
41
|
+
|
42
|
+
// 2) Agrupar ejercicios
|
43
|
+
const map: Record<string, Omit<Exercise, "position">> = {}
|
44
|
+
for (const file of files) {
|
45
|
+
const parts = file.name.split("/")
|
46
|
+
if (!parts.includes("exercises")) continue
|
47
|
+
|
48
|
+
const slug = parts[parts.indexOf("exercises") + 1]
|
49
|
+
if (!map[slug]) {
|
50
|
+
map[slug] = {
|
51
|
+
title: slug,
|
52
|
+
slug,
|
53
|
+
graded: false,
|
54
|
+
files: [],
|
55
|
+
translations: {},
|
56
|
+
}
|
57
|
+
}
|
58
|
+
|
59
|
+
const fname = parts.pop()!
|
60
|
+
const m = fname.match(/^readme(?:\.([a-z]{2}))?\.md$/i)
|
61
|
+
if (m) {
|
62
|
+
const lang = m[1] || "en"
|
63
|
+
map[slug].translations[lang] = fname
|
64
|
+
} else {
|
65
|
+
map[slug].files.push({
|
66
|
+
name: fname,
|
67
|
+
slug: fname,
|
68
|
+
hidden: false,
|
69
|
+
})
|
70
|
+
}
|
71
|
+
}
|
72
|
+
|
73
|
+
const exercises = Object.values(map).map((ex, i) => ({
|
74
|
+
...ex,
|
75
|
+
position: i,
|
76
|
+
}))
|
77
|
+
|
78
|
+
return {
|
79
|
+
config: { ...learnJson },
|
80
|
+
exercises,
|
81
|
+
}
|
82
|
+
}
|
@@ -1,34 +1,34 @@
|
|
1
|
-
import fs from "fs"
|
2
|
-
import path from "path"
|
3
|
-
import clipboardy from "clipboardy"
|
4
|
-
import { fileURLToPath } from "url"
|
5
|
-
import { dirname } from "path"
|
6
|
-
|
7
|
-
const __filename = fileURLToPath(import.meta.url)
|
8
|
-
const __dirname = dirname(__filename)
|
9
|
-
|
10
|
-
const credsPath = path.resolve(process.cwd(), "src/utils/creds.json")
|
11
|
-
|
12
|
-
if (!fs.existsSync(credsPath)) {
|
13
|
-
console.error("❌ File not found:", credsPath)
|
14
|
-
process.exit(1)
|
15
|
-
}
|
16
|
-
|
17
|
-
try {
|
18
|
-
const content = fs.readFileSync(credsPath, "utf-8")
|
19
|
-
const parsed = JSON.parse(content)
|
20
|
-
const stringified = JSON.stringify(parsed)
|
21
|
-
|
22
|
-
await clipboardy.write(stringified)
|
23
|
-
|
24
|
-
console.log("✅ JSON credentials copied to clipboard.")
|
25
|
-
console.log("👉 Paste it into your .env like this:\n")
|
26
|
-
console.log("GCP_CREDENTIALS_JSON='<PASTE_HERE>'\n")
|
27
|
-
|
28
|
-
// Eliminar el archivo creds.json después de copiar
|
29
|
-
fs.unlinkSync(credsPath)
|
30
|
-
console.log("🗑️ creds.json has been deleted to avoid propagation.")
|
31
|
-
} catch (error) {
|
32
|
-
console.error("❌ Error reading or parsing creds.json:", error.message)
|
33
|
-
process.exit(1)
|
34
|
-
}
|
1
|
+
import fs from "fs"
|
2
|
+
import path from "path"
|
3
|
+
import clipboardy from "clipboardy"
|
4
|
+
import { fileURLToPath } from "url"
|
5
|
+
import { dirname } from "path"
|
6
|
+
|
7
|
+
const __filename = fileURLToPath(import.meta.url)
|
8
|
+
const __dirname = dirname(__filename)
|
9
|
+
|
10
|
+
const credsPath = path.resolve(process.cwd(), "src/utils/creds.json")
|
11
|
+
|
12
|
+
if (!fs.existsSync(credsPath)) {
|
13
|
+
console.error("❌ File not found:", credsPath)
|
14
|
+
process.exit(1)
|
15
|
+
}
|
16
|
+
|
17
|
+
try {
|
18
|
+
const content = fs.readFileSync(credsPath, "utf-8")
|
19
|
+
const parsed = JSON.parse(content)
|
20
|
+
const stringified = JSON.stringify(parsed)
|
21
|
+
|
22
|
+
await clipboardy.write(stringified)
|
23
|
+
|
24
|
+
console.log("✅ JSON credentials copied to clipboard.")
|
25
|
+
console.log("👉 Paste it into your .env like this:\n")
|
26
|
+
console.log("GCP_CREDENTIALS_JSON='<PASTE_HERE>'\n")
|
27
|
+
|
28
|
+
// Eliminar el archivo creds.json después de copiar
|
29
|
+
fs.unlinkSync(credsPath)
|
30
|
+
console.log("🗑️ creds.json has been deleted to avoid propagation.")
|
31
|
+
} catch (error) {
|
32
|
+
console.error("❌ Error reading or parsing creds.json:", error.message)
|
33
|
+
process.exit(1)
|
34
|
+
}
|