@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,193 @@
|
|
|
1
|
+
# scaffold-connector
|
|
2
|
+
|
|
3
|
+
A TypeScript scaffold for building API connector CLIs with multi-profile support.
|
|
4
|
+
|
|
5
|
+
## Features
|
|
6
|
+
|
|
7
|
+
- Multi-profile configuration (switch between different API keys/accounts)
|
|
8
|
+
- Bearer token authentication (easily customizable)
|
|
9
|
+
- Clean CLI structure with Commander.js
|
|
10
|
+
- Pretty and JSON output formats
|
|
11
|
+
- TypeScript with strict mode
|
|
12
|
+
|
|
13
|
+
## Quick Start
|
|
14
|
+
|
|
15
|
+
### 1. Clone and Rename
|
|
16
|
+
|
|
17
|
+
```bash
|
|
18
|
+
# Clone for your connector
|
|
19
|
+
git clone https://github.com/hasna/scaffold-connector.git connect-yourapi
|
|
20
|
+
cd connect-yourapi
|
|
21
|
+
|
|
22
|
+
# Update package.json name
|
|
23
|
+
# Change "@hasna/scaffold-connector" to "@hasna/connect-yourapi"
|
|
24
|
+
```
|
|
25
|
+
|
|
26
|
+
### 2. Update Configuration
|
|
27
|
+
|
|
28
|
+
Search for `TODO` comments throughout the codebase and update:
|
|
29
|
+
|
|
30
|
+
- `src/cli/index.ts` - Update `CONNECTOR_NAME` and description
|
|
31
|
+
- `src/utils/config.ts` - Update `CONNECTOR_NAME` and env var names
|
|
32
|
+
- `src/api/client.ts` - Update `DEFAULT_BASE_URL` and authentication method
|
|
33
|
+
- `src/api/index.ts` - Rename `Connector` class to your API name
|
|
34
|
+
- `src/types/index.ts` - Add your API's type definitions
|
|
35
|
+
- `package.json` - Update name, description, bin command
|
|
36
|
+
- `.env.example` - Update environment variable names
|
|
37
|
+
|
|
38
|
+
### 3. Install and Test
|
|
39
|
+
|
|
40
|
+
```bash
|
|
41
|
+
# Install dependencies
|
|
42
|
+
bun install
|
|
43
|
+
|
|
44
|
+
# Run CLI
|
|
45
|
+
bun run dev
|
|
46
|
+
|
|
47
|
+
# Or run specific commands
|
|
48
|
+
bun run dev profile list
|
|
49
|
+
bun run dev config show
|
|
50
|
+
```
|
|
51
|
+
|
|
52
|
+
## CLI Structure
|
|
53
|
+
|
|
54
|
+
```bash
|
|
55
|
+
connector [options] [command]
|
|
56
|
+
|
|
57
|
+
Options:
|
|
58
|
+
-k, --api-key <key> API key (overrides config)
|
|
59
|
+
-f, --format <format> Output format (json, pretty)
|
|
60
|
+
-p, --profile <profile> Use a specific profile
|
|
61
|
+
|
|
62
|
+
Commands:
|
|
63
|
+
profile list List all profiles
|
|
64
|
+
profile use <name> Switch to a profile
|
|
65
|
+
profile create <name> Create a new profile
|
|
66
|
+
profile delete <name> Delete a profile
|
|
67
|
+
profile show [name] Show profile configuration
|
|
68
|
+
|
|
69
|
+
config set-key <key> Set API key for active profile
|
|
70
|
+
config show Show current configuration
|
|
71
|
+
config clear Clear configuration
|
|
72
|
+
|
|
73
|
+
example list Example API command (replace)
|
|
74
|
+
example get <id> Example API command (replace)
|
|
75
|
+
example create Example API command (replace)
|
|
76
|
+
```
|
|
77
|
+
|
|
78
|
+
## Project Structure
|
|
79
|
+
|
|
80
|
+
```
|
|
81
|
+
src/
|
|
82
|
+
├── api/
|
|
83
|
+
│ ├── client.ts # HTTP client with authentication
|
|
84
|
+
│ ├── example.ts # Example API module (replace with your API)
|
|
85
|
+
│ └── index.ts # Main connector class
|
|
86
|
+
├── cli/
|
|
87
|
+
│ └── index.ts # CLI commands
|
|
88
|
+
├── types/
|
|
89
|
+
│ └── index.ts # Type definitions
|
|
90
|
+
├── utils/
|
|
91
|
+
│ ├── config.ts # Multi-profile configuration
|
|
92
|
+
│ └── output.ts # CLI output formatting
|
|
93
|
+
└── index.ts # Library exports
|
|
94
|
+
```
|
|
95
|
+
|
|
96
|
+
## Multi-Profile Configuration
|
|
97
|
+
|
|
98
|
+
Profiles are stored in `~/.connect/{connector-name}/profiles/`:
|
|
99
|
+
|
|
100
|
+
```
|
|
101
|
+
~/.connect/connector/
|
|
102
|
+
├── current_profile # Name of active profile
|
|
103
|
+
└── profiles/
|
|
104
|
+
├── default.json # Default profile
|
|
105
|
+
├── work.json # Named profile
|
|
106
|
+
└── personal.json # Named profile
|
|
107
|
+
```
|
|
108
|
+
|
|
109
|
+
### Profile Commands
|
|
110
|
+
|
|
111
|
+
```bash
|
|
112
|
+
# Create profiles
|
|
113
|
+
connector profile create work --api-key sk-xxx --use
|
|
114
|
+
connector profile create personal --api-key sk-yyy
|
|
115
|
+
|
|
116
|
+
# Switch profiles
|
|
117
|
+
connector profile use work
|
|
118
|
+
|
|
119
|
+
# Use profile for single command
|
|
120
|
+
connector -p personal example list
|
|
121
|
+
|
|
122
|
+
# List profiles
|
|
123
|
+
connector profile list
|
|
124
|
+
```
|
|
125
|
+
|
|
126
|
+
## Customizing Authentication
|
|
127
|
+
|
|
128
|
+
Edit `src/api/client.ts` to change authentication:
|
|
129
|
+
|
|
130
|
+
```typescript
|
|
131
|
+
// Bearer token (default)
|
|
132
|
+
'Authorization': `Bearer ${this.apiKey}`,
|
|
133
|
+
|
|
134
|
+
// API Key header
|
|
135
|
+
'X-API-Key': this.apiKey,
|
|
136
|
+
|
|
137
|
+
// Basic auth
|
|
138
|
+
'Authorization': `Basic ${Buffer.from(`${this.apiKey}:${this.apiSecret}`).toString('base64')}`,
|
|
139
|
+
```
|
|
140
|
+
|
|
141
|
+
## Adding API Endpoints
|
|
142
|
+
|
|
143
|
+
1. Create a new file in `src/api/` (e.g., `users.ts`)
|
|
144
|
+
2. Export it from `src/api/index.ts`
|
|
145
|
+
3. Add types in `src/types/index.ts`
|
|
146
|
+
4. Add CLI commands in `src/cli/index.ts`
|
|
147
|
+
|
|
148
|
+
Example API module:
|
|
149
|
+
|
|
150
|
+
```typescript
|
|
151
|
+
// src/api/users.ts
|
|
152
|
+
import type { ConnectorClient } from './client';
|
|
153
|
+
|
|
154
|
+
export class UsersApi {
|
|
155
|
+
constructor(private readonly client: ConnectorClient) {}
|
|
156
|
+
|
|
157
|
+
async list(options?: { limit?: number }) {
|
|
158
|
+
return this.client.get('/users', { limit: options?.limit });
|
|
159
|
+
}
|
|
160
|
+
|
|
161
|
+
async get(id: string) {
|
|
162
|
+
return this.client.get(`/users/${id}`);
|
|
163
|
+
}
|
|
164
|
+
}
|
|
165
|
+
```
|
|
166
|
+
|
|
167
|
+
## Environment Variables
|
|
168
|
+
|
|
169
|
+
| Variable | Description |
|
|
170
|
+
|----------|-------------|
|
|
171
|
+
| `CONNECTOR_API_KEY` | API key (overrides profile config) |
|
|
172
|
+
| `CONNECTOR_API_SECRET` | API secret (optional) |
|
|
173
|
+
| `CONNECTOR_BASE_URL` | Override base URL (optional) |
|
|
174
|
+
|
|
175
|
+
## Development
|
|
176
|
+
|
|
177
|
+
```bash
|
|
178
|
+
# Install dependencies
|
|
179
|
+
bun install
|
|
180
|
+
|
|
181
|
+
# Run CLI in development
|
|
182
|
+
bun run dev
|
|
183
|
+
|
|
184
|
+
# Build for distribution
|
|
185
|
+
bun run build
|
|
186
|
+
|
|
187
|
+
# Type check
|
|
188
|
+
bun run typecheck
|
|
189
|
+
```
|
|
190
|
+
|
|
191
|
+
## License
|
|
192
|
+
|
|
193
|
+
Apache-2.0
|
|
@@ -0,0 +1,53 @@
|
|
|
1
|
+
{
|
|
2
|
+
"name": "@hasna/connect-trello",
|
|
3
|
+
"version": "0.1.0",
|
|
4
|
+
"description": "Trello connector - Boards, lists, cards, and checklists management",
|
|
5
|
+
"type": "module",
|
|
6
|
+
"bin": {
|
|
7
|
+
"connect-trello": "./bin/index.js"
|
|
8
|
+
},
|
|
9
|
+
"exports": {
|
|
10
|
+
".": {
|
|
11
|
+
"import": "./dist/index.js",
|
|
12
|
+
"types": "./dist/index.d.ts"
|
|
13
|
+
}
|
|
14
|
+
},
|
|
15
|
+
"main": "./dist/index.js",
|
|
16
|
+
"types": "./dist/index.d.ts",
|
|
17
|
+
"scripts": {
|
|
18
|
+
"build": "bun build ./src/index.ts --outdir ./dist --target bun && bun build ./src/cli/index.ts --outdir ./bin --target bun",
|
|
19
|
+
"dev": "bun run ./src/cli/index.ts",
|
|
20
|
+
"typecheck": "tsc --noEmit",
|
|
21
|
+
"prepublishOnly": "bun run build"
|
|
22
|
+
},
|
|
23
|
+
"keywords": [
|
|
24
|
+
"hasna",
|
|
25
|
+
"trello",
|
|
26
|
+
"project-management",
|
|
27
|
+
"boards",
|
|
28
|
+
"cards",
|
|
29
|
+
"kanban",
|
|
30
|
+
"api",
|
|
31
|
+
"connector",
|
|
32
|
+
"cli",
|
|
33
|
+
"typescript",
|
|
34
|
+
"bun"
|
|
35
|
+
],
|
|
36
|
+
"author": "Hasna",
|
|
37
|
+
"license": "Apache-2.0",
|
|
38
|
+
"devDependencies": {
|
|
39
|
+
"@types/bun": "latest",
|
|
40
|
+
"typescript": "^5"
|
|
41
|
+
},
|
|
42
|
+
"dependencies": {
|
|
43
|
+
"commander": "^12.1.0",
|
|
44
|
+
"chalk": "^5.3.0"
|
|
45
|
+
},
|
|
46
|
+
"engines": {
|
|
47
|
+
"bun": ">=1.0.0"
|
|
48
|
+
},
|
|
49
|
+
"repository": {
|
|
50
|
+
"type": "git",
|
|
51
|
+
"url": "git+https://github.com/hasna/connectors.git"
|
|
52
|
+
}
|
|
53
|
+
}
|
|
@@ -0,0 +1,128 @@
|
|
|
1
|
+
import type { TrelloConfig } from '../types';
|
|
2
|
+
import { TrelloApiError } from '../types';
|
|
3
|
+
|
|
4
|
+
const DEFAULT_BASE_URL = 'https://api.trello.com/1';
|
|
5
|
+
|
|
6
|
+
export interface RequestOptions {
|
|
7
|
+
method?: 'GET' | 'POST' | 'PUT' | 'DELETE' | 'PATCH';
|
|
8
|
+
params?: Record<string, string | number | boolean | undefined>;
|
|
9
|
+
body?: Record<string, unknown> | unknown[] | string;
|
|
10
|
+
headers?: Record<string, string>;
|
|
11
|
+
}
|
|
12
|
+
|
|
13
|
+
export class TrelloClient {
|
|
14
|
+
private readonly apiKey: string;
|
|
15
|
+
private readonly token: string;
|
|
16
|
+
private readonly baseUrl: string;
|
|
17
|
+
|
|
18
|
+
constructor(config: TrelloConfig) {
|
|
19
|
+
if (!config.apiKey) {
|
|
20
|
+
throw new Error('API key is required');
|
|
21
|
+
}
|
|
22
|
+
if (!config.token) {
|
|
23
|
+
throw new Error('Token is required');
|
|
24
|
+
}
|
|
25
|
+
this.apiKey = config.apiKey;
|
|
26
|
+
this.token = config.token;
|
|
27
|
+
this.baseUrl = config.baseUrl || DEFAULT_BASE_URL;
|
|
28
|
+
}
|
|
29
|
+
|
|
30
|
+
private buildUrl(path: string, params?: Record<string, string | number | boolean | undefined>): string {
|
|
31
|
+
const url = new URL(`${this.baseUrl}${path}`);
|
|
32
|
+
|
|
33
|
+
// Trello uses key and token as query parameters
|
|
34
|
+
url.searchParams.append('key', this.apiKey);
|
|
35
|
+
url.searchParams.append('token', this.token);
|
|
36
|
+
|
|
37
|
+
if (params) {
|
|
38
|
+
Object.entries(params).forEach(([key, value]) => {
|
|
39
|
+
if (value !== undefined && value !== null && value !== '') {
|
|
40
|
+
url.searchParams.append(key, String(value));
|
|
41
|
+
}
|
|
42
|
+
});
|
|
43
|
+
}
|
|
44
|
+
|
|
45
|
+
return url.toString();
|
|
46
|
+
}
|
|
47
|
+
|
|
48
|
+
async request<T>(path: string, options: RequestOptions = {}): Promise<T> {
|
|
49
|
+
const { method = 'GET', params, body, headers = {} } = options;
|
|
50
|
+
|
|
51
|
+
const url = this.buildUrl(path, params);
|
|
52
|
+
|
|
53
|
+
const requestHeaders: Record<string, string> = {
|
|
54
|
+
'Accept': 'application/json',
|
|
55
|
+
...headers,
|
|
56
|
+
};
|
|
57
|
+
|
|
58
|
+
if (body && ['POST', 'PUT', 'PATCH'].includes(method)) {
|
|
59
|
+
requestHeaders['Content-Type'] = 'application/json';
|
|
60
|
+
}
|
|
61
|
+
|
|
62
|
+
const fetchOptions: RequestInit = {
|
|
63
|
+
method,
|
|
64
|
+
headers: requestHeaders,
|
|
65
|
+
};
|
|
66
|
+
|
|
67
|
+
if (body && ['POST', 'PUT', 'PATCH'].includes(method)) {
|
|
68
|
+
fetchOptions.body = typeof body === 'string' ? body : JSON.stringify(body);
|
|
69
|
+
}
|
|
70
|
+
|
|
71
|
+
const response = await fetch(url, fetchOptions);
|
|
72
|
+
|
|
73
|
+
// Handle 204 No Content
|
|
74
|
+
if (response.status === 204) {
|
|
75
|
+
return {} as T;
|
|
76
|
+
}
|
|
77
|
+
|
|
78
|
+
// Parse response
|
|
79
|
+
let data: unknown;
|
|
80
|
+
const contentType = response.headers.get('content-type') || '';
|
|
81
|
+
|
|
82
|
+
if (contentType.includes('application/json')) {
|
|
83
|
+
const text = await response.text();
|
|
84
|
+
if (text) {
|
|
85
|
+
try {
|
|
86
|
+
data = JSON.parse(text);
|
|
87
|
+
} catch {
|
|
88
|
+
data = text;
|
|
89
|
+
}
|
|
90
|
+
}
|
|
91
|
+
} else {
|
|
92
|
+
data = await response.text();
|
|
93
|
+
}
|
|
94
|
+
|
|
95
|
+
// Handle errors
|
|
96
|
+
if (!response.ok) {
|
|
97
|
+
const errorMessage = typeof data === 'string'
|
|
98
|
+
? data
|
|
99
|
+
: (data as { message?: string })?.message || response.statusText;
|
|
100
|
+
throw new TrelloApiError(errorMessage, response.status);
|
|
101
|
+
}
|
|
102
|
+
|
|
103
|
+
return data as T;
|
|
104
|
+
}
|
|
105
|
+
|
|
106
|
+
async get<T>(path: string, params?: Record<string, string | number | boolean | undefined>): Promise<T> {
|
|
107
|
+
return this.request<T>(path, { method: 'GET', params });
|
|
108
|
+
}
|
|
109
|
+
|
|
110
|
+
async post<T>(path: string, body?: Record<string, unknown> | unknown[] | string | object, params?: Record<string, string | number | boolean | undefined>): Promise<T> {
|
|
111
|
+
return this.request<T>(path, { method: 'POST', body: body as Record<string, unknown>, params });
|
|
112
|
+
}
|
|
113
|
+
|
|
114
|
+
async put<T>(path: string, body?: Record<string, unknown> | object, params?: Record<string, string | number | boolean | undefined>): Promise<T> {
|
|
115
|
+
return this.request<T>(path, { method: 'PUT', body: body as Record<string, unknown>, params });
|
|
116
|
+
}
|
|
117
|
+
|
|
118
|
+
async delete<T>(path: string, params?: Record<string, string | number | boolean | undefined>): Promise<T> {
|
|
119
|
+
return this.request<T>(path, { method: 'DELETE', params });
|
|
120
|
+
}
|
|
121
|
+
|
|
122
|
+
getApiKeyPreview(): string {
|
|
123
|
+
if (this.apiKey.length > 10) {
|
|
124
|
+
return `${this.apiKey.substring(0, 6)}...${this.apiKey.substring(this.apiKey.length - 4)}`;
|
|
125
|
+
}
|
|
126
|
+
return '***';
|
|
127
|
+
}
|
|
128
|
+
}
|
|
@@ -0,0 +1,278 @@
|
|
|
1
|
+
import { TrelloClient } from './client';
|
|
2
|
+
import type {
|
|
3
|
+
TrelloConfig,
|
|
4
|
+
Member,
|
|
5
|
+
Board,
|
|
6
|
+
CreateBoardInput,
|
|
7
|
+
List,
|
|
8
|
+
CreateListInput,
|
|
9
|
+
Card,
|
|
10
|
+
CreateCardInput,
|
|
11
|
+
Label,
|
|
12
|
+
CreateLabelInput,
|
|
13
|
+
Checklist,
|
|
14
|
+
CheckItem,
|
|
15
|
+
CreateChecklistInput,
|
|
16
|
+
CreateCheckItemInput,
|
|
17
|
+
Action,
|
|
18
|
+
Attachment,
|
|
19
|
+
Organization,
|
|
20
|
+
} from '../types';
|
|
21
|
+
|
|
22
|
+
export { TrelloClient } from './client';
|
|
23
|
+
|
|
24
|
+
export class Trello {
|
|
25
|
+
private client: TrelloClient;
|
|
26
|
+
|
|
27
|
+
constructor(config: TrelloConfig) {
|
|
28
|
+
this.client = new TrelloClient(config);
|
|
29
|
+
}
|
|
30
|
+
|
|
31
|
+
// ============================================
|
|
32
|
+
// Members
|
|
33
|
+
// ============================================
|
|
34
|
+
|
|
35
|
+
async getMe(): Promise<Member> {
|
|
36
|
+
return this.client.get<Member>('/members/me');
|
|
37
|
+
}
|
|
38
|
+
|
|
39
|
+
async getMember(idOrUsername: string): Promise<Member> {
|
|
40
|
+
return this.client.get<Member>(`/members/${idOrUsername}`);
|
|
41
|
+
}
|
|
42
|
+
|
|
43
|
+
async getMemberBoards(idOrUsername: string): Promise<Board[]> {
|
|
44
|
+
return this.client.get<Board[]>(`/members/${idOrUsername}/boards`);
|
|
45
|
+
}
|
|
46
|
+
|
|
47
|
+
async getMemberOrganizations(idOrUsername: string): Promise<Organization[]> {
|
|
48
|
+
return this.client.get<Organization[]>(`/members/${idOrUsername}/organizations`);
|
|
49
|
+
}
|
|
50
|
+
|
|
51
|
+
// ============================================
|
|
52
|
+
// Boards
|
|
53
|
+
// ============================================
|
|
54
|
+
|
|
55
|
+
async getBoard(id: string): Promise<Board> {
|
|
56
|
+
return this.client.get<Board>(`/boards/${id}`);
|
|
57
|
+
}
|
|
58
|
+
|
|
59
|
+
async createBoard(input: CreateBoardInput): Promise<Board> {
|
|
60
|
+
return this.client.post<Board>('/boards', undefined, input as unknown as Record<string, string>);
|
|
61
|
+
}
|
|
62
|
+
|
|
63
|
+
async updateBoard(id: string, input: Partial<CreateBoardInput>): Promise<Board> {
|
|
64
|
+
return this.client.put<Board>(`/boards/${id}`, undefined, input as unknown as Record<string, string>);
|
|
65
|
+
}
|
|
66
|
+
|
|
67
|
+
async deleteBoard(id: string): Promise<void> {
|
|
68
|
+
await this.client.delete(`/boards/${id}`);
|
|
69
|
+
}
|
|
70
|
+
|
|
71
|
+
async getBoardLists(boardId: string, filter: 'all' | 'open' | 'closed' = 'open'): Promise<List[]> {
|
|
72
|
+
return this.client.get<List[]>(`/boards/${boardId}/lists`, { filter });
|
|
73
|
+
}
|
|
74
|
+
|
|
75
|
+
async getBoardCards(boardId: string, filter: 'all' | 'open' | 'closed' | 'visible' = 'open'): Promise<Card[]> {
|
|
76
|
+
return this.client.get<Card[]>(`/boards/${boardId}/cards`, { filter });
|
|
77
|
+
}
|
|
78
|
+
|
|
79
|
+
async getBoardLabels(boardId: string): Promise<Label[]> {
|
|
80
|
+
return this.client.get<Label[]>(`/boards/${boardId}/labels`);
|
|
81
|
+
}
|
|
82
|
+
|
|
83
|
+
async getBoardMembers(boardId: string): Promise<Member[]> {
|
|
84
|
+
return this.client.get<Member[]>(`/boards/${boardId}/members`);
|
|
85
|
+
}
|
|
86
|
+
|
|
87
|
+
async addMemberToBoard(boardId: string, email: string, type: 'admin' | 'normal' | 'observer' = 'normal'): Promise<Member> {
|
|
88
|
+
return this.client.put<Member>(`/boards/${boardId}/members`, undefined, { email, type });
|
|
89
|
+
}
|
|
90
|
+
|
|
91
|
+
async removeMemberFromBoard(boardId: string, memberId: string): Promise<void> {
|
|
92
|
+
await this.client.delete(`/boards/${boardId}/members/${memberId}`);
|
|
93
|
+
}
|
|
94
|
+
|
|
95
|
+
// ============================================
|
|
96
|
+
// Lists
|
|
97
|
+
// ============================================
|
|
98
|
+
|
|
99
|
+
async getList(id: string): Promise<List> {
|
|
100
|
+
return this.client.get<List>(`/lists/${id}`);
|
|
101
|
+
}
|
|
102
|
+
|
|
103
|
+
async createList(input: CreateListInput): Promise<List> {
|
|
104
|
+
return this.client.post<List>('/lists', undefined, input as unknown as Record<string, string>);
|
|
105
|
+
}
|
|
106
|
+
|
|
107
|
+
async updateList(id: string, input: Partial<{ name: string; closed: boolean; pos: 'top' | 'bottom' | number }>): Promise<List> {
|
|
108
|
+
return this.client.put<List>(`/lists/${id}`, undefined, input as unknown as Record<string, string>);
|
|
109
|
+
}
|
|
110
|
+
|
|
111
|
+
async archiveList(id: string): Promise<List> {
|
|
112
|
+
return this.client.put<List>(`/lists/${id}/closed`, undefined, { value: 'true' });
|
|
113
|
+
}
|
|
114
|
+
|
|
115
|
+
async getListCards(listId: string): Promise<Card[]> {
|
|
116
|
+
return this.client.get<Card[]>(`/lists/${listId}/cards`);
|
|
117
|
+
}
|
|
118
|
+
|
|
119
|
+
// ============================================
|
|
120
|
+
// Cards
|
|
121
|
+
// ============================================
|
|
122
|
+
|
|
123
|
+
async getCard(id: string): Promise<Card> {
|
|
124
|
+
return this.client.get<Card>(`/cards/${id}`);
|
|
125
|
+
}
|
|
126
|
+
|
|
127
|
+
async createCard(input: CreateCardInput): Promise<Card> {
|
|
128
|
+
return this.client.post<Card>('/cards', undefined, input as unknown as Record<string, string>);
|
|
129
|
+
}
|
|
130
|
+
|
|
131
|
+
async updateCard(id: string, input: Partial<CreateCardInput & { closed: boolean; idList: string }>): Promise<Card> {
|
|
132
|
+
return this.client.put<Card>(`/cards/${id}`, undefined, input as unknown as Record<string, string>);
|
|
133
|
+
}
|
|
134
|
+
|
|
135
|
+
async deleteCard(id: string): Promise<void> {
|
|
136
|
+
await this.client.delete(`/cards/${id}`);
|
|
137
|
+
}
|
|
138
|
+
|
|
139
|
+
async archiveCard(id: string): Promise<Card> {
|
|
140
|
+
return this.client.put<Card>(`/cards/${id}`, undefined, { closed: 'true' });
|
|
141
|
+
}
|
|
142
|
+
|
|
143
|
+
async moveCard(id: string, idList: string, pos?: 'top' | 'bottom' | number): Promise<Card> {
|
|
144
|
+
return this.client.put<Card>(`/cards/${id}`, undefined, {
|
|
145
|
+
idList,
|
|
146
|
+
pos: pos?.toString(),
|
|
147
|
+
});
|
|
148
|
+
}
|
|
149
|
+
|
|
150
|
+
async addLabelToCard(cardId: string, labelId: string): Promise<void> {
|
|
151
|
+
await this.client.post(`/cards/${cardId}/idLabels`, undefined, { value: labelId });
|
|
152
|
+
}
|
|
153
|
+
|
|
154
|
+
async removeLabelFromCard(cardId: string, labelId: string): Promise<void> {
|
|
155
|
+
await this.client.delete(`/cards/${cardId}/idLabels/${labelId}`);
|
|
156
|
+
}
|
|
157
|
+
|
|
158
|
+
async addMemberToCard(cardId: string, memberId: string): Promise<void> {
|
|
159
|
+
await this.client.post(`/cards/${cardId}/idMembers`, undefined, { value: memberId });
|
|
160
|
+
}
|
|
161
|
+
|
|
162
|
+
async removeMemberFromCard(cardId: string, memberId: string): Promise<void> {
|
|
163
|
+
await this.client.delete(`/cards/${cardId}/idMembers/${memberId}`);
|
|
164
|
+
}
|
|
165
|
+
|
|
166
|
+
async getCardAttachments(cardId: string): Promise<Attachment[]> {
|
|
167
|
+
return this.client.get<Attachment[]>(`/cards/${cardId}/attachments`);
|
|
168
|
+
}
|
|
169
|
+
|
|
170
|
+
async getCardActions(cardId: string, filter?: string): Promise<Action[]> {
|
|
171
|
+
return this.client.get<Action[]>(`/cards/${cardId}/actions`, { filter });
|
|
172
|
+
}
|
|
173
|
+
|
|
174
|
+
// ============================================
|
|
175
|
+
// Labels
|
|
176
|
+
// ============================================
|
|
177
|
+
|
|
178
|
+
async getLabel(id: string): Promise<Label> {
|
|
179
|
+
return this.client.get<Label>(`/labels/${id}`);
|
|
180
|
+
}
|
|
181
|
+
|
|
182
|
+
async createLabel(input: CreateLabelInput): Promise<Label> {
|
|
183
|
+
return this.client.post<Label>('/labels', undefined, input as unknown as Record<string, string>);
|
|
184
|
+
}
|
|
185
|
+
|
|
186
|
+
async updateLabel(id: string, input: Partial<{ name: string; color: string }>): Promise<Label> {
|
|
187
|
+
return this.client.put<Label>(`/labels/${id}`, undefined, input as unknown as Record<string, string>);
|
|
188
|
+
}
|
|
189
|
+
|
|
190
|
+
async deleteLabel(id: string): Promise<void> {
|
|
191
|
+
await this.client.delete(`/labels/${id}`);
|
|
192
|
+
}
|
|
193
|
+
|
|
194
|
+
// ============================================
|
|
195
|
+
// Checklists
|
|
196
|
+
// ============================================
|
|
197
|
+
|
|
198
|
+
async getChecklist(id: string): Promise<Checklist> {
|
|
199
|
+
return this.client.get<Checklist>(`/checklists/${id}`);
|
|
200
|
+
}
|
|
201
|
+
|
|
202
|
+
async createChecklist(input: CreateChecklistInput): Promise<Checklist> {
|
|
203
|
+
return this.client.post<Checklist>('/checklists', undefined, input as unknown as Record<string, string>);
|
|
204
|
+
}
|
|
205
|
+
|
|
206
|
+
async updateChecklist(id: string, name: string): Promise<Checklist> {
|
|
207
|
+
return this.client.put<Checklist>(`/checklists/${id}`, undefined, { name });
|
|
208
|
+
}
|
|
209
|
+
|
|
210
|
+
async deleteChecklist(id: string): Promise<void> {
|
|
211
|
+
await this.client.delete(`/checklists/${id}`);
|
|
212
|
+
}
|
|
213
|
+
|
|
214
|
+
async getChecklistItems(checklistId: string): Promise<CheckItem[]> {
|
|
215
|
+
return this.client.get<CheckItem[]>(`/checklists/${checklistId}/checkItems`);
|
|
216
|
+
}
|
|
217
|
+
|
|
218
|
+
async createCheckItem(checklistId: string, input: CreateCheckItemInput): Promise<CheckItem> {
|
|
219
|
+
return this.client.post<CheckItem>(`/checklists/${checklistId}/checkItems`, undefined, input as unknown as Record<string, string>);
|
|
220
|
+
}
|
|
221
|
+
|
|
222
|
+
async updateCheckItem(cardId: string, checkItemId: string, input: Partial<{ name: string; state: 'complete' | 'incomplete'; pos: 'top' | 'bottom' | number }>): Promise<CheckItem> {
|
|
223
|
+
return this.client.put<CheckItem>(`/cards/${cardId}/checkItem/${checkItemId}`, undefined, input as unknown as Record<string, string>);
|
|
224
|
+
}
|
|
225
|
+
|
|
226
|
+
async deleteCheckItem(checklistId: string, checkItemId: string): Promise<void> {
|
|
227
|
+
await this.client.delete(`/checklists/${checklistId}/checkItems/${checkItemId}`);
|
|
228
|
+
}
|
|
229
|
+
|
|
230
|
+
// ============================================
|
|
231
|
+
// Comments
|
|
232
|
+
// ============================================
|
|
233
|
+
|
|
234
|
+
async addComment(cardId: string, text: string): Promise<Action> {
|
|
235
|
+
return this.client.post<Action>(`/cards/${cardId}/actions/comments`, undefined, { text });
|
|
236
|
+
}
|
|
237
|
+
|
|
238
|
+
async updateComment(cardId: string, actionId: string, text: string): Promise<Action> {
|
|
239
|
+
return this.client.put<Action>(`/cards/${cardId}/actions/${actionId}/comments`, undefined, { text });
|
|
240
|
+
}
|
|
241
|
+
|
|
242
|
+
async deleteComment(cardId: string, actionId: string): Promise<void> {
|
|
243
|
+
await this.client.delete(`/cards/${cardId}/actions/${actionId}/comments`);
|
|
244
|
+
}
|
|
245
|
+
|
|
246
|
+
// ============================================
|
|
247
|
+
// Organizations
|
|
248
|
+
// ============================================
|
|
249
|
+
|
|
250
|
+
async getOrganization(id: string): Promise<Organization> {
|
|
251
|
+
return this.client.get<Organization>(`/organizations/${id}`);
|
|
252
|
+
}
|
|
253
|
+
|
|
254
|
+
async getOrganizationBoards(orgId: string): Promise<Board[]> {
|
|
255
|
+
return this.client.get<Board[]>(`/organizations/${orgId}/boards`);
|
|
256
|
+
}
|
|
257
|
+
|
|
258
|
+
async getOrganizationMembers(orgId: string): Promise<Member[]> {
|
|
259
|
+
return this.client.get<Member[]>(`/organizations/${orgId}/members`);
|
|
260
|
+
}
|
|
261
|
+
|
|
262
|
+
// ============================================
|
|
263
|
+
// Search
|
|
264
|
+
// ============================================
|
|
265
|
+
|
|
266
|
+
async search(query: string, options: {
|
|
267
|
+
idBoards?: string;
|
|
268
|
+
modelTypes?: string;
|
|
269
|
+
board_fields?: string;
|
|
270
|
+
boards_limit?: number;
|
|
271
|
+
card_fields?: string;
|
|
272
|
+
cards_limit?: number;
|
|
273
|
+
cards_page?: number;
|
|
274
|
+
partial?: boolean;
|
|
275
|
+
} = {}): Promise<{ boards?: Board[]; cards?: Card[]; members?: Member[]; organizations?: Organization[] }> {
|
|
276
|
+
return this.client.get('/search', { query, ...options });
|
|
277
|
+
}
|
|
278
|
+
}
|