@hasna/connectors 0.3.16 → 0.4.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.
- package/bin/index.js +71 -1
- package/bin/mcp.js +71 -1
- package/bin/serve.js +70 -0
- package/connectors/connect-asana/.env.example +11 -0
- package/connectors/connect-asana/CLAUDE.md +128 -0
- package/connectors/connect-asana/README.md +193 -0
- package/connectors/connect-asana/package.json +52 -0
- package/connectors/connect-asana/src/api/client.ts +119 -0
- package/connectors/connect-asana/src/api/index.ts +319 -0
- package/connectors/connect-asana/src/cli/index.ts +731 -0
- package/connectors/connect-asana/src/index.ts +19 -0
- package/connectors/connect-asana/src/types/index.ts +270 -0
- package/connectors/connect-asana/src/utils/config.ts +171 -0
- package/connectors/connect-asana/src/utils/output.ts +119 -0
- package/connectors/connect-asana/tsconfig.json +16 -0
- package/connectors/connect-clickup/.env.example +11 -0
- package/connectors/connect-clickup/CLAUDE.md +128 -0
- package/connectors/connect-clickup/README.md +193 -0
- package/connectors/connect-clickup/package.json +52 -0
- package/connectors/connect-clickup/src/api/client.ts +116 -0
- package/connectors/connect-clickup/src/api/index.ts +400 -0
- package/connectors/connect-clickup/src/cli/index.ts +625 -0
- package/connectors/connect-clickup/src/index.ts +19 -0
- package/connectors/connect-clickup/src/types/index.ts +591 -0
- package/connectors/connect-clickup/src/utils/config.ts +157 -0
- package/connectors/connect-clickup/src/utils/output.ts +119 -0
- package/connectors/connect-clickup/tsconfig.json +16 -0
- package/connectors/connect-confluence/.env.example +11 -0
- package/connectors/connect-confluence/CLAUDE.md +272 -0
- package/connectors/connect-confluence/README.md +193 -0
- package/connectors/connect-confluence/package.json +53 -0
- package/connectors/connect-confluence/scripts/release.ts +179 -0
- package/connectors/connect-confluence/src/api/client.ts +213 -0
- package/connectors/connect-confluence/src/api/example.ts +48 -0
- package/connectors/connect-confluence/src/api/index.ts +51 -0
- package/connectors/connect-confluence/src/cli/index.ts +254 -0
- package/connectors/connect-confluence/src/index.ts +103 -0
- package/connectors/connect-confluence/src/types/index.ts +237 -0
- package/connectors/connect-confluence/src/utils/auth.ts +274 -0
- package/connectors/connect-confluence/src/utils/bulk.ts +212 -0
- package/connectors/connect-confluence/src/utils/config.ts +326 -0
- package/connectors/connect-confluence/src/utils/output.ts +175 -0
- package/connectors/connect-confluence/src/utils/settings.ts +114 -0
- package/connectors/connect-confluence/src/utils/storage.ts +198 -0
- package/connectors/connect-confluence/tsconfig.json +16 -0
- package/connectors/connect-jira/.env.example +11 -0
- package/connectors/connect-jira/CLAUDE.md +128 -0
- package/connectors/connect-jira/README.md +193 -0
- package/connectors/connect-jira/package.json +53 -0
- package/connectors/connect-jira/src/api/client.ts +131 -0
- package/connectors/connect-jira/src/api/index.ts +266 -0
- package/connectors/connect-jira/src/cli/index.ts +653 -0
- package/connectors/connect-jira/src/index.ts +23 -0
- package/connectors/connect-jira/src/types/index.ts +448 -0
- package/connectors/connect-jira/src/utils/config.ts +179 -0
- package/connectors/connect-jira/src/utils/output.ts +119 -0
- package/connectors/connect-jira/tsconfig.json +16 -0
- package/connectors/connect-linear/CLAUDE.md +88 -0
- package/connectors/connect-linear/README.md +201 -0
- package/connectors/connect-linear/package.json +45 -0
- package/connectors/connect-linear/src/api/client.ts +62 -0
- package/connectors/connect-linear/src/api/index.ts +46 -0
- package/connectors/connect-linear/src/api/issues.ts +247 -0
- package/connectors/connect-linear/src/api/projects.ts +179 -0
- package/connectors/connect-linear/src/api/teams.ts +125 -0
- package/connectors/connect-linear/src/api/users.ts +112 -0
- package/connectors/connect-linear/src/cli/index.ts +560 -0
- package/connectors/connect-linear/src/index.ts +27 -0
- package/connectors/connect-linear/src/types/index.ts +275 -0
- package/connectors/connect-linear/src/utils/config.ts +249 -0
- package/connectors/connect-linear/src/utils/output.ts +119 -0
- package/connectors/connect-linear/tsconfig.json +16 -0
- package/connectors/connect-slack/.env.example +7 -0
- package/connectors/connect-slack/CLAUDE.md +69 -0
- package/connectors/connect-slack/README.md +150 -0
- package/connectors/connect-slack/package.json +44 -0
- package/connectors/connect-slack/src/api/channels.ts +112 -0
- package/connectors/connect-slack/src/api/client.ts +97 -0
- package/connectors/connect-slack/src/api/index.ts +42 -0
- package/connectors/connect-slack/src/api/messages.ts +127 -0
- package/connectors/connect-slack/src/api/users.ts +110 -0
- package/connectors/connect-slack/src/cli/index.ts +494 -0
- package/connectors/connect-slack/src/index.ts +21 -0
- package/connectors/connect-slack/src/types/index.ts +263 -0
- package/connectors/connect-slack/src/utils/config.ts +297 -0
- package/connectors/connect-slack/src/utils/output.ts +119 -0
- package/connectors/connect-slack/tsconfig.json +16 -0
- package/connectors/connect-telegram/.env.example +2 -0
- package/connectors/connect-telegram/package.json +49 -0
- package/connectors/connect-todoist/.env.example +11 -0
- package/connectors/connect-todoist/CLAUDE.md +104 -0
- package/connectors/connect-todoist/README.md +193 -0
- package/connectors/connect-todoist/package.json +52 -0
- package/connectors/connect-todoist/src/api/client.ts +117 -0
- package/connectors/connect-todoist/src/api/index.ts +188 -0
- package/connectors/connect-todoist/src/cli/index.ts +990 -0
- package/connectors/connect-todoist/src/index.ts +21 -0
- package/connectors/connect-todoist/src/types/index.ts +240 -0
- package/connectors/connect-todoist/src/utils/config.ts +157 -0
- package/connectors/connect-todoist/src/utils/output.ts +119 -0
- package/connectors/connect-todoist/tsconfig.json +16 -0
- package/connectors/connect-trello/.env.example +11 -0
- package/connectors/connect-trello/CLAUDE.md +128 -0
- package/connectors/connect-trello/README.md +193 -0
- package/connectors/connect-trello/package.json +53 -0
- package/connectors/connect-trello/src/api/client.ts +128 -0
- package/connectors/connect-trello/src/api/index.ts +278 -0
- package/connectors/connect-trello/src/cli/index.ts +737 -0
- package/connectors/connect-trello/src/index.ts +21 -0
- package/connectors/connect-trello/src/types/index.ts +314 -0
- package/connectors/connect-trello/src/utils/config.ts +182 -0
- package/connectors/connect-trello/src/utils/output.ts +119 -0
- package/connectors/connect-trello/tsconfig.json +16 -0
- package/connectors/connect-whatsapp/.env.example +11 -0
- package/connectors/connect-whatsapp/CLAUDE.md +113 -0
- package/connectors/connect-whatsapp/README.md +193 -0
- package/connectors/connect-whatsapp/package.json +53 -0
- package/connectors/connect-whatsapp/src/api/client.ts +133 -0
- package/connectors/connect-whatsapp/src/api/index.ts +365 -0
- package/connectors/connect-whatsapp/src/cli/index.ts +686 -0
- package/connectors/connect-whatsapp/src/index.ts +25 -0
- package/connectors/connect-whatsapp/src/types/index.ts +502 -0
- package/connectors/connect-whatsapp/src/utils/config.ts +179 -0
- package/connectors/connect-whatsapp/src/utils/output.ts +119 -0
- package/connectors/connect-whatsapp/tsconfig.json +16 -0
- package/dist/index.js +70 -0
- package/package.json +1 -1
|
@@ -0,0 +1,21 @@
|
|
|
1
|
+
// Trello Connector
|
|
2
|
+
// TypeScript wrapper for Trello boards, lists, cards, and checklists API
|
|
3
|
+
|
|
4
|
+
export { Trello } from './api';
|
|
5
|
+
export * from './types';
|
|
6
|
+
export { TrelloClient } from './api';
|
|
7
|
+
|
|
8
|
+
export {
|
|
9
|
+
getApiKey,
|
|
10
|
+
setApiKey,
|
|
11
|
+
getToken,
|
|
12
|
+
setToken,
|
|
13
|
+
getCurrentProfile,
|
|
14
|
+
setCurrentProfile,
|
|
15
|
+
listProfiles,
|
|
16
|
+
createProfile,
|
|
17
|
+
deleteProfile,
|
|
18
|
+
loadProfile,
|
|
19
|
+
saveProfile,
|
|
20
|
+
clearConfig,
|
|
21
|
+
} from './utils/config';
|
|
@@ -0,0 +1,314 @@
|
|
|
1
|
+
// Trello Connector Types
|
|
2
|
+
// Boards, lists, cards, and checklists management
|
|
3
|
+
|
|
4
|
+
// ============================================
|
|
5
|
+
// Configuration
|
|
6
|
+
// ============================================
|
|
7
|
+
|
|
8
|
+
export interface TrelloConfig {
|
|
9
|
+
apiKey: string;
|
|
10
|
+
token: string;
|
|
11
|
+
baseUrl?: string;
|
|
12
|
+
}
|
|
13
|
+
|
|
14
|
+
// ============================================
|
|
15
|
+
// Common Types
|
|
16
|
+
// ============================================
|
|
17
|
+
|
|
18
|
+
export type OutputFormat = 'json' | 'pretty';
|
|
19
|
+
|
|
20
|
+
// ============================================
|
|
21
|
+
// Member Types
|
|
22
|
+
// ============================================
|
|
23
|
+
|
|
24
|
+
export interface Member {
|
|
25
|
+
id: string;
|
|
26
|
+
username: string;
|
|
27
|
+
fullName: string;
|
|
28
|
+
initials?: string;
|
|
29
|
+
avatarHash?: string;
|
|
30
|
+
avatarUrl?: string;
|
|
31
|
+
email?: string;
|
|
32
|
+
url?: string;
|
|
33
|
+
confirmed?: boolean;
|
|
34
|
+
memberType?: 'admin' | 'normal' | 'observer';
|
|
35
|
+
status?: string;
|
|
36
|
+
idBoards?: string[];
|
|
37
|
+
idOrganizations?: string[];
|
|
38
|
+
}
|
|
39
|
+
|
|
40
|
+
// ============================================
|
|
41
|
+
// Board Types
|
|
42
|
+
// ============================================
|
|
43
|
+
|
|
44
|
+
export interface Board {
|
|
45
|
+
id: string;
|
|
46
|
+
name: string;
|
|
47
|
+
desc?: string;
|
|
48
|
+
descData?: object;
|
|
49
|
+
closed?: boolean;
|
|
50
|
+
idOrganization?: string;
|
|
51
|
+
pinned?: boolean;
|
|
52
|
+
url?: string;
|
|
53
|
+
shortUrl?: string;
|
|
54
|
+
prefs?: BoardPrefs;
|
|
55
|
+
labelNames?: Record<string, string>;
|
|
56
|
+
starred?: boolean;
|
|
57
|
+
memberships?: BoardMembership[];
|
|
58
|
+
shortLink?: string;
|
|
59
|
+
dateLastActivity?: string;
|
|
60
|
+
dateLastView?: string;
|
|
61
|
+
}
|
|
62
|
+
|
|
63
|
+
export interface BoardPrefs {
|
|
64
|
+
permissionLevel?: 'private' | 'org' | 'public';
|
|
65
|
+
voting?: 'disabled' | 'members' | 'observers' | 'org' | 'public';
|
|
66
|
+
comments?: 'disabled' | 'members' | 'observers' | 'org' | 'public';
|
|
67
|
+
invitations?: 'admins' | 'members';
|
|
68
|
+
selfJoin?: boolean;
|
|
69
|
+
cardCovers?: boolean;
|
|
70
|
+
background?: string;
|
|
71
|
+
backgroundColor?: string;
|
|
72
|
+
backgroundImage?: string;
|
|
73
|
+
}
|
|
74
|
+
|
|
75
|
+
export interface BoardMembership {
|
|
76
|
+
id: string;
|
|
77
|
+
idMember: string;
|
|
78
|
+
memberType: 'admin' | 'normal' | 'observer';
|
|
79
|
+
unconfirmed?: boolean;
|
|
80
|
+
deactivated?: boolean;
|
|
81
|
+
}
|
|
82
|
+
|
|
83
|
+
export interface CreateBoardInput {
|
|
84
|
+
name: string;
|
|
85
|
+
desc?: string;
|
|
86
|
+
idOrganization?: string;
|
|
87
|
+
defaultLabels?: boolean;
|
|
88
|
+
defaultLists?: boolean;
|
|
89
|
+
prefs_permissionLevel?: 'private' | 'org' | 'public';
|
|
90
|
+
prefs_voting?: 'disabled' | 'members' | 'observers' | 'org' | 'public';
|
|
91
|
+
prefs_comments?: 'disabled' | 'members' | 'observers' | 'org' | 'public';
|
|
92
|
+
prefs_background?: string;
|
|
93
|
+
}
|
|
94
|
+
|
|
95
|
+
// ============================================
|
|
96
|
+
// List Types
|
|
97
|
+
// ============================================
|
|
98
|
+
|
|
99
|
+
export interface List {
|
|
100
|
+
id: string;
|
|
101
|
+
name: string;
|
|
102
|
+
closed?: boolean;
|
|
103
|
+
idBoard?: string;
|
|
104
|
+
pos?: number;
|
|
105
|
+
subscribed?: boolean;
|
|
106
|
+
}
|
|
107
|
+
|
|
108
|
+
export interface CreateListInput {
|
|
109
|
+
name: string;
|
|
110
|
+
idBoard: string;
|
|
111
|
+
pos?: 'top' | 'bottom' | number;
|
|
112
|
+
}
|
|
113
|
+
|
|
114
|
+
// ============================================
|
|
115
|
+
// Card Types
|
|
116
|
+
// ============================================
|
|
117
|
+
|
|
118
|
+
export interface Card {
|
|
119
|
+
id: string;
|
|
120
|
+
name: string;
|
|
121
|
+
desc?: string;
|
|
122
|
+
closed?: boolean;
|
|
123
|
+
idBoard?: string;
|
|
124
|
+
idList?: string;
|
|
125
|
+
idMembers?: string[];
|
|
126
|
+
idLabels?: string[];
|
|
127
|
+
idChecklists?: string[];
|
|
128
|
+
idAttachmentCover?: string;
|
|
129
|
+
manualCoverAttachment?: boolean;
|
|
130
|
+
pos?: number;
|
|
131
|
+
shortLink?: string;
|
|
132
|
+
shortUrl?: string;
|
|
133
|
+
url?: string;
|
|
134
|
+
due?: string;
|
|
135
|
+
dueComplete?: boolean;
|
|
136
|
+
start?: string;
|
|
137
|
+
dateLastActivity?: string;
|
|
138
|
+
labels?: Label[];
|
|
139
|
+
badges?: CardBadges;
|
|
140
|
+
subscribed?: boolean;
|
|
141
|
+
cover?: CardCover;
|
|
142
|
+
}
|
|
143
|
+
|
|
144
|
+
export interface CardBadges {
|
|
145
|
+
votes?: number;
|
|
146
|
+
viewingMemberVoted?: boolean;
|
|
147
|
+
subscribed?: boolean;
|
|
148
|
+
fogbugz?: string;
|
|
149
|
+
checkItems?: number;
|
|
150
|
+
checkItemsChecked?: number;
|
|
151
|
+
comments?: number;
|
|
152
|
+
attachments?: number;
|
|
153
|
+
description?: boolean;
|
|
154
|
+
due?: string;
|
|
155
|
+
dueComplete?: boolean;
|
|
156
|
+
start?: string;
|
|
157
|
+
}
|
|
158
|
+
|
|
159
|
+
export interface CardCover {
|
|
160
|
+
idAttachment?: string;
|
|
161
|
+
color?: string;
|
|
162
|
+
idUploadedBackground?: string;
|
|
163
|
+
size?: 'normal' | 'full';
|
|
164
|
+
brightness?: 'dark' | 'light';
|
|
165
|
+
}
|
|
166
|
+
|
|
167
|
+
export interface CreateCardInput {
|
|
168
|
+
name: string;
|
|
169
|
+
idList: string;
|
|
170
|
+
desc?: string;
|
|
171
|
+
pos?: 'top' | 'bottom' | number;
|
|
172
|
+
due?: string;
|
|
173
|
+
start?: string;
|
|
174
|
+
dueComplete?: boolean;
|
|
175
|
+
idMembers?: string[];
|
|
176
|
+
idLabels?: string[];
|
|
177
|
+
urlSource?: string;
|
|
178
|
+
}
|
|
179
|
+
|
|
180
|
+
// ============================================
|
|
181
|
+
// Label Types
|
|
182
|
+
// ============================================
|
|
183
|
+
|
|
184
|
+
export interface Label {
|
|
185
|
+
id: string;
|
|
186
|
+
idBoard: string;
|
|
187
|
+
name: string;
|
|
188
|
+
color?: 'green' | 'yellow' | 'orange' | 'red' | 'purple' | 'blue' | 'sky' | 'lime' | 'pink' | 'black' | null;
|
|
189
|
+
}
|
|
190
|
+
|
|
191
|
+
export interface CreateLabelInput {
|
|
192
|
+
name: string;
|
|
193
|
+
color: 'green' | 'yellow' | 'orange' | 'red' | 'purple' | 'blue' | 'sky' | 'lime' | 'pink' | 'black';
|
|
194
|
+
idBoard: string;
|
|
195
|
+
}
|
|
196
|
+
|
|
197
|
+
// ============================================
|
|
198
|
+
// Checklist Types
|
|
199
|
+
// ============================================
|
|
200
|
+
|
|
201
|
+
export interface Checklist {
|
|
202
|
+
id: string;
|
|
203
|
+
name: string;
|
|
204
|
+
idBoard?: string;
|
|
205
|
+
idCard?: string;
|
|
206
|
+
pos?: number;
|
|
207
|
+
checkItems?: CheckItem[];
|
|
208
|
+
}
|
|
209
|
+
|
|
210
|
+
export interface CheckItem {
|
|
211
|
+
id: string;
|
|
212
|
+
name: string;
|
|
213
|
+
state?: 'incomplete' | 'complete';
|
|
214
|
+
pos?: number;
|
|
215
|
+
idChecklist?: string;
|
|
216
|
+
due?: string;
|
|
217
|
+
idMember?: string;
|
|
218
|
+
}
|
|
219
|
+
|
|
220
|
+
export interface CreateChecklistInput {
|
|
221
|
+
name: string;
|
|
222
|
+
idCard: string;
|
|
223
|
+
pos?: 'top' | 'bottom' | number;
|
|
224
|
+
}
|
|
225
|
+
|
|
226
|
+
export interface CreateCheckItemInput {
|
|
227
|
+
name: string;
|
|
228
|
+
pos?: 'top' | 'bottom' | number;
|
|
229
|
+
checked?: boolean;
|
|
230
|
+
due?: string;
|
|
231
|
+
idMember?: string;
|
|
232
|
+
}
|
|
233
|
+
|
|
234
|
+
// ============================================
|
|
235
|
+
// Comment Types
|
|
236
|
+
// ============================================
|
|
237
|
+
|
|
238
|
+
export interface Action {
|
|
239
|
+
id: string;
|
|
240
|
+
idMemberCreator: string;
|
|
241
|
+
data?: {
|
|
242
|
+
text?: string;
|
|
243
|
+
card?: { id: string; name: string };
|
|
244
|
+
board?: { id: string; name: string };
|
|
245
|
+
list?: { id: string; name: string };
|
|
246
|
+
};
|
|
247
|
+
type: string;
|
|
248
|
+
date?: string;
|
|
249
|
+
memberCreator?: Member;
|
|
250
|
+
}
|
|
251
|
+
|
|
252
|
+
// ============================================
|
|
253
|
+
// Attachment Types
|
|
254
|
+
// ============================================
|
|
255
|
+
|
|
256
|
+
export interface Attachment {
|
|
257
|
+
id: string;
|
|
258
|
+
name: string;
|
|
259
|
+
url?: string;
|
|
260
|
+
bytes?: number;
|
|
261
|
+
date?: string;
|
|
262
|
+
edgeColor?: string;
|
|
263
|
+
idMember?: string;
|
|
264
|
+
isUpload?: boolean;
|
|
265
|
+
mimeType?: string;
|
|
266
|
+
pos?: number;
|
|
267
|
+
previews?: AttachmentPreview[];
|
|
268
|
+
}
|
|
269
|
+
|
|
270
|
+
export interface AttachmentPreview {
|
|
271
|
+
id?: string;
|
|
272
|
+
url?: string;
|
|
273
|
+
width?: number;
|
|
274
|
+
height?: number;
|
|
275
|
+
bytes?: number;
|
|
276
|
+
scaled?: boolean;
|
|
277
|
+
}
|
|
278
|
+
|
|
279
|
+
// ============================================
|
|
280
|
+
// Organization Types
|
|
281
|
+
// ============================================
|
|
282
|
+
|
|
283
|
+
export interface Organization {
|
|
284
|
+
id: string;
|
|
285
|
+
name: string;
|
|
286
|
+
displayName?: string;
|
|
287
|
+
desc?: string;
|
|
288
|
+
url?: string;
|
|
289
|
+
website?: string;
|
|
290
|
+
logoHash?: string;
|
|
291
|
+
products?: number[];
|
|
292
|
+
powerUps?: number[];
|
|
293
|
+
}
|
|
294
|
+
|
|
295
|
+
// ============================================
|
|
296
|
+
// API Error Types
|
|
297
|
+
// ============================================
|
|
298
|
+
|
|
299
|
+
export interface TrelloErrorDetail {
|
|
300
|
+
message: string;
|
|
301
|
+
error?: string;
|
|
302
|
+
}
|
|
303
|
+
|
|
304
|
+
export class TrelloApiError extends Error {
|
|
305
|
+
public readonly statusCode: number;
|
|
306
|
+
public readonly errorInfo?: string;
|
|
307
|
+
|
|
308
|
+
constructor(message: string, statusCode: number, errorInfo?: string) {
|
|
309
|
+
super(message);
|
|
310
|
+
this.name = 'TrelloApiError';
|
|
311
|
+
this.statusCode = statusCode;
|
|
312
|
+
this.errorInfo = errorInfo;
|
|
313
|
+
}
|
|
314
|
+
}
|
|
@@ -0,0 +1,182 @@
|
|
|
1
|
+
import { existsSync, readFileSync, writeFileSync, mkdirSync, readdirSync, rmSync } from 'fs';
|
|
2
|
+
import { homedir } from 'os';
|
|
3
|
+
import { join } from 'path';
|
|
4
|
+
|
|
5
|
+
const CONNECTOR_NAME = 'connect-trello';
|
|
6
|
+
const DEFAULT_PROFILE = 'default';
|
|
7
|
+
|
|
8
|
+
export interface ProfileConfig {
|
|
9
|
+
apiKey?: string;
|
|
10
|
+
token?: string;
|
|
11
|
+
}
|
|
12
|
+
|
|
13
|
+
// Store for --profile flag override (set by CLI before commands run)
|
|
14
|
+
let profileOverride: string | undefined;
|
|
15
|
+
|
|
16
|
+
// Config directory: ~/.connect/{connector-name}/
|
|
17
|
+
const CONFIG_DIR = join(homedir(), '.connect', CONNECTOR_NAME);
|
|
18
|
+
const PROFILES_DIR = join(CONFIG_DIR, 'profiles');
|
|
19
|
+
const CURRENT_PROFILE_FILE = join(CONFIG_DIR, 'current_profile');
|
|
20
|
+
|
|
21
|
+
// ============================================
|
|
22
|
+
// Profile Management
|
|
23
|
+
// ============================================
|
|
24
|
+
|
|
25
|
+
export function setProfileOverride(profile: string | undefined): void {
|
|
26
|
+
profileOverride = profile;
|
|
27
|
+
}
|
|
28
|
+
|
|
29
|
+
export function ensureConfigDir(): void {
|
|
30
|
+
if (!existsSync(CONFIG_DIR)) {
|
|
31
|
+
mkdirSync(CONFIG_DIR, { recursive: true });
|
|
32
|
+
}
|
|
33
|
+
if (!existsSync(PROFILES_DIR)) {
|
|
34
|
+
mkdirSync(PROFILES_DIR, { recursive: true });
|
|
35
|
+
}
|
|
36
|
+
}
|
|
37
|
+
|
|
38
|
+
function getProfilePath(profile: string): string {
|
|
39
|
+
return join(PROFILES_DIR, `${profile}.json`);
|
|
40
|
+
}
|
|
41
|
+
|
|
42
|
+
export function getCurrentProfile(): string {
|
|
43
|
+
if (profileOverride) {
|
|
44
|
+
return profileOverride;
|
|
45
|
+
}
|
|
46
|
+
|
|
47
|
+
ensureConfigDir();
|
|
48
|
+
|
|
49
|
+
if (existsSync(CURRENT_PROFILE_FILE)) {
|
|
50
|
+
try {
|
|
51
|
+
const profile = readFileSync(CURRENT_PROFILE_FILE, 'utf-8').trim();
|
|
52
|
+
if (profile && profileExists(profile)) {
|
|
53
|
+
return profile;
|
|
54
|
+
}
|
|
55
|
+
} catch {
|
|
56
|
+
// Fall through to default
|
|
57
|
+
}
|
|
58
|
+
}
|
|
59
|
+
|
|
60
|
+
return DEFAULT_PROFILE;
|
|
61
|
+
}
|
|
62
|
+
|
|
63
|
+
export function setCurrentProfile(profile: string): void {
|
|
64
|
+
ensureConfigDir();
|
|
65
|
+
|
|
66
|
+
if (!profileExists(profile) && profile !== DEFAULT_PROFILE) {
|
|
67
|
+
throw new Error(`Profile "${profile}" does not exist`);
|
|
68
|
+
}
|
|
69
|
+
|
|
70
|
+
writeFileSync(CURRENT_PROFILE_FILE, profile);
|
|
71
|
+
}
|
|
72
|
+
|
|
73
|
+
export function profileExists(profile: string): boolean {
|
|
74
|
+
return existsSync(getProfilePath(profile));
|
|
75
|
+
}
|
|
76
|
+
|
|
77
|
+
export function listProfiles(): string[] {
|
|
78
|
+
ensureConfigDir();
|
|
79
|
+
|
|
80
|
+
if (!existsSync(PROFILES_DIR)) {
|
|
81
|
+
return [];
|
|
82
|
+
}
|
|
83
|
+
|
|
84
|
+
return readdirSync(PROFILES_DIR)
|
|
85
|
+
.filter(f => f.endsWith('.json'))
|
|
86
|
+
.map(f => f.replace('.json', ''))
|
|
87
|
+
.sort();
|
|
88
|
+
}
|
|
89
|
+
|
|
90
|
+
export function createProfile(profile: string, config: ProfileConfig = {}): boolean {
|
|
91
|
+
ensureConfigDir();
|
|
92
|
+
|
|
93
|
+
if (profileExists(profile)) {
|
|
94
|
+
return false;
|
|
95
|
+
}
|
|
96
|
+
|
|
97
|
+
if (!/^[a-zA-Z0-9_-]+$/.test(profile)) {
|
|
98
|
+
throw new Error('Profile name can only contain letters, numbers, hyphens, and underscores');
|
|
99
|
+
}
|
|
100
|
+
|
|
101
|
+
writeFileSync(getProfilePath(profile), JSON.stringify(config, null, 2));
|
|
102
|
+
return true;
|
|
103
|
+
}
|
|
104
|
+
|
|
105
|
+
export function deleteProfile(profile: string): boolean {
|
|
106
|
+
if (profile === DEFAULT_PROFILE) {
|
|
107
|
+
return false;
|
|
108
|
+
}
|
|
109
|
+
|
|
110
|
+
if (!profileExists(profile)) {
|
|
111
|
+
return false;
|
|
112
|
+
}
|
|
113
|
+
|
|
114
|
+
if (getCurrentProfile() === profile) {
|
|
115
|
+
setCurrentProfile(DEFAULT_PROFILE);
|
|
116
|
+
}
|
|
117
|
+
|
|
118
|
+
rmSync(getProfilePath(profile));
|
|
119
|
+
return true;
|
|
120
|
+
}
|
|
121
|
+
|
|
122
|
+
export function loadProfile(profile?: string): ProfileConfig {
|
|
123
|
+
ensureConfigDir();
|
|
124
|
+
const profileName = profile || getCurrentProfile();
|
|
125
|
+
const profilePath = getProfilePath(profileName);
|
|
126
|
+
|
|
127
|
+
if (!existsSync(profilePath)) {
|
|
128
|
+
return {};
|
|
129
|
+
}
|
|
130
|
+
|
|
131
|
+
try {
|
|
132
|
+
return JSON.parse(readFileSync(profilePath, 'utf-8'));
|
|
133
|
+
} catch {
|
|
134
|
+
return {};
|
|
135
|
+
}
|
|
136
|
+
}
|
|
137
|
+
|
|
138
|
+
export function saveProfile(config: ProfileConfig, profile?: string): void {
|
|
139
|
+
ensureConfigDir();
|
|
140
|
+
const profileName = profile || getCurrentProfile();
|
|
141
|
+
writeFileSync(getProfilePath(profileName), JSON.stringify(config, null, 2));
|
|
142
|
+
}
|
|
143
|
+
|
|
144
|
+
// ============================================
|
|
145
|
+
// API Key and Token Management
|
|
146
|
+
// ============================================
|
|
147
|
+
|
|
148
|
+
export function getApiKey(): string | undefined {
|
|
149
|
+
return process.env.TRELLO_API_KEY || loadProfile().apiKey;
|
|
150
|
+
}
|
|
151
|
+
|
|
152
|
+
export function setApiKey(apiKey: string): void {
|
|
153
|
+
const config = loadProfile();
|
|
154
|
+
config.apiKey = apiKey;
|
|
155
|
+
saveProfile(config);
|
|
156
|
+
}
|
|
157
|
+
|
|
158
|
+
export function getToken(): string | undefined {
|
|
159
|
+
return process.env.TRELLO_TOKEN || loadProfile().token;
|
|
160
|
+
}
|
|
161
|
+
|
|
162
|
+
export function setToken(token: string): void {
|
|
163
|
+
const config = loadProfile();
|
|
164
|
+
config.token = token;
|
|
165
|
+
saveProfile(config);
|
|
166
|
+
}
|
|
167
|
+
|
|
168
|
+
// ============================================
|
|
169
|
+
// Utility Functions
|
|
170
|
+
// ============================================
|
|
171
|
+
|
|
172
|
+
export function clearConfig(): void {
|
|
173
|
+
saveProfile({});
|
|
174
|
+
}
|
|
175
|
+
|
|
176
|
+
export function getConfigDir(): string {
|
|
177
|
+
return CONFIG_DIR;
|
|
178
|
+
}
|
|
179
|
+
|
|
180
|
+
export function getActiveProfileName(): string {
|
|
181
|
+
return getCurrentProfile();
|
|
182
|
+
}
|
|
@@ -0,0 +1,119 @@
|
|
|
1
|
+
import chalk from 'chalk';
|
|
2
|
+
|
|
3
|
+
export type OutputFormat = 'json' | 'table' | 'pretty';
|
|
4
|
+
|
|
5
|
+
export function formatOutput(data: unknown, format: OutputFormat = 'pretty'): string {
|
|
6
|
+
switch (format) {
|
|
7
|
+
case 'json':
|
|
8
|
+
return JSON.stringify(data, null, 2);
|
|
9
|
+
case 'table':
|
|
10
|
+
return formatAsTable(data);
|
|
11
|
+
case 'pretty':
|
|
12
|
+
default:
|
|
13
|
+
return formatPretty(data);
|
|
14
|
+
}
|
|
15
|
+
}
|
|
16
|
+
|
|
17
|
+
function formatAsTable(data: unknown): string {
|
|
18
|
+
if (!Array.isArray(data)) {
|
|
19
|
+
data = [data];
|
|
20
|
+
}
|
|
21
|
+
|
|
22
|
+
const items = data as Record<string, unknown>[];
|
|
23
|
+
if (items.length === 0) {
|
|
24
|
+
return 'No data';
|
|
25
|
+
}
|
|
26
|
+
|
|
27
|
+
const firstItem = items[0];
|
|
28
|
+
if (!firstItem || typeof firstItem !== 'object') {
|
|
29
|
+
return 'No data';
|
|
30
|
+
}
|
|
31
|
+
|
|
32
|
+
const keys = Object.keys(firstItem);
|
|
33
|
+
const colWidths = keys.map(key => {
|
|
34
|
+
const maxValue = Math.max(
|
|
35
|
+
key.length,
|
|
36
|
+
...items.map(item => String(item[key] ?? '').length)
|
|
37
|
+
);
|
|
38
|
+
return Math.min(maxValue, 40);
|
|
39
|
+
});
|
|
40
|
+
|
|
41
|
+
const header = keys.map((key, i) => key.padEnd(colWidths[i] ?? 10)).join(' | ');
|
|
42
|
+
const separator = colWidths.map(w => '-'.repeat(w)).join('-+-');
|
|
43
|
+
|
|
44
|
+
const rows = items.map(item =>
|
|
45
|
+
keys.map((key, i) => {
|
|
46
|
+
const value = String(item[key] ?? '');
|
|
47
|
+
const width = colWidths[i] ?? 10;
|
|
48
|
+
return value.length > width
|
|
49
|
+
? value.substring(0, width - 3) + '...'
|
|
50
|
+
: value.padEnd(width);
|
|
51
|
+
}).join(' | ')
|
|
52
|
+
);
|
|
53
|
+
|
|
54
|
+
return [header, separator, ...rows].join('\n');
|
|
55
|
+
}
|
|
56
|
+
|
|
57
|
+
function formatPretty(data: unknown): string {
|
|
58
|
+
if (Array.isArray(data)) {
|
|
59
|
+
return data.map((item, i) => `${chalk.cyan(`[${i + 1}]`)} ${formatPrettyItem(item)}`).join('\n\n');
|
|
60
|
+
}
|
|
61
|
+
return formatPrettyItem(data);
|
|
62
|
+
}
|
|
63
|
+
|
|
64
|
+
function formatPrettyItem(item: unknown, indent = 0): string {
|
|
65
|
+
if (item === null || item === undefined) {
|
|
66
|
+
return chalk.gray('null');
|
|
67
|
+
}
|
|
68
|
+
|
|
69
|
+
if (typeof item !== 'object') {
|
|
70
|
+
return String(item);
|
|
71
|
+
}
|
|
72
|
+
|
|
73
|
+
const spaces = ' '.repeat(indent);
|
|
74
|
+
const entries = Object.entries(item as Record<string, unknown>);
|
|
75
|
+
|
|
76
|
+
return entries
|
|
77
|
+
.map(([key, value]) => {
|
|
78
|
+
if (Array.isArray(value)) {
|
|
79
|
+
if (value.length === 0) {
|
|
80
|
+
return `${spaces}${chalk.blue(key)}: ${chalk.gray('[]')}`;
|
|
81
|
+
}
|
|
82
|
+
if (typeof value[0] === 'object') {
|
|
83
|
+
return `${spaces}${chalk.blue(key)}:\n${value.map(v => formatPrettyItem(v, indent + 1)).join('\n')}`;
|
|
84
|
+
}
|
|
85
|
+
return `${spaces}${chalk.blue(key)}: ${value.join(', ')}`;
|
|
86
|
+
}
|
|
87
|
+
|
|
88
|
+
if (typeof value === 'object' && value !== null) {
|
|
89
|
+
return `${spaces}${chalk.blue(key)}:\n${formatPrettyItem(value, indent + 1)}`;
|
|
90
|
+
}
|
|
91
|
+
|
|
92
|
+
return `${spaces}${chalk.blue(key)}: ${chalk.white(String(value))}`;
|
|
93
|
+
})
|
|
94
|
+
.join('\n');
|
|
95
|
+
}
|
|
96
|
+
|
|
97
|
+
export function success(message: string): void {
|
|
98
|
+
console.log(chalk.green('✓'), message);
|
|
99
|
+
}
|
|
100
|
+
|
|
101
|
+
export function error(message: string): void {
|
|
102
|
+
console.error(chalk.red('✗'), message);
|
|
103
|
+
}
|
|
104
|
+
|
|
105
|
+
export function warn(message: string): void {
|
|
106
|
+
console.warn(chalk.yellow('⚠'), message);
|
|
107
|
+
}
|
|
108
|
+
|
|
109
|
+
export function info(message: string): void {
|
|
110
|
+
console.log(chalk.blue('ℹ'), message);
|
|
111
|
+
}
|
|
112
|
+
|
|
113
|
+
export function heading(message: string): void {
|
|
114
|
+
console.log(chalk.bold.cyan(`\n${message}\n`));
|
|
115
|
+
}
|
|
116
|
+
|
|
117
|
+
export function print(data: unknown, format: OutputFormat = 'pretty'): void {
|
|
118
|
+
console.log(formatOutput(data, format));
|
|
119
|
+
}
|
|
@@ -0,0 +1,16 @@
|
|
|
1
|
+
{
|
|
2
|
+
"compilerOptions": {
|
|
3
|
+
"target": "ESNext",
|
|
4
|
+
"module": "ESNext",
|
|
5
|
+
"moduleResolution": "bundler",
|
|
6
|
+
"esModuleInterop": true,
|
|
7
|
+
"strict": true,
|
|
8
|
+
"skipLibCheck": true,
|
|
9
|
+
"declaration": true,
|
|
10
|
+
"outDir": "./dist",
|
|
11
|
+
"rootDir": "./src",
|
|
12
|
+
"types": ["bun-types"]
|
|
13
|
+
},
|
|
14
|
+
"include": ["src/**/*"],
|
|
15
|
+
"exclude": ["node_modules", "dist", "bin"]
|
|
16
|
+
}
|
|
@@ -0,0 +1,11 @@
|
|
|
1
|
+
# API Credentials
|
|
2
|
+
# TODO: Update variable names and instructions for your API
|
|
3
|
+
# Example: PERPLEXITY_API_KEY, OPENAI_API_KEY, etc.
|
|
4
|
+
|
|
5
|
+
CONNECTOR_API_KEY=your-api-key-here
|
|
6
|
+
|
|
7
|
+
# Optional: API secret (if your API requires it)
|
|
8
|
+
# CONNECTOR_API_SECRET=your-api-secret-here
|
|
9
|
+
|
|
10
|
+
# Optional: Custom base URL (if needed)
|
|
11
|
+
# CONNECTOR_BASE_URL=https://api.example.com
|