@dockstat/outline-sync 1.1.2 → 1.1.4

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/README.md +287 -147
  2. package/dist/cli.js +14 -14
  3. package/package.json +1 -1
package/README.md CHANGED
@@ -1,184 +1,191 @@
1
1
  # outline-sync — `@dockstat/outline-sync`
2
2
 
3
3
  Sync Outline (app.getoutline.com) collections ↔ Markdown in your repo.
4
- Designed for multi-collection pipelines.
5
- Features:
6
-
7
- * Two-way sync (pull / push / timestamp-based sync)
8
- * Multi-collection support (`--collection` repeatable)
9
- * Folder-based default storage (each page → `<slug>/README.md`, children inherit folders)
10
- * Per-collection mapping file for custom paths
11
- * Config-driven: `configs/outline-sync.json`, `<collection>.config.json`, `<collection>.pages.json`
12
- * Whitespace/newline-agnostic diffs (formatting-only changes are ignored)
13
- * Uses Git commit time when available (falls back to fs mtime)
14
- * Dry-run mode, backups of overwritten files
4
+ Designed for multi-collection pipelines with intelligent conflict resolution and flexible mapping.
15
5
 
16
- ---
17
-
18
- # Install
19
-
20
- Using `bunx` (recommended):
6
+ ## ✨ Features
21
7
 
22
- ```bash
23
- bunx @dockstat/outline-sync <command> [flags]
24
- ```
25
-
26
- > Security note: prefer `OUTLINE_API_KEY=...` in CI or environment. Passing `--api-key="..."` is supported but may expose the key in process lists or shell history.
8
+ * **Bidirectional Sync**: Pull, push, or intelligent timestamp-based synchronization
9
+ * **Multi-Collection Support**: Handle multiple collections with `--collection` (repeatable)
10
+ * **Smart File Organization**: Folder-based storage (`<slug>/README.md`) with customizable mappings
11
+ * **Intelligent Conflict Resolution**: Whitespace/formatting-agnostic diffs with timestamp-based decisions
12
+ * **Flexible Configuration**: Per-collection mapping files and global configuration
13
+ * **Git Integration**: Uses Git commit timestamps when available (fallback to filesystem mtime)
14
+ * **Safe Operations**: Dry-run mode, automatic backups, and comprehensive logging
15
+ * **Interactive Setup**: Easy collection discovery and configuration
27
16
 
28
17
  ---
29
18
 
30
- # Quickstart (recommended)
19
+ ## 🚀 Quick Start
31
20
 
32
- 1. Set your API key:
21
+ ### 1. Install & Setup
33
22
 
34
23
  ```bash
35
- export OUTLINE_API_KEY="sk_..." # recommended for local / CI usage
36
- ```
37
-
38
- 2. Interactive setup (lists collections and lets you register one):
24
+ # Set your API key (recommended for security)
25
+ export OUTLINE_API_KEY="sk_..."
39
26
 
40
- ```bash
27
+ # Interactive setup - discovers and configures collections
41
28
  bunx @dockstat/outline-sync setup
42
29
  ```
43
30
 
44
- 3. Bootstrap a collection (non-interactive):
31
+ ### 2. Initialize Collections
45
32
 
46
33
  ```bash
34
+ # Bootstrap a single collection
47
35
  bunx @dockstat/outline-sync init --collection="COLLECTION_UUID"
48
- # or multiple:
49
- bunx @dockstat/outline-sync init --collection="id1" --collection="id2"
50
- ```
51
36
 
52
- This creates/updates:
53
-
54
- * `configs/outline-sync.json` — top-level config listing collections
55
- * `configs/<collection-id>.config.json` — per-collection mapping config
56
- * `configs/<collection-id>.pages.json` — assembled manifest of pages (used by sync)
57
- * `docs/...` — markdown files saved folder-based (`<slug>/README.md`)
37
+ # Or multiple collections at once
38
+ bunx @dockstat/outline-sync init \
39
+ --collection="collection-1-id" \
40
+ --collection="collection-2-id"
41
+ ```
58
42
 
59
- 4. Run a dry-run sync:
43
+ ### 3. Sync Your Content
60
44
 
61
45
  ```bash
46
+ # Dry run first (always recommended)
62
47
  bunx @dockstat/outline-sync sync --collection="COLLECTION_UUID" --dry-run
48
+
49
+ # Actual synchronization
50
+ bunx @dockstat/outline-sync sync --collection="COLLECTION_UUID"
51
+
52
+ # Or sync all configured collections
53
+ bunx @dockstat/outline-sync sync
63
54
  ```
64
55
 
65
- 5. Run an actual sync/push/pull:
56
+ ---
66
57
 
58
+ ## 📖 Installation Options
59
+
60
+ ### Using bunx (Recommended)
67
61
  ```bash
68
- bunx @dockstat/outline-sync sync --collection="COLLECTION_UUID"
69
- bunx @dockstat/outline-sync pull --collection="COLLECTION_UUID"
70
- bunx @dockstat/outline-sync push --collection="COLLECTION_UUID"
62
+ bunx @dockstat/outline-sync <command> [flags]
71
63
  ```
72
64
 
73
- You can pass `--collection` multiple times to run against several collections in one invocation:
74
-
65
+ ### Global Installation
75
66
  ```bash
76
- bunx @dockstat/outline-sync sync \
77
- --collection="id-one" --collection="id-two" \
78
- --dry-run
67
+ npm install -g @dockstat/outline-sync
68
+ outline-sync <command> [flags]
79
69
  ```
80
70
 
81
- You can also pass `--api-key="sk_..."` or `--base-url="https://custom.outline"` on the CLI; these override environment variables.
71
+ > **Security Note**: Use `OUTLINE_API_KEY=...` environment variable in CI/production. The `--api-key="..."` flag is supported but may expose keys in process lists.
82
72
 
83
73
  ---
84
74
 
85
- # CLI reference
75
+ ## 🎯 Commands Reference
86
76
 
87
77
  ```
88
78
  Usage:
89
- OUTLINE_API_KEY=... bunx @dockstat/outline-sync [command] [--collection=ID]... [--dry-run] [--api-key="..."]
79
+ OUTLINE_API_KEY=... bunx @dockstat/outline-sync [command] [options]
90
80
 
91
81
  Commands:
92
- setup - interactive setup (list & add a collection)
93
- list-collections - print the collections visible to the API key
94
- init --collection=ID - bootstrap pages.json + markdown for collection (repeatable)
95
- pull --collection=ID - pull remote changes into local files (repeatable)
96
- push --collection=ID - push local changes to remote (repeatable)
97
- sync --collection=ID - bidirectional sync (timestamp-based) (repeatable)
98
-
99
- Flags:
100
- --collection=ID Repeatable; run command for multiple collection ids
101
- --dry-run Preview actions (no writes to Outline or disk)
102
- --api-key="..." Provide Outline API key (overrides env var)
103
- --base-url="..." Custom Outline base URL (overrides default)
104
- --help, -h
82
+ setup Interactive setup: discover and configure collections
83
+ list-collections List all available collections
84
+ init --collection=ID Bootstrap collection (create configs + download content)
85
+ pull --collection=ID Pull remote changes to local files
86
+ push --collection=ID Push local changes to remote
87
+ sync --collection=ID Intelligent bidirectional sync (default)
88
+
89
+ Global Options:
90
+ --collection=ID Target collection (repeatable for multiple collections)
91
+ --dry-run Preview changes without executing
92
+ --api-key="sk_..." Outline API key (overrides OUTLINE_API_KEY env var)
93
+ --base-url="https://..." Custom Outline instance URL
94
+ --verbose Enable debug logging
95
+ --help, -h Show help
96
+
97
+ Examples:
98
+ # Interactive setup
99
+ bunx @dockstat/outline-sync setup
100
+
101
+ # Sync multiple collections with dry-run
102
+ bunx @dockstat/outline-sync sync \
103
+ --collection="id1" --collection="id2" \
104
+ --dry-run
105
+
106
+ # Push changes to custom Outline instance
107
+ bunx @dockstat/outline-sync push \
108
+ --collection="my-collection" \
109
+ --base-url="https://outline.company.com"
105
110
  ```
106
111
 
107
112
  ---
108
113
 
109
- # Config layout & examples
114
+ ## 📁 Configuration Structure
110
115
 
111
- Project layout:
116
+ The tool uses a structured configuration approach:
112
117
 
113
118
  ```
114
- configs/
115
- outline-sync.json # top-level config (collections list)
116
- <collection_id>.config.json # mapping rules + saveDir for a collection
117
- <collection_id>.pages.json # assembled pages manifest used by sync
118
- docs/ # markdown files (default saveDir)
119
+ .config/ # Configuration directory (customizable)
120
+ ├── outline-sync.json # Global configuration
121
+ ├── <collection-id>.config.json # Per-collection mapping rules
122
+ └── <collection-id>.pages.json # Generated page manifest
123
+ docs/ # Content directory (customizable)
124
+ └── collection-name/
125
+ ├── page-slug/
126
+ │ └── README.md # Page content
127
+ └── parent-page/
128
+ ├── README.md
129
+ └── child-page/
130
+ └── README.md
119
131
  ```
120
132
 
121
- ## `configs/outline-sync.json` (top-level)
133
+ ### Global Configuration (`outline-sync.json`)
122
134
 
123
135
  ```json
124
136
  {
125
137
  "collections": [
126
138
  {
127
- "id": "COLLECTION_UUID",
128
- "name": "Support Docs",
129
- "saveDir": "docs",
130
- "pagesFile": "configs/COLLECTION_UUID.pages.json",
131
- "configFile": "configs/COLLECTION_UUID.config.json"
139
+ "id": "collection-uuid-here",
140
+ "name": "Documentation",
141
+ "configDir": ".config",
142
+ "saveDir": "docs/documentation",
143
+ "pagesFile": ".config/collection-uuid-here.pages.json",
144
+ "configFile": ".config/collection-uuid-here.config.json"
132
145
  }
133
146
  ]
134
147
  }
135
148
  ```
136
149
 
137
- ## `configs/<collection_id>.config.json` (mapping rules)
150
+ ### Collection Configuration (`<collection-id>.config.json`)
138
151
 
139
152
  ```json
140
153
  {
141
- "collectionId": "COLLECTION_UUID",
142
- "saveDir": "docs",
154
+ "collectionId": "collection-uuid-here",
155
+ "saveDir": "docs/documentation",
143
156
  "mappings": [
144
157
  {
145
- "match": { "id": "doc-id-123" },
146
- "path": "guides/setup/" // directory mapping → will place doc as guides/setup/README.md
158
+ "match": { "id": "specific-doc-id" },
159
+ "path": "guides/getting-started/"
147
160
  },
148
161
  {
149
162
  "match": { "title": "API Reference" },
150
- "path": "reference/README.md" // explicit filename mapping
163
+ "path": "reference/api.md"
164
+ },
165
+ {
166
+ "match": { "title": "FAQ" },
167
+ "path": "support/"
151
168
  }
152
169
  ]
153
170
  }
154
171
  ```
155
172
 
156
- Rules:
157
-
158
- * Match by `id` (preferred) or `title` (exact match).
159
- * `path` can be:
160
-
161
- * directory-like (`guides/setup/` or any path without `.md`) → page becomes `<path>/README.md` and children inherit that directory,
162
- * explicit file (`reference/README.md`) → used verbatim (relative to project root unless you give an absolute path),
163
- * bare filename → placed under parent directory or `saveDir`.
164
-
165
- ## `configs/<collection_id>.pages.json` (generated)
166
-
167
- This is a manifest of pages used by the sync engine. Example:
173
+ ### Page Manifest (`<collection-id>.pages.json`)
174
+ *Generated automatically - contains the document tree structure*
168
175
 
169
176
  ```json
170
177
  {
171
- "collectionId": "COLLECTION_UUID",
178
+ "collectionId": "collection-uuid-here",
172
179
  "pages": [
173
180
  {
174
- "title": "Product",
175
- "file": "docs/product/README.md",
176
- "id": "doc-product-id",
181
+ "title": "Getting Started",
182
+ "file": "docs/getting-started/README.md",
183
+ "id": "doc-id-123",
177
184
  "children": [
178
185
  {
179
- "title": "Getting Started",
180
- "file": "docs/product/getting-started/README.md",
181
- "id": "doc-getting-started-id",
186
+ "title": "Installation",
187
+ "file": "docs/getting-started/installation/README.md",
188
+ "id": "doc-id-456",
182
189
  "children": []
183
190
  }
184
191
  ]
@@ -187,85 +194,218 @@ This is a manifest of pages used by the sync engine. Example:
187
194
  }
188
195
  ```
189
196
 
190
- `pages.json` is updated when new remote documents are created (IDs get written back).
191
-
192
197
  ---
193
198
 
194
- # How syncing & conflict resolution works
199
+ ## 🗂️ File Organization & Mapping
195
200
 
196
- * The tool uses Git commit timestamp (if file is tracked) as the authoritative local timestamp. If Git info isn't available, it falls back to filesystem modification time.
197
- * For `sync` (default bidirectional flow): the *newer* version wins (remote.updatedAt vs local timestamp). The tool **ignores whitespace/newline-only differences** when deciding whether content actually differs — if the only difference is formatting, no update is performed.
198
- * For `pull`: remote always wins and overwrites the local file if content differs (ignoring whitespace).
199
- * For `push`: local always wins; remote is updated if content differs (ignoring whitespace).
200
- * When writing to existing local files, a backup is created: `path/to/file.md.outline-sync.bak.<timestamp>`.
201
+ ### Default Behavior: Folder-Based Structure
201
202
 
202
- ---
203
+ Each Outline document becomes a folder with `README.md`:
204
+ - Clean URLs when served
205
+ - Natural hierarchy representation
206
+ - Child documents inherit parent folders
203
207
 
204
- # File layout style
208
+ **Example Outline Structure:**
209
+ ```
210
+ Product Documentation
211
+ ├── Getting Started
212
+ │ ├── Installation
213
+ │ └── Configuration
214
+ └── API Reference
215
+ └── Authentication
216
+ ```
205
217
 
206
- Default behavior: folder-based.
218
+ **Generated File Structure:**
219
+ ```
220
+ docs/
221
+ ├── product-documentation/README.md
222
+ ├── product-documentation/getting-started/README.md
223
+ ├── product-documentation/getting-started/installation/README.md
224
+ ├── product-documentation/getting-started/configuration/README.md
225
+ ├── product-documentation/api-reference/README.md
226
+ └── product-documentation/api-reference/authentication/README.md
227
+ ```
228
+
229
+ ### Custom Mapping Rules
207
230
 
208
- For each page:
231
+ Override default paths using mapping rules in collection configuration:
209
232
 
233
+ ```json
234
+ {
235
+ "mappings": [
236
+ {
237
+ "match": { "id": "doc-123" },
238
+ "path": "guides/setup/"
239
+ },
240
+ {
241
+ "match": { "title": "API Reference" },
242
+ "path": "reference/README.md"
243
+ }
244
+ ]
245
+ }
210
246
  ```
211
- <saveDir>/<ancestor-slug>/<page-slug>/README.md
247
+
248
+ **Mapping Rule Types:**
249
+ - **Directory mapping** (`path/to/dir/`): Document becomes `path/to/dir/README.md`
250
+ - **File mapping** (`path/to/file.md`): Document saved as specified file
251
+ - **ID matching**: Exact document ID match (preferred)
252
+ - **Title matching**: Exact title match (fallback)
253
+
254
+ ---
255
+
256
+ ## 🔄 Sync Modes & Conflict Resolution
257
+
258
+ ### Sync Modes
259
+
260
+ 1. **`sync` (default)**: Intelligent bidirectional synchronization
261
+ - Compares timestamps: Git commit time vs Outline `updatedAt`
262
+ - Newer version wins
263
+ - Ignores whitespace-only changes
264
+
265
+ 2. **`pull`**: Remote → Local (one-way)
266
+ - Outline content overwrites local files
267
+ - Creates backups of existing files
268
+
269
+ 3. **`push`**: Local → Remote (one-way)
270
+ - Local files overwrite Outline content
271
+ - No local file modifications
272
+
273
+ ### Conflict Resolution Logic
274
+
275
+ ```mermaid
276
+ flowchart TD
277
+ A[Compare Content] --> B{Content Different?}
278
+ B -->|No| C[Skip - No Changes]
279
+ B -->|Yes| D[Compare Timestamps]
280
+ D --> E{Remote Newer?}
281
+ E -->|Yes| F[Pull Remote → Local]
282
+ E -->|No| G{Local Newer?}
283
+ G -->|Yes| H[Push Local → Remote]
284
+ G -->|No| I[Skip - Same Timestamp]
212
285
  ```
213
286
 
214
- Example Outline structure:
287
+ ### Content Comparison
215
288
 
216
- * Product
289
+ The tool performs **whitespace-agnostic** comparisons:
290
+ - Ignores formatting differences (spaces, tabs, newlines)
291
+ - Focuses on actual content changes
292
+ - Prevents unnecessary sync operations
217
293
 
218
- * Getting Started
294
+ ---
219
295
 
220
- * Install
296
+ ## 🛡️ Safety Features
221
297
 
222
- Results in:
298
+ ### Backup System
299
+ - Automatic backups before overwriting files: `file.md.outline-sync.bak.timestamp`
300
+ - Preserves original content for recovery
223
301
 
302
+ ### Dry Run Mode
303
+ ```bash
304
+ # Preview all changes without executing
305
+ bunx @dockstat/outline-sync sync --dry-run
224
306
  ```
225
- docs/product/README.md
226
- docs/product/getting-started/README.md
227
- docs/product/getting-started/install/README.md
307
+
308
+ ### Comprehensive Logging
309
+ ```bash
310
+ # Enable detailed debug logging
311
+ bunx @dockstat/outline-sync sync --verbose
228
312
  ```
229
313
 
230
- You can override per-page locations in the `<collection_id>.config.json` mappings (see above).
314
+ ### Git Integration
315
+ - Uses Git commit timestamps when available
316
+ - Falls back to filesystem modification time
317
+ - Respects version control history
231
318
 
232
319
  ---
233
320
 
234
- # Advanced notes
321
+ ## 🔧 Advanced Usage
235
322
 
236
- * **Multiple collections:** pass `--collection` multiple times to operate on multiple collections in order. If `--collection` is not provided, the tool will operate on all collections listed in `configs/outline-sync.json`.
237
- * **API key sources:** first CLI `--api-key`, then `OUTLINE_API_KEY` env var. Passing `--api-key` sets `process.env.OUTLINE_API_KEY` early so imports read it properly.
238
- * **Base URL:** `--base-url` for self-hosted/alternate Outline instances.
239
- * **Dry-run:** always test with `--dry-run` before doing real `push`/`sync`.
240
- * **Mappings precedence:** `id` match wins over `title` match; explicit mapping wins and children will inherit directories if mapping points to a directory.
241
- * **Mapping templates:** not supported by default; you can use direct paths and directories via `path` in mapping rules.
323
+ ### Multiple Collections
324
+ ```bash
325
+ # Sync specific collections
326
+ bunx @dockstat/outline-sync sync \
327
+ --collection="docs-UUID1" \
328
+ --collection="guides-UUID2" \
329
+ --collection="api-ref-UUID3"
242
330
 
243
- ---
331
+ # Sync all configured collections (default when no --collection specified)
332
+ bunx @dockstat/outline-sync sync
333
+ ```
244
334
 
245
- # Troubleshooting
335
+ ### Custom Outline Instance
336
+ ```bash
337
+ # Self-hosted or enterprise Outline
338
+ bunx @dockstat/outline-sync sync \
339
+ --base-url="https://docs.company.com" \
340
+ --api-key="your-api-key"
341
+ ```
246
342
 
247
- * "Manifest not found": run `init --collection=ID` or `setup` to create `configs/<collection_id>.pages.json`.
248
- * API errors / auth: check `OUTLINE_API_KEY` (or pass `--api-key`) and `--base-url`. Ensure token has access to the collection.
249
- * Permissions / writing files: make sure the process has write permissions for `docs/` and `configs/`.
250
- * Large collections: the Outline API is paginated (the client already pages with limit=100). If you hit rate limits, re-run with fewer collections at a time or add retries in your CI.
343
+ ### CI/CD Integration
344
+ ```yaml
345
+ # GitHub Actions example
346
+ - name: Sync Documentation
347
+ env:
348
+ OUTLINE_API_KEY: ${{ secrets.OUTLINE_API_KEY }}
349
+ run: |
350
+ bunx @dockstat/outline-sync sync --collection="$COLLECTION_ID"
351
+ ```
251
352
 
252
353
  ---
253
354
 
254
- # Contributing & development
355
+ ## 🐛 Troubleshooting
255
356
 
256
- Contributions welcome!
357
+ ### Common Issues
257
358
 
258
- * Repo layout is modular (`bin/cli.ts`, `lib/*.ts`) — feel free to add features:
359
+ **"Manifest not found"**
360
+ ```bash
361
+ # Solution: Initialize the collection first
362
+ bunx @dockstat/outline-sync init --collection="COLLECTION_ID"
363
+ ```
259
364
 
260
- * mapping templates (`{{slug}}`) or glob rules
261
- * parallel collection sync with a `--concurrent` flag
262
- * a `--api-key-file` option to read secrets from a file
263
- * richer conflict resolution (merge or interactive prompts)
365
+ **API Authentication Errors**
366
+ ```bash
367
+ # Verify your API key
368
+ export OUTLINE_API_KEY="sk_your_actual_key"
369
+ bunx @dockstat/outline-sync list-collections
370
+ ```
264
371
 
265
- When developing locally:
372
+ **Permission Issues**
373
+ - Ensure write permissions for `docs/` and configuration directories
374
+ - Check that API key has access to target collections
266
375
 
376
+ **Rate Limiting**
377
+ - Tool includes automatic retry logic with exponential backoff
378
+ - For large collections, process fewer collections simultaneously
379
+
380
+ ### Debug Mode
267
381
  ```bash
268
- # run CLI against a local checkout
382
+ # Get detailed operation logs
383
+ bunx @dockstat/outline-sync sync --verbose --dry-run
384
+ ```
385
+
386
+ ---
387
+
388
+ ### Development Setup
389
+ ```bash
390
+ # Clone and setup
391
+ git clone https://github.com/Its4Nik/DockStat.git
392
+ cd packages/outline-sync
393
+
394
+ # Run locally
269
395
  bun run bin/cli.ts setup
270
- bun run bin/cli.ts init --collection="..."
396
+ bun run bin/cli.ts sync --collection="test-collection" --dry-run
271
397
  ```
398
+
399
+ ---
400
+
401
+ ## 📄 License
402
+
403
+ Mozilla Public License 2.0 (MPL-2.0)
404
+
405
+ ---
406
+
407
+ ## 🔗 Links
408
+
409
+ - [Repository](https://github.com/Its4Nik/DockStat/tree/main/packages/outline-sync)
410
+ - [Issues](https://github.com/Its4Nik/DockStat/issues)
411
+ - [Outline API Documentation](https://www.getoutline.com/developers)
package/dist/cli.js CHANGED
@@ -1,16 +1,18 @@
1
1
  #!/usr/bin/env bun
2
2
  // @bun
3
- var Gj=Object.defineProperty;var f=($,u)=>{for(var j in u)Gj($,j,{get:u[j],enumerable:!0,configurable:!0,set:(w)=>u[j]=()=>w})};var y=($,u)=>()=>($&&(u=$($=0)),u);var wj={};f(wj,{sleep:()=>Oj,saveTopConfig:()=>Nj,saveCollectionConfig:()=>Rj,loadTopConfig:()=>s,loadCollectionConfig:()=>_j,getCollectionTopConfig:()=>xj,getCollectionFilesBase:()=>g,ensureConfigDirs:()=>uj,ensureConfigDir:()=>$j,TOP_CONFIG_FILE:()=>T});import b from"node:fs/promises";import{existsSync as m}from"node:fs";import U from"node:path";async function $j($){if(!m($))await b.mkdir($,{recursive:!0})}async function uj($){let u=$.collections;for(let j of u){let w=j.configDir;if(!m(w))await b.mkdir(w,{recursive:!0})}}async function s(){if(!m(T))return null;let $=await b.readFile(T,"utf8");return JSON.parse($)}async function Nj($){await uj($),await b.writeFile(T,`${JSON.stringify($,null,2)}
4
- `,"utf8")}async function g($){let u=await s(),j=u.collections.find((J)=>J.id===$).configDir;if(!u||!u.collections)return{pagesFile:U.join(j,`${$}.pages.json`),configFile:U.join(j,`${$}.config.json`),saveDir:"docs",configDir:j};let w=u.collections.find((J)=>J.id===$);if(!w)return{pagesFile:U.join(j,`${$}.pages.json`),configFile:U.join(j,`${$}.config.json`),saveDir:"docs",configDir:".config"};return{pagesFile:w.pagesFile||U.join(j,`${$}.pages.json`),configFile:w.configFile||U.join(j,`${$}.config.json`),saveDir:w.saveDir||"docs",configDir:w.configFile||".config"}}async function _j($){let{configFile:u}=await g($);if(!m(u))return null;let j=await b.readFile(u,"utf8");return JSON.parse(j)}async function xj($){let u=await s();if(!u||!u.collections)return null;return u.collections.find((j)=>j.id===$)||null}async function Rj($){let{configFile:u,configDir:j}=await g($.collectionId);await $j(j),await b.writeFile(u,`${JSON.stringify($,null,2)}
5
- `,"utf8")}var T,Oj=($)=>new Promise((u)=>setTimeout(u,$));var Jj=y(()=>{T=U.join("outline-sync.json")});async function G($,u,j=3){let w=`${Kj}/api/${$}`;for(let J=0;J<j;J++)try{let K=await fetch(w,{method:"POST",headers:Tj,body:JSON.stringify(u)});if(K.status===429){let H=1000*(J+1);console.warn(`Rate limited. backing off ${H}ms`),await new Promise((V)=>setTimeout(V,H));continue}let A=await K.json();if(!K.ok)throw console.error(`[Outline@${Kj}/api/${$}] ${K.status} ${$} payload=`,u,"response=",A),new Error(`Outline API error ${K.status}: ${JSON.stringify(A)}`);return A}catch(K){if(J===j-1)throw K;console.warn(`Request failed (attempt ${J+1}): ${K}. Retrying...`),await new Promise((A)=>setTimeout(A,500*(J+1)))}throw new Error("outlineRequest: unreachable")}async function qj(){let $=[],u=0,j=100;while(!0){let J=(await G("collections.list",{offset:u,limit:j})).data||[];for(let K of J)$.push({id:K.id,name:K.name});if(J.length<j)break;u+=J.length}return $}async function Ej($){let u=[],j=0,w=100;while(!0){let K=(await G("documents.list",{collectionId:$,offset:j,limit:w})).data||[];for(let A of K)u.push(A);if(K.length<w)break;j+=K.length}return u}async function Hj($){return(await G("documents.info",{id:$})).data??null}async function Aj($,u,j,w){return(await G("documents.create",{title:$,text:u,collectionId:j,parentDocumentId:w||null,publish:!0})).data}async function d($,u,j){let w={id:$,text:j,publish:!0};if(u)w.title=u;return(await G("documents.update",w)).data}var Kj,Pj,Tj;var p=y(()=>{Kj=process.env.OUTLINE_BASE_URL||"https://app.getoutline.com",Pj=process.env.OUTLINE_API_KEY||"",Tj={Authorization:`Bearer ${Pj}`,"Content-Type":"application/json"}});import O from"node:fs/promises";import{existsSync as D}from"node:fs";import M from"node:path";async function Qj($){if(!D($))await O.mkdir($,{recursive:!0})}async function t($){let u=$.collections;for(let j of u){let w=j.configDir;if(!D(w))await O.mkdir(w,{recursive:!0})}}async function N(){if(!D(a))return null;let $=await O.readFile(a,"utf8");return JSON.parse($)}async function n($){await t($),await O.writeFile(a,`${JSON.stringify($,null,2)}
6
- `,"utf8")}async function S($){let u=await N(),j=u.collections.find((J)=>J.id===$).configDir;if(!u||!u.collections)return{pagesFile:M.join(j,`${$}.pages.json`),configFile:M.join(j,`${$}.config.json`),saveDir:"docs",configDir:j};let w=u.collections.find((J)=>J.id===$);if(!w)return{pagesFile:M.join(j,`${$}.pages.json`),configFile:M.join(j,`${$}.config.json`),saveDir:"docs",configDir:".config"};return{pagesFile:w.pagesFile||M.join(j,`${$}.pages.json`),configFile:w.configFile||M.join(j,`${$}.config.json`),saveDir:w.saveDir||"docs",configDir:w.configFile||".config"}}async function Vj($){let{configFile:u}=await S($);if(!D(u))return null;let j=await O.readFile(u,"utf8");return JSON.parse(j)}var a;var c=y(()=>{a=M.join("outline-sync.json")});import F from"node:fs/promises";import{existsSync as mj}from"node:fs";import Xj from"node:path";import{spawnSync as Dj}from"node:child_process";function i($){return $.replace(/\s+/g,"")}function _($){return $.toString().normalize("NFKD").replace(/\p{M}/gu,"").toLowerCase().replace(/[^a-z0-9]+/g,"-").replace(/(^-|-$)+/g,"").slice(0,120)}function Fj($){try{let u=Xj.resolve($),j=Dj("git",["log","-1","--format=%ct","--",u],{cwd:process.cwd(),encoding:"utf8"});if(j.status!==0)return null;let w=(j.stdout||"").trim();if(!w)return null;let J=Number(w);if(Number.isNaN(J))return null;return J*1000}catch{return null}}async function Zj($){let u=Fj($);if(u)return u;return(await F.stat($)).mtimeMs}async function h($,u,j=!1){if(mj($)){let w=`${$}.outline-sync.bak.${Date.now()}`;if(!j)await F.copyFile($,w);console.log(`Backed up existing file to ${w}`)}else if(!j)await F.mkdir(Xj.dirname($),{recursive:!0});if(!j)await F.writeFile($,u,"utf8");else console.log(`[dry-run] would write file ${$} (${u.length} bytes)`)}var l=()=>{};var zj={};f(zj,{question:()=>o,listCollectionsPrompt:()=>Cj,bootstrapCollection:()=>Ij});import C from"node:fs/promises";import{existsSync as hj}from"node:fs";import v from"node:path";async function Cj($){let u=await qj();if(!u.length){console.log("No collections found for this API key.");return}if(console.log("Collections:"),u.forEach((V,Q)=>console.log(`${Q+1}) ${V.id} ${V.name}`)),$.nonInteractive)return;let j=await o("Select a collection by number (or press Enter to cancel): "),w=Number(j.trim());if(!w||w<1||w>u.length){console.log("Cancelled.");return}let J=u[w-1];console.log(`You chose: ${J.name} (${J.id})`),await t(await N());let K=await N()||{collections:[]},A=K.collections.find((V)=>V.id===J.id),H=await o("Enter a base folder path for the collections config files (or press enter for default `.config`): ");if(H.trim().length<=1)H=".config";if(!A)K.collections.unshift({id:J.id,name:J.name,configDir:H,saveDir:"docs",pagesFile:v.join("configs",`${J.id}.pages.json`),configFile:v.join("configs",`${J.id}.config.json`)}),await n(K),console.log(`Added collection to ${v.join("configs","outline-sync.json")}`);else console.log("Collection already configured.")}function o($){return new Promise((u)=>{process.stdout.write($),process.stdin.resume(),process.stdin.setEncoding("utf8"),process.stdin.once("data",(j)=>{process.stdin.pause(),u(j.toString())})})}async function Ij($){let{collectionId:u,dryRun:j=!1}=$;console.log(`Bootstrapping collection ${u} (dryRun=${j})...`);let w=await Ej(u);console.log(`Fetched ${w.length} documents from Outline.`);let J=new Map;for(let q of w)J.set(q.id,{title:q.title,file:"",id:q.id,children:[],raw:q});let K=[];for(let q of J.values()){let L=q.raw||{},W=L.parentDocumentId??L.parentId??null;if(W&&J.has(W))J.get(W).children.push(q);else K.push(q)}let{saveDir:A}=await S(u);function H(q,L){let W=_(q.title||"untitled"),r=v.join(L,W),bj=v.join(r,"README.md");if(q.file=bj,q.children?.length)for(let vj of q.children)H(vj,r)}for(let q of K)H(q,A);for(let q of J.values()){let L=q.file,W=q.raw?.text??`# ${q.title}
3
+ var DJ=Object.defineProperty;var t=(J,X)=>{for(var Q in X)DJ(J,Q,{get:X[Q],enumerable:!0,configurable:!0,set:($)=>X[Q]=()=>$})};var S=(J,X)=>()=>(J&&(X=J(J=0)),X);class f{minLevel;colors;showTimestamp;name;constructor(J){let{level:X="INFO",colors:Q=!0,timestamp:$=!0,name:K}=J??{};this.minLevel=X,this.showTimestamp=$,this.name=K;let H=typeof process!=="undefined"&&!!process.env.NO_COLOR,z=typeof process!=="undefined"&&!!process.stdout&&!!process.stdout.isTTY;this.colors=Q&&!H&&z}shouldLog(J){return T[J]>=0&&T[J]>=T[this.minLevel]&&T[this.minLevel]<T.NONE}levelMeta(J){switch(J){case"DEBUG":return{tag:"DEBUG",color:E.debug,emoji:"\uD83D\uDC1B"};case"INFO":return{tag:"INFO",color:E.info,emoji:"ℹ️"};case"WARN":return{tag:"WARN",color:E.warn,emoji:"⚠️"};case"ERROR":return{tag:"ERROR",color:E.error,emoji:"❌"}}}timestamp(){if(!this.showTimestamp)return"";return new Date().toISOString()}padLevelTag(J){return J.padEnd(5," ")}colorize(J,X){if(!this.colors||!X)return J;return`${X}${J}${E.reset}`}format(J,X){let Q=this.levelMeta(J),$=this.timestamp(),K=this.name?`[${this.name}] `:"",H=`[ ${this.padLevelTag(Q.tag)} ]`,z=Q.emoji;if(this.colors){let W=this.colorize(H,Q.color),j=this.colorize($?`${$} `:"",E.gray),q=this.colorize(K,E.dim);return`${j}${q}${W} ${z} ${X}`}return`${$?`${$} `:""}${K}${H} ${z} ${X}`}write(J,X){if(!this.shouldLog(J))return;let Q=this.format(J,X);switch(J){case"DEBUG":return console.debug?console.debug(Q):console.log(Q);case"INFO":return console.info?console.info(Q):console.log(Q);case"WARN":return console.warn?console.warn(Q):console.log(Q);case"ERROR":return console.error?console.error(Q):console.log(Q)}}debug(J){this.write("DEBUG",J)}info(J){this.write("INFO",J)}warn(J){this.write("WARN",J)}error(J){this.write("ERROR",J)}child(J){return new f({level:this.minLevel,colors:this.colors,timestamp:this.showTimestamp,name:J})}setLevel(J){this.minLevel=J}}var T,E;var VJ=S(()=>{T={DEBUG:0,INFO:1,WARN:2,ERROR:3,NONE:4},E={reset:"\x1B[0m",bold:"\x1B[1m",dim:"\x1B[2m",debug:"\x1B[36m",info:"\x1B[32m",warn:"\x1B[33m",error:"\x1B[31m",gray:"\x1B[90m"}});var BJ={};t(BJ,{sleep:()=>uJ,saveTopConfig:()=>CJ,saveCollectionConfig:()=>IJ,loadTopConfig:()=>l,loadCollectionConfig:()=>hJ,getCollectionTopConfig:()=>mJ,getCollectionFilesBase:()=>o,ensureConfigDirs:()=>qJ,ensureConfigDir:()=>YJ,TOP_CONFIG_FILE:()=>R});import _ from"node:fs/promises";import{existsSync as p}from"node:fs";import O from"node:path";async function YJ(J){if(Z.debug(`Ensuring config dir: ${J}`),!p(J))Z.warn(`Dir (${J}) does not exist, creating...`),await _.mkdir(J,{recursive:!0})}async function qJ(J){let X=J.collections;console.debug(`Ensuring config dir for ${X.length} collection(s)`);for(let Q of X){let $=Q.configDir;if(Z.debug(`Ensuring config dir: ${$}`),!p($))Z.warn(`Dir (${$}) does not exist, creating...`),await _.mkdir($,{recursive:!0})}}async function l(){if(Z.debug(`Loading top config from: ${R}`),!p(R))return Z.warn("Top config file not found"),null;let J=await _.readFile(R,"utf8");return JSON.parse(J)}async function CJ(J){Z.debug("Saving top config"),await qJ(J),await _.writeFile(R,`${JSON.stringify(J,null,2)}
4
+ `,"utf8")}async function o(J){Z.debug("Getting Collection file base");let X=await l(),Q=X.collections.find((K)=>K.id===J).configDir;if(!X||!X.collections)return Z.warn(`No Top config found for ${J}, returning default`),{pagesFile:O.join(Q,`${J}.pages.json`),configFile:O.join(Q,`${J}.config.json`),saveDir:"docs",configDir:Q};let $=X.collections.find((K)=>K.id===J);if(!$)return Z.warn(`Collection config for ${J} not found, returning default paths`),{pagesFile:O.join(Q,`${J}.pages.json`),configFile:O.join(Q,`${J}.config.json`),saveDir:"docs",configDir:".config"};return{pagesFile:$.pagesFile||O.join(Q,`${J}.pages.json`),configFile:$.configFile||O.join(Q,`${J}.config.json`),saveDir:$.saveDir||"docs",configDir:$.configDir||".config"}}async function hJ(J){Z.debug(`Loading collection config for ${J}`);let{configFile:X}=await o(J);if(!p(X))return Z.warn(`Collection config file for ${J} not found`),null;let Q=await _.readFile(X,"utf8");return JSON.parse(Q)}async function mJ(J){Z.debug(`Getting Top Collection Config for ${J}`);let X=await l();if(!X||!X.collections)return Z.debug(`No Top Config found for ${J}`),null;return X.collections.find((Q)=>Q.id===J)||null}async function IJ(J){Z.debug(`Saving Collection config for ${J.collectionId}`);let{configFile:X,configDir:Q}=await o(J.collectionId);await YJ(Q),await _.writeFile(X,`${JSON.stringify(J,null,2)}
5
+ `,"utf8")}var R,uJ=(J)=>new Promise((X)=>setTimeout(X,J));var MJ=S(async()=>{await N();R=O.join("outline-sync.json")});import F from"node:fs/promises";import{existsSync as s}from"node:fs";import y from"node:path";async function jJ(J){if(Z.debug(`Ensuring config dir: ${J}`),!s(J))Z.warn(`Dir (${J}) does not exist, creating...`),await F.mkdir(J,{recursive:!0})}async function JJ(J){let X=J.collections;console.debug(`Ensuring config dir for ${X.length} collection(s)`);for(let Q of X){let $=Q.configDir;if(Z.debug(`Ensuring config dir: ${$}`),!s($))Z.warn(`Dir (${$}) does not exist, creating...`),await F.mkdir($,{recursive:!0})}}async function P(){if(Z.debug(`Loading top config from: ${d}`),!s(d))return Z.warn("Top config file not found"),null;let J=await F.readFile(d,"utf8");return JSON.parse(J)}async function QJ(J){Z.debug("Saving top config"),await JJ(J),await F.writeFile(d,`${JSON.stringify(J,null,2)}
6
+ `,"utf8")}async function x(J){Z.debug("Getting Collection file base");let X=await P(),Q=X.collections.find((K)=>K.id===J).configDir;if(!X||!X.collections)return Z.warn(`No Top config found for ${J}, returning default`),{pagesFile:y.join(Q,`${J}.pages.json`),configFile:y.join(Q,`${J}.config.json`),saveDir:"docs",configDir:Q};let $=X.collections.find((K)=>K.id===J);if(!$)return Z.warn(`Collection config for ${J} not found, returning default paths`),{pagesFile:y.join(Q,`${J}.pages.json`),configFile:y.join(Q,`${J}.config.json`),saveDir:"docs",configDir:".config"};return{pagesFile:$.pagesFile||y.join(Q,`${J}.pages.json`),configFile:$.configFile||y.join(Q,`${J}.config.json`),saveDir:$.saveDir||"docs",configDir:$.configDir||".config"}}async function UJ(J){Z.debug(`Loading collection config for ${J}`);let{configFile:X}=await x(J);if(!s(X))return Z.warn(`Collection config file for ${J} not found`),null;let Q=await F.readFile(X,"utf8");return JSON.parse(Q)}var d,e=(J)=>new Promise((X)=>setTimeout(X,J));var r=S(async()=>{await N();d=y.join("outline-sync.json")});async function D(J,X,Q=3){let $=`${AJ}/api/${J}`;for(let K=0;K<Q;K++)try{Z.debug(`Outline request: POST ${$} (attempt ${K+1})`),Z.debug(`Payload (trimmed): ${JSON.stringify(X,null,2)}`);let H=await fetch($,{method:"POST",headers:pJ,body:JSON.stringify(X)});if(H.status===429){let W=1000*(K+1);Z.warn(`Rate limited by Outline API. Backing off ${W}ms (attempt ${K+1}).`),await e(W);continue}let z;try{z=await H.json()}catch(W){throw Z.error(`Failed to parse JSON response from ${J}: ${W}`),W}if(!H.ok)throw Z.error(`[Outline@${AJ}/api/${J}] HTTP ${H.status} - payload=${JSON.stringify(X)} response=${JSON.stringify(z)}`),new Error(`Outline API error ${H.status}: ${JSON.stringify(z)}`);return Z.debug(`Outline response for ${J} (attempt ${K+1}): ${JSON.stringify(z).slice(0,200)}${JSON.stringify(z).length>200?"...":""}`),z}catch(H){if(K===Q-1)throw Z.error(`Request to Outline failed after ${Q} attempts: ${H?.message??H}`),H;let z=500*(K+1);Z.warn(`Request failed (attempt ${K+1}): ${H?.message??H}. Retrying after ${z}ms...`),await e(z)}throw new Error("outlineRequest: unreachable")}async function kJ(){let J=[],X=0,Q=100;while(!0){let K=(await D("collections.list",{offset:X,limit:Q})).data||[];for(let H of K)J.push({id:H.id,name:H.name});if(K.length<Q)break;X+=K.length}return Z.debug(`listCollectionsPaged: returned ${J.length} collections`),J}async function wJ(J){let X=[],Q=0,$=100;while(!0){let H=(await D("documents.list",{collectionId:J,offset:Q,limit:$})).data||[];for(let z of H)X.push(z);if(H.length<$)break;Q+=H.length}return Z.debug(`listDocumentsInCollection(${J}): returned ${X.length} documents`),X}async function GJ(J){let X=await D("documents.info",{id:J});return Z.debug(`fetchDocumentInfo(${J}) -> ${X?"ok":"null"}`),X.data??null}async function LJ(J,X,Q,$){let H=await D("documents.create",{title:J,text:X,collectionId:Q,parentDocumentId:$||null,publish:!0});return Z.info(`Created document "${J}" in collection ${Q} (id=${H?.id??"unknown"})`),H.data}async function XJ(J,X,Q){let $={id:J,text:Q,publish:!0};if(X)$.title=X;let K=await D("documents.update",$);return Z.info(`Updated document id=${J}${X?` title="${X}"`:""}`),K.data}var AJ,fJ,pJ;var ZJ=S(async()=>{await r();await N();AJ=process.env.OUTLINE_BASE_URL||"https://app.getoutline.com",fJ=process.env.OUTLINE_API_KEY||"",pJ={Authorization:`Bearer ${fJ}`,"Content-Type":"application/json"}});import a from"node:fs/promises";import{existsSync as dJ}from"node:fs";import u from"node:path";import{spawnSync as sJ}from"node:child_process";function $J(J){return J.replace(/\s+/g,"")}function C(J){return J.toString().normalize("NFKD").replace(/\p{M}/gu,"").toLowerCase().replace(/[^a-z0-9]+/g,"-").replace(/(^-|-$)+/g,"").slice(0,120)}function rJ(J){try{let X=u.resolve(J),Q=sJ("git",["log","-1","--format=%ct","--",X],{cwd:process.cwd(),encoding:"utf8"});if(Q.status!==0)return Z.debug(`git log returned non-zero status for ${X}: ${Q.status}`),null;let $=(Q.stdout||"").trim();if(!$)return Z.debug(`git log returned no output for ${X}`),null;let K=Number($);if(Number.isNaN(K))return Z.debug(`git log output not a number for ${X}: "${$}"`),null;return K*1000}catch(X){return Z.debug(`getGitTimestampMs error for ${J}: ${X}`),null}}async function SJ(J){let X=rJ(J);if(X)return Z.debug(`Using git timestamp for ${J}: ${X}`),X;let Q=await a.stat(J);return Z.debug(`Using FS mtime for ${J}: ${Q.mtimeMs}`),Q.mtimeMs}async function g(J,X,Q=!1){if(dJ(J)){let $=`${J}.outline-sync.bak.${Date.now()}`;if(!Q)try{await a.copyFile(J,$),Z.info(`Backed up existing file to ${$}`)}catch(K){Z.warn(`Failed to back up ${J} to ${$}: ${K}`)}else Z.debug(`[dry-run] would backup existing file ${J} -> ${$}`)}else if(!Q)try{await a.mkdir(u.dirname(J),{recursive:!0}),Z.debug(`Ensured directory ${u.dirname(J)}`)}catch($){Z.warn(`Failed to create directory ${u.dirname(J)}: ${$}`)}else Z.debug(`[dry-run] would ensure directory ${u.dirname(J)}`);if(!Q)try{await a.writeFile(J,X,"utf8"),Z.info(`Wrote file ${J} (${X.length} bytes)`)}catch($){throw Z.error(`Failed to write file ${J}: ${$}`),$}else Z.debug(`[dry-run] would write file ${J} (${X.length} bytes)`)}var HJ=S(async()=>{await N()});var EJ={};t(EJ,{question:()=>n,listCollectionsPrompt:()=>gJ,bootstrapCollection:()=>cJ});import c from"node:fs/promises";import{existsSync as aJ}from"node:fs";import b from"node:path";async function gJ(J){let X=await kJ();if(!X.length){Z.warn("No collections found for this API key.");return}if(console.info("Collections:"),X.forEach((q,V)=>console.info(`${V+1}) ${q.id} ${q.name}`)),J.nonInteractive)return;let Q=await n("Select a collection by number (or press Enter to cancel): "),$=Number(Q.trim());if(!$||$<1||$>X.length){Z.warn("Cancelled collection selection.");return}let K=X[$-1];Z.info(`You chose: ${K.name} (${K.id})`),await JJ(await P()||{collections:[]});let H=await P()||{collections:[]},z=H.collections.find((q)=>q.id===K.id),W=(await n("Enter a base folder path for the collections config files (or press enter for default `.config`): ")).replaceAll(`
7
+ `,"");if(W.trim().length<=1)W=".config",Z.debug("Using default configDir `.config`");let j=(await n("Enter a base folder path for the collections markdown files (or press enter for default `docs`): ")).replaceAll(`
8
+ `,"");if(j.trim().length<=1)j="docs",Z.debug("Using default saveDir `docs`");if(!z)H.collections.unshift({id:K.id,name:K.name,configDir:W,saveDir:j,pagesFile:b.join(W,`${K.id}.pages.json`),configFile:b.join(W,`${K.id}.config.json`)}),await QJ(H),Z.info(`Added collection to ${b.join("configs","outline-sync.json")}`);else Z.warn("Collection already configured.")}function n(J){return new Promise((X)=>{process.stdout.write(J),process.stdin.resume(),process.stdin.setEncoding("utf8"),process.stdin.once("data",(Q)=>{process.stdin.pause(),X(Q.toString())})})}async function cJ(J){let{collectionId:X,dryRun:Q=!1}=J;Z.info(`Bootstrapping collection ${X} (dryRun=${Q})...`);let $=await wJ(X);Z.info(`Fetched ${$.length} documents from Outline.`),Z.debug(`First 3 documents: ${JSON.stringify($.slice(0,3),null,2)}`);let K=new Map;for(let Y of $)K.set(Y.id,{title:Y.title,file:"",id:Y.id,children:[],raw:Y});let H=[];for(let Y of K.values()){let w=Y.raw||{},G=w.parentDocumentId??w.parentId??null;if(G&&K.has(G))K.get(G).children.push(Y);else H.push(Y)}Z.debug(`Built document tree with ${H.length} root(s)`);let{saveDir:z}=await x(X);function W(Y,w){let G=C(Y.title||"untitled"),zJ=b.join(w,G),FJ=b.join(zJ,"README.md");if(Y.file=FJ,Y.children?.length)for(let PJ of Y.children)W(PJ,zJ)}for(let Y of H)W(Y,z);for(let Y of K.values()){let w=Y.file,G=Y.raw?.text??`# ${Y.title}
7
9
 
8
- `;if(!j)await C.mkdir(v.dirname(L),{recursive:!0}),await C.writeFile(L,W,"utf8");else console.log(`[dry-run] would write ${L} (${W.length} bytes)`)}function V(q){return{title:q.title,file:q.file,id:q.id,children:(q.children||[]).map(V)}}let Q={collectionId:u,pages:K.map(V)},{pagesFile:E,configFile:Z,saveDir:k,configDir:z}=await S(u);if(await Qj(z),!j){if(await C.writeFile(E,`${JSON.stringify(Q,null,2)}
9
- `,"utf8"),!hj(Z))await C.writeFile(Z,`${JSON.stringify({collectionId:u,saveDir:k,mappings:[]},null,2)}
10
- `,"utf8");let q=await N()||{collections:[]};if(!q.collections.find((W)=>W.id===u))q.collections.unshift({id:u,saveDir:k,pagesFile:E,configFile:Z}),await n(q)}else console.log(`[dry-run] would save pages to ${E} and config to ${Z}`);console.log(`Bootstrap complete: wrote ${E} and ${Z}`)}var Bj=y(()=>{p();c();l()});var Uj={};f(Uj,{syncPage:()=>e,runSync:()=>rj,persistPagesManifest:()=>Lj,loadPagesManifest:()=>Yj,contentsEqualIgnoringWhitespace:()=>x,applyMappingsToManifest:()=>kj});import R from"node:fs/promises";import{existsSync as Wj}from"node:fs";import X from"node:path";async function Yj($){let{pagesFile:u}=await S($);if(!Wj(u))throw new Error(`${u} not found. Run init/setup to create it`);let j=await R.readFile(u,"utf8");return JSON.parse(j)}async function Lj($,u,j=!1){let{pagesFile:w}=await S($);if(j){console.log(`[dry-run] would persist manifest to ${w}`);return}await R.writeFile(w,`${JSON.stringify(u,null,2)}
11
- `,"utf8")}function kj($,u){let j=u?.mappings||[];function w(K){if(!K)return!1;if(K.endsWith("/")||K.endsWith(X.sep))return!0;return X.extname(K).toLowerCase()!==".md"}function J(K,A){let H=!1;for(let Q of j)if(Q.match?.id&&K.id===Q.match.id){let E=Q.path;if(w(E)){let Z=E.endsWith("/")?E:E;K.file=X.join(Z,"README.md")}else K.file=E;H=!0;break}if(!H){for(let Q of j)if(Q.match?.title&&K.title===Q.match.title){let E=Q.path;if(w(E)){let Z=E.endsWith("/")?E:E;K.file=X.join(Z,"README.md")}else K.file=E;H=!0;break}}if(!K.file){let Q=_(K.title||"untitled"),E=A?X.join(A,Q):X.join(u?.saveDir||"docs",Q);K.file=X.join(E,"README.md")}else if(!(X.dirname(K.file)&&X.dirname(K.file)!==".")){let E=A||u?.saveDir||"docs";K.file=X.join(E,K.file)}let V=X.dirname(K.file);if(K.children?.length)for(let Q of K.children)J(Q,V)}for(let K of $.pages)J(K,null);return $}function x($,u){return i($)===i(u)}async function e($,u,j,w,J){let K=j.file,A=X.resolve(K),H=Wj(A),V=0;if(H)try{V=await Zj(A)}catch{V=0}let Q=null;if(j.id)try{Q=await Hj(j.id)}catch(z){console.warn(`Failed to fetch remote info for ${j.title} (${j.id}): ${z}`),Q=null}let E=Q?.text??null,Z=Q?.updatedAt?new Date(Q.updatedAt).getTime():0;if(!H){if(J.mode==="pull"||J.mode==="sync"||J.mode==="push"){let z=E!=null?E:`# ${j.title}
10
+ `;if(!Q)await c.mkdir(b.dirname(w),{recursive:!0}),await c.writeFile(w,G,"utf8"),Z.debug(`Wrote file: ${w}`);else Z.debug(`[dry-run] would write ${w} (${G.length} bytes)`)}function j(Y){return{title:Y.title,file:Y.file,id:Y.id,children:(Y.children||[]).map(j)}}let q={collectionId:X,pages:H.map(j)},{pagesFile:V,configFile:B,saveDir:L,configDir:U}=await x(X);if(Z.debug(`Collection (${X}) files: pagesFile=${V}, configFile=${B}, saveDir=${L}, configDir=${U}`),await jJ(U),!Q){if(await c.writeFile(V,`${JSON.stringify(q,null,2)}
11
+ `,"utf8"),Z.info(`Saved manifest: ${V}`),!aJ(B))await c.writeFile(B,`${JSON.stringify({collectionId:X,saveDir:L,mappings:[]},null,2)}
12
+ `,"utf8"),Z.info(`Created new config: ${B}`);Z.info(`Saved config: ${B}`);let Y=await P()||{collections:[]};if(!Y.collections.find((G)=>G.id===X))Y.collections.unshift({id:X,saveDir:L,pagesFile:V,configFile:B}),await QJ(Y),Z.debug(`Updated top config with collection ${X}`)}else Z.debug(`[dry-run] would save pages to ${V} and config to ${B}`);Z.info("Bootstrap complete")}var OJ=S(async()=>{await ZJ();await r();await HJ();await N()});var bJ={};t(bJ,{syncPage:()=>KJ,runSync:()=>nJ,persistPagesManifest:()=>xJ,loadPagesManifest:()=>yJ,contentsEqualIgnoringWhitespace:()=>h,applyMappingsToManifest:()=>_J});import m from"node:fs/promises";import{existsSync as NJ}from"node:fs";import M from"node:path";async function yJ(J){let{pagesFile:X}=await x(J);if(!NJ(X))throw Z.error(`${X} not found. Run init/setup to create it`),new Error(`${X} not found. Run init/setup to create it`);let Q=await m.readFile(X,"utf8");return Z.debug(`Loaded pages manifest from ${X} (${Q.length} bytes)`),JSON.parse(Q)}async function xJ(J,X,Q=!1){let{pagesFile:$}=await x(J);if(Q){Z.debug(`[dry-run] would persist manifest to ${$}`);return}await m.writeFile($,`${JSON.stringify(X,null,2)}
13
+ `,"utf8"),Z.info(`Persisted manifest to ${$}`)}function _J(J,X){let Q=X?.mappings||[];function $(H){if(!H)return!1;if(H.endsWith("/")||H.endsWith(M.sep))return!0;return M.extname(H).toLowerCase()!==".md"}function K(H,z){let W=!1;for(let q of Q)if(q.match?.id&&H.id===q.match.id){let V=q.path;if($(V)){let B=V.endsWith("/")?V:V;H.file=M.join(B,"README.md")}else H.file=V;W=!0,Z.debug(`Mapping applied by id for "${H.title}" -> ${H.file}`);break}if(!W){for(let q of Q)if(q.match?.title&&H.title===q.match.title){let V=q.path;if($(V)){let B=V.endsWith("/")?V:V;H.file=M.join(B,"README.md")}else H.file=V;W=!0,Z.debug(`Mapping applied by title for "${H.title}" -> ${H.file}`);break}}if(!H.file){let q=C(H.title||"untitled"),V=z?M.join(z,q):M.join(X?.saveDir||"docs",q);H.file=M.join(V,"README.md"),Z.debug(`Inherited path for "${H.title}" -> ${H.file}`)}else if(!(M.dirname(H.file)&&M.dirname(H.file)!==".")){let V=z||X?.saveDir||"docs";H.file=M.join(V,H.file),Z.debug(`Normalized bare filename for "${H.title}" -> ${H.file}`)}else Z.debug(`Using mapped path for "${H.title}" -> ${H.file}`);let j=M.dirname(H.file);if(H.children?.length)for(let q of H.children)K(q,j)}for(let H of J.pages)K(H,null);return Z.debug(`Applied mappings to manifest (rules=${Q.length})`),J}function h(J,X){return $J(J)===$J(X)}async function KJ(J,X,Q,$,K){let H=Q.file,z=M.resolve(H),W=NJ(z),j=0;if(W)try{j=await SJ(z)}catch(U){Z.warn(`Failed to get local timestamp for ${z}: ${U}`),j=0}let q=null;if(Q.id)try{q=await GJ(Q.id)}catch(U){Z.warn(`Failed to fetch remote info for ${Q.title} (${Q.id}): ${U}`),q=null}let V=q?.text??null,B=q?.updatedAt?new Date(q.updatedAt).getTime():0;if(Z.debug(`syncPage("${Q.title}") localExists=${W} localTs=${j} remoteExists=${!!q} remoteUpdatedAt=${B}`),!W){if(K.mode==="pull"||K.mode==="sync"||K.mode==="push"){let U=V!=null?V:`# ${Q.title}
12
14
 
13
- `;await h(A,z,J.dryRun||!1)}}if(!j.id){let z=await R.readFile(A,"utf8");if(J.dryRun)console.log(`[dry-run][CREATE] Would create remote doc for "${j.title}" in collection ${$}`);else try{let q=await Aj(j.title,z,$,w);j.id=q?.id??j.id,console.log(`[CREATE] Created remote "${j.title}" id=${j.id}`)}catch(q){console.error(`[CREATE] Failed to create remote for ${j.title}: ${q}`)}}else{let z=await R.readFile(A,"utf8");if(J.mode==="pull")if(E!=null&&!x(z,E))console.log(`[PULL] Remote applied to local for "${j.title}"`),await h(A,E??"",J.dryRun||!1);else console.log(`[SKIP] No change (pull) for "${j.title}"`);else if(J.mode==="push")if(E==null||!x(z,E))if(J.dryRun)console.log(`[dry-run][PUSH] Would update remote "${j.title}" id=${j.id}`);else try{await d(j.id,j.title,z),console.log(`[PUSH] Updated remote "${j.title}" id=${j.id}`)}catch(q){console.error(`[PUSH] Failed to update remote for ${j.title}: ${q}`)}else console.log(`[SKIP] No change (push) for "${j.title}"`);else if(Z>V+500)if(!x(z,E??""))console.log(`[PULL] Remote newer -> overwrite local for "${j.title}"`),await h(A,E??"",J.dryRun||!1);else console.log(`[SKIP] equal after normalizing (remote newer timestamp but content same) "${j.title}"`);else if(V>Z+500)if(!x(z,E??""))if(console.log(`[PUSH] Local newer -> update remote for "${j.title}"`),J.dryRun)console.log(`[dry-run] would update remote ${j.title}`);else try{await d(j.id,j.title,z),console.log(`[PUSH] Updated remote "${j.title}" id=${j.id}`)}catch(q){console.error(`[PUSH] Failed to update remote for ${j.title}: ${q}`)}else console.log(`[SKIP] equal after normalizing (local newer timestamp but content same) "${j.title}"`);else console.log(`[SKIP] No changes for "${j.title}"`)}let k=j.id||w;if(j.children?.length)for(let z of j.children)await e($,u,z,k,J)}async function rj($){let{collectionId:u,mode:j,dryRun:w=!1}=$;console.log(`Starting ${j} for collection ${u} (dryRun=${w})`);let J=await Yj(u),K=await Vj(u)||{saveDir:"docs",mappings:[]};kj(J,K);async function A(H,V){if(!H.file){let Z=_(H.title||"untitled"),k=V?X.join(V,Z):X.join(K.saveDir||"docs",Z);H.file=X.join(k,"README.md")}else{let Z=X.dirname(H.file);if(!Z||Z==="."){let k=V||K.saveDir||"docs";H.file=X.join(k,H.file)}}let Q=X.dirname(H.file);if(!w)try{await R.mkdir(Q,{recursive:!0})}catch{}else console.log(`[dry-run] would ensure directory ${Q}`);let E=X.dirname(H.file);if(H.children?.length)for(let Z of H.children)await A(Z,E)}for(let H of J.pages)await A(H,null);for(let H of J.pages)await e(u,J,H,null,{mode:j,dryRun:w});await Lj(u,J,w),console.log("Done.")}var Mj=y(()=>{c();l();p()});var jj=process.argv.slice(2),B={},I=[];for(let $=0;$<jj.length;$++){let u=jj[$];if(u==="--help"||u==="-h")I.push("--help");else if(u.startsWith("--collection=")||u.startsWith("--collection:")){let j=u.split(/[:=]/)[1]||"";if(!B.collection)B.collection=[];B.collection.push(j)}else if(u==="--collection"){let j=jj[$+1];if(j&&!j.startsWith("--")){if(!B.collection)B.collection=[];B.collection.push(j),$++}}else if(u.startsWith("--")){let[j,w]=u.replace(/^--/,"").split("=");B[j]=w===void 0?!0:w}else I.push(u)}if(B["api-key"])process.env.OUTLINE_API_KEY=String(B["api-key"]);if(B["base-url"])process.env.OUTLINE_BASE_URL=String(B["base-url"]);if(I.includes("--help")||B.help||B.h)console.log(`
15
+ `;await g(z,U,K.dryRun||!1),Z.info(`[INIT] Ensured local file for "${Q.title}" -> ${z}`)}}if(!Q.id){let U=await m.readFile(z,"utf8");if(K.dryRun)Z.info(`[dry-run][CREATE] Would create remote doc for "${Q.title}" in collection ${J}`);else try{let Y=await LJ(Q.title,U,J,$);Q.id=Y?.id??Q.id,Z.info(`[CREATE] Created remote "${Q.title}" id=${Q.id}`)}catch(Y){Z.error(`[CREATE] Failed to create remote for ${Q.title}: ${Y}`)}}else{let U=await m.readFile(z,"utf8");if(K.mode==="pull")if(V!=null&&!h(U,V))Z.info(`[PULL] Remote applied to local for "${Q.title}"`),await g(z,V??"",K.dryRun||!1);else Z.debug(`[SKIP] No change (pull) for "${Q.title}"`);else if(K.mode==="push")if(V==null||!h(U,V))if(K.dryRun)Z.info(`[dry-run][PUSH] Would update remote "${Q.title}" id=${Q.id}`);else try{await XJ(Q.id,Q.title,U),Z.info(`[PUSH] Updated remote "${Q.title}" id=${Q.id}`)}catch(Y){Z.error(`[PUSH] Failed to update remote for ${Q.title}: ${Y}`)}else Z.debug(`[SKIP] No change (push) for "${Q.title}"`);else if(B>j+500)if(!h(U,V??""))Z.info(`[PULL] Remote newer -> overwrite local for "${Q.title}"`),await g(z,V??"",K.dryRun||!1);else Z.debug(`[SKIP] equal after normalizing (remote newer timestamp but content same) "${Q.title}"`);else if(j>B+500)if(!h(U,V??""))if(Z.info(`[PUSH] Local newer -> update remote for "${Q.title}"`),K.dryRun)Z.info(`[dry-run] would update remote ${Q.title}`);else try{await XJ(Q.id,Q.title,U),Z.info(`[PUSH] Updated remote "${Q.title}" id=${Q.id}`)}catch(Y){Z.error(`[PUSH] Failed to update remote for ${Q.title}: ${Y}`)}else Z.debug(`[SKIP] equal after normalizing (local newer timestamp but content same) "${Q.title}"`);else Z.debug(`[SKIP] No changes for "${Q.title}"`)}let L=Q.id||$;if(Q.children?.length)for(let U of Q.children)await KJ(J,X,U,L,K)}async function nJ(J){let{collectionId:X,mode:Q,dryRun:$=!1}=J;Z.info(`Starting ${Q} for collection ${X} (dryRun=${$})`);let K=await yJ(X),H=await UJ(X)||{saveDir:"docs",mappings:[]};_J(K,H);async function z(W,j){if(!W.file){let B=C(W.title||"untitled"),L=j?M.join(j,B):M.join(H.saveDir||"docs",B);W.file=M.join(L,"README.md")}else{let B=M.dirname(W.file);if(!B||B==="."){let L=j||H.saveDir||"docs";W.file=M.join(L,W.file)}}let q=M.dirname(W.file);if(!$)try{await m.mkdir(q,{recursive:!0}),Z.debug(`Ensured directory ${q}`)}catch(B){Z.warn(`Failed to ensure directory ${q}: ${B}`)}else Z.debug(`[dry-run] would ensure directory ${q}`);let V=M.dirname(W.file);if(W.children?.length)for(let B of W.children)await z(B,V)}for(let W of K.pages)await z(W,null);Z.debug("Completed path normalization for manifest");for(let W of K.pages)await KJ(X,K,W,null,{mode:Q,dryRun:$});await xJ(X,K,$),Z.info("Done.")}var vJ=S(async()=>{await r();await HJ();await ZJ();await N()});var WJ,RJ=!1,A,v,Z,iJ,TJ,tJ,lJ,k,I,i;var N=S(async()=>{VJ();WJ=process.argv.slice(2),A={},v=[];for(let J=0;J<WJ.length;J++){let X=WJ[J];if(X==="--help"||X==="-h")v.push("--help");else if(X==="--verbose")v.push("--verbose");else if(X.startsWith("--collection=")||X.startsWith("--collection:")){let Q=X.split(/[:=]/)[1]||"";if(!A.collection)A.collection=[];A.collection.push(Q)}else if(X==="--collection"){let Q=WJ[J+1];if(Q&&!Q.startsWith("--")){if(!A.collection)A.collection=[];A.collection.push(Q),J++}}else if(X.startsWith("--")){let[Q,$]=X.replace(/^--/,"").split("=");A[Q]=$===void 0?!0:$}else v.push(X)}if(A["api-key"])process.env.OUTLINE_API_KEY=String(A["api-key"]);if(A["base-url"])process.env.OUTLINE_BASE_URL=String(A["base-url"]);if(v.includes("--verbose"))RJ=!0;Z=new f({level:RJ?"DEBUG":"INFO"});if(v.includes("--help")||A.help||A.h)console.log(`
14
16
  Usage:
15
17
  OUTLINE_API_KEY=... bun run bin/cli.ts [command] [--collection=ID]... [--dry-run] [--api-key="..."]
16
18
 
@@ -31,9 +33,7 @@ Flags:
31
33
  Examples:
32
34
  OUTLINE_API_KEY=... bunx @dockstat/outline-sync --collection="id1" --collection="id2" sync --dry-run
33
35
  bun run bin/cli.ts sync --api-key="sk_xxx" --collection="id1"
34
- `),process.exit(0);var{loadTopConfig:fj}=await Promise.resolve().then(() => (Jj(),wj)),{listCollectionsPrompt:Sj,bootstrapCollection:sj}=await Promise.resolve().then(() => (Bj(),zj)),{runSync:gj}=await Promise.resolve().then(() => (Mj(),Uj)),Y=I[0]||"sync",P=Boolean(B["dry-run"]),yj=B.collection??[];try{let $=await fj()||{collections:[]},u=()=>{if(yj.length>0)return yj;if($.collections&&$.collections.length>0)return $.collections.map((j)=>j.id);return[]};if(Y==="list-collections")await Sj({dryRun:P,nonInteractive:!1}),process.exit(0);if(Y==="setup")await Sj({dryRun:P,nonInteractive:!1}),process.exit(0);if(Y==="init"){let j=u();if(!j.length)throw new Error("Init requires at least one collection. Provide --collection or run setup.");for(let w of j)console.log(`
35
- ==> bootstrapping collection ${w} (dryRun=${P})`),await sj({collectionId:w,dryRun:P});process.exit(0)}if(Y==="pull"||Y==="push"||Y==="sync"){let j=u();if(!j.length)throw new Error(`Command "${Y}" requires at least one collection id. Provide with --collection=ID or run setup.`);let w=Y==="pull"?"pull":Y==="push"?"push":"sync";for(let J of j)console.log(`
36
- ==> Running ${w} for collection ${J}`),await gj({collectionId:J,mode:w,dryRun:P});process.exit(0)}console.error("Unknown command:",Y),process.exit(1)}catch($){console.error("ERROR:",$?.message||$),process.exit(1)}
36
+ `),process.exit(0);({loadTopConfig:iJ}=await MJ().then(() => BJ)),{listCollectionsPrompt:TJ,bootstrapCollection:tJ}=await OJ().then(() => EJ),{runSync:lJ}=await vJ().then(() => bJ);Z.debug("Parsing positionals");k=v[0]||"sync",I=Boolean(A["dry-run"]),i=A.collection??[];Z.debug(`Parsed cmd=${k} DRY_RUN=${I} collectionsFromCli=${i}`);try{Z.debug("Loading top config");let J=await iJ()||{collections:[]};Z.debug(`Loaded: ${JSON.stringify(J)}`);let X=()=>{if(Z.debug("Resolving targets"),i.length>0)return Z.debug(`Found Collection from cli: ${JSON.stringify(i)}`),i;if(J.collections&&J.collections.length>0)return Z.debug(`Found collections in Top Config: ${JSON.stringify(J)}`),J.collections.map((Q)=>Q.id);return Z.warn("Couldn't resolve targets"),[]};if(k==="list-collections")Z.debug("Listing collections"),await TJ({dryRun:I,nonInteractive:!1}),process.exit(0);if(k==="setup")Z.debug("Running setup"),await TJ({dryRun:I,nonInteractive:!1}),process.exit(0);if(k==="init"){Z.debug("Running init");let Q=X();if(!Q.length)throw new Error("Init requires at least one collection. Provide --collection or run setup.");for(let $ of Q)await tJ({collectionId:$,dryRun:I});process.exit(0)}if(k==="pull"||k==="push"||k==="sync"){Z.debug("Parsing CMD (pull/push/sync)");let Q=X();if(!Q.length)throw new Error(`Command "${k}" requires at least one collection id. Provide with --collection=ID or run setup.`);let $=k==="pull"?"pull":k==="push"?"push":"sync";for(let K of Q)await lJ({collectionId:K,mode:$,dryRun:I});process.exit(0)}Z.error(`Unknown command: ${k}`),process.exit(1)}catch(J){console.error("ERROR:",J?.message||J),process.exit(1)}});await N();export{Z as logger};
37
37
 
38
- //# debugId=FD181150AA38169B64756E2164756E21
39
- //# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../lib/config.ts", "../lib/outlineApi.ts", "../lib/config.ts", "../lib/utils.ts", "../lib/init.ts", "../lib/syncEngine.ts", "../bin/cli.ts"],
  "sourcesContent": [
    "import fs from \"node:fs/promises\";\nimport { existsSync } from \"node:fs\";\nimport path from \"node:path\";\nimport type { TopConfig, TopCollectionConfig, CollectionConfig } from \"./types\";\n\nexport const TOP_CONFIG_FILE = path.join(\"outline-sync.json\");\n\nexport const sleep = (ms: number) => new Promise((r) => setTimeout(r, ms));\n\nexport async function ensureConfigDir(dir: string) {\n  if (!existsSync(dir)) {\n    await fs.mkdir(dir, { recursive: true });\n  }\n}\n\nexport async function ensureConfigDirs(cfg: TopConfig) {\n  const collections = cfg.collections;\n  for (const collection of collections) {\n    const dir = collection.configDir;\n    if (!existsSync(dir)) {\n      await fs.mkdir(dir, { recursive: true });\n    }\n  }\n}\n\nexport async function loadTopConfig(): Promise<TopConfig | null> {\n  if (!existsSync(TOP_CONFIG_FILE)) return null;\n  const raw = await fs.readFile(TOP_CONFIG_FILE, \"utf8\");\n  return JSON.parse(raw) as TopConfig;\n}\n\nexport async function saveTopConfig(cfg: TopConfig) {\n  await ensureConfigDirs(cfg);\n  await fs.writeFile(\n    TOP_CONFIG_FILE,\n    `${JSON.stringify(cfg, null, 2)}\\n`,\n    \"utf8\",\n  );\n}\n\nexport async function getCollectionFilesBase(collectionId: string): Promise<{\n  pagesFile: string;\n  configFile: string;\n  saveDir: string;\n  configDir: string;\n}> {\n  const topConfig = await loadTopConfig();\n  const CONFIG_DIR = topConfig.collections.find(\n    (collection) => collection.id === collectionId,\n  ).configDir;\n\n  // If no top config exists, use default paths\n  if (!topConfig || !topConfig.collections) {\n    return {\n      pagesFile: path.join(CONFIG_DIR, `${collectionId}.pages.json`),\n      configFile: path.join(CONFIG_DIR, `${collectionId}.config.json`),\n      saveDir: \"docs\",\n      configDir: CONFIG_DIR,\n    };\n  }\n\n  // Find the collection configuration\n  const collectionConfig = topConfig.collections.find(\n    (c) => c.id === collectionId,\n  );\n\n  // If no specific collection config found, use default paths\n  if (!collectionConfig) {\n    return {\n      pagesFile: path.join(CONFIG_DIR, `${collectionId}.pages.json`),\n      configFile: path.join(CONFIG_DIR, `${collectionId}.config.json`),\n      saveDir: \"docs\",\n      configDir: \".config\",\n    };\n  }\n\n  // Use custom paths if provided, otherwise fall back to defaults\n  return {\n    pagesFile:\n      collectionConfig.pagesFile ||\n      path.join(CONFIG_DIR, `${collectionId}.pages.json`),\n    configFile:\n      collectionConfig.configFile ||\n      path.join(CONFIG_DIR, `${collectionId}.config.json`),\n    saveDir: collectionConfig.saveDir || \"docs\",\n    configDir: collectionConfig.configFile || \".config\",\n  };\n}\n\nexport async function loadCollectionConfig(\n  collectionId: string,\n): Promise<CollectionConfig | null> {\n  const { configFile } = await getCollectionFilesBase(collectionId);\n  if (!existsSync(configFile)) return null;\n  const raw = await fs.readFile(configFile, \"utf8\");\n  return JSON.parse(raw) as CollectionConfig;\n}\n\nexport async function getCollectionTopConfig(\n  collectionId: string,\n): Promise<TopCollectionConfig | null> {\n  const topConfig = await loadTopConfig();\n  if (!topConfig || !topConfig.collections) {\n    return null;\n  }\n  return topConfig.collections.find((c) => c.id === collectionId) || null;\n}\n\nexport async function saveCollectionConfig(c: CollectionConfig) {\n  const { configFile, configDir } = await getCollectionFilesBase(\n    c.collectionId,\n  );\n  await ensureConfigDir(configDir);\n  await fs.writeFile(configFile, `${JSON.stringify(c, null, 2)}\\n`, \"utf8\");\n}\n",
    "import { sleep } from \"./config\";\nconst BASE_URL = process.env.OUTLINE_BASE_URL || \"https://app.getoutline.com\";\nconst API_KEY = process.env.OUTLINE_API_KEY || \"\";\nconst HEADERS = {\n  Authorization: `Bearer ${API_KEY}`,\n  \"Content-Type\": \"application/json\",\n};\n\nasync function outlineRequest(\n  endpoint: string,\n  body: any,\n  retries = 3,\n): Promise<any> {\n  const url = `${BASE_URL}/api/${endpoint}`;\n  for (let attempt = 0; attempt < retries; attempt++) {\n    try {\n      const res = await fetch(url, {\n        method: \"POST\",\n        headers: HEADERS,\n        body: JSON.stringify(body),\n      });\n      if (res.status === 429) {\n        const backoff = 1000 * (attempt + 1);\n        console.warn(`Rate limited. backing off ${backoff}ms`);\n        await new Promise((r) => setTimeout(r, backoff));\n        continue;\n      }\n      const json = await res.json();\n      if (!res.ok) {\n        console.error(\n          `[Outline@${BASE_URL}/api/${endpoint}] ${res.status} ${endpoint} payload=`,\n          body,\n          \"response=\",\n          json,\n        );\n        throw new Error(\n          `Outline API error ${res.status}: ${JSON.stringify(json)}`,\n        );\n      }\n      return json;\n    } catch (err) {\n      if (attempt === retries - 1) throw err;\n      console.warn(\n        `Request failed (attempt ${attempt + 1}): ${err}. Retrying...`,\n      );\n      await new Promise((r) => setTimeout(r, 500 * (attempt + 1)));\n    }\n  }\n  throw new Error(\"outlineRequest: unreachable\");\n}\n\nexport async function listCollectionsPaged(): Promise<\n  { id: string; name: string }[]\n> {\n  const out: { id: string; name: string }[] = [];\n  let offset = 0;\n  const limit = 100;\n  while (true) {\n    const json = await outlineRequest(\"collections.list\", { offset, limit });\n    const data = json.data || [];\n    for (const c of data) out.push({ id: c.id, name: c.name });\n    if (data.length < limit) break;\n    offset += data.length;\n  }\n  return out;\n}\n\nexport async function listDocumentsInCollection(\n  collectionId: string,\n): Promise<any[]> {\n  const out: any[] = [];\n  let offset = 0;\n  const limit = 100;\n  while (true) {\n    const json = await outlineRequest(\"documents.list\", {\n      collectionId,\n      offset,\n      limit,\n    });\n    const data = json.data || [];\n    for (const d of data) out.push(d);\n    if (data.length < limit) break;\n    offset += data.length;\n  }\n  return out;\n}\n\nexport async function fetchDocumentInfo(documentId: string) {\n  const json = await outlineRequest(\"documents.info\", { id: documentId });\n  return json.data ?? null;\n}\n\nexport async function createDocument(\n  title: string,\n  text: string,\n  collectionId: string,\n  parentDocumentId: string | null,\n) {\n  const payload = {\n    title,\n    text,\n    collectionId,\n    parentDocumentId: parentDocumentId || null,\n    publish: true,\n  };\n  const json = await outlineRequest(\"documents.create\", payload);\n  return json.data;\n}\n\nexport async function updateDocument(\n  id: string,\n  title: string | undefined,\n  text: string,\n) {\n  const payload: any = { id, text, publish: true };\n  if (title) payload.title = title;\n  const json = await outlineRequest(\"documents.update\", payload);\n  return json.data;\n}\n",
    "import fs from \"node:fs/promises\";\nimport { existsSync } from \"node:fs\";\nimport path from \"node:path\";\nimport type { TopConfig, TopCollectionConfig, CollectionConfig } from \"./types\";\n\nexport const TOP_CONFIG_FILE = path.join(\"outline-sync.json\");\n\nexport const sleep = (ms: number) => new Promise((r) => setTimeout(r, ms));\n\nexport async function ensureConfigDir(dir: string) {\n  if (!existsSync(dir)) {\n    await fs.mkdir(dir, { recursive: true });\n  }\n}\n\nexport async function ensureConfigDirs(cfg: TopConfig) {\n  const collections = cfg.collections;\n  for (const collection of collections) {\n    const dir = collection.configDir;\n    if (!existsSync(dir)) {\n      await fs.mkdir(dir, { recursive: true });\n    }\n  }\n}\n\nexport async function loadTopConfig(): Promise<TopConfig | null> {\n  if (!existsSync(TOP_CONFIG_FILE)) return null;\n  const raw = await fs.readFile(TOP_CONFIG_FILE, \"utf8\");\n  return JSON.parse(raw) as TopConfig;\n}\n\nexport async function saveTopConfig(cfg: TopConfig) {\n  await ensureConfigDirs(cfg);\n  await fs.writeFile(\n    TOP_CONFIG_FILE,\n    `${JSON.stringify(cfg, null, 2)}\\n`,\n    \"utf8\",\n  );\n}\n\nexport async function getCollectionFilesBase(collectionId: string): Promise<{\n  pagesFile: string;\n  configFile: string;\n  saveDir: string;\n  configDir: string;\n}> {\n  const topConfig = await loadTopConfig();\n  const CONFIG_DIR = topConfig.collections.find(\n    (collection) => collection.id === collectionId,\n  ).configDir;\n\n  // If no top config exists, use default paths\n  if (!topConfig || !topConfig.collections) {\n    return {\n      pagesFile: path.join(CONFIG_DIR, `${collectionId}.pages.json`),\n      configFile: path.join(CONFIG_DIR, `${collectionId}.config.json`),\n      saveDir: \"docs\",\n      configDir: CONFIG_DIR,\n    };\n  }\n\n  // Find the collection configuration\n  const collectionConfig = topConfig.collections.find(\n    (c) => c.id === collectionId,\n  );\n\n  // If no specific collection config found, use default paths\n  if (!collectionConfig) {\n    return {\n      pagesFile: path.join(CONFIG_DIR, `${collectionId}.pages.json`),\n      configFile: path.join(CONFIG_DIR, `${collectionId}.config.json`),\n      saveDir: \"docs\",\n      configDir: \".config\",\n    };\n  }\n\n  // Use custom paths if provided, otherwise fall back to defaults\n  return {\n    pagesFile:\n      collectionConfig.pagesFile ||\n      path.join(CONFIG_DIR, `${collectionId}.pages.json`),\n    configFile:\n      collectionConfig.configFile ||\n      path.join(CONFIG_DIR, `${collectionId}.config.json`),\n    saveDir: collectionConfig.saveDir || \"docs\",\n    configDir: collectionConfig.configFile || \".config\",\n  };\n}\n\nexport async function loadCollectionConfig(\n  collectionId: string,\n): Promise<CollectionConfig | null> {\n  const { configFile } = await getCollectionFilesBase(collectionId);\n  if (!existsSync(configFile)) return null;\n  const raw = await fs.readFile(configFile, \"utf8\");\n  return JSON.parse(raw) as CollectionConfig;\n}\n\nexport async function getCollectionTopConfig(\n  collectionId: string,\n): Promise<TopCollectionConfig | null> {\n  const topConfig = await loadTopConfig();\n  if (!topConfig || !topConfig.collections) {\n    return null;\n  }\n  return topConfig.collections.find((c) => c.id === collectionId) || null;\n}\n\nexport async function saveCollectionConfig(c: CollectionConfig) {\n  const { configFile, configDir } = await getCollectionFilesBase(\n    c.collectionId,\n  );\n  await ensureConfigDir(configDir);\n  await fs.writeFile(configFile, `${JSON.stringify(c, null, 2)}\\n`, \"utf8\");\n}\n",
    "import fs from \"node:fs/promises\";\nimport { existsSync } from \"node:fs\";\nimport path from \"node:path\";\nimport { spawnSync } from \"node:child_process\";\n\n/**\n * Remove ALL whitespace (space, tab, newline, CR) for comparison.\n * This makes diffs tolerant to formatting differences.\n */\nexport function normalizeContentIgnoreWhitespace(s: string) {\n  return s.replace(/\\s+/g, \"\");\n}\n\nexport function slugifyTitle(title: string) {\n  return title\n    .toString()\n    .normalize(\"NFKD\")\n    .replace(/\\p{M}/gu, \"\")\n    .toLowerCase()\n    .replace(/[^a-z0-9]+/g, \"-\")\n    .replace(/(^-|-$)+/g, \"\")\n    .slice(0, 120);\n}\n\nexport function getGitTimestampMs(filePath: string): number | null {\n  try {\n    const resolved = path.resolve(filePath);\n    const out = spawnSync(\n      \"git\",\n      [\"log\", \"-1\", \"--format=%ct\", \"--\", resolved],\n      {\n        cwd: process.cwd(),\n        encoding: \"utf8\",\n      },\n    );\n    if (out.status !== 0) return null;\n    const txt = (out.stdout || \"\").trim();\n    if (!txt) return null;\n    const sec = Number(txt);\n    if (Number.isNaN(sec)) return null;\n    return sec * 1000;\n  } catch {\n    return null;\n  }\n}\n\nexport async function getLocalTimestampMs(filePath: string): Promise<number> {\n  const gitTs = getGitTimestampMs(filePath);\n  if (gitTs) return gitTs;\n  const st = await fs.stat(filePath);\n  return st.mtimeMs;\n}\n\nexport async function safeWriteFile(\n  filePath: string,\n  content: string,\n  dryRun = false,\n) {\n  if (existsSync(filePath)) {\n    const bak = `${filePath}.outline-sync.bak.${Date.now()}`;\n    if (!dryRun) {\n      await fs.copyFile(filePath, bak);\n    }\n    console.log(`Backed up existing file to ${bak}`);\n  } else {\n    if (!dryRun) {\n      await fs.mkdir(path.dirname(filePath), { recursive: true });\n    }\n  }\n  if (!dryRun) {\n    await fs.writeFile(filePath, content, \"utf8\");\n  } else {\n    console.log(\n      `[dry-run] would write file ${filePath} (${content.length} bytes)`,\n    );\n  }\n}\n",
    "import { listCollectionsPaged, listDocumentsInCollection } from \"./outlineApi\";\nimport {\n  loadTopConfig,\n  saveTopConfig,\n  getCollectionFilesBase,\n  saveCollectionConfig,\n  ensureConfigDir,\n  ensureConfigDirs,\n} from \"./config\";\nimport fs from \"node:fs/promises\";\nimport { existsSync } from \"node:fs\";\nimport path from \"node:path\";\nimport { slugifyTitle } from \"./utils\";\nimport type { PageEntry, Manifest } from \"./types\";\n\n/**\n * Prompt helper (simple): prints numbered list and reads a line from stdin.\n */\nexport async function listCollectionsPrompt(opts: {\n  dryRun?: boolean;\n  nonInteractive?: boolean;\n}) {\n  const cols = await listCollectionsPaged();\n  if (!cols.length) {\n    console.log(\"No collections found for this API key.\");\n    return;\n  }\n  console.log(\"Collections:\");\n  cols.forEach((c, i) => console.log(`${i + 1}) ${c.id}\\t${c.name}`));\n  if (opts.nonInteractive) return;\n  const selection = await question(\n    \"Select a collection by number (or press Enter to cancel): \",\n  );\n  const idx = Number(selection.trim());\n  if (!idx || idx < 1 || idx > cols.length) {\n    console.log(\"Cancelled.\");\n    return;\n  }\n  const chosen = cols[idx - 1];\n  console.log(`You chose: ${chosen.name} (${chosen.id})`);\n  await ensureConfigDirs(await loadTopConfig());\n  const top = (await loadTopConfig()) || { collections: [] };\n  const exists = top.collections.find((c) => c.id === chosen.id);\n  let configDir = await question(\n    \"Enter a base folder path for the collections config files (or press enter for default `.config`): \",\n  );\n\n  if (configDir.trim().length <= 1) {\n    configDir = \".config\";\n  }\n\n  if (!exists) {\n    const defaultSaveDir = \"docs\";\n    top.collections.unshift({\n      id: chosen.id,\n      name: chosen.name,\n      configDir: configDir,\n      saveDir: defaultSaveDir,\n      pagesFile: path.join(\"configs\", `${chosen.id}.pages.json`),\n      configFile: path.join(\"configs\", `${chosen.id}.config.json`),\n    });\n    await saveTopConfig(top);\n    console.log(\n      `Added collection to ${path.join(\"configs\", \"outline-sync.json\")}`,\n    );\n  } else {\n    console.log(\"Collection already configured.\");\n  }\n}\n\n/**\n * Simple stdin line reader\n */\nexport function question(promptText: string): Promise<string> {\n  return new Promise((resolve) => {\n    process.stdout.write(promptText);\n    process.stdin.resume();\n    process.stdin.setEncoding(\"utf8\");\n    process.stdin.once(\"data\", (data) => {\n      process.stdin.pause();\n      resolve(data.toString());\n    });\n  });\n}\n\n/**\n * Bootstrap: fetch documents from collection, create pages.json and write markdown files.\n * Now: folder-based structure by inheritance. Each page gets a directory named from the slug,\n * and the page itself is saved as `README.md` inside that directory. Children become subdirectories.\n */\nexport async function bootstrapCollection(opts: {\n  collectionId: string;\n  dryRun?: boolean;\n}) {\n  const { collectionId, dryRun = false } = opts;\n  console.log(`Bootstrapping collection ${collectionId} (dryRun=${dryRun})...`);\n  const docs = await listDocumentsInCollection(collectionId);\n  console.log(`Fetched ${docs.length} documents from Outline.`);\n\n  // 1) build flat map of nodes with raw\n  const map = new Map<string, PageEntry & { raw?: any }>();\n  for (const d of docs) {\n    map.set(d.id, {\n      title: d.title,\n      file: \"\",\n      id: d.id,\n      children: [],\n      raw: d,\n    });\n  }\n\n  // 2) attach children using parentDocumentId or parentId\n  const roots: (PageEntry & { raw?: any })[] = [];\n  for (const node of map.values()) {\n    const raw = node.raw || {};\n    const parentId = raw.parentDocumentId ?? raw.parentId ?? null;\n    if (parentId && map.has(parentId)) {\n      const parent = map.get(parentId);\n      parent.children.push(node);\n    } else {\n      roots.push(node);\n    }\n  }\n\n  // 3) assign folder-based file paths recursively\n  // pattern: <saveDir>/<ancestor-slug-1>/<ancestor-slug-2>/<this-slug>/README.md\n  // Get the configured saveDir\n  const { saveDir } = await getCollectionFilesBase(collectionId);\n\n  function assignPaths(node: any, parentDir: string) {\n    const slug = slugifyTitle(node.title || \"untitled\");\n    const dir = path.join(parentDir, slug);\n    const filePath = path.join(dir, \"README.md\");\n    node.file = filePath;\n    if (node.children?.length) {\n      for (const c of node.children) {\n        assignPaths(c, dir);\n      }\n    }\n  }\n\n  for (const r of roots) {\n    assignPaths(r, saveDir);\n  }\n\n  // 4) write files to disk\n  for (const n of map.values()) {\n    const filePath = n.file;\n    const content = n.raw?.text ?? `# ${n.title}\\n\\n`;\n    if (!dryRun) {\n      await fs.mkdir(path.dirname(filePath), { recursive: true });\n      await fs.writeFile(filePath, content, \"utf8\");\n    } else {\n      console.log(\n        `[dry-run] would write ${filePath} (${content.length} bytes)`,\n      );\n    }\n  }\n\n  // 5) strip raw and build manifest\n  function strip(n: any): PageEntry {\n    return {\n      title: n.title,\n      file: n.file,\n      id: n.id,\n      children: (n.children || []).map(strip),\n    };\n  }\n  const manifest: Manifest = { collectionId, pages: roots.map(strip) };\n\n  // save manifest and config files\n  const {\n    pagesFile,\n    configFile,\n    saveDir: configuredSaveDir,\n    configDir,\n  } = await getCollectionFilesBase(collectionId);\n  await ensureConfigDir(configDir);\n  if (!dryRun) {\n    await fs.writeFile(\n      pagesFile,\n      `${JSON.stringify(manifest, null, 2)}\\n`,\n      \"utf8\",\n    );\n    if (!existsSync(configFile)) {\n      await fs.writeFile(\n        configFile,\n        `${JSON.stringify({ collectionId, saveDir: configuredSaveDir, mappings: [] }, null, 2)}\\n`,\n        \"utf8\",\n      );\n    }\n    const top = (await loadTopConfig()) || { collections: [] };\n    const existsTop = top.collections.find((c) => c.id === collectionId);\n    if (!existsTop) {\n      top.collections.unshift({\n        id: collectionId,\n        saveDir: configuredSaveDir,\n        pagesFile,\n        configFile,\n      });\n      await saveTopConfig(top);\n    }\n  } else {\n    console.log(\n      `[dry-run] would save pages to ${pagesFile} and config to ${configFile}`,\n    );\n  }\n  console.log(`Bootstrap complete: wrote ${pagesFile} and ${configFile}`);\n}\n",
    "import fs from \"node:fs/promises\";\nimport { existsSync } from \"node:fs\";\nimport path from \"node:path\";\nimport {\n  loadTopConfig,\n  getCollectionFilesBase,\n  loadCollectionConfig,\n} from \"./config\";\nimport {\n  normalizeContentIgnoreWhitespace,\n  getLocalTimestampMs,\n  safeWriteFile,\n  slugifyTitle,\n} from \"./utils\";\nimport {\n  fetchDocumentInfo,\n  updateDocument,\n  createDocument,\n} from \"./outlineApi\";\nimport type { Manifest, PageEntry } from \"./types\";\n\n/**\n * Load pages.json for a collection (if missing, error/ask to init)\n */\nexport async function loadPagesManifest(\n  collectionId: string,\n): Promise<Manifest> {\n  const { pagesFile } = await getCollectionFilesBase(collectionId);\n  if (!existsSync(pagesFile)) {\n    throw new Error(`${pagesFile} not found. Run init/setup to create it`);\n  }\n  const raw = await fs.readFile(pagesFile, \"utf8\");\n  return JSON.parse(raw) as Manifest;\n}\n\nexport async function persistPagesManifest(\n  collectionId: string,\n  manifest: Manifest,\n  dryRun = false,\n) {\n  const { pagesFile } = await getCollectionFilesBase(collectionId);\n  if (dryRun) {\n    console.log(`[dry-run] would persist manifest to ${pagesFile}`);\n    return;\n  }\n  await fs.writeFile(\n    pagesFile,\n    `${JSON.stringify(manifest, null, 2)}\\n`,\n    \"utf8\",\n  );\n}\n\n/**\n * Resolve destination file path for a document using collection config mappings.\n *\n * Rules:\n * - mapping with id exact match wins\n * - mapping with title exact match next\n * - mapping.path may be:\n *    - a file path (ending with `.md`) -> set node.file to that path\n *    - a directory path (ends with `/` or no extension) -> place node in that dir as README.md\n * - if no mapping, fall back to inherited folder approach:\n *    parentDir + slug/README.md\n */\nexport function applyMappingsToManifest(\n  manifest: Manifest,\n  collectionConfig: any,\n) {\n  const rules = (collectionConfig?.mappings || []) as {\n    match: any;\n    path: string;\n  }[];\n\n  function isDirPath(p: string) {\n    // treat trailing slash as directory OR lack of .md extension as directory\n    if (!p) return false;\n    if (p.endsWith(\"/\") || p.endsWith(path.sep)) return true;\n    return path.extname(p).toLowerCase() !== \".md\";\n  }\n\n  function applyToNode(node: any, parentDir: string | null) {\n    // try id match then title match\n    let matched = false;\n    for (const r of rules) {\n      if (r.match?.id && node.id === r.match.id) {\n        const p = r.path;\n        if (isDirPath(p)) {\n          const dir = p.endsWith(\"/\") ? p : p;\n          node.file = path.join(dir, \"README.md\");\n        } else {\n          node.file = p;\n        }\n        matched = true;\n        break;\n      }\n    }\n    if (!matched) {\n      for (const r of rules) {\n        if (r.match?.title && node.title === r.match.title) {\n          const p = r.path;\n          if (isDirPath(p)) {\n            const dir = p.endsWith(\"/\") ? p : p;\n            node.file = path.join(dir, \"README.md\");\n          } else {\n            node.file = p;\n          }\n          matched = true;\n          break;\n        }\n      }\n    }\n\n    // if still not assigned, inherit from parentDir by using slug -> README.md\n    if (!node.file) {\n      const slug = slugifyTitle(node.title || \"untitled\");\n      const dir = parentDir\n        ? path.join(parentDir, slug)\n        : path.join(collectionConfig?.saveDir || \"docs\", slug);\n      node.file = path.join(dir, \"README.md\");\n    } else {\n      // If node.file is a bare filename (no directory) => put under parentDir or saveDir\n      const hasDir = path.dirname(node.file) && path.dirname(node.file) !== \".\";\n      if (!hasDir) {\n        const baseDir = parentDir || collectionConfig?.saveDir || \"docs\";\n        node.file = path.join(baseDir, node.file);\n      } else {\n        // if mapping provided a directory-like path (no .md) we already handled; otherwise path may be relative, keep as-is\n      }\n    }\n\n    // compute this node's dir to pass to children (directory containing README.md)\n    const nodeDir = path.dirname(node.file);\n\n    if (node.children?.length) {\n      for (const c of node.children) {\n        applyToNode(c, nodeDir);\n      }\n    }\n  }\n\n  for (const p of manifest.pages) {\n    applyToNode(p as any, null);\n  }\n  return manifest;\n}\n\n/**\n * Compare content ignoring whitespace/newlines\n */\nexport function contentsEqualIgnoringWhitespace(a: string, b: string) {\n  return (\n    normalizeContentIgnoreWhitespace(a) === normalizeContentIgnoreWhitespace(b)\n  );\n}\n\n/**\n * Decide whether to pull or push or skip for one page.\n */\nexport async function syncPage(\n  collectionId: string,\n  manifest: Manifest,\n  page: PageEntry,\n  parentId: string | null,\n  opts: { mode: \"pull\" | \"push\" | \"sync\"; dryRun?: boolean },\n) {\n  const filePath = page.file;\n  const absPath = path.resolve(filePath);\n  const fileExists = existsSync(absPath);\n\n  let localTs = 0;\n  if (fileExists) {\n    try {\n      localTs = await getLocalTimestampMs(absPath);\n    } catch {\n      localTs = 0;\n    }\n  }\n\n  let remoteDoc: any = null;\n  if (page.id) {\n    try {\n      remoteDoc = await fetchDocumentInfo(page.id);\n    } catch (err) {\n      console.warn(\n        `Failed to fetch remote info for ${page.title} (${page.id}): ${err}`,\n      );\n      remoteDoc = null;\n    }\n  }\n\n  const remoteText = remoteDoc?.text ?? null;\n  const remoteUpdatedAt = remoteDoc?.updatedAt\n    ? new Date(remoteDoc.updatedAt).getTime()\n    : 0;\n\n  // ensure local file exists when needed - create parent dirs\n  if (!fileExists) {\n    if (opts.mode === \"pull\" || opts.mode === \"sync\" || opts.mode === \"push\") {\n      const dataToWrite =\n        remoteText != null ? remoteText : `# ${page.title}\\n\\n`;\n      await safeWriteFile(absPath, dataToWrite, opts.dryRun || false);\n    }\n  }\n\n  // create remote if missing\n  if (!page.id) {\n    const localContent = await fs.readFile(absPath, \"utf8\");\n    if (opts.dryRun) {\n      console.log(\n        `[dry-run][CREATE] Would create remote doc for \"${page.title}\" in collection ${collectionId}`,\n      );\n    } else {\n      try {\n        const created = await createDocument(\n          page.title,\n          localContent,\n          collectionId,\n          parentId,\n        );\n        page.id = created?.id ?? page.id;\n        console.log(`[CREATE] Created remote \"${page.title}\" id=${page.id}`);\n      } catch (err) {\n        console.error(\n          `[CREATE] Failed to create remote for ${page.title}: ${err}`,\n        );\n      }\n    }\n  } else {\n    const localContent = await fs.readFile(absPath, \"utf8\");\n    if (opts.mode === \"pull\") {\n      if (\n        remoteText != null &&\n        !contentsEqualIgnoringWhitespace(localContent, remoteText)\n      ) {\n        console.log(`[PULL] Remote applied to local for \"${page.title}\"`);\n        await safeWriteFile(absPath, remoteText ?? \"\", opts.dryRun || false);\n      } else {\n        console.log(`[SKIP] No change (pull) for \"${page.title}\"`);\n      }\n    } else if (opts.mode === \"push\") {\n      if (\n        remoteText == null ||\n        !contentsEqualIgnoringWhitespace(localContent, remoteText)\n      ) {\n        if (opts.dryRun) {\n          console.log(\n            `[dry-run][PUSH] Would update remote \"${page.title}\" id=${page.id}`,\n          );\n        } else {\n          try {\n            await updateDocument(page.id, page.title, localContent);\n            console.log(`[PUSH] Updated remote \"${page.title}\" id=${page.id}`);\n          } catch (err) {\n            console.error(\n              `[PUSH] Failed to update remote for ${page.title}: ${err}`,\n            );\n          }\n        }\n      } else {\n        console.log(`[SKIP] No change (push) for \"${page.title}\"`);\n      }\n    } else {\n      // sync: timestamp-based but skip if only whitespace differs\n      if (remoteUpdatedAt > localTs + 500) {\n        if (!contentsEqualIgnoringWhitespace(localContent, remoteText ?? \"\")) {\n          console.log(\n            `[PULL] Remote newer -> overwrite local for \"${page.title}\"`,\n          );\n          await safeWriteFile(absPath, remoteText ?? \"\", opts.dryRun || false);\n        } else {\n          console.log(\n            `[SKIP] equal after normalizing (remote newer timestamp but content same) \"${page.title}\"`,\n          );\n        }\n      } else if (localTs > remoteUpdatedAt + 500) {\n        if (!contentsEqualIgnoringWhitespace(localContent, remoteText ?? \"\")) {\n          console.log(\n            `[PUSH] Local newer -> update remote for \"${page.title}\"`,\n          );\n          if (opts.dryRun) {\n            console.log(`[dry-run] would update remote ${page.title}`);\n          } else {\n            try {\n              await updateDocument(page.id, page.title, localContent);\n              console.log(\n                `[PUSH] Updated remote \"${page.title}\" id=${page.id}`,\n              );\n            } catch (err) {\n              console.error(\n                `[PUSH] Failed to update remote for ${page.title}: ${err}`,\n              );\n            }\n          }\n        } else {\n          console.log(\n            `[SKIP] equal after normalizing (local newer timestamp but content same) \"${page.title}\"`,\n          );\n        }\n      } else {\n        console.log(`[SKIP] No changes for \"${page.title}\"`);\n      }\n    }\n  }\n\n  // recurse through children\n  const nextParentId = page.id || parentId;\n  if (page.children?.length) {\n    for (const child of page.children) {\n      await syncPage(collectionId, manifest, child, nextParentId, opts);\n    }\n  }\n}\n\n/**\n * Run sync for a collection\n */\nexport async function runSync(opts: {\n  collectionId: string;\n  mode: \"pull\" | \"push\" | \"sync\";\n  dryRun?: boolean;\n}) {\n  const { collectionId, mode, dryRun = false } = opts;\n  console.log(\n    `Starting ${mode} for collection ${collectionId} (dryRun=${dryRun})`,\n  );\n  const pagesManifest = await loadPagesManifest(collectionId);\n  const collCfg = (await loadCollectionConfig(collectionId)) || {\n    saveDir: \"docs\",\n    mappings: [],\n  };\n\n  // apply mappings (this will set folder-based `file` fields)\n  applyMappingsToManifest(pagesManifest, collCfg);\n\n  // ensure local files/folders exist and normalize any relative filenames\n  async function normalizePaths(node: any, parentDir: string | null) {\n    // if node.file is absent, apply inheritance (applyMappingsToManifest should have set it)\n    if (!node.file) {\n      const slug = slugifyTitle(node.title || \"untitled\");\n      const dir = parentDir\n        ? path.join(parentDir, slug)\n        : path.join(collCfg.saveDir || \"docs\", slug);\n      node.file = path.join(dir, \"README.md\");\n    } else {\n      // If node.file is bare filename without dir -> put into parentDir or saveDir\n      const dirnameOfFile = path.dirname(node.file);\n      if (!dirnameOfFile || dirnameOfFile === \".\") {\n        const baseDir = parentDir || collCfg.saveDir || \"docs\";\n        node.file = path.join(baseDir, node.file);\n      }\n    }\n\n    // create directory if needed (not writing file contents here, just ensuring structure)\n    const dirToMake = path.dirname(node.file);\n    if (!dryRun) {\n      try {\n        await fs.mkdir(dirToMake, { recursive: true });\n      } catch {\n        // ignore\n      }\n    } else {\n      console.log(`[dry-run] would ensure directory ${dirToMake}`);\n    }\n\n    const nodeDir = path.dirname(node.file);\n    if (node.children?.length) {\n      for (const c of node.children) {\n        await normalizePaths(c, nodeDir);\n      }\n    }\n  }\n\n  // normalize for each root\n  for (const root of pagesManifest.pages) {\n    await normalizePaths(root as any, null);\n  }\n\n  // run sync recursion\n  for (const p of pagesManifest.pages) {\n    await syncPage(collectionId, pagesManifest, p, null, { mode, dryRun });\n  }\n\n  // persist manifest (write any created ids back)\n  await persistPagesManifest(collectionId, pagesManifest, dryRun);\n  console.log(\"Done.\");\n}\n",
    "#!/usr/bin/env bun\n\n/* bin/cli.ts\n   - Parse CLI flags first (so --api-key is applied before modules import env)\n   - Support repeatable --collection flags\n   - Then dynamically import the rest of the app\n*/\n\nconst rawArgs = process.argv.slice(2);\n\n// parse flags (repeatable --collection)\nconst flags: Record<string, string | boolean | string[]> = {};\nconst positionals: string[] = [];\nfor (let i = 0; i < rawArgs.length; i++) {\n  const a = rawArgs[i];\n  if (a === \"--help\" || a === \"-h\") {\n    positionals.push(\"--help\");\n  } else if (a.startsWith(\"--collection=\") || a.startsWith(\"--collection:\")) {\n    const val = a.split(/[:=]/)[1] || \"\";\n    if (!flags.collection) flags.collection = [];\n    (flags.collection as string[]).push(val);\n  } else if (a === \"--collection\") {\n    // support `--collection <value>`\n    const val = rawArgs[i + 1];\n    if (val && !val.startsWith(\"--\")) {\n      if (!flags.collection) flags.collection = [];\n      (flags.collection as string[]).push(val);\n      i++; // consume next arg\n    }\n  } else if (a.startsWith(\"--\")) {\n    const [k, v] = a.replace(/^--/, \"\").split(\"=\");\n    flags[k] = v === undefined ? true : v;\n  } else {\n    positionals.push(a);\n  }\n}\n\n// If user passed --api-key or --base-url, set them immediately so dynamic imports see them.\nif (flags[\"api-key\"]) {\n  // do NOT log the key to avoid accidental leakage in logs\n  process.env.OUTLINE_API_KEY = String(flags[\"api-key\"]);\n}\nif (flags[\"base-url\"]) {\n  process.env.OUTLINE_BASE_URL = String(flags[\"base-url\"]);\n}\n\nif (positionals.includes(\"--help\") || flags.help || flags.h) {\n  console.log(`\nUsage:\n  OUTLINE_API_KEY=... bun run bin/cli.ts [command] [--collection=ID]... [--dry-run] [--api-key=\"...\"]\n\nCommands:\n  setup                    - interactive setup: list collections, choose one\n  list-collections         - print collections\n  init --collection=ID     - bootstrap pages.json + markdown (repeatable)\n  pull --collection=ID     - pull remote changes (repeatable)\n  push --collection=ID     - push local changes (repeatable)\n  sync --collection=ID     - bidirectional sync (repeatable)\n\nFlags:\n  --collection=ID          Repeatable; run command against multiple collections\n  --api-key=\"...\"          Provide Outline API key (overrides env var)\n  --base-url=\"...\"         Provide Outline base URL (overrides env var)\n  --dry-run                Preview only\n  --help, -h\nExamples:\n  OUTLINE_API_KEY=... bunx @dockstat/outline-sync --collection=\"id1\" --collection=\"id2\" sync --dry-run\n  bun run bin/cli.ts sync --api-key=\"sk_xxx\" --collection=\"id1\"\n`);\n  process.exit(0);\n}\n\nconst { loadTopConfig } = await import(\"../lib/config\");\nconst { listCollectionsPrompt, bootstrapCollection } = await import(\n  \"../lib/init\"\n);\nconst { runSync } = await import(\"../lib/syncEngine\");\n\nconst cmd = positionals[0] || \"sync\";\nconst DRY_RUN = Boolean(flags[\"dry-run\"]);\nconst collectionsFromCli = (flags.collection as string[] | undefined) ?? [];\n\ntry {\n  const topConfig = (await loadTopConfig()) || { collections: [] };\n\n  const resolveTargets = (): string[] => {\n    if (collectionsFromCli.length > 0) return collectionsFromCli;\n    if (topConfig.collections && topConfig.collections.length > 0) {\n      return topConfig.collections.map((c) => c.id);\n    }\n    return [];\n  };\n\n  if (cmd === \"list-collections\") {\n    await listCollectionsPrompt({ dryRun: DRY_RUN, nonInteractive: false });\n    process.exit(0);\n  }\n\n  if (cmd === \"setup\") {\n    await listCollectionsPrompt({ dryRun: DRY_RUN, nonInteractive: false });\n    process.exit(0);\n  }\n\n  if (cmd === \"init\") {\n    const targets = resolveTargets();\n    if (!targets.length)\n      throw new Error(\n        \"Init requires at least one collection. Provide --collection or run setup.\",\n      );\n    for (const collectionId of targets) {\n      console.log(\n        `\\n==> bootstrapping collection ${collectionId} (dryRun=${DRY_RUN})`,\n      );\n      await bootstrapCollection({ collectionId, dryRun: DRY_RUN });\n    }\n    process.exit(0);\n  }\n\n  if (cmd === \"pull\" || cmd === \"push\" || cmd === \"sync\") {\n    const targets = resolveTargets();\n    if (!targets.length)\n      throw new Error(\n        `Command \"${cmd}\" requires at least one collection id. Provide with --collection=ID or run setup.`,\n      );\n    const mode = cmd === \"pull\" ? \"pull\" : cmd === \"push\" ? \"push\" : \"sync\";\n    for (const collectionId of targets) {\n      console.log(`\\n==> Running ${mode} for collection ${collectionId}`);\n      await runSync({ collectionId, mode: mode as any, dryRun: DRY_RUN });\n    }\n    process.exit(0);\n  }\n\n  console.error(\"Unknown command:\", cmd);\n  process.exit(1);\n} catch (err: any) {\n  console.error(\"ERROR:\", err?.message || err);\n  process.exit(1);\n}\n\nexport {};\n"
  ],
  "mappings": "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",
  "debugId": "FD181150AA38169B64756E2164756E21",
  "names": []
}
38
+ //# debugId=C9F1296C2DE7230364756E2164756E21
39
+ //# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../lib/logger.ts", "../lib/config.ts", "../lib/config.ts", "../lib/outlineApi.ts", "../lib/utils.ts", "../lib/init.ts", "../lib/syncEngine.ts", "../bin/cli.ts"],
  "sourcesContent": [
    "export type LogLevel = \"DEBUG\" | \"INFO\" | \"WARN\" | \"ERROR\" | \"NONE\";\n\nconst LEVEL_ORDER: Record<LogLevel, number> = {\n  DEBUG: 0,\n  INFO: 1,\n  WARN: 2,\n  ERROR: 3,\n  NONE: 4,\n};\n\nconst ANSI = {\n  reset: \"\\x1b[0m\",\n  bold: \"\\x1b[1m\",\n  dim: \"\\x1b[2m\",\n  debug: \"\\x1b[36m\", // cyan\n  info: \"\\x1b[32m\", // green\n  warn: \"\\x1b[33m\", // yellow\n  error: \"\\x1b[31m\", // red\n  gray: \"\\x1b[90m\",\n};\n\nexport class Logger {\n  private minLevel: LogLevel;\n  private colors: boolean;\n  private showTimestamp: boolean;\n  private name?: string;\n\n  /**\n   * new Logger({ level: 'DEBUG', colors: true, timestamp: true, name: 'api' })\n   */\n  constructor(opts?: {\n    level?: LogLevel;\n    colors?: boolean; // allow forcing on/off\n    timestamp?: boolean;\n    name?: string; // optional logger name e.g. \"api\"\n  }) {\n    const {\n      level = \"INFO\",\n      colors = true,\n      timestamp = true,\n      name,\n    } = opts ?? {};\n\n    this.minLevel = level;\n    this.showTimestamp = timestamp;\n    this.name = name;\n\n    // auto-detect if we should colorize: respect NO_COLOR and non-tty (CI)\n    const envNoColor = typeof process !== \"undefined\" && !!process.env.NO_COLOR;\n    const isTTY =\n      typeof process !== \"undefined\" &&\n      !!process.stdout &&\n      !!process.stdout.isTTY;\n    this.colors = colors && !envNoColor && isTTY;\n  }\n\n  private shouldLog(level: LogLevel) {\n    return (\n      LEVEL_ORDER[level] >= 0 &&\n      LEVEL_ORDER[level] >= LEVEL_ORDER[this.minLevel] &&\n      LEVEL_ORDER[this.minLevel] < LEVEL_ORDER.NONE\n    );\n  }\n\n  private levelMeta(level: Exclude<LogLevel, \"NONE\">) {\n    switch (level) {\n      case \"DEBUG\":\n        return { tag: \"DEBUG\", color: ANSI.debug, emoji: \"🐛\" };\n      case \"INFO\":\n        return { tag: \"INFO\", color: ANSI.info, emoji: \"ℹ️\" };\n      case \"WARN\":\n        return { tag: \"WARN\", color: ANSI.warn, emoji: \"⚠️\" };\n      case \"ERROR\":\n        return { tag: \"ERROR\", color: ANSI.error, emoji: \"❌\" };\n    }\n  }\n\n  private timestamp() {\n    if (!this.showTimestamp) return \"\";\n    // ISO-like with ms, or leave short local time:\n    return new Date().toISOString();\n  }\n\n  private padLevelTag(tag: string) {\n    // keep width consistent\n    return tag.padEnd(5, \" \");\n  }\n\n  private colorize(text: string, colorCode?: string) {\n    if (!this.colors || !colorCode) return text;\n    return `${colorCode}${text}${ANSI.reset}`;\n  }\n\n  private format(level: Exclude<LogLevel, \"NONE\">, msg: string) {\n    const meta = this.levelMeta(level);\n    const when = this.timestamp();\n    const namePart = this.name ? `[${this.name}] ` : \"\";\n    const levelTag = `[ ${this.padLevelTag(meta.tag)} ]`;\n    const emoji = meta.emoji;\n    if (this.colors) {\n      // Color level tag and dim the timestamp + name\n      const coloredTag = this.colorize(levelTag, meta.color);\n      const dimMeta = this.colorize(when ? `${when} ` : \"\", ANSI.gray);\n      const dimName = this.colorize(namePart, ANSI.dim);\n      return `${dimMeta}${dimName}${coloredTag} ${emoji} ${msg}`;\n    }\n    return `${when ? `${when} ` : \"\"}${namePart}${levelTag} ${emoji} ${msg}`;\n  }\n\n  private write(level: Exclude<LogLevel, \"NONE\">, msg: string) {\n    if (!this.shouldLog(level)) return;\n\n    const out = this.format(level, msg);\n\n    switch (level) {\n      case \"DEBUG\":\n        return console.debug ? console.debug(out) : console.log(out);\n      case \"INFO\":\n        return console.info ? console.info(out) : console.log(out);\n      case \"WARN\":\n        return console.warn ? console.warn(out) : console.log(out);\n      case \"ERROR\":\n        return console.error ? console.error(out) : console.log(out);\n    }\n  }\n\n  debug(msg: string) {\n    this.write(\"DEBUG\", msg);\n  }\n  info(msg: string) {\n    this.write(\"INFO\", msg);\n  }\n  warn(msg: string) {\n    this.write(\"WARN\", msg);\n  }\n  error(msg: string) {\n    this.write(\"ERROR\", msg);\n  }\n\n  /** Return a child logger sharing options but with a name (useful for modules) */\n  child(name: string) {\n    return new Logger({\n      level: this.minLevel,\n      colors: this.colors,\n      timestamp: this.showTimestamp,\n      name,\n    });\n  }\n\n  /** Update minimum level at runtime */\n  setLevel(level: LogLevel) {\n    this.minLevel = level;\n  }\n}\n",
    "import fs from \"node:fs/promises\";\nimport { existsSync } from \"node:fs\";\nimport path from \"node:path\";\nimport type { TopConfig, TopCollectionConfig, CollectionConfig } from \"./types\";\nimport { logger } from \"../bin/cli\";\n\nexport const TOP_CONFIG_FILE = path.join(\"outline-sync.json\");\n\nexport const sleep = (ms: number) => new Promise((r) => setTimeout(r, ms));\n\nexport async function ensureConfigDir(dir: string) {\n  logger.debug(`Ensuring config dir: ${dir}`);\n  if (!existsSync(dir)) {\n    logger.warn(`Dir (${dir}) does not exist, creating...`);\n    await fs.mkdir(dir, { recursive: true });\n  }\n}\n\nexport async function ensureConfigDirs(cfg: TopConfig) {\n  const collections = cfg.collections;\n  console.debug(`Ensuring config dir for ${collections.length} collection(s)`);\n  for (const collection of collections) {\n    const dir = collection.configDir;\n    logger.debug(`Ensuring config dir: ${dir}`);\n    if (!existsSync(dir)) {\n      logger.warn(`Dir (${dir}) does not exist, creating...`);\n      await fs.mkdir(dir, { recursive: true });\n    }\n  }\n}\n\nexport async function loadTopConfig(): Promise<TopConfig | null> {\n  logger.debug(`Loading top config from: ${TOP_CONFIG_FILE}`);\n  if (!existsSync(TOP_CONFIG_FILE)) {\n    logger.warn(\"Top config file not found\");\n    return null;\n  }\n  const raw = await fs.readFile(TOP_CONFIG_FILE, \"utf8\");\n  return JSON.parse(raw) as TopConfig;\n}\n\nexport async function saveTopConfig(cfg: TopConfig) {\n  logger.debug(\"Saving top config\");\n  await ensureConfigDirs(cfg);\n  await fs.writeFile(\n    TOP_CONFIG_FILE,\n    `${JSON.stringify(cfg, null, 2)}\\n`,\n    \"utf8\",\n  );\n}\n\nexport async function getCollectionFilesBase(collectionId: string): Promise<{\n  pagesFile: string;\n  configFile: string;\n  saveDir: string;\n  configDir: string;\n}> {\n  logger.debug(\"Getting Collection file base\");\n  const topConfig = await loadTopConfig();\n  const CONFIG_DIR = topConfig.collections.find(\n    (collection) => collection.id === collectionId,\n  ).configDir;\n\n  // If no top config exists, use default paths\n  if (!topConfig || !topConfig.collections) {\n    logger.warn(`No Top config found for ${collectionId}, returning default`);\n    return {\n      pagesFile: path.join(CONFIG_DIR, `${collectionId}.pages.json`),\n      configFile: path.join(CONFIG_DIR, `${collectionId}.config.json`),\n      saveDir: \"docs\",\n      configDir: CONFIG_DIR,\n    };\n  }\n\n  // Find the collection configuration\n  const collectionConfig = topConfig.collections.find(\n    (c) => c.id === collectionId,\n  );\n\n  // If no specific collection config found, use default paths\n  if (!collectionConfig) {\n    logger.warn(\n      `Collection config for ${collectionId} not found, returning default paths`,\n    );\n    return {\n      pagesFile: path.join(CONFIG_DIR, `${collectionId}.pages.json`),\n      configFile: path.join(CONFIG_DIR, `${collectionId}.config.json`),\n      saveDir: \"docs\",\n      configDir: \".config\",\n    };\n  }\n\n  // Use custom paths if provided, otherwise fall back to defaults\n  return {\n    pagesFile:\n      collectionConfig.pagesFile ||\n      path.join(CONFIG_DIR, `${collectionId}.pages.json`),\n    configFile:\n      collectionConfig.configFile ||\n      path.join(CONFIG_DIR, `${collectionId}.config.json`),\n    saveDir: collectionConfig.saveDir || \"docs\",\n    configDir: collectionConfig.configDir || \".config\",\n  };\n}\n\nexport async function loadCollectionConfig(\n  collectionId: string,\n): Promise<CollectionConfig | null> {\n  logger.debug(`Loading collection config for ${collectionId}`);\n  const { configFile } = await getCollectionFilesBase(collectionId);\n  if (!existsSync(configFile)) {\n    logger.warn(`Collection config file for ${collectionId} not found`);\n    return null;\n  }\n  const raw = await fs.readFile(configFile, \"utf8\");\n  return JSON.parse(raw) as CollectionConfig;\n}\n\nexport async function getCollectionTopConfig(\n  collectionId: string,\n): Promise<TopCollectionConfig | null> {\n  logger.debug(`Getting Top Collection Config for ${collectionId}`);\n  const topConfig = await loadTopConfig();\n  if (!topConfig || !topConfig.collections) {\n    logger.debug(`No Top Config found for ${collectionId}`);\n    return null;\n  }\n  return topConfig.collections.find((c) => c.id === collectionId) || null;\n}\n\nexport async function saveCollectionConfig(c: CollectionConfig) {\n  logger.debug(`Saving Collection config for ${c.collectionId}`);\n  const { configFile, configDir } = await getCollectionFilesBase(\n    c.collectionId,\n  );\n  await ensureConfigDir(configDir);\n  await fs.writeFile(configFile, `${JSON.stringify(c, null, 2)}\\n`, \"utf8\");\n}\n",
    "import fs from \"node:fs/promises\";\nimport { existsSync } from \"node:fs\";\nimport path from \"node:path\";\nimport type { TopConfig, TopCollectionConfig, CollectionConfig } from \"./types\";\nimport { logger } from \"../bin/cli\";\n\nexport const TOP_CONFIG_FILE = path.join(\"outline-sync.json\");\n\nexport const sleep = (ms: number) => new Promise((r) => setTimeout(r, ms));\n\nexport async function ensureConfigDir(dir: string) {\n  logger.debug(`Ensuring config dir: ${dir}`);\n  if (!existsSync(dir)) {\n    logger.warn(`Dir (${dir}) does not exist, creating...`);\n    await fs.mkdir(dir, { recursive: true });\n  }\n}\n\nexport async function ensureConfigDirs(cfg: TopConfig) {\n  const collections = cfg.collections;\n  console.debug(`Ensuring config dir for ${collections.length} collection(s)`);\n  for (const collection of collections) {\n    const dir = collection.configDir;\n    logger.debug(`Ensuring config dir: ${dir}`);\n    if (!existsSync(dir)) {\n      logger.warn(`Dir (${dir}) does not exist, creating...`);\n      await fs.mkdir(dir, { recursive: true });\n    }\n  }\n}\n\nexport async function loadTopConfig(): Promise<TopConfig | null> {\n  logger.debug(`Loading top config from: ${TOP_CONFIG_FILE}`);\n  if (!existsSync(TOP_CONFIG_FILE)) {\n    logger.warn(\"Top config file not found\");\n    return null;\n  }\n  const raw = await fs.readFile(TOP_CONFIG_FILE, \"utf8\");\n  return JSON.parse(raw) as TopConfig;\n}\n\nexport async function saveTopConfig(cfg: TopConfig) {\n  logger.debug(\"Saving top config\");\n  await ensureConfigDirs(cfg);\n  await fs.writeFile(\n    TOP_CONFIG_FILE,\n    `${JSON.stringify(cfg, null, 2)}\\n`,\n    \"utf8\",\n  );\n}\n\nexport async function getCollectionFilesBase(collectionId: string): Promise<{\n  pagesFile: string;\n  configFile: string;\n  saveDir: string;\n  configDir: string;\n}> {\n  logger.debug(\"Getting Collection file base\");\n  const topConfig = await loadTopConfig();\n  const CONFIG_DIR = topConfig.collections.find(\n    (collection) => collection.id === collectionId,\n  ).configDir;\n\n  // If no top config exists, use default paths\n  if (!topConfig || !topConfig.collections) {\n    logger.warn(`No Top config found for ${collectionId}, returning default`);\n    return {\n      pagesFile: path.join(CONFIG_DIR, `${collectionId}.pages.json`),\n      configFile: path.join(CONFIG_DIR, `${collectionId}.config.json`),\n      saveDir: \"docs\",\n      configDir: CONFIG_DIR,\n    };\n  }\n\n  // Find the collection configuration\n  const collectionConfig = topConfig.collections.find(\n    (c) => c.id === collectionId,\n  );\n\n  // If no specific collection config found, use default paths\n  if (!collectionConfig) {\n    logger.warn(\n      `Collection config for ${collectionId} not found, returning default paths`,\n    );\n    return {\n      pagesFile: path.join(CONFIG_DIR, `${collectionId}.pages.json`),\n      configFile: path.join(CONFIG_DIR, `${collectionId}.config.json`),\n      saveDir: \"docs\",\n      configDir: \".config\",\n    };\n  }\n\n  // Use custom paths if provided, otherwise fall back to defaults\n  return {\n    pagesFile:\n      collectionConfig.pagesFile ||\n      path.join(CONFIG_DIR, `${collectionId}.pages.json`),\n    configFile:\n      collectionConfig.configFile ||\n      path.join(CONFIG_DIR, `${collectionId}.config.json`),\n    saveDir: collectionConfig.saveDir || \"docs\",\n    configDir: collectionConfig.configDir || \".config\",\n  };\n}\n\nexport async function loadCollectionConfig(\n  collectionId: string,\n): Promise<CollectionConfig | null> {\n  logger.debug(`Loading collection config for ${collectionId}`);\n  const { configFile } = await getCollectionFilesBase(collectionId);\n  if (!existsSync(configFile)) {\n    logger.warn(`Collection config file for ${collectionId} not found`);\n    return null;\n  }\n  const raw = await fs.readFile(configFile, \"utf8\");\n  return JSON.parse(raw) as CollectionConfig;\n}\n\nexport async function getCollectionTopConfig(\n  collectionId: string,\n): Promise<TopCollectionConfig | null> {\n  logger.debug(`Getting Top Collection Config for ${collectionId}`);\n  const topConfig = await loadTopConfig();\n  if (!topConfig || !topConfig.collections) {\n    logger.debug(`No Top Config found for ${collectionId}`);\n    return null;\n  }\n  return topConfig.collections.find((c) => c.id === collectionId) || null;\n}\n\nexport async function saveCollectionConfig(c: CollectionConfig) {\n  logger.debug(`Saving Collection config for ${c.collectionId}`);\n  const { configFile, configDir } = await getCollectionFilesBase(\n    c.collectionId,\n  );\n  await ensureConfigDir(configDir);\n  await fs.writeFile(configFile, `${JSON.stringify(c, null, 2)}\\n`, \"utf8\");\n}\n",
    "import { sleep } from \"./config\";\nconst BASE_URL = process.env.OUTLINE_BASE_URL || \"https://app.getoutline.com\";\nconst API_KEY = process.env.OUTLINE_API_KEY || \"\";\nconst HEADERS = {\n  Authorization: `Bearer ${API_KEY}`,\n  \"Content-Type\": \"application/json\",\n};\n\nimport { logger } from \"../bin/cli\";\n\n/**\n * Make a POST request to the Outline API with simple retry/backoff logic.\n */\nasync function outlineRequest(\n  endpoint: string,\n  body: any,\n  retries = 3,\n): Promise<any> {\n  const url = `${BASE_URL}/api/${endpoint}`;\n\n  for (let attempt = 0; attempt < retries; attempt++) {\n    try {\n      logger.debug(`Outline request: POST ${url} (attempt ${attempt + 1})`);\n      logger.debug(`Payload (trimmed): ${JSON.stringify(body, null, 2)}`);\n\n      const res = await fetch(url, {\n        method: \"POST\",\n        headers: HEADERS,\n        body: JSON.stringify(body),\n      });\n\n      if (res.status === 429) {\n        const backoff = 1000 * (attempt + 1);\n        logger.warn(\n          `Rate limited by Outline API. Backing off ${backoff}ms (attempt ${attempt + 1}).`,\n        );\n        await sleep(backoff);\n        continue;\n      }\n\n      let json: any;\n      try {\n        json = await res.json();\n      } catch (parseErr) {\n        logger.error(\n          `Failed to parse JSON response from ${endpoint}: ${parseErr}`,\n        );\n        throw parseErr;\n      }\n\n      if (!res.ok) {\n        // Log useful context but avoid leaking sensitive headers/keys\n        logger.error(\n          `[Outline@${BASE_URL}/api/${endpoint}] HTTP ${res.status} - payload=${JSON.stringify(body)} response=${JSON.stringify(json)}`,\n        );\n        throw new Error(\n          `Outline API error ${res.status}: ${JSON.stringify(json)}`,\n        );\n      }\n\n      logger.debug(\n        `Outline response for ${endpoint} (attempt ${attempt + 1}): ${JSON.stringify(json).slice(0, 200)}${JSON.stringify(json).length > 200 ? \"...\" : \"\"}`,\n      );\n      return json;\n    } catch (err: any) {\n      // last attempt -> rethrow\n      if (attempt === retries - 1) {\n        logger.error(\n          `Request to Outline failed after ${retries} attempts: ${err?.message ?? err}`,\n        );\n        throw err;\n      }\n      const backoff = 500 * (attempt + 1);\n      logger.warn(\n        `Request failed (attempt ${attempt + 1}): ${err?.message ?? err}. Retrying after ${backoff}ms...`,\n      );\n      await sleep(backoff);\n    }\n  }\n\n  // Should be unreachable, but keep the error for TypeScript\n  throw new Error(\"outlineRequest: unreachable\");\n}\n\nexport async function listCollectionsPaged(): Promise<\n  { id: string; name: string }[]\n> {\n  const out: { id: string; name: string }[] = [];\n  let offset = 0;\n  const limit = 100;\n  while (true) {\n    const json = await outlineRequest(\"collections.list\", { offset, limit });\n    const data = json.data || [];\n    for (const c of data) out.push({ id: c.id, name: c.name });\n    if (data.length < limit) break;\n    offset += data.length;\n  }\n  logger.debug(`listCollectionsPaged: returned ${out.length} collections`);\n  return out;\n}\n\nexport async function listDocumentsInCollection(\n  collectionId: string,\n): Promise<any[]> {\n  const out: any[] = [];\n  let offset = 0;\n  const limit = 100;\n  while (true) {\n    const json = await outlineRequest(\"documents.list\", {\n      collectionId,\n      offset,\n      limit,\n    });\n    const data = json.data || [];\n    for (const d of data) out.push(d);\n    if (data.length < limit) break;\n    offset += data.length;\n  }\n  logger.debug(\n    `listDocumentsInCollection(${collectionId}): returned ${out.length} documents`,\n  );\n  return out;\n}\n\nexport async function fetchDocumentInfo(documentId: string) {\n  const json = await outlineRequest(\"documents.info\", { id: documentId });\n  logger.debug(`fetchDocumentInfo(${documentId}) -> ${json ? \"ok\" : \"null\"}`);\n  return json.data ?? null;\n}\n\nexport async function createDocument(\n  title: string,\n  text: string,\n  collectionId: string,\n  parentDocumentId: string | null,\n) {\n  const payload = {\n    title,\n    text,\n    collectionId,\n    parentDocumentId: parentDocumentId || null,\n    publish: true,\n  };\n  const json = await outlineRequest(\"documents.create\", payload);\n  logger.info(\n    `Created document \"${title}\" in collection ${collectionId} (id=${json?.id ?? \"unknown\"})`,\n  );\n  return json.data;\n}\n\nexport async function updateDocument(\n  id: string,\n  title: string | undefined,\n  text: string,\n) {\n  const payload: any = { id, text, publish: true };\n  if (title) payload.title = title;\n  const json = await outlineRequest(\"documents.update\", payload);\n  logger.info(`Updated document id=${id}${title ? ` title=\"${title}\"` : \"\"}`);\n  return json.data;\n}\n",
    "import fs from \"node:fs/promises\";\nimport { existsSync } from \"node:fs\";\nimport path from \"node:path\";\nimport { spawnSync } from \"node:child_process\";\nimport { logger } from \"../bin/cli\";\n\n/**\n * Remove ALL whitespace (space, tab, newline, CR) for comparison.\n * This makes diffs tolerant to formatting differences.\n */\nexport function normalizeContentIgnoreWhitespace(s: string) {\n  return s.replace(/\\s+/g, \"\");\n}\n\nexport function slugifyTitle(title: string) {\n  return title\n    .toString()\n    .normalize(\"NFKD\")\n    .replace(/\\p{M}/gu, \"\")\n    .toLowerCase()\n    .replace(/[^a-z0-9]+/g, \"-\")\n    .replace(/(^-|-$)+/g, \"\")\n    .slice(0, 120);\n}\n\nexport function getGitTimestampMs(filePath: string): number | null {\n  try {\n    const resolved = path.resolve(filePath);\n    const out = spawnSync(\n      \"git\",\n      [\"log\", \"-1\", \"--format=%ct\", \"--\", resolved],\n      {\n        cwd: process.cwd(),\n        encoding: \"utf8\",\n      },\n    );\n    if (out.status !== 0) {\n      logger.debug(\n        `git log returned non-zero status for ${resolved}: ${out.status}`,\n      );\n      return null;\n    }\n    const txt = (out.stdout || \"\").trim();\n    if (!txt) {\n      logger.debug(`git log returned no output for ${resolved}`);\n      return null;\n    }\n    const sec = Number(txt);\n    if (Number.isNaN(sec)) {\n      logger.debug(`git log output not a number for ${resolved}: \"${txt}\"`);\n      return null;\n    }\n    return sec * 1000;\n  } catch (err) {\n    logger.debug(`getGitTimestampMs error for ${filePath}: ${err}`);\n    return null;\n  }\n}\n\nexport async function getLocalTimestampMs(filePath: string): Promise<number> {\n  const gitTs = getGitTimestampMs(filePath);\n  if (gitTs) {\n    logger.debug(`Using git timestamp for ${filePath}: ${gitTs}`);\n    return gitTs;\n  }\n  const st = await fs.stat(filePath);\n  logger.debug(`Using FS mtime for ${filePath}: ${st.mtimeMs}`);\n  return st.mtimeMs;\n}\n\nexport async function safeWriteFile(\n  filePath: string,\n  content: string,\n  dryRun = false,\n) {\n  if (existsSync(filePath)) {\n    const bak = `${filePath}.outline-sync.bak.${Date.now()}`;\n    if (!dryRun) {\n      try {\n        await fs.copyFile(filePath, bak);\n        logger.info(`Backed up existing file to ${bak}`);\n      } catch (err) {\n        logger.warn(`Failed to back up ${filePath} to ${bak}: ${err}`);\n      }\n    } else {\n      logger.debug(\n        `[dry-run] would backup existing file ${filePath} -> ${bak}`,\n      );\n    }\n  } else {\n    if (!dryRun) {\n      try {\n        await fs.mkdir(path.dirname(filePath), { recursive: true });\n        logger.debug(`Ensured directory ${path.dirname(filePath)}`);\n      } catch (err) {\n        logger.warn(\n          `Failed to create directory ${path.dirname(filePath)}: ${err}`,\n        );\n      }\n    } else {\n      logger.debug(\n        `[dry-run] would ensure directory ${path.dirname(filePath)}`,\n      );\n    }\n  }\n\n  if (!dryRun) {\n    try {\n      await fs.writeFile(filePath, content, \"utf8\");\n      logger.info(`Wrote file ${filePath} (${content.length} bytes)`);\n    } catch (err) {\n      logger.error(`Failed to write file ${filePath}: ${err}`);\n      throw err;\n    }\n  } else {\n    logger.debug(\n      `[dry-run] would write file ${filePath} (${content.length} bytes)`,\n    );\n  }\n}\n",
    "import { listCollectionsPaged, listDocumentsInCollection } from \"./outlineApi\";\nimport {\n  loadTopConfig,\n  saveTopConfig,\n  getCollectionFilesBase,\n  saveCollectionConfig,\n  ensureConfigDir,\n  ensureConfigDirs,\n} from \"./config\";\nimport fs from \"node:fs/promises\";\nimport { existsSync } from \"node:fs\";\nimport path from \"node:path\";\nimport { slugifyTitle } from \"./utils\";\nimport type { PageEntry, Manifest } from \"./types\";\nimport { logger } from \"../bin/cli\";\n\n/**\n * Prompt helper (simple): prints numbered list and reads a line from stdin.\n */\nexport async function listCollectionsPrompt(opts: {\n  dryRun?: boolean;\n  nonInteractive?: boolean;\n}) {\n  const cols = await listCollectionsPaged();\n  if (!cols.length) {\n    logger.warn(\"No collections found for this API key.\");\n    return;\n  }\n  console.info(\"Collections:\");\n  cols.forEach((c, i) => console.info(`${i + 1}) ${c.id}\\t${c.name}`));\n  if (opts.nonInteractive) {\n    return;\n  }\n\n  const selection = await question(\n    \"Select a collection by number (or press Enter to cancel): \",\n  );\n  const idx = Number(selection.trim());\n  if (!idx || idx < 1 || idx > cols.length) {\n    logger.warn(\"Cancelled collection selection.\");\n    return;\n  }\n\n  const chosen = cols[idx - 1];\n  logger.info(`You chose: ${chosen.name} (${chosen.id})`);\n\n  await ensureConfigDirs((await loadTopConfig()) || { collections: [] });\n  const top = (await loadTopConfig()) || { collections: [] };\n  const exists = top.collections.find((c) => c.id === chosen.id);\n\n  let configDir = (\n    await question(\n      \"Enter a base folder path for the collections config files (or press enter for default `.config`): \",\n    )\n  ).replaceAll(\"\\n\", \"\");\n\n  if (configDir.trim().length <= 1) {\n    configDir = \".config\";\n    logger.debug(\"Using default configDir `.config`\");\n  }\n\n  let saveDir = (\n    await question(\n      \"Enter a base folder path for the collections markdown files (or press enter for default `docs`): \",\n    )\n  ).replaceAll(\"\\n\", \"\");\n\n  if (saveDir.trim().length <= 1) {\n    saveDir = \"docs\";\n    logger.debug(\"Using default saveDir `docs`\");\n  }\n\n  if (!exists) {\n    top.collections.unshift({\n      id: chosen.id,\n      name: chosen.name,\n      configDir: configDir,\n      saveDir: saveDir,\n      pagesFile: path.join(configDir, `${chosen.id}.pages.json`),\n      configFile: path.join(configDir, `${chosen.id}.config.json`),\n    });\n    await saveTopConfig(top);\n    logger.info(\n      `Added collection to ${path.join(\"configs\", \"outline-sync.json\")}`,\n    );\n  } else {\n    logger.warn(\"Collection already configured.\");\n  }\n}\n\n/**\n * Simple stdin line reader\n */\nexport function question(promptText: string): Promise<string> {\n  return new Promise((resolve) => {\n    process.stdout.write(promptText);\n    process.stdin.resume();\n    process.stdin.setEncoding(\"utf8\");\n    process.stdin.once(\"data\", (data) => {\n      process.stdin.pause();\n      resolve(data.toString());\n    });\n  });\n}\n\n/**\n * Bootstrap: fetch documents from collection, create pages.json and write markdown files.\n */\nexport async function bootstrapCollection(opts: {\n  collectionId: string;\n  dryRun?: boolean;\n}) {\n  const { collectionId, dryRun = false } = opts;\n  logger.info(`Bootstrapping collection ${collectionId} (dryRun=${dryRun})...`);\n\n  const docs = await listDocumentsInCollection(collectionId);\n  logger.info(`Fetched ${docs.length} documents from Outline.`);\n  logger.debug(\n    `First 3 documents: ${JSON.stringify(docs.slice(0, 3), null, 2)}`,\n  );\n\n  // 1) build flat map of nodes\n  const map = new Map<string, PageEntry & { raw?: any }>();\n  for (const d of docs) {\n    map.set(d.id, {\n      title: d.title,\n      file: \"\",\n      id: d.id,\n      children: [],\n      raw: d,\n    });\n  }\n\n  // 2) attach children\n  const roots: (PageEntry & { raw?: any })[] = [];\n  for (const node of map.values()) {\n    const raw = node.raw || {};\n    const parentId = raw.parentDocumentId ?? raw.parentId ?? null;\n    if (parentId && map.has(parentId)) {\n      map.get(parentId).children.push(node);\n    } else {\n      roots.push(node);\n    }\n  }\n  logger.debug(`Built document tree with ${roots.length} root(s)`);\n\n  // 3) assign paths\n  const { saveDir } = await getCollectionFilesBase(collectionId);\n  function assignPaths(node: any, parentDir: string) {\n    const slug = slugifyTitle(node.title || \"untitled\");\n    const dir = path.join(parentDir, slug);\n    const filePath = path.join(dir, \"README.md\");\n    node.file = filePath;\n    if (node.children?.length) {\n      for (const c of node.children) {\n        assignPaths(c, dir);\n      }\n    }\n  }\n  for (const r of roots) {\n    assignPaths(r, saveDir);\n  }\n\n  // 4) write files\n  for (const n of map.values()) {\n    const filePath = n.file;\n    const content = n.raw?.text ?? `# ${n.title}\\n\\n`;\n    if (!dryRun) {\n      await fs.mkdir(path.dirname(filePath), { recursive: true });\n      await fs.writeFile(filePath, content, \"utf8\");\n      logger.debug(`Wrote file: ${filePath}`);\n    } else {\n      logger.debug(\n        `[dry-run] would write ${filePath} (${content.length} bytes)`,\n      );\n    }\n  }\n\n  // 5) build manifest\n  function strip(n: any): PageEntry {\n    return {\n      title: n.title,\n      file: n.file,\n      id: n.id,\n      children: (n.children || []).map(strip),\n    };\n  }\n  const manifest: Manifest = { collectionId, pages: roots.map(strip) };\n\n  // save manifest/config\n  const {\n    pagesFile,\n    configFile,\n    saveDir: configuredSaveDir,\n    configDir,\n  } = await getCollectionFilesBase(collectionId);\n  logger.debug(\n    `Collection (${collectionId}) files: pagesFile=${pagesFile}, configFile=${configFile}, saveDir=${configuredSaveDir}, configDir=${configDir}`,\n  );\n\n  await ensureConfigDir(configDir);\n  if (!dryRun) {\n    await fs.writeFile(\n      pagesFile,\n      `${JSON.stringify(manifest, null, 2)}\\n`,\n      \"utf8\",\n    );\n    logger.info(`Saved manifest: ${pagesFile}`);\n\n    if (!existsSync(configFile)) {\n      await fs.writeFile(\n        configFile,\n        `${JSON.stringify({ collectionId, saveDir: configuredSaveDir, mappings: [] }, null, 2)}\\n`,\n        \"utf8\",\n      );\n      logger.info(`Created new config: ${configFile}`);\n    }\n    logger.info(`Saved config: ${configFile}`);\n\n    const top = (await loadTopConfig()) || { collections: [] };\n    const existsTop = top.collections.find((c) => c.id === collectionId);\n    if (!existsTop) {\n      top.collections.unshift({\n        id: collectionId,\n        saveDir: configuredSaveDir,\n        pagesFile,\n        configFile,\n      });\n      await saveTopConfig(top);\n      logger.debug(`Updated top config with collection ${collectionId}`);\n    }\n  } else {\n    logger.debug(\n      `[dry-run] would save pages to ${pagesFile} and config to ${configFile}`,\n    );\n  }\n\n  logger.info(\"Bootstrap complete\");\n}\n",
    "import fs from \"node:fs/promises\";\nimport { existsSync } from \"node:fs\";\nimport path from \"node:path\";\nimport {\n  loadTopConfig,\n  getCollectionFilesBase,\n  loadCollectionConfig,\n} from \"./config\";\nimport {\n  normalizeContentIgnoreWhitespace,\n  getLocalTimestampMs,\n  safeWriteFile,\n  slugifyTitle,\n} from \"./utils\";\nimport {\n  fetchDocumentInfo,\n  updateDocument,\n  createDocument,\n} from \"./outlineApi\";\nimport type { Manifest, PageEntry } from \"./types\";\nimport { logger } from \"../bin/cli\";\n\n/**\n * Load pages.json for a collection (if missing, error/ask to init)\n */\nexport async function loadPagesManifest(\n  collectionId: string,\n): Promise<Manifest> {\n  const { pagesFile } = await getCollectionFilesBase(collectionId);\n  if (!existsSync(pagesFile)) {\n    logger.error(`${pagesFile} not found. Run init/setup to create it`);\n    throw new Error(`${pagesFile} not found. Run init/setup to create it`);\n  }\n  const raw = await fs.readFile(pagesFile, \"utf8\");\n  logger.debug(`Loaded pages manifest from ${pagesFile} (${raw.length} bytes)`);\n  return JSON.parse(raw) as Manifest;\n}\n\nexport async function persistPagesManifest(\n  collectionId: string,\n  manifest: Manifest,\n  dryRun = false,\n) {\n  const { pagesFile } = await getCollectionFilesBase(collectionId);\n  if (dryRun) {\n    logger.debug(`[dry-run] would persist manifest to ${pagesFile}`);\n    return;\n  }\n  await fs.writeFile(\n    pagesFile,\n    `${JSON.stringify(manifest, null, 2)}\\n`,\n    \"utf8\",\n  );\n  logger.info(`Persisted manifest to ${pagesFile}`);\n}\n\n/**\n * Resolve destination file path for a document using collection config mappings.\n *\n * Rules:\n * - mapping with id exact match wins\n * - mapping with title exact match next\n * - mapping.path may be:\n *    - a file path (ending with `.md`) -> set node.file to that path\n *    - a directory path (ends with `/` or no extension) -> place node in that dir as README.md\n * - if no mapping, fall back to inherited folder approach:\n *    parentDir + slug/README.md\n */\nexport function applyMappingsToManifest(\n  manifest: Manifest,\n  collectionConfig: any,\n) {\n  const rules = (collectionConfig?.mappings || []) as {\n    match: any;\n    path: string;\n  }[];\n\n  function isDirPath(p: string) {\n    // treat trailing slash as directory OR lack of .md extension as directory\n    if (!p) return false;\n    if (p.endsWith(\"/\") || p.endsWith(path.sep)) return true;\n    return path.extname(p).toLowerCase() !== \".md\";\n  }\n\n  function applyToNode(node: any, parentDir: string | null) {\n    // try id match then title match\n    let matched = false;\n    for (const r of rules) {\n      if (r.match?.id && node.id === r.match.id) {\n        const p = r.path;\n        if (isDirPath(p)) {\n          const dir = p.endsWith(\"/\") ? p : p;\n          node.file = path.join(dir, \"README.md\");\n        } else {\n          node.file = p;\n        }\n        matched = true;\n        logger.debug(\n          `Mapping applied by id for \"${node.title}\" -> ${node.file}`,\n        );\n        break;\n      }\n    }\n    if (!matched) {\n      for (const r of rules) {\n        if (r.match?.title && node.title === r.match.title) {\n          const p = r.path;\n          if (isDirPath(p)) {\n            const dir = p.endsWith(\"/\") ? p : p;\n            node.file = path.join(dir, \"README.md\");\n          } else {\n            node.file = p;\n          }\n          matched = true;\n          logger.debug(\n            `Mapping applied by title for \"${node.title}\" -> ${node.file}`,\n          );\n          break;\n        }\n      }\n    }\n\n    // if still not assigned, inherit from parentDir by using slug -> README.md\n    if (!node.file) {\n      const slug = slugifyTitle(node.title || \"untitled\");\n      const dir = parentDir\n        ? path.join(parentDir, slug)\n        : path.join(collectionConfig?.saveDir || \"docs\", slug);\n      node.file = path.join(dir, \"README.md\");\n      logger.debug(`Inherited path for \"${node.title}\" -> ${node.file}`);\n    } else {\n      // If node.file is a bare filename (no directory) => put under parentDir or saveDir\n      const hasDir = path.dirname(node.file) && path.dirname(node.file) !== \".\";\n      if (!hasDir) {\n        const baseDir = parentDir || collectionConfig?.saveDir || \"docs\";\n        node.file = path.join(baseDir, node.file);\n        logger.debug(\n          `Normalized bare filename for \"${node.title}\" -> ${node.file}`,\n        );\n      } else {\n        // mapping may be relative; keep as-is\n        logger.debug(`Using mapped path for \"${node.title}\" -> ${node.file}`);\n      }\n    }\n\n    // compute this node's dir to pass to children (directory containing README.md)\n    const nodeDir = path.dirname(node.file);\n\n    if (node.children?.length) {\n      for (const c of node.children) {\n        applyToNode(c, nodeDir);\n      }\n    }\n  }\n\n  for (const p of manifest.pages) {\n    applyToNode(p as any, null);\n  }\n  logger.debug(`Applied mappings to manifest (rules=${rules.length})`);\n  return manifest;\n}\n\n/**\n * Compare content ignoring whitespace/newlines\n */\nexport function contentsEqualIgnoringWhitespace(a: string, b: string) {\n  return (\n    normalizeContentIgnoreWhitespace(a) === normalizeContentIgnoreWhitespace(b)\n  );\n}\n\n/**\n * Decide whether to pull or push or skip for one page.\n */\nexport async function syncPage(\n  collectionId: string,\n  manifest: Manifest,\n  page: PageEntry,\n  parentId: string | null,\n  opts: { mode: \"pull\" | \"push\" | \"sync\"; dryRun?: boolean },\n) {\n  const filePath = page.file;\n  const absPath = path.resolve(filePath);\n  const fileExists = existsSync(absPath);\n\n  let localTs = 0;\n  if (fileExists) {\n    try {\n      localTs = await getLocalTimestampMs(absPath);\n    } catch (err) {\n      logger.warn(`Failed to get local timestamp for ${absPath}: ${err}`);\n      localTs = 0;\n    }\n  }\n\n  let remoteDoc: any = null;\n  if (page.id) {\n    try {\n      remoteDoc = await fetchDocumentInfo(page.id);\n    } catch (err) {\n      logger.warn(\n        `Failed to fetch remote info for ${page.title} (${page.id}): ${err}`,\n      );\n      remoteDoc = null;\n    }\n  }\n\n  const remoteText = remoteDoc?.text ?? null;\n  const remoteUpdatedAt = remoteDoc?.updatedAt\n    ? new Date(remoteDoc.updatedAt).getTime()\n    : 0;\n\n  logger.debug(\n    `syncPage(\"${page.title}\") localExists=${fileExists} localTs=${localTs} remoteExists=${!!remoteDoc} remoteUpdatedAt=${remoteUpdatedAt}`,\n  );\n\n  // ensure local file exists when needed - create parent dirs\n  if (!fileExists) {\n    if (opts.mode === \"pull\" || opts.mode === \"sync\" || opts.mode === \"push\") {\n      const dataToWrite =\n        remoteText != null ? remoteText : `# ${page.title}\\n\\n`;\n      await safeWriteFile(absPath, dataToWrite, opts.dryRun || false);\n      logger.info(\n        `[INIT] Ensured local file for \"${page.title}\" -> ${absPath}`,\n      );\n    }\n  }\n\n  // create remote if missing\n  if (!page.id) {\n    const localContent = await fs.readFile(absPath, \"utf8\");\n    if (opts.dryRun) {\n      logger.info(\n        `[dry-run][CREATE] Would create remote doc for \"${page.title}\" in collection ${collectionId}`,\n      );\n    } else {\n      try {\n        const created = await createDocument(\n          page.title,\n          localContent,\n          collectionId,\n          parentId,\n        );\n        page.id = created?.id ?? page.id;\n        logger.info(`[CREATE] Created remote \"${page.title}\" id=${page.id}`);\n      } catch (err) {\n        logger.error(\n          `[CREATE] Failed to create remote for ${page.title}: ${err}`,\n        );\n      }\n    }\n  } else {\n    const localContent = await fs.readFile(absPath, \"utf8\");\n    if (opts.mode === \"pull\") {\n      if (\n        remoteText != null &&\n        !contentsEqualIgnoringWhitespace(localContent, remoteText)\n      ) {\n        logger.info(`[PULL] Remote applied to local for \"${page.title}\"`);\n        await safeWriteFile(absPath, remoteText ?? \"\", opts.dryRun || false);\n      } else {\n        logger.debug(`[SKIP] No change (pull) for \"${page.title}\"`);\n      }\n    } else if (opts.mode === \"push\") {\n      if (\n        remoteText == null ||\n        !contentsEqualIgnoringWhitespace(localContent, remoteText)\n      ) {\n        if (opts.dryRun) {\n          logger.info(\n            `[dry-run][PUSH] Would update remote \"${page.title}\" id=${page.id}`,\n          );\n        } else {\n          try {\n            await updateDocument(page.id, page.title, localContent);\n            logger.info(`[PUSH] Updated remote \"${page.title}\" id=${page.id}`);\n          } catch (err) {\n            logger.error(\n              `[PUSH] Failed to update remote for ${page.title}: ${err}`,\n            );\n          }\n        }\n      } else {\n        logger.debug(`[SKIP] No change (push) for \"${page.title}\"`);\n      }\n    } else {\n      // sync: timestamp-based but skip if only whitespace differs\n      if (remoteUpdatedAt > localTs + 500) {\n        if (!contentsEqualIgnoringWhitespace(localContent, remoteText ?? \"\")) {\n          logger.info(\n            `[PULL] Remote newer -> overwrite local for \"${page.title}\"`,\n          );\n          await safeWriteFile(absPath, remoteText ?? \"\", opts.dryRun || false);\n        } else {\n          logger.debug(\n            `[SKIP] equal after normalizing (remote newer timestamp but content same) \"${page.title}\"`,\n          );\n        }\n      } else if (localTs > remoteUpdatedAt + 500) {\n        if (!contentsEqualIgnoringWhitespace(localContent, remoteText ?? \"\")) {\n          logger.info(\n            `[PUSH] Local newer -> update remote for \"${page.title}\"`,\n          );\n          if (opts.dryRun) {\n            logger.info(`[dry-run] would update remote ${page.title}`);\n          } else {\n            try {\n              await updateDocument(page.id, page.title, localContent);\n              logger.info(\n                `[PUSH] Updated remote \"${page.title}\" id=${page.id}`,\n              );\n            } catch (err) {\n              logger.error(\n                `[PUSH] Failed to update remote for ${page.title}: ${err}`,\n              );\n            }\n          }\n        } else {\n          logger.debug(\n            `[SKIP] equal after normalizing (local newer timestamp but content same) \"${page.title}\"`,\n          );\n        }\n      } else {\n        logger.debug(`[SKIP] No changes for \"${page.title}\"`);\n      }\n    }\n  }\n\n  // recurse through children\n  const nextParentId = page.id || parentId;\n  if (page.children?.length) {\n    for (const child of page.children) {\n      await syncPage(collectionId, manifest, child, nextParentId, opts);\n    }\n  }\n}\n\n/**\n * Run sync for a collection\n */\nexport async function runSync(opts: {\n  collectionId: string;\n  mode: \"pull\" | \"push\" | \"sync\";\n  dryRun?: boolean;\n}) {\n  const { collectionId, mode, dryRun = false } = opts;\n  logger.info(\n    `Starting ${mode} for collection ${collectionId} (dryRun=${dryRun})`,\n  );\n\n  const pagesManifest = await loadPagesManifest(collectionId);\n  const collCfg = (await loadCollectionConfig(collectionId)) || {\n    saveDir: \"docs\",\n    mappings: [],\n  };\n\n  // apply mappings (this will set folder-based `file` fields)\n  applyMappingsToManifest(pagesManifest, collCfg);\n\n  // ensure local files/folders exist and normalize any relative filenames\n  async function normalizePaths(node: any, parentDir: string | null) {\n    // if node.file is absent, apply inheritance (applyMappingsToManifest should have set it)\n    if (!node.file) {\n      const slug = slugifyTitle(node.title || \"untitled\");\n      const dir = parentDir\n        ? path.join(parentDir, slug)\n        : path.join(collCfg.saveDir || \"docs\", slug);\n      node.file = path.join(dir, \"README.md\");\n    } else {\n      // If node.file is bare filename without dir -> put into parentDir or saveDir\n      const dirnameOfFile = path.dirname(node.file);\n      if (!dirnameOfFile || dirnameOfFile === \".\") {\n        const baseDir = parentDir || collCfg.saveDir || \"docs\";\n        node.file = path.join(baseDir, node.file);\n      }\n    }\n\n    // create directory if needed (not writing file contents here, just ensuring structure)\n    const dirToMake = path.dirname(node.file);\n    if (!dryRun) {\n      try {\n        await fs.mkdir(dirToMake, { recursive: true });\n        logger.debug(`Ensured directory ${dirToMake}`);\n      } catch (err) {\n        logger.warn(`Failed to ensure directory ${dirToMake}: ${err}`);\n      }\n    } else {\n      logger.debug(`[dry-run] would ensure directory ${dirToMake}`);\n    }\n\n    const nodeDir = path.dirname(node.file);\n    if (node.children?.length) {\n      for (const c of node.children) {\n        await normalizePaths(c, nodeDir);\n      }\n    }\n  }\n\n  // normalize for each root\n  for (const root of pagesManifest.pages) {\n    await normalizePaths(root as any, null);\n  }\n  logger.debug(\"Completed path normalization for manifest\");\n\n  // run sync recursion\n  for (const p of pagesManifest.pages) {\n    await syncPage(collectionId, pagesManifest, p, null, { mode, dryRun });\n  }\n\n  // persist manifest (write any created ids back)\n  await persistPagesManifest(collectionId, pagesManifest, dryRun);\n  logger.info(\"Done.\");\n}\n",
    "#!/usr/bin/env bun\n\n/* bin/cli.ts\n   - Parse CLI flags first (so --api-key is applied before modules import env)\n   - Support repeatable --collection flags\n   - Then dynamically import the rest of the app\n*/\n\nimport { Logger } from \"../lib/logger\";\n\nconst rawArgs = process.argv.slice(2);\nlet DEBUG = false;\n\n// parse flags (repeatable --collection)\nconst flags: Record<string, string | boolean | string[]> = {};\nconst positionals: string[] = [];\nfor (let i = 0; i < rawArgs.length; i++) {\n  const a = rawArgs[i];\n  if (a === \"--help\" || a === \"-h\") {\n    positionals.push(\"--help\");\n  } else if (a === \"--verbose\") {\n    positionals.push(\"--verbose\");\n  } else if (a.startsWith(\"--collection=\") || a.startsWith(\"--collection:\")) {\n    const val = a.split(/[:=]/)[1] || \"\";\n    if (!flags.collection) flags.collection = [];\n    (flags.collection as string[]).push(val);\n  } else if (a === \"--collection\") {\n    // support `--collection <value>`\n    const val = rawArgs[i + 1];\n    if (val && !val.startsWith(\"--\")) {\n      if (!flags.collection) flags.collection = [];\n      (flags.collection as string[]).push(val);\n      i++; // consume next arg\n    }\n  } else if (a.startsWith(\"--\")) {\n    const [k, v] = a.replace(/^--/, \"\").split(\"=\");\n    flags[k] = v === undefined ? true : v;\n  } else {\n    positionals.push(a);\n  }\n}\n\n// If user passed --api-key or --base-url, set them immediately so dynamic imports see them.\nif (flags[\"api-key\"]) {\n  // do NOT log the key to avoid accidental leakage in logs\n  process.env.OUTLINE_API_KEY = String(flags[\"api-key\"]);\n}\nif (flags[\"base-url\"]) {\n  process.env.OUTLINE_BASE_URL = String(flags[\"base-url\"]);\n}\n\nif (positionals.includes(\"--verbose\")) {\n  DEBUG = true;\n}\n\nexport const logger = new Logger({\n  level: DEBUG ? \"DEBUG\" : \"INFO\",\n});\n\nif (positionals.includes(\"--help\") || flags.help || flags.h) {\n  console.log(`\nUsage:\n  OUTLINE_API_KEY=... bun run bin/cli.ts [command] [--collection=ID]... [--dry-run] [--api-key=\"...\"]\n\nCommands:\n  setup                    - interactive setup: list collections, choose one\n  list-collections         - print collections\n  init --collection=ID     - bootstrap pages.json + markdown (repeatable)\n  pull --collection=ID     - pull remote changes (repeatable)\n  push --collection=ID     - push local changes (repeatable)\n  sync --collection=ID     - bidirectional sync (repeatable)\n\nFlags:\n  --collection=ID          Repeatable; run command against multiple collections\n  --api-key=\"...\"          Provide Outline API key (overrides env var)\n  --base-url=\"...\"         Provide Outline base URL (overrides env var)\n  --dry-run                Preview only\n  --help, -h\nExamples:\n  OUTLINE_API_KEY=... bunx @dockstat/outline-sync --collection=\"id1\" --collection=\"id2\" sync --dry-run\n  bun run bin/cli.ts sync --api-key=\"sk_xxx\" --collection=\"id1\"\n`);\n  process.exit(0);\n}\n\nconst { loadTopConfig } = await import(\"../lib/config\");\nconst { listCollectionsPrompt, bootstrapCollection } = await import(\n  \"../lib/init\"\n);\nconst { runSync } = await import(\"../lib/syncEngine\");\n\nlogger.debug(\"Parsing positionals\");\nconst cmd = positionals[0] || \"sync\";\nconst DRY_RUN = Boolean(flags[\"dry-run\"]);\nconst collectionsFromCli = (flags.collection as string[] | undefined) ?? [];\nlogger.debug(\n  `Parsed cmd=${cmd} DRY_RUN=${DRY_RUN} collectionsFromCli=${collectionsFromCli}`,\n);\n\ntry {\n  logger.debug(\"Loading top config\");\n  const topConfig = (await loadTopConfig()) || { collections: [] };\n  logger.debug(`Loaded: ${JSON.stringify(topConfig)}`);\n\n  const resolveTargets = (): string[] => {\n    logger.debug(\"Resolving targets\");\n    if (collectionsFromCli.length > 0) {\n      logger.debug(\n        `Found Collection from cli: ${JSON.stringify(collectionsFromCli)}`,\n      );\n      return collectionsFromCli;\n    }\n    if (topConfig.collections && topConfig.collections.length > 0) {\n      logger.debug(\n        `Found collections in Top Config: ${JSON.stringify(topConfig)}`,\n      );\n      return topConfig.collections.map((c) => c.id);\n    }\n    logger.warn(\"Couldn't resolve targets\");\n    return [];\n  };\n\n  if (cmd === \"list-collections\") {\n    logger.debug(\"Listing collections\");\n    await listCollectionsPrompt({ dryRun: DRY_RUN, nonInteractive: false });\n    process.exit(0);\n  }\n\n  if (cmd === \"setup\") {\n    logger.debug(\"Running setup\");\n    await listCollectionsPrompt({ dryRun: DRY_RUN, nonInteractive: false });\n    process.exit(0);\n  }\n\n  if (cmd === \"init\") {\n    logger.debug(\"Running init\");\n    const targets = resolveTargets();\n    if (!targets.length)\n      throw new Error(\n        \"Init requires at least one collection. Provide --collection or run setup.\",\n      );\n    for (const collectionId of targets) {\n      await bootstrapCollection({ collectionId, dryRun: DRY_RUN });\n    }\n    process.exit(0);\n  }\n\n  if (cmd === \"pull\" || cmd === \"push\" || cmd === \"sync\") {\n    logger.debug(\"Parsing CMD (pull/push/sync)\");\n    const targets = resolveTargets();\n    if (!targets.length)\n      throw new Error(\n        `Command \"${cmd}\" requires at least one collection id. Provide with --collection=ID or run setup.`,\n      );\n    const mode = cmd === \"pull\" ? \"pull\" : cmd === \"push\" ? \"push\" : \"sync\";\n    for (const collectionId of targets) {\n      await runSync({ collectionId, mode: mode as any, dryRun: DRY_RUN });\n    }\n    process.exit(0);\n  }\n\n  logger.error(`Unknown command: ${cmd}`);\n  process.exit(1);\n} catch (err: any) {\n  console.error(\"ERROR:\", err?.message || err);\n  process.exit(1);\n}\n"
  ],
  "mappings": "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",
  "debugId": "C9F1296C2DE7230364756E2164756E21",
  "names": []
}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@dockstat/outline-sync",
3
- "version": "1.1.2",
3
+ "version": "1.1.4",
4
4
  "description": "A simple outline git-sync library",
5
5
  "type": "module",
6
6
  "main": "./dist/cli.js",