hvac-marketing-skills/tools/clis/postmark.js
Corey Haines 3a85964305 feat: add 23 new CLI tools and integration guides
New tools across 13 categories:
- Email/Newsletter: beehiiv, klaviyo, postmark, brevo, activecampaign
- Data Enrichment: clearbit, apollo
- CRO/Testing: hotjar, optimizely
- Analytics: plausible
- Scheduling: calendly, savvycal
- Forms: typeform
- Messaging: intercom
- Social: buffer
- Video: wistia
- Payments: paddle
- Affiliate: partnerstack
- Reviews: trustpilot, g2
- Push: onesignal
- Webinar: demio, livestorm

Each tool includes a zero-dependency CLI and integration guide.
Registry and CLI README updated with all new entries.

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

366 lines
13 KiB
JavaScript
Executable file

#!/usr/bin/env node
const API_KEY = process.env.POSTMARK_API_KEY
const BASE_URL = 'https://api.postmarkapp.com'
if (!API_KEY) {
console.error(JSON.stringify({ error: 'POSTMARK_API_KEY environment variable required' }))
process.exit(1)
}
async function api(method, path, body, useAccountToken) {
const headers = {
'Content-Type': 'application/json',
'Accept': 'application/json',
}
if (useAccountToken) {
headers['X-Postmark-Account-Token'] = API_KEY
} else {
headers['X-Postmark-Server-Token'] = API_KEY
}
const res = await fetch(`${BASE_URL}${path}`, {
method,
headers,
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 'email':
switch (sub) {
case 'send': {
const from = args.from
const to = args.to
const subject = args.subject
if (!from) { result = { error: '--from required' }; break }
if (!to) { result = { error: '--to required' }; break }
if (!subject) { result = { error: '--subject required' }; break }
const body = {
From: from,
To: to,
Subject: subject,
}
if (args.html) body.HtmlBody = args.html
if (args.text) body.TextBody = args.text
if (!args.html && !args.text) body.TextBody = ''
if (args.tag) body.Tag = args.tag
if (args.stream) body.MessageStream = args.stream
if (args['track-opens']) body.TrackOpens = true
if (args['track-links']) body.TrackLinks = args['track-links']
if (args.cc) body.Cc = args.cc
if (args.bcc) body.Bcc = args.bcc
if (args['reply-to']) body.ReplyTo = args['reply-to']
result = await api('POST', '/email', body)
break
}
case 'send-template': {
const from = args.from
const to = args.to
const template = args.template
if (!from) { result = { error: '--from required' }; break }
if (!to) { result = { error: '--to required' }; break }
if (!template) { result = { error: '--template required (template ID or alias)' }; break }
const body = {
From: from,
To: to,
TemplateModel: {},
}
const templateNum = Number(template)
if (!isNaN(templateNum)) {
body.TemplateId = templateNum
} else {
body.TemplateAlias = template
}
if (args.model) {
const pairs = args.model.split(',')
for (const pair of pairs) {
const [k, v] = pair.split(':')
if (k && v) body.TemplateModel[k] = v
}
}
if (args.stream) body.MessageStream = args.stream
if (args.tag) body.Tag = args.tag
result = await api('POST', '/email/withTemplate', body)
break
}
case 'send-batch': {
const from = args.from
const to = args.to
const subject = args.subject
if (!from || !to || !subject) {
result = { error: '--from, --to (comma-separated), and --subject required' }; break
}
const recipients = to.split(',')
const messages = recipients.map(recipient => ({
From: from,
To: recipient.trim(),
Subject: subject,
TextBody: args.text || '',
HtmlBody: args.html || undefined,
MessageStream: args.stream || undefined,
Tag: args.tag || undefined,
}))
result = await api('POST', '/email/batch', messages)
break
}
default:
result = { error: 'Unknown email subcommand. Use: send, send-template, send-batch' }
}
break
case 'templates':
switch (sub) {
case 'list': {
const params = new URLSearchParams()
params.set('Count', args.count || '100')
params.set('Offset', args.offset || '0')
if (args.type) params.set('TemplateType', args.type)
result = await api('GET', `/templates?${params.toString()}`)
break
}
case 'get': {
const id = args.id
if (!id) { result = { error: '--id required (template ID or alias)' }; break }
result = await api('GET', `/templates/${id}`)
break
}
case 'create': {
const name = args.name
if (!name) { result = { error: '--name required' }; break }
const body = {
Name: name,
Subject: args.subject || '',
}
if (args.html) body.HtmlBody = args.html
if (args.text) body.TextBody = args.text
if (args.alias) body.Alias = args.alias
if (args.type) body.TemplateType = args.type
result = await api('POST', '/templates', body)
break
}
case 'delete': {
const id = args.id
if (!id) { result = { error: '--id required (template ID or alias)' }; break }
result = await api('DELETE', `/templates/${id}`)
break
}
default:
result = { error: 'Unknown templates subcommand. Use: list, get, create, delete' }
}
break
case 'bounces':
switch (sub) {
case 'list': {
const params = new URLSearchParams()
params.set('count', args.count || '50')
params.set('offset', args.offset || '0')
if (args.type) params.set('type', args.type)
if (args.inactive) params.set('inactive', args.inactive)
if (args.email) params.set('emailFilter', args.email)
result = await api('GET', `/bounces?${params.toString()}`)
break
}
case 'get': {
const id = args.id
if (!id) { result = { error: '--id required' }; break }
result = await api('GET', `/bounces/${id}`)
break
}
case 'stats':
result = await api('GET', '/deliverystats')
break
case 'activate': {
const id = args.id
if (!id) { result = { error: '--id required' }; break }
result = await api('PUT', `/bounces/${id}/activate`)
break
}
default:
result = { error: 'Unknown bounces subcommand. Use: list, get, stats, activate' }
}
break
case 'messages':
switch (sub) {
case 'outbound': {
const params = new URLSearchParams()
params.set('count', args.count || '50')
params.set('offset', args.offset || '0')
if (args.recipient) params.set('recipient', args.recipient)
if (args.tag) params.set('tag', args.tag)
if (args.status) params.set('status', args.status)
result = await api('GET', `/messages/outbound?${params.toString()}`)
break
}
case 'inbound': {
const params = new URLSearchParams()
params.set('count', args.count || '50')
params.set('offset', args.offset || '0')
if (args.recipient) params.set('recipient', args.recipient)
if (args.status) params.set('status', args.status)
result = await api('GET', `/messages/inbound?${params.toString()}`)
break
}
case 'get': {
const id = args.id
if (!id) { result = { error: '--id required (message ID)' }; break }
result = await api('GET', `/messages/outbound/${id}/details`)
break
}
default:
result = { error: 'Unknown messages subcommand. Use: outbound, inbound, get' }
}
break
case 'stats':
switch (sub) {
case 'overview': {
const params = new URLSearchParams()
if (args.tag) params.set('tag', args.tag)
if (args.from) params.set('fromdate', args.from)
if (args.to) params.set('todate', args.to)
result = await api('GET', `/stats/outbound?${params.toString()}`)
break
}
case 'sends': {
const params = new URLSearchParams()
if (args.tag) params.set('tag', args.tag)
if (args.from) params.set('fromdate', args.from)
if (args.to) params.set('todate', args.to)
result = await api('GET', `/stats/outbound/sends?${params.toString()}`)
break
}
case 'bounces': {
const params = new URLSearchParams()
if (args.tag) params.set('tag', args.tag)
if (args.from) params.set('fromdate', args.from)
if (args.to) params.set('todate', args.to)
result = await api('GET', `/stats/outbound/bounces?${params.toString()}`)
break
}
case 'opens': {
const params = new URLSearchParams()
if (args.tag) params.set('tag', args.tag)
if (args.from) params.set('fromdate', args.from)
if (args.to) params.set('todate', args.to)
result = await api('GET', `/stats/outbound/opens?${params.toString()}`)
break
}
case 'clicks': {
const params = new URLSearchParams()
if (args.tag) params.set('tag', args.tag)
if (args.from) params.set('fromdate', args.from)
if (args.to) params.set('todate', args.to)
result = await api('GET', `/stats/outbound/clicks?${params.toString()}`)
break
}
case 'spam': {
const params = new URLSearchParams()
if (args.tag) params.set('tag', args.tag)
if (args.from) params.set('fromdate', args.from)
if (args.to) params.set('todate', args.to)
result = await api('GET', `/stats/outbound/spam?${params.toString()}`)
break
}
default:
result = { error: 'Unknown stats subcommand. Use: overview, sends, bounces, opens, clicks, spam' }
}
break
case 'server':
switch (sub) {
case 'get':
result = await api('GET', '/server')
break
default:
result = { error: 'Unknown server subcommand. Use: get' }
}
break
case 'suppressions':
switch (sub) {
case 'list': {
const params = new URLSearchParams()
if (args.stream) params.set('MessageStream', args.stream)
result = await api('GET', `/message-streams/${args.stream || 'outbound'}/suppressions/dump`)
break
}
case 'create': {
const email = args.email
if (!email) { result = { error: '--email required' }; break }
const stream = args.stream || 'outbound'
result = await api('POST', `/message-streams/${stream}/suppressions`, {
Suppressions: email.split(',').map(e => ({ EmailAddress: e.trim() }))
})
break
}
case 'delete': {
const email = args.email
if (!email) { result = { error: '--email required' }; break }
const stream = args.stream || 'outbound'
result = await api('POST', `/message-streams/${stream}/suppressions/delete`, {
Suppressions: email.split(',').map(e => ({ EmailAddress: e.trim() }))
})
break
}
default:
result = { error: 'Unknown suppressions subcommand. Use: list, create, delete' }
}
break
default:
result = {
error: 'Unknown command',
usage: {
email: 'email [send --from <from> --to <to> --subject <subj> | send-template --from <from> --to <to> --template <id> | send-batch --from <from> --to <to1,to2> --subject <subj>]',
templates: 'templates [list | get --id <id> | create --name <name> | delete --id <id>]',
bounces: 'bounces [list | get --id <id> | stats | activate --id <id>]',
messages: 'messages [outbound | inbound | get --id <id>]',
stats: 'stats [overview | sends | bounces | opens | clicks | spam]',
server: 'server [get]',
suppressions: 'suppressions [list | create --email <email> | delete --email <email>]',
options: '--tag <tag> --from <date> --to <date> --stream <stream-id>',
}
}
}
console.log(JSON.stringify(result, null, 2))
}
main().catch(err => {
console.error(JSON.stringify({ error: err.message }))
process.exit(1)
})