@dockstat/outline-sync 1.1.3 → 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.
- package/README.md +287 -147
- package/dist/cli.js +13 -14
- 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
|
-
|
|
23
|
-
|
|
24
|
-
|
|
25
|
-
|
|
26
|
-
|
|
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
|
-
|
|
19
|
+
## 🚀 Quick Start
|
|
31
20
|
|
|
32
|
-
1.
|
|
21
|
+
### 1. Install & Setup
|
|
33
22
|
|
|
34
23
|
```bash
|
|
35
|
-
|
|
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
|
-
|
|
27
|
+
# Interactive setup - discovers and configures collections
|
|
41
28
|
bunx @dockstat/outline-sync setup
|
|
42
29
|
```
|
|
43
30
|
|
|
44
|
-
|
|
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
|
-
|
|
53
|
-
|
|
54
|
-
|
|
55
|
-
|
|
56
|
-
|
|
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
|
-
|
|
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
|
-
|
|
56
|
+
---
|
|
66
57
|
|
|
58
|
+
## 📖 Installation Options
|
|
59
|
+
|
|
60
|
+
### Using bunx (Recommended)
|
|
67
61
|
```bash
|
|
68
|
-
bunx @dockstat/outline-sync
|
|
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
|
-
|
|
74
|
-
|
|
65
|
+
### Global Installation
|
|
75
66
|
```bash
|
|
76
|
-
|
|
77
|
-
|
|
78
|
-
--dry-run
|
|
67
|
+
npm install -g @dockstat/outline-sync
|
|
68
|
+
outline-sync <command> [flags]
|
|
79
69
|
```
|
|
80
70
|
|
|
81
|
-
|
|
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
|
-
|
|
75
|
+
## 🎯 Commands Reference
|
|
86
76
|
|
|
87
77
|
```
|
|
88
78
|
Usage:
|
|
89
|
-
OUTLINE_API_KEY=... bunx @dockstat/outline-sync [command] [
|
|
79
|
+
OUTLINE_API_KEY=... bunx @dockstat/outline-sync [command] [options]
|
|
90
80
|
|
|
91
81
|
Commands:
|
|
92
|
-
setup
|
|
93
|
-
list-collections
|
|
94
|
-
init --collection=ID
|
|
95
|
-
pull --collection=ID
|
|
96
|
-
push --collection=ID
|
|
97
|
-
sync --collection=ID
|
|
98
|
-
|
|
99
|
-
|
|
100
|
-
--collection=ID
|
|
101
|
-
--dry-run Preview
|
|
102
|
-
--api-key="..."
|
|
103
|
-
--base-url="
|
|
104
|
-
--
|
|
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
|
-
|
|
114
|
+
## 📁 Configuration Structure
|
|
110
115
|
|
|
111
|
-
|
|
116
|
+
The tool uses a structured configuration approach:
|
|
112
117
|
|
|
113
118
|
```
|
|
114
|
-
|
|
115
|
-
|
|
116
|
-
|
|
117
|
-
|
|
118
|
-
docs/
|
|
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
|
-
|
|
133
|
+
### Global Configuration (`outline-sync.json`)
|
|
122
134
|
|
|
123
135
|
```json
|
|
124
136
|
{
|
|
125
137
|
"collections": [
|
|
126
138
|
{
|
|
127
|
-
"id": "
|
|
128
|
-
"name": "
|
|
129
|
-
"
|
|
130
|
-
"
|
|
131
|
-
"
|
|
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
|
-
|
|
150
|
+
### Collection Configuration (`<collection-id>.config.json`)
|
|
138
151
|
|
|
139
152
|
```json
|
|
140
153
|
{
|
|
141
|
-
"collectionId": "
|
|
142
|
-
"saveDir": "docs",
|
|
154
|
+
"collectionId": "collection-uuid-here",
|
|
155
|
+
"saveDir": "docs/documentation",
|
|
143
156
|
"mappings": [
|
|
144
157
|
{
|
|
145
|
-
"match": { "id": "doc-id
|
|
146
|
-
"path": "guides/
|
|
158
|
+
"match": { "id": "specific-doc-id" },
|
|
159
|
+
"path": "guides/getting-started/"
|
|
147
160
|
},
|
|
148
161
|
{
|
|
149
162
|
"match": { "title": "API Reference" },
|
|
150
|
-
"path": "reference/
|
|
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
|
-
|
|
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": "
|
|
178
|
+
"collectionId": "collection-uuid-here",
|
|
172
179
|
"pages": [
|
|
173
180
|
{
|
|
174
|
-
"title": "
|
|
175
|
-
"file": "docs/
|
|
176
|
-
"id": "doc-
|
|
181
|
+
"title": "Getting Started",
|
|
182
|
+
"file": "docs/getting-started/README.md",
|
|
183
|
+
"id": "doc-id-123",
|
|
177
184
|
"children": [
|
|
178
185
|
{
|
|
179
|
-
"title": "
|
|
180
|
-
"file": "docs/
|
|
181
|
-
"id": "doc-
|
|
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
|
-
|
|
199
|
+
## 🗂️ File Organization & Mapping
|
|
195
200
|
|
|
196
|
-
|
|
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
|
-
|
|
208
|
+
**Example Outline Structure:**
|
|
209
|
+
```
|
|
210
|
+
Product Documentation
|
|
211
|
+
├── Getting Started
|
|
212
|
+
│ ├── Installation
|
|
213
|
+
│ └── Configuration
|
|
214
|
+
└── API Reference
|
|
215
|
+
└── Authentication
|
|
216
|
+
```
|
|
205
217
|
|
|
206
|
-
|
|
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
|
-
|
|
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
|
-
|
|
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
|
-
|
|
287
|
+
### Content Comparison
|
|
215
288
|
|
|
216
|
-
|
|
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
|
-
|
|
294
|
+
---
|
|
219
295
|
|
|
220
|
-
|
|
296
|
+
## 🛡️ Safety Features
|
|
221
297
|
|
|
222
|
-
|
|
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
|
-
|
|
226
|
-
|
|
227
|
-
|
|
307
|
+
|
|
308
|
+
### Comprehensive Logging
|
|
309
|
+
```bash
|
|
310
|
+
# Enable detailed debug logging
|
|
311
|
+
bunx @dockstat/outline-sync sync --verbose
|
|
228
312
|
```
|
|
229
313
|
|
|
230
|
-
|
|
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
|
-
|
|
321
|
+
## 🔧 Advanced Usage
|
|
235
322
|
|
|
236
|
-
|
|
237
|
-
|
|
238
|
-
|
|
239
|
-
|
|
240
|
-
|
|
241
|
-
|
|
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
|
-
|
|
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
|
-
|
|
248
|
-
|
|
249
|
-
|
|
250
|
-
|
|
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
|
-
|
|
355
|
+
## 🐛 Troubleshooting
|
|
255
356
|
|
|
256
|
-
|
|
357
|
+
### Common Issues
|
|
257
358
|
|
|
258
|
-
|
|
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
|
-
|
|
261
|
-
|
|
262
|
-
|
|
263
|
-
|
|
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
|
-
|
|
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
|
-
#
|
|
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
|
|
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,18 +1,18 @@
|
|
|
1
1
|
#!/usr/bin/env bun
|
|
2
2
|
// @bun
|
|
3
|
-
var DJ=Object.defineProperty;var
|
|
4
|
-
`,"utf8")}async function o(J){Z.debug("Getting Collection file base");let X=await
|
|
5
|
-
`,"utf8")}var
|
|
6
|
-
`,"utf8")}async function x(J){Z.debug("Getting Collection file base");let X=await
|
|
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
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:
|
|
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}
|
|
9
9
|
|
|
10
|
-
`;if(!Q)await c.mkdir(
|
|
11
|
-
`,"utf8"),Z.info(`Saved manifest: ${V}`),!
|
|
12
|
-
`,"utf8"),Z.info(`Created new config: ${B}`);let
|
|
13
|
-
`,"utf8"),Z.info(`Persisted manifest to ${$}`)}function
|
|
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}
|
|
14
14
|
|
|
15
|
-
`;await
|
|
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(`
|
|
16
16
|
Usage:
|
|
17
17
|
OUTLINE_API_KEY=... bun run bin/cli.ts [command] [--collection=ID]... [--dry-run] [--api-key="..."]
|
|
18
18
|
|
|
@@ -33,8 +33,7 @@ Flags:
|
|
|
33
33
|
Examples:
|
|
34
34
|
OUTLINE_API_KEY=... bunx @dockstat/outline-sync --collection="id1" --collection="id2" sync --dry-run
|
|
35
35
|
bun run bin/cli.ts sync --api-key="sk_xxx" --collection="id1"
|
|
36
|
-
`),process.exit(0);({loadTopConfig:iJ}=await MJ().then(() => BJ)),{listCollectionsPrompt:TJ,bootstrapCollection:
|
|
37
|
-
==> bootstrapping collection ${$} (dryRun=${v})`),await lJ({collectionId:$,dryRun:v});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 tJ({collectionId:K,mode:$,dryRun:v});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};
|
|
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};
|
|
38
37
|
|
|
39
|
-
//# debugId=
|
|
40
|
-
//# 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.configFile || \".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.configFile || \".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  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\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: 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\";\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    console.debug(\"Listing collections\");\n    await listCollectionsPrompt({ dryRun: DRY_RUN, nonInteractive: false });\n    process.exit(0);\n  }\n\n  if (cmd === \"setup\") {\n    console.debug(\"Running setup\");\n    await listCollectionsPrompt({ dryRun: DRY_RUN, nonInteractive: false });\n    process.exit(0);\n  }\n\n  if (cmd === \"init\") {\n    console.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      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    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": "E7F6B78036B50AF664756E2164756E21",
  "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": []
}
|