@hasna/connectors 0.5.0 → 0.5.1
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 +92 -1
- package/bin/mcp.js +92 -1
- package/bin/serve.js +91 -0
- package/connectors/connect-ably/.env.example +11 -0
- package/connectors/connect-ably/CLAUDE.md +111 -0
- package/connectors/connect-ably/README.md +193 -0
- package/connectors/connect-ably/package.json +54 -0
- package/connectors/connect-ably/scripts/release.ts +179 -0
- package/connectors/connect-ably/src/api/channels.ts +33 -0
- package/connectors/connect-ably/src/api/client.ts +203 -0
- package/connectors/connect-ably/src/api/index.ts +59 -0
- package/connectors/connect-ably/src/api/messages.ts +48 -0
- package/connectors/connect-ably/src/api/presence.ts +39 -0
- package/connectors/connect-ably/src/api/stats.ts +29 -0
- package/connectors/connect-ably/src/cli/index.ts +397 -0
- package/connectors/connect-ably/src/index.ts +102 -0
- package/connectors/connect-ably/src/types/index.ts +294 -0
- package/connectors/connect-ably/src/utils/auth.ts +274 -0
- package/connectors/connect-ably/src/utils/bulk.ts +212 -0
- package/connectors/connect-ably/src/utils/config.ts +323 -0
- package/connectors/connect-ably/src/utils/output.ts +175 -0
- package/connectors/connect-ably/src/utils/settings.ts +114 -0
- package/connectors/connect-ably/src/utils/storage.ts +198 -0
- package/connectors/connect-ably/tsconfig.json +16 -0
- package/connectors/connect-box/.env.example +11 -0
- package/connectors/connect-box/CLAUDE.md +272 -0
- package/connectors/connect-box/README.md +193 -0
- package/connectors/connect-box/package.json +51 -0
- package/connectors/connect-box/scripts/release.ts +179 -0
- package/connectors/connect-box/src/api/client.ts +213 -0
- package/connectors/connect-box/src/api/example.ts +48 -0
- package/connectors/connect-box/src/api/index.ts +51 -0
- package/connectors/connect-box/src/cli/index.ts +254 -0
- package/connectors/connect-box/src/index.ts +103 -0
- package/connectors/connect-box/src/types/index.ts +237 -0
- package/connectors/connect-box/src/utils/auth.ts +274 -0
- package/connectors/connect-box/src/utils/bulk.ts +212 -0
- package/connectors/connect-box/src/utils/config.ts +326 -0
- package/connectors/connect-box/src/utils/output.ts +175 -0
- package/connectors/connect-box/src/utils/settings.ts +114 -0
- package/connectors/connect-box/src/utils/storage.ts +198 -0
- package/connectors/connect-box/tsconfig.json +16 -0
- package/connectors/connect-clearbit/.env.example +11 -0
- package/connectors/connect-clearbit/CLAUDE.md +272 -0
- package/connectors/connect-clearbit/README.md +193 -0
- package/connectors/connect-clearbit/package.json +51 -0
- package/connectors/connect-clearbit/scripts/release.ts +179 -0
- package/connectors/connect-clearbit/src/api/client.ts +213 -0
- package/connectors/connect-clearbit/src/api/example.ts +48 -0
- package/connectors/connect-clearbit/src/api/index.ts +51 -0
- package/connectors/connect-clearbit/src/cli/index.ts +254 -0
- package/connectors/connect-clearbit/src/index.ts +103 -0
- package/connectors/connect-clearbit/src/types/index.ts +237 -0
- package/connectors/connect-clearbit/src/utils/auth.ts +274 -0
- package/connectors/connect-clearbit/src/utils/bulk.ts +212 -0
- package/connectors/connect-clearbit/src/utils/config.ts +326 -0
- package/connectors/connect-clearbit/src/utils/output.ts +175 -0
- package/connectors/connect-clearbit/src/utils/settings.ts +114 -0
- package/connectors/connect-clearbit/src/utils/storage.ts +198 -0
- package/connectors/connect-clearbit/tsconfig.json +16 -0
- package/connectors/connect-coda/.env.example +11 -0
- package/connectors/connect-coda/CLAUDE.md +272 -0
- package/connectors/connect-coda/README.md +193 -0
- package/connectors/connect-coda/package.json +51 -0
- package/connectors/connect-coda/scripts/release.ts +179 -0
- package/connectors/connect-coda/src/api/client.ts +213 -0
- package/connectors/connect-coda/src/api/example.ts +48 -0
- package/connectors/connect-coda/src/api/index.ts +51 -0
- package/connectors/connect-coda/src/cli/index.ts +254 -0
- package/connectors/connect-coda/src/index.ts +103 -0
- package/connectors/connect-coda/src/types/index.ts +237 -0
- package/connectors/connect-coda/src/utils/auth.ts +274 -0
- package/connectors/connect-coda/src/utils/bulk.ts +212 -0
- package/connectors/connect-coda/src/utils/config.ts +326 -0
- package/connectors/connect-coda/src/utils/output.ts +175 -0
- package/connectors/connect-coda/src/utils/settings.ts +114 -0
- package/connectors/connect-coda/src/utils/storage.ts +198 -0
- package/connectors/connect-coda/tsconfig.json +16 -0
- package/connectors/connect-dropbox/.env.example +11 -0
- package/connectors/connect-dropbox/CLAUDE.md +119 -0
- package/connectors/connect-dropbox/README.md +193 -0
- package/connectors/connect-dropbox/package.json +51 -0
- package/connectors/connect-dropbox/src/api/client.ts +222 -0
- package/connectors/connect-dropbox/src/api/index.ts +395 -0
- package/connectors/connect-dropbox/src/cli/index.ts +627 -0
- package/connectors/connect-dropbox/src/index.ts +20 -0
- package/connectors/connect-dropbox/src/types/index.ts +516 -0
- package/connectors/connect-dropbox/src/utils/config.ts +197 -0
- package/connectors/connect-dropbox/tsconfig.json +16 -0
- package/connectors/connect-linode/.env.example +11 -0
- package/connectors/connect-linode/CLAUDE.md +272 -0
- package/connectors/connect-linode/README.md +193 -0
- package/connectors/connect-linode/package.json +51 -0
- package/connectors/connect-linode/scripts/release.ts +179 -0
- package/connectors/connect-linode/src/api/client.ts +213 -0
- package/connectors/connect-linode/src/api/example.ts +48 -0
- package/connectors/connect-linode/src/api/index.ts +51 -0
- package/connectors/connect-linode/src/cli/index.ts +254 -0
- package/connectors/connect-linode/src/index.ts +103 -0
- package/connectors/connect-linode/src/types/index.ts +237 -0
- package/connectors/connect-linode/src/utils/auth.ts +274 -0
- package/connectors/connect-linode/src/utils/bulk.ts +212 -0
- package/connectors/connect-linode/src/utils/config.ts +326 -0
- package/connectors/connect-linode/src/utils/output.ts +175 -0
- package/connectors/connect-linode/src/utils/settings.ts +114 -0
- package/connectors/connect-linode/src/utils/storage.ts +198 -0
- package/connectors/connect-linode/tsconfig.json +16 -0
- package/connectors/connect-mailgun/.env.example +11 -0
- package/connectors/connect-mailgun/CLAUDE.md +272 -0
- package/connectors/connect-mailgun/README.md +193 -0
- package/connectors/connect-mailgun/package.json +51 -0
- package/connectors/connect-mailgun/scripts/release.ts +179 -0
- package/connectors/connect-mailgun/src/api/client.ts +213 -0
- package/connectors/connect-mailgun/src/api/example.ts +48 -0
- package/connectors/connect-mailgun/src/api/index.ts +51 -0
- package/connectors/connect-mailgun/src/cli/index.ts +254 -0
- package/connectors/connect-mailgun/src/index.ts +103 -0
- package/connectors/connect-mailgun/src/types/index.ts +237 -0
- package/connectors/connect-mailgun/src/utils/auth.ts +274 -0
- package/connectors/connect-mailgun/src/utils/bulk.ts +212 -0
- package/connectors/connect-mailgun/src/utils/config.ts +326 -0
- package/connectors/connect-mailgun/src/utils/output.ts +175 -0
- package/connectors/connect-mailgun/src/utils/settings.ts +114 -0
- package/connectors/connect-mailgun/src/utils/storage.ts +198 -0
- package/connectors/connect-mailgun/tsconfig.json +16 -0
- package/connectors/connect-messagebird/.env.example +11 -0
- package/connectors/connect-messagebird/CLAUDE.md +272 -0
- package/connectors/connect-messagebird/README.md +193 -0
- package/connectors/connect-messagebird/package.json +51 -0
- package/connectors/connect-messagebird/scripts/release.ts +179 -0
- package/connectors/connect-messagebird/src/api/client.ts +213 -0
- package/connectors/connect-messagebird/src/api/example.ts +48 -0
- package/connectors/connect-messagebird/src/api/index.ts +51 -0
- package/connectors/connect-messagebird/src/cli/index.ts +254 -0
- package/connectors/connect-messagebird/src/index.ts +103 -0
- package/connectors/connect-messagebird/src/types/index.ts +237 -0
- package/connectors/connect-messagebird/src/utils/auth.ts +274 -0
- package/connectors/connect-messagebird/src/utils/bulk.ts +212 -0
- package/connectors/connect-messagebird/src/utils/config.ts +326 -0
- package/connectors/connect-messagebird/src/utils/output.ts +175 -0
- package/connectors/connect-messagebird/src/utils/settings.ts +114 -0
- package/connectors/connect-messagebird/src/utils/storage.ts +198 -0
- package/connectors/connect-messagebird/tsconfig.json +16 -0
- package/connectors/connect-miro/.env.example +11 -0
- package/connectors/connect-miro/CLAUDE.md +272 -0
- package/connectors/connect-miro/README.md +193 -0
- package/connectors/connect-miro/package.json +51 -0
- package/connectors/connect-miro/scripts/release.ts +179 -0
- package/connectors/connect-miro/src/api/client.ts +213 -0
- package/connectors/connect-miro/src/api/example.ts +48 -0
- package/connectors/connect-miro/src/api/index.ts +51 -0
- package/connectors/connect-miro/src/cli/index.ts +254 -0
- package/connectors/connect-miro/src/index.ts +103 -0
- package/connectors/connect-miro/src/types/index.ts +237 -0
- package/connectors/connect-miro/src/utils/auth.ts +274 -0
- package/connectors/connect-miro/src/utils/bulk.ts +212 -0
- package/connectors/connect-miro/src/utils/config.ts +326 -0
- package/connectors/connect-miro/src/utils/output.ts +175 -0
- package/connectors/connect-miro/src/utils/settings.ts +114 -0
- package/connectors/connect-miro/src/utils/storage.ts +198 -0
- package/connectors/connect-miro/tsconfig.json +16 -0
- package/connectors/connect-monday/.env.example +11 -0
- package/connectors/connect-monday/CLAUDE.md +128 -0
- package/connectors/connect-monday/README.md +193 -0
- package/connectors/connect-monday/package.json +52 -0
- package/connectors/connect-monday/src/api/client.ts +59 -0
- package/connectors/connect-monday/src/api/index.ts +539 -0
- package/connectors/connect-monday/src/cli/index.ts +479 -0
- package/connectors/connect-monday/src/index.ts +19 -0
- package/connectors/connect-monday/src/types/index.ts +274 -0
- package/connectors/connect-monday/src/utils/config.ts +197 -0
- package/connectors/connect-monday/src/utils/output.ts +119 -0
- package/connectors/connect-monday/tsconfig.json +16 -0
- package/connectors/connect-pipedrive/.env.example +11 -0
- package/connectors/connect-pipedrive/CLAUDE.md +128 -0
- package/connectors/connect-pipedrive/README.md +193 -0
- package/connectors/connect-pipedrive/package.json +52 -0
- package/connectors/connect-pipedrive/src/api/client.ts +121 -0
- package/connectors/connect-pipedrive/src/api/index.ts +306 -0
- package/connectors/connect-pipedrive/src/cli/index.ts +824 -0
- package/connectors/connect-pipedrive/src/index.ts +19 -0
- package/connectors/connect-pipedrive/src/types/index.ts +335 -0
- package/connectors/connect-pipedrive/src/utils/config.ts +171 -0
- package/connectors/connect-pipedrive/src/utils/output.ts +119 -0
- package/connectors/connect-pipedrive/tsconfig.json +16 -0
- package/connectors/connect-pusher/.env.example +11 -0
- package/connectors/connect-pusher/CLAUDE.md +272 -0
- package/connectors/connect-pusher/README.md +193 -0
- package/connectors/connect-pusher/package.json +51 -0
- package/connectors/connect-pusher/scripts/release.ts +179 -0
- package/connectors/connect-pusher/src/api/client.ts +213 -0
- package/connectors/connect-pusher/src/api/example.ts +48 -0
- package/connectors/connect-pusher/src/api/index.ts +51 -0
- package/connectors/connect-pusher/src/cli/index.ts +254 -0
- package/connectors/connect-pusher/src/index.ts +103 -0
- package/connectors/connect-pusher/src/types/index.ts +237 -0
- package/connectors/connect-pusher/src/utils/auth.ts +274 -0
- package/connectors/connect-pusher/src/utils/bulk.ts +212 -0
- package/connectors/connect-pusher/src/utils/config.ts +326 -0
- package/connectors/connect-pusher/src/utils/output.ts +175 -0
- package/connectors/connect-pusher/src/utils/settings.ts +114 -0
- package/connectors/connect-pusher/src/utils/storage.ts +198 -0
- package/connectors/connect-pusher/tsconfig.json +16 -0
- package/connectors/connect-vonage/.env.example +11 -0
- package/connectors/connect-vonage/CLAUDE.md +272 -0
- package/connectors/connect-vonage/README.md +193 -0
- package/connectors/connect-vonage/package.json +51 -0
- package/connectors/connect-vonage/scripts/release.ts +179 -0
- package/connectors/connect-vonage/src/api/client.ts +213 -0
- package/connectors/connect-vonage/src/api/example.ts +48 -0
- package/connectors/connect-vonage/src/api/index.ts +51 -0
- package/connectors/connect-vonage/src/cli/index.ts +254 -0
- package/connectors/connect-vonage/src/index.ts +103 -0
- package/connectors/connect-vonage/src/types/index.ts +237 -0
- package/connectors/connect-vonage/src/utils/auth.ts +274 -0
- package/connectors/connect-vonage/src/utils/bulk.ts +212 -0
- package/connectors/connect-vonage/src/utils/config.ts +326 -0
- package/connectors/connect-vonage/src/utils/output.ts +175 -0
- package/connectors/connect-vonage/src/utils/settings.ts +114 -0
- package/connectors/connect-vonage/src/utils/storage.ts +198 -0
- package/connectors/connect-vonage/tsconfig.json +16 -0
- package/dist/index.js +91 -0
- package/package.json +1 -1
|
@@ -0,0 +1,179 @@
|
|
|
1
|
+
#!/usr/bin/env bun
|
|
2
|
+
/**
|
|
3
|
+
* Release script for publishing to npm
|
|
4
|
+
*
|
|
5
|
+
* Usage:
|
|
6
|
+
* bun run release # Build, bump version, and publish
|
|
7
|
+
* bun run release:dry # Dry run (preview only)
|
|
8
|
+
*
|
|
9
|
+
* Features:
|
|
10
|
+
* - Auto-fetches current npm version
|
|
11
|
+
* - Bumps patch version if needed
|
|
12
|
+
* - Creates git tag
|
|
13
|
+
* - Publishes to npm
|
|
14
|
+
*/
|
|
15
|
+
|
|
16
|
+
import { readFileSync, writeFileSync } from 'fs';
|
|
17
|
+
import { execSync } from 'child_process';
|
|
18
|
+
|
|
19
|
+
const isDryRun = process.argv.includes('--dry-run');
|
|
20
|
+
|
|
21
|
+
interface PackageJson {
|
|
22
|
+
name: string;
|
|
23
|
+
version: string;
|
|
24
|
+
[key: string]: unknown;
|
|
25
|
+
}
|
|
26
|
+
|
|
27
|
+
function exec(command: string, silent = false): string {
|
|
28
|
+
try {
|
|
29
|
+
const result = execSync(command, {
|
|
30
|
+
encoding: 'utf-8',
|
|
31
|
+
stdio: silent ? 'pipe' : 'inherit',
|
|
32
|
+
});
|
|
33
|
+
return result?.trim() || '';
|
|
34
|
+
} catch (err) {
|
|
35
|
+
if (silent) {
|
|
36
|
+
return '';
|
|
37
|
+
}
|
|
38
|
+
throw err;
|
|
39
|
+
}
|
|
40
|
+
}
|
|
41
|
+
|
|
42
|
+
function log(message: string): void {
|
|
43
|
+
console.log(`\x1b[36m▸\x1b[0m ${message}`);
|
|
44
|
+
}
|
|
45
|
+
|
|
46
|
+
function success(message: string): void {
|
|
47
|
+
console.log(`\x1b[32m✓\x1b[0m ${message}`);
|
|
48
|
+
}
|
|
49
|
+
|
|
50
|
+
function warn(message: string): void {
|
|
51
|
+
console.log(`\x1b[33m⚠\x1b[0m ${message}`);
|
|
52
|
+
}
|
|
53
|
+
|
|
54
|
+
function error(message: string): void {
|
|
55
|
+
console.error(`\x1b[31m✗\x1b[0m ${message}`);
|
|
56
|
+
}
|
|
57
|
+
|
|
58
|
+
function parseVersion(version: string): { major: number; minor: number; patch: number } {
|
|
59
|
+
const [major, minor, patch] = version.split('.').map(Number);
|
|
60
|
+
return { major, minor, patch };
|
|
61
|
+
}
|
|
62
|
+
|
|
63
|
+
function bumpPatch(version: string): string {
|
|
64
|
+
const { major, minor, patch } = parseVersion(version);
|
|
65
|
+
return `${major}.${minor}.${patch + 1}`;
|
|
66
|
+
}
|
|
67
|
+
|
|
68
|
+
async function main(): Promise<void> {
|
|
69
|
+
// Read package.json
|
|
70
|
+
const packageJsonPath = 'package.json';
|
|
71
|
+
const packageJson: PackageJson = JSON.parse(readFileSync(packageJsonPath, 'utf-8'));
|
|
72
|
+
const { name, version: localVersion } = packageJson;
|
|
73
|
+
|
|
74
|
+
log(`Package: ${name}`);
|
|
75
|
+
log(`Local version: ${localVersion}`);
|
|
76
|
+
|
|
77
|
+
// Get current npm version
|
|
78
|
+
let npmVersion = '';
|
|
79
|
+
try {
|
|
80
|
+
npmVersion = exec(`npm view ${name} version 2>/dev/null`, true);
|
|
81
|
+
log(`npm version: ${npmVersion || 'not published'}`);
|
|
82
|
+
} catch {
|
|
83
|
+
log('npm version: not published yet');
|
|
84
|
+
}
|
|
85
|
+
|
|
86
|
+
// Determine new version
|
|
87
|
+
let newVersion = localVersion;
|
|
88
|
+
if (npmVersion) {
|
|
89
|
+
const localParsed = parseVersion(localVersion);
|
|
90
|
+
const npmParsed = parseVersion(npmVersion);
|
|
91
|
+
|
|
92
|
+
// If local version <= npm version, bump from npm version
|
|
93
|
+
if (
|
|
94
|
+
localParsed.major < npmParsed.major ||
|
|
95
|
+
(localParsed.major === npmParsed.major && localParsed.minor < npmParsed.minor) ||
|
|
96
|
+
(localParsed.major === npmParsed.major &&
|
|
97
|
+
localParsed.minor === npmParsed.minor &&
|
|
98
|
+
localParsed.patch <= npmParsed.patch)
|
|
99
|
+
) {
|
|
100
|
+
newVersion = bumpPatch(npmVersion);
|
|
101
|
+
}
|
|
102
|
+
}
|
|
103
|
+
|
|
104
|
+
if (newVersion !== localVersion) {
|
|
105
|
+
log(`Bumping version: ${localVersion} → ${newVersion}`);
|
|
106
|
+
|
|
107
|
+
if (!isDryRun) {
|
|
108
|
+
packageJson.version = newVersion;
|
|
109
|
+
writeFileSync(packageJsonPath, JSON.stringify(packageJson, null, 2) + '\n');
|
|
110
|
+
}
|
|
111
|
+
} else {
|
|
112
|
+
log(`Version unchanged: ${newVersion}`);
|
|
113
|
+
}
|
|
114
|
+
|
|
115
|
+
// Run typecheck
|
|
116
|
+
log('Running typecheck...');
|
|
117
|
+
if (!isDryRun) {
|
|
118
|
+
exec('bun run typecheck');
|
|
119
|
+
success('Typecheck passed');
|
|
120
|
+
}
|
|
121
|
+
|
|
122
|
+
// Build
|
|
123
|
+
log('Building...');
|
|
124
|
+
if (!isDryRun) {
|
|
125
|
+
exec('bun run build');
|
|
126
|
+
success('Build completed');
|
|
127
|
+
}
|
|
128
|
+
|
|
129
|
+
// Git operations
|
|
130
|
+
const gitStatus = exec('git status --porcelain', true);
|
|
131
|
+
if (gitStatus) {
|
|
132
|
+
log('Staging changes...');
|
|
133
|
+
if (!isDryRun) {
|
|
134
|
+
exec('git add package.json');
|
|
135
|
+
exec(`git commit -m "chore: release v${newVersion}"`);
|
|
136
|
+
}
|
|
137
|
+
}
|
|
138
|
+
|
|
139
|
+
// Create git tag
|
|
140
|
+
const tagName = `v${newVersion}`;
|
|
141
|
+
log(`Creating tag: ${tagName}`);
|
|
142
|
+
if (!isDryRun) {
|
|
143
|
+
try {
|
|
144
|
+
exec(`git tag ${tagName}`);
|
|
145
|
+
success(`Tag ${tagName} created`);
|
|
146
|
+
} catch {
|
|
147
|
+
warn(`Tag ${tagName} already exists`);
|
|
148
|
+
}
|
|
149
|
+
}
|
|
150
|
+
|
|
151
|
+
// Publish
|
|
152
|
+
log('Publishing to npm...');
|
|
153
|
+
if (isDryRun) {
|
|
154
|
+
warn('Dry run - skipping publish');
|
|
155
|
+
exec('npm publish --dry-run');
|
|
156
|
+
} else {
|
|
157
|
+
exec('npm publish');
|
|
158
|
+
success(`Published ${name}@${newVersion}`);
|
|
159
|
+
}
|
|
160
|
+
|
|
161
|
+
// Push tags
|
|
162
|
+
if (!isDryRun) {
|
|
163
|
+
log('Pushing tags...');
|
|
164
|
+
try {
|
|
165
|
+
exec('git push --tags');
|
|
166
|
+
success('Tags pushed');
|
|
167
|
+
} catch {
|
|
168
|
+
warn('Failed to push tags (you may need to push manually)');
|
|
169
|
+
}
|
|
170
|
+
}
|
|
171
|
+
|
|
172
|
+
console.log('');
|
|
173
|
+
success(`Release ${isDryRun ? '(dry run) ' : ''}complete: ${name}@${newVersion}`);
|
|
174
|
+
}
|
|
175
|
+
|
|
176
|
+
main().catch((err) => {
|
|
177
|
+
error(String(err));
|
|
178
|
+
process.exit(1);
|
|
179
|
+
});
|
|
@@ -0,0 +1,33 @@
|
|
|
1
|
+
import type { ConnectorClient } from './client';
|
|
2
|
+
import type { ChannelDetails, ListChannelsParams } from '../types';
|
|
3
|
+
|
|
4
|
+
export class ChannelsApi {
|
|
5
|
+
constructor(private readonly client: ConnectorClient) {}
|
|
6
|
+
|
|
7
|
+
/**
|
|
8
|
+
* List active channels
|
|
9
|
+
* GET /channels
|
|
10
|
+
*/
|
|
11
|
+
async list(params?: ListChannelsParams): Promise<ChannelDetails[]> {
|
|
12
|
+
const queryParams: Record<string, string | number | boolean | undefined> = {};
|
|
13
|
+
if (params?.limit) queryParams.limit = params.limit;
|
|
14
|
+
if (params?.prefix) queryParams.prefix = params.prefix;
|
|
15
|
+
if (params?.by) queryParams.by = params.by;
|
|
16
|
+
|
|
17
|
+
const result = await this.client.get<ChannelDetails[] | unknown>('/channels', queryParams);
|
|
18
|
+
|
|
19
|
+
if (Array.isArray(result)) {
|
|
20
|
+
return result;
|
|
21
|
+
}
|
|
22
|
+
|
|
23
|
+
return [];
|
|
24
|
+
}
|
|
25
|
+
|
|
26
|
+
/**
|
|
27
|
+
* Get channel details
|
|
28
|
+
* GET /channels/{channelId}
|
|
29
|
+
*/
|
|
30
|
+
async get(channelId: string): Promise<ChannelDetails> {
|
|
31
|
+
return this.client.get<ChannelDetails>(`/channels/${encodeURIComponent(channelId)}`);
|
|
32
|
+
}
|
|
33
|
+
}
|
|
@@ -0,0 +1,203 @@
|
|
|
1
|
+
import type { ConnectorConfig, OutputFormat } from '../types';
|
|
2
|
+
import { ConnectorApiError, parseApiError } from '../types';
|
|
3
|
+
|
|
4
|
+
const DEFAULT_BASE_URL = 'https://rest.ably.io';
|
|
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
|
+
format?: OutputFormat;
|
|
12
|
+
/** Number of retries for failed requests (default: 3) */
|
|
13
|
+
retries?: number;
|
|
14
|
+
/** Timeout in milliseconds (default: 30000) */
|
|
15
|
+
timeout?: number;
|
|
16
|
+
}
|
|
17
|
+
|
|
18
|
+
export class ConnectorClient {
|
|
19
|
+
private readonly apiKey: string;
|
|
20
|
+
private readonly baseUrl: string;
|
|
21
|
+
|
|
22
|
+
constructor(config: ConnectorConfig) {
|
|
23
|
+
const key = config.apiKey || config.token || config.accessToken;
|
|
24
|
+
if (!key) {
|
|
25
|
+
throw new Error('Ably API key is required (format: appId.keyId:keySecret)');
|
|
26
|
+
}
|
|
27
|
+
this.apiKey = key;
|
|
28
|
+
this.baseUrl = config.baseUrl || DEFAULT_BASE_URL;
|
|
29
|
+
}
|
|
30
|
+
|
|
31
|
+
private buildUrl(path: string, params?: Record<string, string | number | boolean | undefined>): string {
|
|
32
|
+
const url = new URL(`${this.baseUrl}${path}`);
|
|
33
|
+
|
|
34
|
+
if (params) {
|
|
35
|
+
Object.entries(params).forEach(([key, value]) => {
|
|
36
|
+
if (value !== undefined && value !== null && value !== '') {
|
|
37
|
+
url.searchParams.append(key, String(value));
|
|
38
|
+
}
|
|
39
|
+
});
|
|
40
|
+
}
|
|
41
|
+
|
|
42
|
+
return url.toString();
|
|
43
|
+
}
|
|
44
|
+
|
|
45
|
+
/**
|
|
46
|
+
* Sleep for a given number of milliseconds
|
|
47
|
+
*/
|
|
48
|
+
private sleep(ms: number): Promise<void> {
|
|
49
|
+
return new Promise(resolve => setTimeout(resolve, ms));
|
|
50
|
+
}
|
|
51
|
+
|
|
52
|
+
/**
|
|
53
|
+
* Calculate delay for exponential backoff
|
|
54
|
+
*/
|
|
55
|
+
private getRetryDelay(attempt: number, baseDelay: number = 1000): number {
|
|
56
|
+
// Exponential backoff with jitter: base * 2^attempt + random(0-1000)ms
|
|
57
|
+
return baseDelay * Math.pow(2, attempt) + Math.random() * 1000;
|
|
58
|
+
}
|
|
59
|
+
|
|
60
|
+
/**
|
|
61
|
+
* Check if error is retryable
|
|
62
|
+
*/
|
|
63
|
+
private isRetryableStatus(status: number): boolean {
|
|
64
|
+
// Retry on rate limit (429) and server errors (5xx)
|
|
65
|
+
return status === 429 || (status >= 500 && status < 600);
|
|
66
|
+
}
|
|
67
|
+
|
|
68
|
+
/**
|
|
69
|
+
* Make an authenticated request to the Ably REST API
|
|
70
|
+
*/
|
|
71
|
+
async request<T>(path: string, options: RequestOptions = {}): Promise<T> {
|
|
72
|
+
const { method = 'GET', params, body, headers = {}, retries = 3, timeout = 30000 } = options;
|
|
73
|
+
|
|
74
|
+
const url = this.buildUrl(path, params);
|
|
75
|
+
|
|
76
|
+
const requestHeaders: Record<string, string> = {
|
|
77
|
+
'Authorization': `Basic ${btoa(this.apiKey)}`,
|
|
78
|
+
'Accept': 'application/json',
|
|
79
|
+
...headers,
|
|
80
|
+
};
|
|
81
|
+
|
|
82
|
+
if (body && ['POST', 'PUT', 'PATCH'].includes(method)) {
|
|
83
|
+
requestHeaders['Content-Type'] = 'application/json';
|
|
84
|
+
}
|
|
85
|
+
|
|
86
|
+
const fetchOptions: RequestInit = {
|
|
87
|
+
method,
|
|
88
|
+
headers: requestHeaders,
|
|
89
|
+
};
|
|
90
|
+
|
|
91
|
+
if (body && ['POST', 'PUT', 'PATCH'].includes(method)) {
|
|
92
|
+
fetchOptions.body = typeof body === 'string' ? body : JSON.stringify(body);
|
|
93
|
+
}
|
|
94
|
+
|
|
95
|
+
let lastError: Error | null = null;
|
|
96
|
+
let lastStatus: number = 0;
|
|
97
|
+
|
|
98
|
+
for (let attempt = 0; attempt <= retries; attempt++) {
|
|
99
|
+
try {
|
|
100
|
+
// Create abort controller for timeout
|
|
101
|
+
const controller = new AbortController();
|
|
102
|
+
const timeoutId = setTimeout(() => controller.abort(), timeout);
|
|
103
|
+
|
|
104
|
+
const response = await fetch(url, {
|
|
105
|
+
...fetchOptions,
|
|
106
|
+
signal: controller.signal,
|
|
107
|
+
});
|
|
108
|
+
|
|
109
|
+
clearTimeout(timeoutId);
|
|
110
|
+
lastStatus = response.status;
|
|
111
|
+
|
|
112
|
+
// Handle 204 No Content
|
|
113
|
+
if (response.status === 204) {
|
|
114
|
+
return {} as T;
|
|
115
|
+
}
|
|
116
|
+
|
|
117
|
+
// Parse response
|
|
118
|
+
let data: unknown;
|
|
119
|
+
const contentType = response.headers.get('content-type') || '';
|
|
120
|
+
|
|
121
|
+
if (contentType.includes('application/json')) {
|
|
122
|
+
const text = await response.text();
|
|
123
|
+
if (text) {
|
|
124
|
+
try {
|
|
125
|
+
data = JSON.parse(text);
|
|
126
|
+
} catch {
|
|
127
|
+
data = text;
|
|
128
|
+
}
|
|
129
|
+
}
|
|
130
|
+
} else {
|
|
131
|
+
data = await response.text();
|
|
132
|
+
}
|
|
133
|
+
|
|
134
|
+
// Handle errors
|
|
135
|
+
if (!response.ok) {
|
|
136
|
+
// Check if we should retry
|
|
137
|
+
if (this.isRetryableStatus(response.status) && attempt < retries) {
|
|
138
|
+
// Check for Retry-After header
|
|
139
|
+
const retryAfter = response.headers.get('retry-after');
|
|
140
|
+
const delay = retryAfter
|
|
141
|
+
? parseInt(retryAfter, 10) * 1000
|
|
142
|
+
: this.getRetryDelay(attempt);
|
|
143
|
+
|
|
144
|
+
await this.sleep(delay);
|
|
145
|
+
continue;
|
|
146
|
+
}
|
|
147
|
+
|
|
148
|
+
throw parseApiError(data, response.status);
|
|
149
|
+
}
|
|
150
|
+
|
|
151
|
+
return data as T;
|
|
152
|
+
} catch (err) {
|
|
153
|
+
lastError = err instanceof Error ? err : new Error(String(err));
|
|
154
|
+
|
|
155
|
+
// Handle timeout errors
|
|
156
|
+
if (lastError.name === 'AbortError') {
|
|
157
|
+
lastError = new Error(`Request timeout after ${timeout}ms`);
|
|
158
|
+
}
|
|
159
|
+
|
|
160
|
+
// Retry on network errors
|
|
161
|
+
if (attempt < retries && !(err instanceof ConnectorApiError)) {
|
|
162
|
+
await this.sleep(this.getRetryDelay(attempt));
|
|
163
|
+
continue;
|
|
164
|
+
}
|
|
165
|
+
|
|
166
|
+
throw err;
|
|
167
|
+
}
|
|
168
|
+
}
|
|
169
|
+
|
|
170
|
+
// Should not reach here, but just in case
|
|
171
|
+
throw lastError || new ConnectorApiError('Request failed', lastStatus);
|
|
172
|
+
}
|
|
173
|
+
|
|
174
|
+
async get<T>(path: string, params?: Record<string, string | number | boolean | undefined>): Promise<T> {
|
|
175
|
+
return this.request<T>(path, { method: 'GET', params });
|
|
176
|
+
}
|
|
177
|
+
|
|
178
|
+
async post<T>(path: string, body?: Record<string, unknown> | unknown[] | string | object, params?: Record<string, string | number | boolean | undefined>): Promise<T> {
|
|
179
|
+
return this.request<T>(path, { method: 'POST', body: body as Record<string, unknown>, params });
|
|
180
|
+
}
|
|
181
|
+
|
|
182
|
+
async put<T>(path: string, body?: Record<string, unknown> | object, params?: Record<string, string | number | boolean | undefined>): Promise<T> {
|
|
183
|
+
return this.request<T>(path, { method: 'PUT', body: body as Record<string, unknown>, params });
|
|
184
|
+
}
|
|
185
|
+
|
|
186
|
+
async patch<T>(path: string, body?: Record<string, unknown> | object, params?: Record<string, string | number | boolean | undefined>): Promise<T> {
|
|
187
|
+
return this.request<T>(path, { method: 'PATCH', body: body as Record<string, unknown>, params });
|
|
188
|
+
}
|
|
189
|
+
|
|
190
|
+
async delete<T>(path: string, params?: Record<string, string | number | boolean | undefined>): Promise<T> {
|
|
191
|
+
return this.request<T>(path, { method: 'DELETE', params });
|
|
192
|
+
}
|
|
193
|
+
|
|
194
|
+
/**
|
|
195
|
+
* Get a preview of the API key (for display/debugging)
|
|
196
|
+
*/
|
|
197
|
+
getApiKeyPreview(): string {
|
|
198
|
+
if (this.apiKey.length > 10) {
|
|
199
|
+
return `${this.apiKey.substring(0, 6)}...${this.apiKey.substring(this.apiKey.length - 4)}`;
|
|
200
|
+
}
|
|
201
|
+
return '***';
|
|
202
|
+
}
|
|
203
|
+
}
|
|
@@ -0,0 +1,59 @@
|
|
|
1
|
+
import type { ConnectorConfig } from '../types';
|
|
2
|
+
import { ConnectorClient } from './client';
|
|
3
|
+
import { MessagesApi } from './messages';
|
|
4
|
+
import { ChannelsApi } from './channels';
|
|
5
|
+
import { PresenceApi } from './presence';
|
|
6
|
+
import { StatsApi } from './stats';
|
|
7
|
+
|
|
8
|
+
/**
|
|
9
|
+
* Ably REST API Connector
|
|
10
|
+
*/
|
|
11
|
+
export class Connector {
|
|
12
|
+
private readonly client: ConnectorClient;
|
|
13
|
+
|
|
14
|
+
public readonly messages: MessagesApi;
|
|
15
|
+
public readonly channels: ChannelsApi;
|
|
16
|
+
public readonly presence: PresenceApi;
|
|
17
|
+
public readonly stats: StatsApi;
|
|
18
|
+
|
|
19
|
+
constructor(config: ConnectorConfig) {
|
|
20
|
+
this.client = new ConnectorClient(config);
|
|
21
|
+
this.messages = new MessagesApi(this.client);
|
|
22
|
+
this.channels = new ChannelsApi(this.client);
|
|
23
|
+
this.presence = new PresenceApi(this.client);
|
|
24
|
+
this.stats = new StatsApi(this.client);
|
|
25
|
+
}
|
|
26
|
+
|
|
27
|
+
/**
|
|
28
|
+
* Create a client from environment variables
|
|
29
|
+
* Looks for ABLY_API_KEY
|
|
30
|
+
*/
|
|
31
|
+
static fromEnv(): Connector {
|
|
32
|
+
const apiKey = process.env.ABLY_API_KEY;
|
|
33
|
+
|
|
34
|
+
if (!apiKey) {
|
|
35
|
+
throw new Error('ABLY_API_KEY environment variable is required');
|
|
36
|
+
}
|
|
37
|
+
return new Connector({ apiKey });
|
|
38
|
+
}
|
|
39
|
+
|
|
40
|
+
/**
|
|
41
|
+
* Get a preview of the API key (for debugging)
|
|
42
|
+
*/
|
|
43
|
+
getApiKeyPreview(): string {
|
|
44
|
+
return this.client.getApiKeyPreview();
|
|
45
|
+
}
|
|
46
|
+
|
|
47
|
+
/**
|
|
48
|
+
* Get the underlying client for direct API access
|
|
49
|
+
*/
|
|
50
|
+
getClient(): ConnectorClient {
|
|
51
|
+
return this.client;
|
|
52
|
+
}
|
|
53
|
+
}
|
|
54
|
+
|
|
55
|
+
export { ConnectorClient } from './client';
|
|
56
|
+
export { MessagesApi } from './messages';
|
|
57
|
+
export { ChannelsApi } from './channels';
|
|
58
|
+
export { PresenceApi } from './presence';
|
|
59
|
+
export { StatsApi } from './stats';
|
|
@@ -0,0 +1,48 @@
|
|
|
1
|
+
import type { ConnectorClient } from './client';
|
|
2
|
+
import type {
|
|
3
|
+
Message,
|
|
4
|
+
PublishMessageParams,
|
|
5
|
+
PublishMessageResult,
|
|
6
|
+
MessageHistoryParams,
|
|
7
|
+
} from '../types';
|
|
8
|
+
|
|
9
|
+
export class MessagesApi {
|
|
10
|
+
constructor(private readonly client: ConnectorClient) {}
|
|
11
|
+
|
|
12
|
+
/**
|
|
13
|
+
* Publish a message to a channel
|
|
14
|
+
* POST /channels/{channelId}/messages
|
|
15
|
+
*/
|
|
16
|
+
async publish(channelId: string, params: PublishMessageParams): Promise<PublishMessageResult> {
|
|
17
|
+
const body: Record<string, unknown> = {};
|
|
18
|
+
if (params.name !== undefined) body.name = params.name;
|
|
19
|
+
if (params.data !== undefined) body.data = params.data;
|
|
20
|
+
if (params.id !== undefined) body.id = params.id;
|
|
21
|
+
if (params.clientId !== undefined) body.clientId = params.clientId;
|
|
22
|
+
if (params.extras !== undefined) body.extras = params.extras;
|
|
23
|
+
|
|
24
|
+
await this.client.post(`/channels/${encodeURIComponent(channelId)}/messages`, body);
|
|
25
|
+
|
|
26
|
+
return {
|
|
27
|
+
channel: channelId,
|
|
28
|
+
messageId: params.id || '',
|
|
29
|
+
};
|
|
30
|
+
}
|
|
31
|
+
|
|
32
|
+
/**
|
|
33
|
+
* Get message history for a channel
|
|
34
|
+
* GET /channels/{channelId}/messages
|
|
35
|
+
*/
|
|
36
|
+
async history(channelId: string, params?: MessageHistoryParams): Promise<Message[]> {
|
|
37
|
+
const queryParams: Record<string, string | number | boolean | undefined> = {};
|
|
38
|
+
if (params?.start) queryParams.start = params.start;
|
|
39
|
+
if (params?.end) queryParams.end = params.end;
|
|
40
|
+
if (params?.limit) queryParams.limit = params.limit;
|
|
41
|
+
if (params?.direction) queryParams.direction = params.direction;
|
|
42
|
+
|
|
43
|
+
return this.client.get<Message[]>(
|
|
44
|
+
`/channels/${encodeURIComponent(channelId)}/messages`,
|
|
45
|
+
queryParams,
|
|
46
|
+
);
|
|
47
|
+
}
|
|
48
|
+
}
|
|
@@ -0,0 +1,39 @@
|
|
|
1
|
+
import type { ConnectorClient } from './client';
|
|
2
|
+
import type { PresenceMember, PresenceParams, PresenceHistoryParams } from '../types';
|
|
3
|
+
|
|
4
|
+
export class PresenceApi {
|
|
5
|
+
constructor(private readonly client: ConnectorClient) {}
|
|
6
|
+
|
|
7
|
+
/**
|
|
8
|
+
* Get current presence members for a channel
|
|
9
|
+
* GET /channels/{channelId}/presence
|
|
10
|
+
*/
|
|
11
|
+
async get(channelId: string, params?: PresenceParams): Promise<PresenceMember[]> {
|
|
12
|
+
const queryParams: Record<string, string | number | boolean | undefined> = {};
|
|
13
|
+
if (params?.clientId) queryParams.clientId = params.clientId;
|
|
14
|
+
if (params?.connectionId) queryParams.connectionId = params.connectionId;
|
|
15
|
+
if (params?.limit) queryParams.limit = params.limit;
|
|
16
|
+
|
|
17
|
+
return this.client.get<PresenceMember[]>(
|
|
18
|
+
`/channels/${encodeURIComponent(channelId)}/presence`,
|
|
19
|
+
queryParams,
|
|
20
|
+
);
|
|
21
|
+
}
|
|
22
|
+
|
|
23
|
+
/**
|
|
24
|
+
* Get presence history for a channel
|
|
25
|
+
* GET /channels/{channelId}/presence/history
|
|
26
|
+
*/
|
|
27
|
+
async history(channelId: string, params?: PresenceHistoryParams): Promise<PresenceMember[]> {
|
|
28
|
+
const queryParams: Record<string, string | number | boolean | undefined> = {};
|
|
29
|
+
if (params?.start) queryParams.start = params.start;
|
|
30
|
+
if (params?.end) queryParams.end = params.end;
|
|
31
|
+
if (params?.limit) queryParams.limit = params.limit;
|
|
32
|
+
if (params?.direction) queryParams.direction = params.direction;
|
|
33
|
+
|
|
34
|
+
return this.client.get<PresenceMember[]>(
|
|
35
|
+
`/channels/${encodeURIComponent(channelId)}/presence/history`,
|
|
36
|
+
queryParams,
|
|
37
|
+
);
|
|
38
|
+
}
|
|
39
|
+
}
|
|
@@ -0,0 +1,29 @@
|
|
|
1
|
+
import type { ConnectorClient } from './client';
|
|
2
|
+
import type { Stats, StatsParams } from '../types';
|
|
3
|
+
|
|
4
|
+
export class StatsApi {
|
|
5
|
+
constructor(private readonly client: ConnectorClient) {}
|
|
6
|
+
|
|
7
|
+
/**
|
|
8
|
+
* Get application statistics
|
|
9
|
+
* GET /stats
|
|
10
|
+
*/
|
|
11
|
+
async get(params?: StatsParams): Promise<Stats[]> {
|
|
12
|
+
const queryParams: Record<string, string | number | boolean | undefined> = {};
|
|
13
|
+
if (params?.start) queryParams.start = params.start;
|
|
14
|
+
if (params?.end) queryParams.end = params.end;
|
|
15
|
+
if (params?.limit) queryParams.limit = params.limit;
|
|
16
|
+
if (params?.direction) queryParams.direction = params.direction;
|
|
17
|
+
if (params?.unit) queryParams.unit = params.unit;
|
|
18
|
+
|
|
19
|
+
return this.client.get<Stats[]>('/stats', queryParams);
|
|
20
|
+
}
|
|
21
|
+
|
|
22
|
+
/**
|
|
23
|
+
* Get server time
|
|
24
|
+
* GET /time
|
|
25
|
+
*/
|
|
26
|
+
async time(): Promise<number[]> {
|
|
27
|
+
return this.client.get<number[]>('/time');
|
|
28
|
+
}
|
|
29
|
+
}
|