skool-cli 1.0.2 → 1.0.3

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (3) hide show
  1. package/package.json +1 -1
  2. package/CLAUDE.md +0 -104
  3. package/planning.md +0 -135
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "skool-cli",
3
- "version": "1.0.2",
3
+ "version": "1.0.3",
4
4
  "description": "CLI and programmatic API for Skool.com automation — create lessons, posts, and manage communities via browser automation",
5
5
  "type": "module",
6
6
  "main": "dist/core/skool-client.js",
package/CLAUDE.md DELETED
@@ -1,104 +0,0 @@
1
- # CLAUDE.md - skool-cli
2
-
3
- ## Sobre este proyecto
4
-
5
- CLI, Skill y MCP Server para automatizar Skool.com. Publicado en npm como `skool-cli` (v1.0.1) y en GitHub como `unikprompt/skool-cli`.
6
-
7
- Skool no tiene API publica. Usamos Playwright para auth y la API interna `api2.skool.com` (descubierta via interceptacion de red) para operaciones de contenido.
8
-
9
- ## Links
10
-
11
- - **npm**: npmjs.com/package/skool-cli
12
- - **GitHub**: github.com/unikprompt/skool-cli
13
- - **Skool group**: skool.com/operadores-aumentados
14
-
15
- ## Estructura
16
-
17
- ```
18
- skool-cli/
19
- src/
20
- cli.ts # Entry point CLI (Commander.js)
21
- mcp/server.ts # Entry point MCP Server
22
- commands/ # 10 comandos CLI
23
- core/
24
- skool-client.ts # API publica (high-level)
25
- skool-api.ts # HTTP client + HTML-to-TipTap converter
26
- page-ops.ts # Playwright browser ops (nav, posts, auth)
27
- browser-manager.ts # Playwright lifecycle + session persistence
28
- html-generator.ts # Markdown/JSON → HTML (port de Content-Pipeline)
29
- config.ts # Constantes y env vars
30
- types.ts # TypeScript interfaces
31
- skill/SKILL.md # Claude Code skill (alternativa liviana a MCP)
32
- dist/ # Compiled JS (npm publish from here)
33
- ```
34
-
35
- ## Comandos
36
-
37
- ```bash
38
- skool login -e EMAIL -p PASSWORD # Auth (guarda cookies en ~/.skool-cli/)
39
- skool whoami -g GROUP # Verifica sesion
40
- skool create-lesson -g GROUP --course COURSE -t TITULO --markdown "contenido"
41
- skool create-lesson -g GROUP --course COURSE --folder-id ID -t TITULO -f archivo.md
42
- skool create-folder -g GROUP --course COURSE -t "Nombre Modulo"
43
- skool delete-lesson --id PAGE_ID
44
- skool list-lessons -g GROUP --course COURSE
45
- skool create-post -g GROUP -t TITULO -b "body" -c CATEGORIA
46
- skool get-posts -g GROUP
47
- skool get-categories -g GROUP
48
- skool get-members -g GROUP
49
- ```
50
-
51
- ## API Interna de Skool (api2.skool.com)
52
-
53
- | Endpoint | Metodo | Que hace |
54
- |----------|--------|----------|
55
- | `/courses` | POST | Crea pagina (`unit_type: "module"`) o folder (`unit_type: "set"`) |
56
- | `/courses/{id}` | PUT | Actualiza titulo + contenido (`desc: "[v2]" + TipTap JSON`) |
57
- | `/courses/{id}` | DELETE | Elimina pagina o folder |
58
-
59
- **Auth**: Cookies del browser (`auth_token` JWT).
60
-
61
- **IDs necesarios**: `group_id` (de assets URL), `user_id` (de JWT payload), `root_id` (interceptando POST response al crear pagina temporal).
62
-
63
- **Formato de contenido**: TipTap JSON con prefijo `[v2]`. Ejemplo:
64
- ```json
65
- "[v2][{\"type\":\"heading\",\"attrs\":{\"level\":2},\"content\":[{\"type\":\"text\",\"text\":\"Titulo\"}]},{\"type\":\"paragraph\",\"content\":[{\"type\":\"text\",\"text\":\"Contenido con \"},{\"type\":\"text\",\"text\":\"bold\",\"marks\":[{\"type\":\"bold\"}]}]}]"
66
- ```
67
-
68
- ## Build y publish
69
-
70
- ```bash
71
- npm run build # Compila TypeScript
72
- npm version patch # Bump version
73
- npm publish --access public # Publica en npm
74
- git push origin main # Push a GitHub
75
- ```
76
-
77
- **npm account**: `unikprompt` (token en `~/.npmrc`)
78
-
79
- ## Env vars
80
-
81
- | Variable | Default | Uso |
82
- |----------|---------|-----|
83
- | `SKOOL_EMAIL` | - | Email default para login |
84
- | `SKOOL_PASSWORD` | - | Password default |
85
- | `SKOOL_GROUP` | - | Group slug default |
86
- | `SKOOL_CLI_HEADLESS` | `true` | `false` para ver el browser |
87
- | `SKOOL_CLI_DATA_DIR` | `~/.skool-cli/` | Donde se guardan cookies |
88
- | `SKOOL_CLI_TIMEOUT` | `30000` | Timeout en ms |
89
-
90
- ## Decisiones tecnicas
91
-
92
- - **API directa sobre UI automation**: Intentamos 10+ veces automatizar el editor TipTap via Playwright (pencil icon click, setContent, dirty trigger). React synthetic events no responden a dispatchEvent ni a clicks de Playwright en SVGs. La solucion fue descubrir la API interna via interceptacion de red.
93
- - **Playwright solo para auth**: Login requiere browser real (CAPTCHA, WAF). Despues de auth, todas las operaciones son HTTP puro.
94
- - **TipTap JSON, no HTML**: Skool almacena contenido como `[v2]` + TipTap JSON, no HTML. El `htmlToSkoolDesc()` en skool-api.ts convierte HTML a este formato.
95
- - **Session en disco**: `~/.skool-cli/auth-state.json` persiste cookies entre invocaciones del CLI.
96
-
97
- ## Pendiente
98
-
99
- - Fix `--folder` por nombre (endpoint GET para listar items no descubierto, fallback via __NEXT_DATA__)
100
- - Limpiar dist/ de archivos debug viejos (debug.js, test-api.js siguen en dist/ de v1.0.1)
101
- - Mejorar parser de inline marks (espacios entre bold/italic y texto adyacente)
102
- - Soporte para editar lecciones existentes (PUT con ID conocido)
103
- - Soporte para mover lecciones entre folders
104
- - Tests automatizados
package/planning.md DELETED
@@ -1,135 +0,0 @@
1
- # skool-cli - Planning Document
2
-
3
- ## Vision Original (2026-04-01)
4
-
5
- Construir la primera herramienta de automatizacion para Skool.com publicable en npm. Skool no tiene API publica, lo que representa una oportunidad: ser el unico CLI/MCP funcional para Skool en todo el ecosistema.
6
-
7
- ### Modelo de negocio: Picos y Palas
8
-
9
- La herramienta gratuita en npm/GitHub alimenta el funnel de UnikPrompt:
10
- ```
11
- GitHub/npm (descubrimiento gratuito)
12
- → TikTok/Instagram (demos en accion)
13
- → Facebook Group "Claude Code en Espanol" (soporte gratuito)
14
- → Skool "Operadores Aumentados" (comunidad premium)
15
- ```
16
-
17
- ### Comparacion de enfoques
18
-
19
- Se evaluaron 3 opciones:
20
-
21
- | Criterio | MCP Server solo | CLI solo | CLI + MCP (elegido) |
22
- |----------|----------------|----------|---------------------|
23
- | Audiencia | Solo MCP clients | Todos con terminal | Todos |
24
- | Scriptable | No | Si | Si |
25
- | AI integration | Nativa (tools) | Via bash | Ambos |
26
- | Paquetes npm | 1 | 1 | 2 |
27
-
28
- **Decision: CLI + MCP + Skill** (maximo alcance, una sola base de codigo)
29
-
30
- Inspirado en opencli (github.com/jackwener/opencli, 10.6K stars) como modelo de CLI universal.
31
-
32
- ## API Interna Descubierta
33
-
34
- Skool usa `api2.skool.com` como backend. Descubierto via interceptacion de red con Playwright `page.on('request')` y `page.on('response')`.
35
-
36
- ### Endpoints
37
-
38
- | Endpoint | Metodo | Uso |
39
- |----------|--------|-----|
40
- | `/courses` | POST | Crear pagina (`unit_type: "module"`) o folder (`unit_type: "set"`) |
41
- | `/courses/{id}` | PUT | Actualizar titulo + contenido |
42
- | `/courses/{id}` | DELETE | Eliminar pagina o folder |
43
-
44
- ### Formato de contenido
45
-
46
- Skool almacena contenido como TipTap JSON con prefijo `[v2]`:
47
- ```json
48
- {
49
- "title": "Titulo",
50
- "desc": "[v2][{\"type\":\"heading\",\"attrs\":{\"level\":2},\"content\":[{\"type\":\"text\",\"text\":\"Mi Heading\"}]},{\"type\":\"paragraph\",\"content\":[{\"type\":\"text\",\"text\":\"Texto normal \"},{\"type\":\"text\",\"text\":\"bold\",\"marks\":[{\"type\":\"bold\"}]}]}]"
51
- }
52
- ```
53
-
54
- ### Autenticacion
55
-
56
- - Login via Playwright (browser headless)
57
- - Cookies persistidas en `~/.skool-cli/auth-state.json`
58
- - `user_id` extraido del JWT `auth_token` cookie (campo `user_id` del payload)
59
- - `group_id` extraido de assets URL (patron `assets.skool.com/f/{group_id}`)
60
- - `root_id` descubierto interceptando el POST response al crear pagina temporal
61
-
62
- ## Decisiones Tecnicas
63
-
64
- ### Por que API directa sobre Playwright UI automation
65
-
66
- Se intento 10+ veces automatizar el editor TipTap de Skool via Playwright:
67
- 1. `dispatchEvent(new MouseEvent(...))` no activa React synthetic events
68
- 2. Click real de Playwright en SVGs (pencil icon) es inconsistente
69
- 3. Incluso detectando "edit mode" (SAVE button), titulo y contenido no se llenaban
70
- 4. 6 workarounds de produccion necesarios (dirty trigger, overlay fix, ref volatility, SPA timing, HTML conversion, title limit)
71
-
72
- La API directa elimina toda esa complejidad: un POST crea la pagina, un PUT le pone contenido.
73
-
74
- ### Stack tecnologico
75
-
76
- - TypeScript + Commander.js (CLI)
77
- - Playwright (solo para login, una vez)
78
- - MCP SDK + Zod (MCP server)
79
- - Node.js >= 18
80
-
81
- ## Cronologia
82
-
83
- | Fecha | Hito |
84
- |-------|------|
85
- | 2026-04-01 | Inicio: evaluacion MCP vs CLI vs opencli |
86
- | 2026-04-01 | Scaffolding del proyecto, setup TypeScript + Commander |
87
- | 2026-04-01 | Login funcional, get-posts, get-categories, get-members |
88
- | 2026-04-01 | create-post funcional con categoria |
89
- | 2026-04-01/02 | 10+ intentos de UI automation para create-lesson (fallido) |
90
- | 2026-04-02 | Pivot a API-first: interceptacion de red |
91
- | 2026-04-03 | Descubrimiento de api2.skool.com/courses |
92
- | 2026-04-03 | create-lesson funcional via API (titulo + contenido) |
93
- | 2026-04-03 | Descubrimiento de folders (unit_type "set") |
94
- | 2026-04-03 | create-folder + create-lesson --folder-id funcional |
95
- | 2026-04-04 | Cleanup, delete-lesson, list-lessons |
96
- | 2026-04-04 | npm publish v1.0.0 y v1.0.1 (npmjs.com/package/skool-cli) |
97
- | 2026-04-04 | MCP server wrapper (10 tools) |
98
- | 2026-04-04 | Claude Code Skill (skill/SKILL.md) |
99
- | 2026-04-04 | GitHub repo (github.com/unikprompt/skool-cli) |
100
- | 2026-04-04 | Integracion con Content-Pipeline (reemplazo de Playwright) |
101
-
102
- ## Estado Actual (v1.0.1)
103
-
104
- ### 10 Comandos
105
-
106
- | Comando | Estado |
107
- |---------|--------|
108
- | `login` | OK |
109
- | `whoami` | OK |
110
- | `create-lesson` | OK (con --folder-id, --course, --file, --markdown) |
111
- | `create-folder` | OK |
112
- | `delete-lesson` | OK |
113
- | `list-lessons` | OK (via sidebar DOM) |
114
- | `create-post` | OK (con --category) |
115
- | `get-posts` | OK |
116
- | `get-categories` | OK |
117
- | `get-members` | OK |
118
-
119
- ### 3 Canales de distribucion
120
-
121
- | Canal | URL |
122
- |-------|-----|
123
- | npm | npmjs.com/package/skool-cli |
124
- | GitHub | github.com/unikprompt/skool-cli |
125
- | Skill | skill/SKILL.md (para Claude Code) |
126
-
127
- ## Pendiente
128
-
129
- - Fix `--folder` por nombre (endpoint GET no descubierto, fallback via __NEXT_DATA__)
130
- - Limpiar dist/ de archivos debug viejos del build anterior
131
- - Mejorar parser de inline marks (espacios entre bold/italic y texto adyacente)
132
- - Soporte para editar lecciones existentes (PUT con ID conocido)
133
- - Soporte para mover lecciones entre folders
134
- - Tests automatizados
135
- - Agregar a landing page unikprompt.com