#!/usr/bin/env node const API_KEY = process.env.DUB_API_KEY const BASE_URL = 'https://api.dub.co' if (!API_KEY) { console.error(JSON.stringify({ error: 'DUB_API_KEY environment variable required' })) process.exit(1) } async function api(method, path, body) { if (args['dry-run']) { return { _dry_run: true, method, url: `${BASE_URL}${path}`, headers: { Authorization: '***', 'Content-Type': 'application/json' }, body: body || undefined } } const res = await fetch(`${BASE_URL}${path}`, { method, headers: { 'Authorization': `Bearer ${API_KEY}`, 'Content-Type': 'application/json', }, body: body ? JSON.stringify(body) : undefined, }) const text = await res.text() try { return JSON.parse(text) } catch { return { status: res.status, body: text } } } function parseArgs(args) { const result = { _: [] } for (let i = 0; i < args.length; i++) { const arg = args[i] if (arg.startsWith('--')) { const key = arg.slice(2) const next = args[i + 1] if (next && !next.startsWith('--')) { result[key] = next i++ } else { result[key] = true } } else { result._.push(arg) } } return result } const args = parseArgs(process.argv.slice(2)) const [cmd, sub, ...rest] = args._ async function main() { let result switch (cmd) { case 'links': switch (sub) { case 'create': { if (!args.url) { result = { error: '--url required' }; break } const body = {} if (args.url) body.url = args.url if (args.domain) body.domain = args.domain if (args.key) body.key = args.key if (args.tags) body.tags = args.tags.split(',') result = await api('POST', '/links', body) break } case 'list': { const params = new URLSearchParams() if (args.domain) params.set('domain', args.domain) if (args.page) params.set('page', args.page) result = await api('GET', `/links?${params}`) break } case 'get': { const params = new URLSearchParams() if (args.domain) params.set('domain', args.domain) if (args.key) params.set('key', args.key) if (args['link-id']) params.set('linkId', args['link-id']) if (args['external-id']) params.set('externalId', args['external-id']) result = await api('GET', `/links/info?${params}`) break } case 'update': { if (!args.id) { result = { error: '--id required (link ID)' }; break } const body = {} if (args.url) body.url = args.url if (args.tags) body.tags = args.tags.split(',') result = await api('PATCH', `/links/${args.id}`, body) break } case 'delete': if (!args.id) { result = { error: '--id required (link ID)' }; break } result = await api('DELETE', `/links/${args.id}`) break case 'bulk-create': { let links try { links = JSON.parse(args.links || '[]') } catch { result = { error: 'Invalid JSON in --links' }; break } result = await api('POST', '/links/bulk', links) break } default: result = { error: 'Unknown links subcommand. Use: create, list, get, update, delete, bulk-create' } } break case 'analytics': switch (sub) { case 'get': { const params = new URLSearchParams() if (args.domain) params.set('domain', args.domain) if (args.key) params.set('key', args.key) if (args.interval) params.set('interval', args.interval) result = await api('GET', `/analytics?${params}`) break } case 'country': { const params = new URLSearchParams() if (args.domain) params.set('domain', args.domain) if (args.key) params.set('key', args.key) result = await api('GET', `/analytics/country?${params}`) break } case 'device': { const params = new URLSearchParams() if (args.domain) params.set('domain', args.domain) if (args.key) params.set('key', args.key) result = await api('GET', `/analytics/device?${params}`) break } default: result = { error: 'Unknown analytics subcommand. Use: get, country, device' } } break default: result = { error: 'Unknown command', usage: { links: 'links [create|list|get|update|delete|bulk-create] [--url ] [--domain ] [--key ] [--tags ] [--id ] [--page ] [--links ]', analytics: 'analytics [get|country|device] [--domain ] [--key ] [--interval ]', } } } console.log(JSON.stringify(result, null, 2)) } main().catch(err => { console.error(JSON.stringify({ error: err.message })) process.exit(1) })