Single-file Node.js scripts for every tool in the registry that lacked
a CLI. All follow the same pattern: env var auth, JSON output, consistent
`{tool} <resource> <action>` command structure, zero npm dependencies.
CLIs added: resend, sendgrid, mailchimp, kit, customer-io, ahrefs,
semrush, google-search-console, ga4, mixpanel, amplitude, segment,
adobe-analytics, rewardful, tolt, mention-me, dub, google-ads,
meta-ads, linkedin-ads, tiktok-ads, zapier.
Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
181 lines
5.7 KiB
JavaScript
Executable file
181 lines
5.7 KiB
JavaScript
Executable file
#!/usr/bin/env node
|
|
|
|
const ACCESS_TOKEN = process.env.GA4_ACCESS_TOKEN
|
|
const DATA_API = 'https://analyticsdata.googleapis.com/v1beta'
|
|
const ADMIN_API = 'https://analyticsadmin.googleapis.com/v1beta'
|
|
const MP_URL = 'https://www.google-analytics.com/mp/collect'
|
|
|
|
if (!ACCESS_TOKEN) {
|
|
console.error(JSON.stringify({ error: 'GA4_ACCESS_TOKEN environment variable required' }))
|
|
process.exit(1)
|
|
}
|
|
|
|
async function api(method, baseUrl, path, body) {
|
|
const res = await fetch(`${baseUrl}${path}`, {
|
|
method,
|
|
headers: {
|
|
'Authorization': `Bearer ${ACCESS_TOKEN}`,
|
|
'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 }
|
|
}
|
|
}
|
|
|
|
async function mpApi(measurementId, apiSecret, body) {
|
|
const params = new URLSearchParams({ measurement_id: measurementId, api_secret: apiSecret })
|
|
const res = await fetch(`${MP_URL}?${params}`, {
|
|
method: 'POST',
|
|
headers: { 'Content-Type': 'application/json' },
|
|
body: JSON.stringify(body),
|
|
})
|
|
const text = await res.text()
|
|
if (!text) return { status: res.status, success: res.ok }
|
|
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 'reports':
|
|
switch (sub) {
|
|
case 'run': {
|
|
const property = args.property
|
|
if (!property) { result = { error: '--property required' }; break }
|
|
const body = {
|
|
dateRanges: [{
|
|
startDate: args['start-date'] || '30daysAgo',
|
|
endDate: args['end-date'] || 'today',
|
|
}],
|
|
}
|
|
if (args.dimensions) {
|
|
body.dimensions = args.dimensions.split(',').map(d => ({ name: d.trim() }))
|
|
}
|
|
if (args.metrics) {
|
|
body.metrics = args.metrics.split(',').map(m => ({ name: m.trim() }))
|
|
}
|
|
result = await api('POST', DATA_API, `/properties/${property}:runReport`, body)
|
|
break
|
|
}
|
|
default:
|
|
result = { error: 'Unknown reports subcommand. Use: run' }
|
|
}
|
|
break
|
|
|
|
case 'realtime':
|
|
switch (sub) {
|
|
case 'run': {
|
|
const property = args.property
|
|
if (!property) { result = { error: '--property required' }; break }
|
|
const body = {}
|
|
if (args.dimensions) {
|
|
body.dimensions = args.dimensions.split(',').map(d => ({ name: d.trim() }))
|
|
}
|
|
if (args.metrics) {
|
|
body.metrics = args.metrics.split(',').map(m => ({ name: m.trim() }))
|
|
}
|
|
result = await api('POST', DATA_API, `/properties/${property}:runRealtimeReport`, body)
|
|
break
|
|
}
|
|
default:
|
|
result = { error: 'Unknown realtime subcommand. Use: run' }
|
|
}
|
|
break
|
|
|
|
case 'conversions':
|
|
switch (sub) {
|
|
case 'list': {
|
|
const property = args.property
|
|
if (!property) { result = { error: '--property required' }; break }
|
|
result = await api('GET', ADMIN_API, `/properties/${property}/conversionEvents`)
|
|
break
|
|
}
|
|
case 'create': {
|
|
const property = args.property
|
|
if (!property) { result = { error: '--property required' }; break }
|
|
if (!args['event-name']) { result = { error: '--event-name required' }; break }
|
|
result = await api('POST', ADMIN_API, `/properties/${property}/conversionEvents`, {
|
|
eventName: args['event-name'],
|
|
})
|
|
break
|
|
}
|
|
default:
|
|
result = { error: 'Unknown conversions subcommand. Use: list, create' }
|
|
}
|
|
break
|
|
|
|
case 'events':
|
|
switch (sub) {
|
|
case 'send': {
|
|
if (!args['measurement-id']) { result = { error: '--measurement-id required' }; break }
|
|
if (!args['api-secret']) { result = { error: '--api-secret required' }; break }
|
|
if (!args['client-id']) { result = { error: '--client-id required' }; break }
|
|
if (!args['event-name']) { result = { error: '--event-name required' }; break }
|
|
const eventParams = args.params ? JSON.parse(args.params) : {}
|
|
const body = {
|
|
client_id: args['client-id'],
|
|
events: [{
|
|
name: args['event-name'],
|
|
params: eventParams,
|
|
}],
|
|
}
|
|
result = await mpApi(args['measurement-id'], args['api-secret'], body)
|
|
break
|
|
}
|
|
default:
|
|
result = { error: 'Unknown events subcommand. Use: send' }
|
|
}
|
|
break
|
|
|
|
default:
|
|
result = {
|
|
error: 'Unknown command',
|
|
usage: {
|
|
reports: 'reports run --property <id> [--start-date <date>] [--end-date <date>] [--dimensions <dims>] [--metrics <metrics>]',
|
|
realtime: 'realtime run --property <id> [--dimensions <dims>] [--metrics <metrics>]',
|
|
conversions: 'conversions [list|create] --property <id> [--event-name <name>]',
|
|
events: 'events send --measurement-id <id> --api-secret <secret> --client-id <id> --event-name <name> [--params <json>]',
|
|
}
|
|
}
|
|
}
|
|
|
|
console.log(JSON.stringify(result, null, 2))
|
|
}
|
|
|
|
main().catch(err => {
|
|
console.error(JSON.stringify({ error: err.message }))
|
|
process.exit(1)
|
|
})
|