hvac-marketing-skills/tools/clis/partnerstack.js
Corey Haines 2c26f8497b feat: add --dry-run flag to all 47 CLI tools
When --dry-run is passed, each CLI prints the HTTP request it would
make (method, URL, headers, body) without actually calling fetch().
Auth credentials are masked as "***" in the output.

Useful for verifying request shape and API endpoints without needing
real API keys or making actual API calls.

Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
2026-02-17 11:47:12 -08:00

382 lines
14 KiB
JavaScript
Executable file

#!/usr/bin/env node
const PUBLIC_KEY = process.env.PARTNERSTACK_PUBLIC_KEY
const SECRET_KEY = process.env.PARTNERSTACK_SECRET_KEY
const BASE_URL = 'https://api.partnerstack.com/api/v2'
if (!PUBLIC_KEY || !SECRET_KEY) {
console.error(JSON.stringify({ error: 'PARTNERSTACK_PUBLIC_KEY and PARTNERSTACK_SECRET_KEY environment variables required' }))
process.exit(1)
}
const AUTH = 'Basic ' + Buffer.from(`${PUBLIC_KEY}:${SECRET_KEY}`).toString('base64')
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', 'Accept': 'application/json' }, body: body || undefined }
}
const res = await fetch(`${BASE_URL}${path}`, {
method,
headers: {
'Authorization': AUTH,
'Content-Type': 'application/json',
'Accept': '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._
const limit = args.limit ? Number(args.limit) : 10
function buildQuery() {
const params = new URLSearchParams()
if (args.limit) params.set('limit', args.limit)
if (args.after) params.set('starting_after', args.after)
if (args.before) params.set('ending_before', args.before)
if (args['order-by']) params.set('order_by', args['order-by'])
return params.toString() ? `?${params.toString()}` : ''
}
async function main() {
let result
switch (cmd) {
case 'partnerships':
switch (sub) {
case 'list':
result = await api('GET', `/partnerships${buildQuery()}`)
break
case 'get': {
const key = args.key
if (!key) { result = { error: '--key required (partnership key)' }; break }
result = await api('GET', `/partnerships/${key}`)
break
}
case 'create': {
const email = args.email
const group = args.group
if (!email) { result = { error: '--email required' }; break }
if (!group) { result = { error: '--group required (group key)' }; break }
const body = { email, group_key: group }
if (args.name) body.name = args.name
if (args['first-name']) body.first_name = args['first-name']
if (args['last-name']) body.last_name = args['last-name']
result = await api('POST', '/partnerships', body)
break
}
case 'update': {
const key = args.key
if (!key) { result = { error: '--key required (partnership key)' }; break }
const body = {}
if (args.name) body.name = args.name
if (args.group) body.group_key = args.group
result = await api('PATCH', `/partnerships/${key}`, body)
break
}
default:
result = { error: 'Unknown partnerships subcommand. Use: list, get, create, update' }
}
break
case 'customers':
switch (sub) {
case 'list':
result = await api('GET', `/customers${buildQuery()}`)
break
case 'get': {
const key = args.key
if (!key) { result = { error: '--key required (customer key)' }; break }
result = await api('GET', `/customers/${key}`)
break
}
case 'create': {
const email = args.email
const partnerKey = args['partner-key']
if (!email) { result = { error: '--email required' }; break }
if (!partnerKey) { result = { error: '--partner-key required' }; break }
const body = { email, partner_key: partnerKey }
if (args.name) body.name = args.name
if (args['customer-key']) body.customer_key = args['customer-key']
result = await api('POST', '/customers', body)
break
}
case 'update': {
const key = args.key
if (!key) { result = { error: '--key required (customer key)' }; break }
const body = {}
if (args.name) body.name = args.name
if (args.email) body.email = args.email
if (args['partner-key']) body.partner_key = args['partner-key']
result = await api('PATCH', `/customers/${key}`, body)
break
}
case 'delete': {
const key = args.key
if (!key) { result = { error: '--key required (customer key)' }; break }
result = await api('DELETE', `/customers/${key}`)
break
}
default:
result = { error: 'Unknown customers subcommand. Use: list, get, create, update, delete' }
}
break
case 'transactions':
switch (sub) {
case 'list':
result = await api('GET', `/transactions${buildQuery()}`)
break
case 'get': {
const key = args.key
if (!key) { result = { error: '--key required (transaction key)' }; break }
result = await api('GET', `/transactions/${key}`)
break
}
case 'create': {
const customerKey = args['customer-key']
const amount = args.amount
if (!customerKey) { result = { error: '--customer-key required' }; break }
if (!amount) { result = { error: '--amount required (in cents)' }; break }
const body = {
customer_key: customerKey,
amount: Number(amount),
}
if (args.currency) body.currency = args.currency
if (args.category) body.category = args.category
if (args['product-key']) body.product_key = args['product-key']
result = await api('POST', '/transactions', body)
break
}
case 'delete': {
const key = args.key
if (!key) { result = { error: '--key required (transaction key)' }; break }
result = await api('DELETE', `/transactions/${key}`)
break
}
default:
result = { error: 'Unknown transactions subcommand. Use: list, get, create, delete' }
}
break
case 'deals':
switch (sub) {
case 'list':
result = await api('GET', `/deals${buildQuery()}`)
break
case 'get': {
const key = args.key
if (!key) { result = { error: '--key required (deal key)' }; break }
result = await api('GET', `/deals/${key}`)
break
}
case 'create': {
const partnerKey = args['partner-key']
const name = args.name
if (!partnerKey) { result = { error: '--partner-key required' }; break }
if (!name) { result = { error: '--name required' }; break }
const body = { partner_key: partnerKey, name }
if (args.amount) body.amount = Number(args.amount)
if (args.currency) body.currency = args.currency
if (args.stage) body.stage = args.stage
result = await api('POST', '/deals', body)
break
}
case 'update': {
const key = args.key
if (!key) { result = { error: '--key required (deal key)' }; break }
const body = {}
if (args.name) body.name = args.name
if (args.amount) body.amount = Number(args.amount)
if (args.stage) body.stage = args.stage
if (args.status) body.status = args.status
result = await api('PATCH', `/deals/${key}`, body)
break
}
case 'archive': {
const key = args.key
if (!key) { result = { error: '--key required (deal key)' }; break }
result = await api('DELETE', `/deals/${key}`)
break
}
default:
result = { error: 'Unknown deals subcommand. Use: list, get, create, update, archive' }
}
break
case 'actions':
switch (sub) {
case 'list':
result = await api('GET', `/actions${buildQuery()}`)
break
case 'create': {
const customerKey = args['customer-key']
const actionKey = args['action-key']
if (!customerKey) { result = { error: '--customer-key required' }; break }
if (!actionKey) { result = { error: '--action-key required' }; break }
const body = {
customer_key: customerKey,
key: actionKey,
}
if (args.value) body.value = Number(args.value)
result = await api('POST', '/actions', body)
break
}
default:
result = { error: 'Unknown actions subcommand. Use: list, create' }
}
break
case 'rewards':
switch (sub) {
case 'list':
result = await api('GET', `/rewards${buildQuery()}`)
break
case 'create': {
const partnerKey = args['partner-key']
const amount = args.amount
if (!partnerKey) { result = { error: '--partner-key required' }; break }
if (!amount) { result = { error: '--amount required (in cents)' }; break }
const body = {
partner_key: partnerKey,
amount: Number(amount),
}
if (args.description) body.description = args.description
if (args.currency) body.currency = args.currency
result = await api('POST', '/rewards', body)
break
}
default:
result = { error: 'Unknown rewards subcommand. Use: list, create' }
}
break
case 'leads':
switch (sub) {
case 'list':
result = await api('GET', `/leads${buildQuery()}`)
break
case 'get': {
const key = args.key
if (!key) { result = { error: '--key required (lead key)' }; break }
result = await api('GET', `/leads/${key}`)
break
}
case 'create': {
const partnerKey = args['partner-key']
const email = args.email
if (!partnerKey) { result = { error: '--partner-key required' }; break }
if (!email) { result = { error: '--email required' }; break }
const body = { partner_key: partnerKey, email }
if (args.name) body.name = args.name
if (args.company) body.company = args.company
result = await api('POST', '/leads', body)
break
}
case 'update': {
const key = args.key
if (!key) { result = { error: '--key required (lead key)' }; break }
const body = {}
if (args.email) body.email = args.email
if (args.name) body.name = args.name
if (args.status) body.status = args.status
result = await api('PATCH', `/leads/${key}`, body)
break
}
default:
result = { error: 'Unknown leads subcommand. Use: list, get, create, update' }
}
break
case 'groups':
switch (sub) {
case 'list':
result = await api('GET', `/groups${buildQuery()}`)
break
default:
result = { error: 'Unknown groups subcommand. Use: list' }
}
break
case 'webhooks':
switch (sub) {
case 'list':
result = await api('GET', `/webhooks${buildQuery()}`)
break
case 'get': {
const key = args.key
if (!key) { result = { error: '--key required (webhook key)' }; break }
result = await api('GET', `/webhooks/${key}`)
break
}
case 'create': {
const target = args.target
if (!target) { result = { error: '--target required (webhook URL)' }; break }
const body = { target }
if (args.events) body.events = args.events.split(',')
result = await api('POST', '/webhooks', body)
break
}
case 'delete': {
const key = args.key
if (!key) { result = { error: '--key required (webhook key)' }; break }
result = await api('DELETE', `/webhooks/${key}`)
break
}
default:
result = { error: 'Unknown webhooks subcommand. Use: list, get, create, delete' }
}
break
default:
result = {
error: 'Unknown command',
usage: {
partnerships: 'partnerships [list | get --key <key> | create --email <email> --group <group-key> | update --key <key>]',
customers: 'customers [list | get --key <key> | create --email <email> --partner-key <key> | update --key <key> | delete --key <key>]',
transactions: 'transactions [list | get --key <key> | create --customer-key <key> --amount <cents> | delete --key <key>]',
deals: 'deals [list | get --key <key> | create --partner-key <key> --name <name> | update --key <key> | archive --key <key>]',
actions: 'actions [list | create --customer-key <key> --action-key <key> [--value <n>]]',
rewards: 'rewards [list | create --partner-key <key> --amount <cents>]',
leads: 'leads [list | get --key <key> | create --partner-key <key> --email <email> | update --key <key>]',
groups: 'groups [list]',
webhooks: 'webhooks [list | get --key <key> | create --target <url> [--events <evt1,evt2>] | delete --key <key>]',
options: '--limit <n> --after <cursor> --before <cursor> --order-by <field>',
}
}
}
console.log(JSON.stringify(result, null, 2))
}
main().catch(err => {
console.error(JSON.stringify({ error: err.message }))
process.exit(1)
})