hvac-marketing-skills/tools/clis/instantly.js
Corey Haines aad399682c feat: add email outreach CLIs for backlink building
Add 4 new zero-dependency CLI tools for email outreach:
- hunter.js: Email finding/verification via Hunter.io (query param auth)
- snov.js: Email finding + drip campaigns via Snov.io (OAuth2 auth)
- lemlist.js: Cold email campaigns via Lemlist (Basic auth)
- instantly.js: Cold email at scale via Instantly.ai (query param auth)

Includes integration guides and registry/README updates.

Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
2026-02-17 14:09:50 -08:00

270 lines
10 KiB
JavaScript
Executable file

#!/usr/bin/env node
const API_KEY = process.env.INSTANTLY_API_KEY
const BASE_URL = 'https://api.instantly.ai/api/v1'
if (!API_KEY) {
console.error(JSON.stringify({ error: 'INSTANTLY_API_KEY environment variable required' }))
process.exit(1)
}
async function api(method, path, body) {
const separator = path.includes('?') ? '&' : '?'
const url = `${BASE_URL}${path}${separator}api_key=${API_KEY}`
if (args['dry-run']) {
const maskedUrl = url.replace(API_KEY, '***')
const maskedBody = body ? JSON.parse(JSON.stringify(body)) : undefined
if (maskedBody && maskedBody.api_key) maskedBody.api_key = '***'
return { _dry_run: true, method, url: maskedUrl, headers: { 'Content-Type': 'application/json', 'Accept': 'application/json' }, body: maskedBody }
}
const res = await fetch(url, {
method,
headers: {
'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._
async function main() {
let result
switch (cmd) {
case 'campaigns':
switch (sub) {
case 'list': {
const params = new URLSearchParams()
if (args.limit) params.set('limit', args.limit)
if (args.skip) params.set('skip', args.skip)
const qs = params.toString()
result = await api('GET', `/campaign/list${qs ? '?' + qs : ''}`)
break
}
case 'get': {
const id = args.id
if (!id) { result = { error: '--id required' }; break }
const params = new URLSearchParams({ campaign_id: id })
result = await api('GET', `/campaign/get?${params.toString()}`)
break
}
case 'status': {
const id = args.id
if (!id) { result = { error: '--id required' }; break }
const params = new URLSearchParams({ campaign_id: id })
result = await api('GET', `/campaign/get/status?${params.toString()}`)
break
}
case 'launch': {
const id = args.id
if (!id) { result = { error: '--id required' }; break }
result = await api('POST', '/campaign/launch', { api_key: API_KEY, campaign_id: id })
break
}
case 'pause': {
const id = args.id
if (!id) { result = { error: '--id required' }; break }
result = await api('POST', '/campaign/pause', { api_key: API_KEY, campaign_id: id })
break
}
default:
result = { error: 'Unknown campaigns subcommand. Use: list, get, status, launch, pause' }
}
break
case 'leads':
switch (sub) {
case 'list': {
const campaignId = args['campaign-id']
if (!campaignId) { result = { error: '--campaign-id required' }; break }
const params = new URLSearchParams({ campaign_id: campaignId })
if (args.limit) params.set('limit', args.limit)
if (args.skip) params.set('skip', args.skip)
result = await api('GET', `/lead/get?${params.toString()}`)
break
}
case 'add': {
const campaignId = args['campaign-id']
const email = args.email
if (!campaignId) { result = { error: '--campaign-id required' }; break }
if (!email) { result = { error: '--email required' }; break }
const lead = { email }
if (args['first-name']) lead.first_name = args['first-name']
if (args['last-name']) lead.last_name = args['last-name']
if (args.company) lead.company_name = args.company
result = await api('POST', '/lead/add', { api_key: API_KEY, campaign_id: campaignId, leads: [lead] })
break
}
case 'delete': {
const campaignId = args['campaign-id']
const email = args.email
if (!campaignId) { result = { error: '--campaign-id required' }; break }
if (!email) { result = { error: '--email required' }; break }
result = await api('POST', '/lead/delete', { api_key: API_KEY, campaign_id: campaignId, delete_list: [email] })
break
}
case 'status': {
const campaignId = args['campaign-id']
const email = args.email
if (!campaignId) { result = { error: '--campaign-id required' }; break }
if (!email) { result = { error: '--email required' }; break }
const params = new URLSearchParams({ campaign_id: campaignId, email })
result = await api('GET', `/lead/get/status?${params.toString()}`)
break
}
default:
result = { error: 'Unknown leads subcommand. Use: list, add, delete, status' }
}
break
case 'accounts':
switch (sub) {
case 'list': {
const params = new URLSearchParams()
if (args.limit) params.set('limit', args.limit)
if (args.skip) params.set('skip', args.skip)
const qs = params.toString()
result = await api('GET', `/account/list${qs ? '?' + qs : ''}`)
break
}
case 'status': {
const accountId = args['account-id']
if (!accountId) { result = { error: '--account-id required' }; break }
const params = new URLSearchParams({ email: accountId })
result = await api('GET', `/account/get/status?${params.toString()}`)
break
}
case 'warmup-status': {
const accountId = args['account-id']
if (!accountId) { result = { error: '--account-id required' }; break }
const params = new URLSearchParams({ email: accountId })
result = await api('GET', `/account/get/warmup?${params.toString()}`)
break
}
default:
result = { error: 'Unknown accounts subcommand. Use: list, status, warmup-status' }
}
break
case 'analytics':
switch (sub) {
case 'campaign': {
const campaignId = args['campaign-id']
if (!campaignId) { result = { error: '--campaign-id required' }; break }
const body = { api_key: API_KEY, campaign_id: campaignId }
if (args['start-date']) body.start_date = args['start-date']
if (args['end-date']) body.end_date = args['end-date']
result = await api('POST', '/analytics/campaign/summary', body)
break
}
case 'steps': {
const campaignId = args['campaign-id']
if (!campaignId) { result = { error: '--campaign-id required' }; break }
const body = { api_key: API_KEY, campaign_id: campaignId }
if (args['start-date']) body.start_date = args['start-date']
if (args['end-date']) body.end_date = args['end-date']
result = await api('POST', '/analytics/campaign/step', body)
break
}
case 'account': {
const startDate = args['start-date']
const endDate = args['end-date']
if (!startDate) { result = { error: '--start-date required' }; break }
if (!endDate) { result = { error: '--end-date required' }; break }
result = await api('POST', '/analytics/campaign/count', { api_key: API_KEY, start_date: startDate, end_date: endDate })
break
}
default:
result = { error: 'Unknown analytics subcommand. Use: campaign, steps, account' }
}
break
case 'blocklist':
switch (sub) {
case 'list':
result = await api('GET', '/blocklist')
break
case 'add': {
const entries = args.entries
if (!entries) { result = { error: '--entries required (comma-separated emails or domains)' }; break }
const entryList = entries.split(',').map(e => e.trim())
result = await api('POST', '/blocklist/add', { api_key: API_KEY, entries: entryList })
break
}
default:
result = { error: 'Unknown blocklist subcommand. Use: list, add' }
}
break
default:
result = {
error: 'Unknown command',
usage: {
campaigns: {
list: 'campaigns list [--limit <n>] [--skip <n>]',
get: 'campaigns get --id <id>',
status: 'campaigns status --id <id>',
launch: 'campaigns launch --id <id>',
pause: 'campaigns pause --id <id>',
},
leads: {
list: 'leads list --campaign-id <id> [--limit <n>] [--skip <n>]',
add: 'leads add --campaign-id <id> --email <email> [--first-name <name>] [--last-name <name>] [--company <name>]',
delete: 'leads delete --campaign-id <id> --email <email>',
status: 'leads status --campaign-id <id> --email <email>',
},
accounts: {
list: 'accounts list [--limit <n>] [--skip <n>]',
status: 'accounts status --account-id <email>',
'warmup-status': 'accounts warmup-status --account-id <email>',
},
analytics: {
campaign: 'analytics campaign --campaign-id <id> [--start-date YYYY-MM-DD] [--end-date YYYY-MM-DD]',
steps: 'analytics steps --campaign-id <id> [--start-date YYYY-MM-DD] [--end-date YYYY-MM-DD]',
account: 'analytics account --start-date YYYY-MM-DD --end-date YYYY-MM-DD',
},
blocklist: {
list: 'blocklist list',
add: 'blocklist add --entries <email-or-domain,email-or-domain>',
},
options: '--dry-run (show request without executing)',
}
}
}
console.log(JSON.stringify(result, null, 2))
}
main().catch(err => {
console.error(JSON.stringify({ error: err.message }))
process.exit(1)
})