@sascha384/tic 1.1.0 → 1.2.0

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.
@@ -0,0 +1,164 @@
1
+ import { writeWorkItem, deleteWorkItem as removeWorkItemFile, } from '../backends/local/items.js';
2
+ export class SyncManager {
3
+ local;
4
+ remote;
5
+ queue;
6
+ status;
7
+ listeners = [];
8
+ constructor(local, remote, queue) {
9
+ this.local = local;
10
+ this.remote = remote;
11
+ this.queue = queue;
12
+ this.status = {
13
+ state: 'idle',
14
+ pendingCount: queue.read().pending.length,
15
+ lastSyncTime: null,
16
+ errors: [],
17
+ };
18
+ }
19
+ getStatus() {
20
+ return { ...this.status };
21
+ }
22
+ onStatusChange(cb) {
23
+ this.listeners.push(cb);
24
+ }
25
+ updateStatus(partial) {
26
+ this.status = { ...this.status, ...partial };
27
+ for (const cb of this.listeners) {
28
+ cb(this.getStatus());
29
+ }
30
+ }
31
+ async pushPending() {
32
+ const { pending } = this.queue.read();
33
+ let pushed = 0;
34
+ const errors = [];
35
+ for (const entry of pending) {
36
+ try {
37
+ const resolvedId = await this.pushEntry(entry);
38
+ this.queue.remove(resolvedId, entry.action);
39
+ pushed++;
40
+ }
41
+ catch (e) {
42
+ errors.push({
43
+ entry,
44
+ message: e instanceof Error ? e.message : String(e),
45
+ timestamp: new Date().toISOString(),
46
+ });
47
+ }
48
+ }
49
+ this.updateStatus({
50
+ pendingCount: this.queue.read().pending.length,
51
+ errors,
52
+ });
53
+ return { pushed, failed: errors.length, errors };
54
+ }
55
+ // eslint-disable-next-line @typescript-eslint/require-await
56
+ async pushEntry(entry) {
57
+ switch (entry.action) {
58
+ case 'create': {
59
+ const localItem = this.local.getWorkItem(entry.itemId);
60
+ const remoteItem = this.remote.createWorkItem({
61
+ title: localItem.title,
62
+ type: localItem.type,
63
+ status: localItem.status,
64
+ priority: localItem.priority,
65
+ assignee: localItem.assignee,
66
+ labels: localItem.labels,
67
+ iteration: localItem.iteration,
68
+ description: localItem.description,
69
+ parent: localItem.parent,
70
+ dependsOn: localItem.dependsOn,
71
+ });
72
+ if (remoteItem.id !== entry.itemId) {
73
+ this.renameLocalItem(entry.itemId, remoteItem.id);
74
+ this.queue.renameItem(entry.itemId, remoteItem.id);
75
+ return remoteItem.id;
76
+ }
77
+ return entry.itemId;
78
+ }
79
+ case 'update': {
80
+ const localItem = this.local.getWorkItem(entry.itemId);
81
+ this.remote.updateWorkItem(entry.itemId, {
82
+ title: localItem.title,
83
+ type: localItem.type,
84
+ status: localItem.status,
85
+ priority: localItem.priority,
86
+ assignee: localItem.assignee,
87
+ labels: localItem.labels,
88
+ iteration: localItem.iteration,
89
+ description: localItem.description,
90
+ parent: localItem.parent,
91
+ dependsOn: localItem.dependsOn,
92
+ });
93
+ return entry.itemId;
94
+ }
95
+ case 'delete': {
96
+ this.remote.deleteWorkItem(entry.itemId);
97
+ return entry.itemId;
98
+ }
99
+ case 'comment': {
100
+ if (entry.commentData) {
101
+ this.remote.addComment(entry.itemId, {
102
+ author: entry.commentData.author,
103
+ body: entry.commentData.body,
104
+ });
105
+ }
106
+ return entry.itemId;
107
+ }
108
+ default:
109
+ return entry.itemId;
110
+ }
111
+ }
112
+ renameLocalItem(oldId, newId) {
113
+ const item = this.local.getWorkItem(oldId);
114
+ const root = this.local.getRoot();
115
+ const renamedItem = { ...item, id: newId };
116
+ writeWorkItem(root, renamedItem);
117
+ removeWorkItemFile(root, oldId);
118
+ const allItems = this.local.listWorkItems();
119
+ for (const other of allItems) {
120
+ let changed = false;
121
+ if (other.parent === oldId) {
122
+ other.parent = newId;
123
+ changed = true;
124
+ }
125
+ if (other.dependsOn.includes(oldId)) {
126
+ other.dependsOn = other.dependsOn.map((d) => (d === oldId ? newId : d));
127
+ changed = true;
128
+ }
129
+ if (changed) {
130
+ writeWorkItem(root, other);
131
+ }
132
+ }
133
+ }
134
+ async sync() {
135
+ this.updateStatus({ state: 'syncing' });
136
+ const push = await this.pushPending();
137
+ const pullCount = await this.pull();
138
+ this.updateStatus({
139
+ state: push.errors.length > 0 ? 'error' : 'idle',
140
+ pendingCount: this.queue.read().pending.length,
141
+ lastSyncTime: new Date(),
142
+ });
143
+ return { push, pullCount };
144
+ }
145
+ // eslint-disable-next-line @typescript-eslint/require-await
146
+ async pull() {
147
+ const remoteItems = this.remote.listWorkItems();
148
+ const root = this.local.getRoot();
149
+ const pendingIds = new Set(this.queue.read().pending.map((e) => e.itemId));
150
+ const localItems = this.local.listWorkItems();
151
+ const localIds = new Set(localItems.map((i) => i.id));
152
+ const remoteIds = new Set(remoteItems.map((i) => i.id));
153
+ for (const item of remoteItems) {
154
+ writeWorkItem(root, item);
155
+ }
156
+ for (const localId of localIds) {
157
+ if (!remoteIds.has(localId) && !pendingIds.has(localId)) {
158
+ removeWorkItemFile(root, localId);
159
+ }
160
+ }
161
+ return remoteItems.length;
162
+ }
163
+ }
164
+ //# sourceMappingURL=SyncManager.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"SyncManager.js","sourceRoot":"","sources":["../../src/sync/SyncManager.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,11 @@
1
+ import type { QueueAction, QueueEntry, SyncQueue } from './types.js';
2
+ export declare class SyncQueueStore {
3
+ private filePath;
4
+ constructor(root: string);
5
+ read(): SyncQueue;
6
+ private write;
7
+ append(entry: QueueEntry): void;
8
+ remove(itemId: string, action: QueueAction): void;
9
+ clear(): void;
10
+ renameItem(oldId: string, newId: string): void;
11
+ }
@@ -0,0 +1,52 @@
1
+ import fs from 'node:fs';
2
+ import path from 'node:path';
3
+ export class SyncQueueStore {
4
+ filePath;
5
+ constructor(root) {
6
+ this.filePath = path.join(root, '.tic', 'sync-queue.json');
7
+ }
8
+ read() {
9
+ try {
10
+ if (!fs.existsSync(this.filePath))
11
+ return { pending: [] };
12
+ const raw = fs.readFileSync(this.filePath, 'utf-8');
13
+ const data = JSON.parse(raw);
14
+ if (!Array.isArray(data.pending))
15
+ return { pending: [] };
16
+ return data;
17
+ }
18
+ catch {
19
+ return { pending: [] };
20
+ }
21
+ }
22
+ write(queue) {
23
+ const dir = path.dirname(this.filePath);
24
+ fs.mkdirSync(dir, { recursive: true });
25
+ fs.writeFileSync(this.filePath, JSON.stringify(queue, null, 2));
26
+ }
27
+ append(entry) {
28
+ const queue = this.read();
29
+ // Collapse: remove existing entry with same itemId + action
30
+ queue.pending = queue.pending.filter((e) => !(e.itemId === entry.itemId && e.action === entry.action));
31
+ queue.pending.push(entry);
32
+ this.write(queue);
33
+ }
34
+ remove(itemId, action) {
35
+ const queue = this.read();
36
+ queue.pending = queue.pending.filter((e) => !(e.itemId === itemId && e.action === action));
37
+ this.write(queue);
38
+ }
39
+ clear() {
40
+ this.write({ pending: [] });
41
+ }
42
+ renameItem(oldId, newId) {
43
+ const queue = this.read();
44
+ for (const entry of queue.pending) {
45
+ if (entry.itemId === oldId) {
46
+ entry.itemId = newId;
47
+ }
48
+ }
49
+ this.write(queue);
50
+ }
51
+ }
52
+ //# sourceMappingURL=queue.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"queue.js","sourceRoot":"","sources":["../../src/sync/queue.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,MAAM,SAAS,CAAC;AACzB,OAAO,IAAI,MAAM,WAAW,CAAC;AAG7B,MAAM,OAAO,cAAc;IACjB,QAAQ,CAAS;IAEzB,YAAY,IAAY;QACtB,IAAI,CAAC,QAAQ,GAAG,IAAI,CAAC,IAAI,CAAC,IAAI,EAAE,MAAM,EAAE,iBAAiB,CAAC,CAAC;IAC7D,CAAC;IAED,IAAI;QACF,IAAI,CAAC;YACH,IAAI,CAAC,EAAE,CAAC,UAAU,CAAC,IAAI,CAAC,QAAQ,CAAC;gBAAE,OAAO,EAAE,OAAO,EAAE,EAAE,EAAE,CAAC;YAC1D,MAAM,GAAG,GAAG,EAAE,CAAC,YAAY,CAAC,IAAI,CAAC,QAAQ,EAAE,OAAO,CAAC,CAAC;YACpD,MAAM,IAAI,GAAG,IAAI,CAAC,KAAK,CAAC,GAAG,CAAc,CAAC;YAC1C,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,IAAI,CAAC,OAAO,CAAC;gBAAE,OAAO,EAAE,OAAO,EAAE,EAAE,EAAE,CAAC;YACzD,OAAO,IAAI,CAAC;QACd,CAAC;QAAC,MAAM,CAAC;YACP,OAAO,EAAE,OAAO,EAAE,EAAE,EAAE,CAAC;QACzB,CAAC;IACH,CAAC;IAEO,KAAK,CAAC,KAAgB;QAC5B,MAAM,GAAG,GAAG,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC;QACxC,EAAE,CAAC,SAAS,CAAC,GAAG,EAAE,EAAE,SAAS,EAAE,IAAI,EAAE,CAAC,CAAC;QACvC,EAAE,CAAC,aAAa,CAAC,IAAI,CAAC,QAAQ,EAAE,IAAI,CAAC,SAAS,CAAC,KAAK,EAAE,IAAI,EAAE,CAAC,CAAC,CAAC,CAAC;IAClE,CAAC;IAED,MAAM,CAAC,KAAiB;QACtB,MAAM,KAAK,GAAG,IAAI,CAAC,IAAI,EAAE,CAAC;QAC1B,4DAA4D;QAC5D,KAAK,CAAC,OAAO,GAAG,KAAK,CAAC,OAAO,CAAC,MAAM,CAClC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,CAAC,CAAC,MAAM,KAAK,KAAK,CAAC,MAAM,IAAI,CAAC,CAAC,MAAM,KAAK,KAAK,CAAC,MAAM,CAAC,CACjE,CAAC;QACF,KAAK,CAAC,OAAO,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;QAC1B,IAAI,CAAC,KAAK,CAAC,KAAK,CAAC,CAAC;IACpB,CAAC;IAED,MAAM,CAAC,MAAc,EAAE,MAAmB;QACxC,MAAM,KAAK,GAAG,IAAI,CAAC,IAAI,EAAE,CAAC;QAC1B,KAAK,CAAC,OAAO,GAAG,KAAK,CAAC,OAAO,CAAC,MAAM,CAClC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,CAAC,CAAC,MAAM,KAAK,MAAM,IAAI,CAAC,CAAC,MAAM,KAAK,MAAM,CAAC,CACrD,CAAC;QACF,IAAI,CAAC,KAAK,CAAC,KAAK,CAAC,CAAC;IACpB,CAAC;IAED,KAAK;QACH,IAAI,CAAC,KAAK,CAAC,EAAE,OAAO,EAAE,EAAE,EAAE,CAAC,CAAC;IAC9B,CAAC;IAED,UAAU,CAAC,KAAa,EAAE,KAAa;QACrC,MAAM,KAAK,GAAG,IAAI,CAAC,IAAI,EAAE,CAAC;QAC1B,KAAK,MAAM,KAAK,IAAI,KAAK,CAAC,OAAO,EAAE,CAAC;YAClC,IAAI,KAAK,CAAC,MAAM,KAAK,KAAK,EAAE,CAAC;gBAC3B,KAAK,CAAC,MAAM,GAAG,KAAK,CAAC;YACvB,CAAC;QACH,CAAC;QACD,IAAI,CAAC,KAAK,CAAC,KAAK,CAAC,CAAC;IACpB,CAAC;CACF"}
@@ -0,0 +1,34 @@
1
+ export type QueueAction = 'create' | 'update' | 'delete' | 'comment';
2
+ export interface QueueEntry {
3
+ action: QueueAction;
4
+ itemId: string;
5
+ timestamp: string;
6
+ /** For comments: the comment body and author */
7
+ commentData?: {
8
+ author: string;
9
+ body: string;
10
+ };
11
+ }
12
+ export interface SyncQueue {
13
+ pending: QueueEntry[];
14
+ }
15
+ export interface SyncError {
16
+ entry: QueueEntry;
17
+ message: string;
18
+ timestamp: string;
19
+ }
20
+ export interface SyncStatus {
21
+ state: 'idle' | 'syncing' | 'error';
22
+ pendingCount: number;
23
+ lastSyncTime: Date | null;
24
+ errors: SyncError[];
25
+ }
26
+ export interface PushResult {
27
+ pushed: number;
28
+ failed: number;
29
+ errors: SyncError[];
30
+ }
31
+ export interface SyncResult {
32
+ push: PushResult;
33
+ pullCount: number;
34
+ }
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=types.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"types.js","sourceRoot":"","sources":["../../src/sync/types.ts"],"names":[],"mappings":""}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@sascha384/tic",
3
- "version": "1.1.0",
3
+ "version": "1.2.0",
4
4
  "description": "Terminal UI for issue tracking",
5
5
  "type": "module",
6
6
  "bin": {