@sendly/cli 3.0.1 → 3.0.3
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +2 -3
- package/dist/commands/credits/history.js +4 -4
- package/dist/commands/keys/create.js +3 -3
- package/dist/commands/keys/revoke.js +13 -10
- package/dist/commands/send.d.ts +13 -0
- package/dist/commands/send.js +75 -0
- package/dist/commands/status.d.ts +10 -0
- package/dist/commands/status.js +112 -0
- package/dist/commands/webhooks/create.js +7 -5
- package/dist/commands/webhooks/deliveries.js +17 -15
- package/dist/commands/webhooks/get.js +17 -15
- package/dist/commands/webhooks/list.js +3 -3
- package/dist/commands/webhooks/listen.js +5 -18
- package/dist/commands/webhooks/rotate-secret.js +6 -6
- package/dist/commands/webhooks/test.js +17 -14
- package/dist/commands/webhooks/update.js +16 -11
- package/dist/commands/whoami.js +14 -10
- package/dist/hooks/command-not-found.d.ts +3 -0
- package/dist/hooks/command-not-found.js +142 -0
- package/dist/lib/api-client.js +66 -15
- package/oclif.manifest.json +173 -72
- package/package.json +4 -1
|
@@ -40,16 +40,18 @@ export default class WebhooksDeliveries extends AuthenticatedCommand {
|
|
|
40
40
|
return;
|
|
41
41
|
}
|
|
42
42
|
if (deliveries.length === 0) {
|
|
43
|
-
info(flags["failed-only"]
|
|
43
|
+
info(flags["failed-only"]
|
|
44
|
+
? "No failed deliveries found"
|
|
45
|
+
: "No deliveries found");
|
|
44
46
|
return;
|
|
45
47
|
}
|
|
46
48
|
console.log();
|
|
47
49
|
console.log(colors.dim(`Showing ${deliveries.length} deliveries for webhook ${args.id}`));
|
|
48
50
|
console.log();
|
|
49
51
|
// Add computed fields for better display
|
|
50
|
-
const deliveriesWithComputed = deliveries.map(d => ({
|
|
52
|
+
const deliveriesWithComputed = deliveries.map((d) => ({
|
|
51
53
|
...d,
|
|
52
|
-
attemptDisplay: `${d.
|
|
54
|
+
attemptDisplay: `${d.attempt_number}/${d.max_attempts}`,
|
|
53
55
|
}));
|
|
54
56
|
table(deliveriesWithComputed, [
|
|
55
57
|
{
|
|
@@ -60,7 +62,7 @@ export default class WebhooksDeliveries extends AuthenticatedCommand {
|
|
|
60
62
|
},
|
|
61
63
|
{
|
|
62
64
|
header: "Event",
|
|
63
|
-
key: "
|
|
65
|
+
key: "event_type",
|
|
64
66
|
width: 15,
|
|
65
67
|
formatter: (v) => String(v).replace("message.", ""),
|
|
66
68
|
},
|
|
@@ -90,7 +92,7 @@ export default class WebhooksDeliveries extends AuthenticatedCommand {
|
|
|
90
92
|
},
|
|
91
93
|
{
|
|
92
94
|
header: "Status Code",
|
|
93
|
-
key: "
|
|
95
|
+
key: "response_status_code",
|
|
94
96
|
width: 12,
|
|
95
97
|
formatter: (v) => {
|
|
96
98
|
if (!v)
|
|
@@ -103,35 +105,35 @@ export default class WebhooksDeliveries extends AuthenticatedCommand {
|
|
|
103
105
|
},
|
|
104
106
|
{
|
|
105
107
|
header: "Response Time",
|
|
106
|
-
key: "
|
|
108
|
+
key: "response_time",
|
|
107
109
|
width: 14,
|
|
108
|
-
formatter: (v) => v ? `${v}ms` : colors.dim("-"),
|
|
110
|
+
formatter: (v) => (v ? `${v}ms` : colors.dim("-")),
|
|
109
111
|
},
|
|
110
112
|
{
|
|
111
113
|
header: "Created",
|
|
112
|
-
key: "
|
|
114
|
+
key: "created_at",
|
|
113
115
|
width: 16,
|
|
114
116
|
formatter: (v) => formatDate(String(v)),
|
|
115
117
|
},
|
|
116
118
|
]);
|
|
117
119
|
// Show failed delivery details
|
|
118
|
-
const failed = deliveries.filter(d => d.status === "failed" && d.
|
|
120
|
+
const failed = deliveries.filter((d) => d.status === "failed" && d.error_message);
|
|
119
121
|
if (failed.length > 0 && !flags["failed-only"]) {
|
|
120
122
|
console.log();
|
|
121
123
|
console.log(colors.error("Failed deliveries:"));
|
|
122
|
-
failed.forEach(delivery => {
|
|
123
|
-
console.log(colors.dim(` ${delivery.id.slice(0, 15)}...:`), delivery.
|
|
124
|
+
failed.forEach((delivery) => {
|
|
125
|
+
console.log(colors.dim(` ${delivery.id.slice(0, 15)}...:`), delivery.error_message || "Unknown error");
|
|
124
126
|
});
|
|
125
127
|
}
|
|
126
128
|
// Show pending retries
|
|
127
|
-
const pending = deliveries.filter(d => d.status === "pending" && d.
|
|
129
|
+
const pending = deliveries.filter((d) => d.status === "pending" && d.next_retry_at);
|
|
128
130
|
if (pending.length > 0) {
|
|
129
131
|
console.log();
|
|
130
132
|
console.log(colors.warning("Pending retries:"));
|
|
131
|
-
pending.forEach(delivery => {
|
|
132
|
-
console.log(colors.dim(` ${delivery.id.slice(0, 15)}...:`), `Next retry at ${formatDate(delivery.
|
|
133
|
+
pending.forEach((delivery) => {
|
|
134
|
+
console.log(colors.dim(` ${delivery.id.slice(0, 15)}...:`), `Next retry at ${formatDate(delivery.next_retry_at)}`);
|
|
133
135
|
});
|
|
134
136
|
}
|
|
135
137
|
}
|
|
136
138
|
}
|
|
137
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"deliveries.js","sourceRoot":"","sources":["../../../src/commands/webhooks/deliveries.ts"],"names":[],"mappings":"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","sourcesContent":["import { Args, Flags } from \"@oclif/core\";\nimport { AuthenticatedCommand } from \"../../lib/base-command.js\";\nimport { apiClient } from \"../../lib/api-client.js\";\nimport {\n  table,\n  json,\n  info,\n  colors,\n  formatDate,\n  isJsonMode,\n} from \"../../lib/output.js\";\n\ninterface WebhookDelivery {\n  id: string;\n  eventType: string;\n  attemptNumber: number;\n  maxAttempts: number;\n  status: \"pending\" | \"delivered\" | \"failed\" | \"cancelled\";\n  responseStatusCode?: number;\n  responseTime?: number;\n  errorMessage?: string;\n  errorCode?: string;\n  nextRetryAt?: string;\n  createdAt: string;\n  deliveredAt?: string;\n}\n\nexport default class WebhooksDeliveries extends AuthenticatedCommand {\n  static description = \"View webhook delivery history\";\n\n  static examples = [\n    \"<%= config.bin %> webhooks deliveries whk_abc123\",\n    \"<%= config.bin %> webhooks deliveries whk_abc123 --limit 20\",\n    \"<%= config.bin %> webhooks deliveries whk_abc123 --failed-only\",\n    \"<%= config.bin %> webhooks deliveries whk_abc123 --json\",\n  ];\n\n  static args = {\n    id: Args.string({\n      description: \"Webhook ID\",\n      required: true,\n    }),\n  };\n\n  static flags = {\n    ...AuthenticatedCommand.baseFlags,\n    limit: Flags.integer({\n      char: \"l\",\n      description: \"Number of deliveries to show\",\n      default: 10,\n    }),\n    \"failed-only\": Flags.boolean({\n      description: \"Show only failed deliveries\",\n      default: false,\n    }),\n  };\n\n  async run(): Promise<void> {\n    const { args, flags } = await this.parse(WebhooksDeliveries);\n\n    const params = new URLSearchParams({\n      limit: String(flags.limit),\n      ...(flags[\"failed-only\"] && { status: \"failed\" }),\n    });\n\n    const deliveries = await apiClient.get<WebhookDelivery[]>(\n      `/api/v1/webhooks/${args.id}/deliveries?${params.toString()}`\n    );\n\n    if (isJsonMode()) {\n      json(deliveries);\n      return;\n    }\n\n    if (deliveries.length === 0) {\n      info(flags[\"failed-only\"] ? \"No failed deliveries found\" : \"No deliveries found\");\n      return;\n    }\n\n    console.log();\n    console.log(colors.dim(`Showing ${deliveries.length} deliveries for webhook ${args.id}`));\n    console.log();\n\n    // Add computed fields for better display\n    const deliveriesWithComputed = deliveries.map(d => ({\n      ...d,\n      attemptDisplay: `${d.attemptNumber}/${d.maxAttempts}`,\n    }));\n\n    table(deliveriesWithComputed, [\n      {\n        header: \"Delivery ID\",\n        key: \"id\",\n        width: 18,\n        formatter: (v) => colors.dim(String(v).slice(0, 15) + \"...\"),\n      },\n      {\n        header: \"Event\",\n        key: \"eventType\",\n        width: 15,\n        formatter: (v) => String(v).replace(\"message.\", \"\"),\n      },\n      {\n        header: \"Status\",\n        key: \"status\",\n        width: 12,\n        formatter: (v) => {\n          switch (v) {\n            case \"delivered\":\n              return colors.success(\"delivered\");\n            case \"failed\":\n              return colors.error(\"failed\");\n            case \"pending\":\n              return colors.warning(\"pending\");\n            case \"cancelled\":\n              return colors.dim(\"cancelled\");\n            default:\n              return String(v);\n          }\n        },\n      },\n      {\n        header: \"Attempt\",\n        key: \"attemptDisplay\",\n        width: 10,\n      },\n      {\n        header: \"Status Code\",\n        key: \"responseStatusCode\",\n        width: 12,\n        formatter: (v) => {\n          if (!v) return colors.dim(\"-\");\n          const code = Number(v);\n          return code >= 200 && code < 300 \n            ? colors.success(String(code))\n            : colors.error(String(code));\n        },\n      },\n      {\n        header: \"Response Time\",\n        key: \"responseTime\",\n        width: 14,\n        formatter: (v) => v ? `${v}ms` : colors.dim(\"-\"),\n      },\n      {\n        header: \"Created\",\n        key: \"createdAt\",\n        width: 16,\n        formatter: (v) => formatDate(String(v)),\n      },\n    ]);\n\n    // Show failed delivery details\n    const failed = deliveries.filter(d => d.status === \"failed\" && d.errorMessage);\n    if (failed.length > 0 && !flags[\"failed-only\"]) {\n      console.log();\n      console.log(colors.error(\"Failed deliveries:\"));\n      failed.forEach(delivery => {\n        console.log(colors.dim(`  ${delivery.id.slice(0, 15)}...:`), delivery.errorMessage || \"Unknown error\");\n      });\n    }\n\n    // Show pending retries\n    const pending = deliveries.filter(d => d.status === \"pending\" && d.nextRetryAt);\n    if (pending.length > 0) {\n      console.log();\n      console.log(colors.warning(\"Pending retries:\"));\n      pending.forEach(delivery => {\n        console.log(colors.dim(`  ${delivery.id.slice(0, 15)}...:`), `Next retry at ${formatDate(delivery.nextRetryAt!)}`);\n      });\n    }\n  }\n}"]}
|
|
139
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"deliveries.js","sourceRoot":"","sources":["../../../src/commands/webhooks/deliveries.ts"],"names":[],"mappings":"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","sourcesContent":["import { Args, Flags } from \"@oclif/core\";\nimport { AuthenticatedCommand } from \"../../lib/base-command.js\";\nimport { apiClient } from \"../../lib/api-client.js\";\nimport {\n  table,\n  json,\n  info,\n  colors,\n  formatDate,\n  isJsonMode,\n} from \"../../lib/output.js\";\n\ninterface WebhookDelivery {\n  id: string;\n  event_type: string;\n  attempt_number: number;\n  max_attempts: number;\n  status: \"pending\" | \"delivered\" | \"failed\" | \"cancelled\";\n  response_status_code?: number;\n  response_time?: number;\n  error_message?: string;\n  error_code?: string;\n  next_retry_at?: string;\n  created_at: string;\n  delivered_at?: string;\n}\n\nexport default class WebhooksDeliveries extends AuthenticatedCommand {\n  static description = \"View webhook delivery history\";\n\n  static examples = [\n    \"<%= config.bin %> webhooks deliveries whk_abc123\",\n    \"<%= config.bin %> webhooks deliveries whk_abc123 --limit 20\",\n    \"<%= config.bin %> webhooks deliveries whk_abc123 --failed-only\",\n    \"<%= config.bin %> webhooks deliveries whk_abc123 --json\",\n  ];\n\n  static args = {\n    id: Args.string({\n      description: \"Webhook ID\",\n      required: true,\n    }),\n  };\n\n  static flags = {\n    ...AuthenticatedCommand.baseFlags,\n    limit: Flags.integer({\n      char: \"l\",\n      description: \"Number of deliveries to show\",\n      default: 10,\n    }),\n    \"failed-only\": Flags.boolean({\n      description: \"Show only failed deliveries\",\n      default: false,\n    }),\n  };\n\n  async run(): Promise<void> {\n    const { args, flags } = await this.parse(WebhooksDeliveries);\n\n    const params = new URLSearchParams({\n      limit: String(flags.limit),\n      ...(flags[\"failed-only\"] && { status: \"failed\" }),\n    });\n\n    const deliveries = await apiClient.get<WebhookDelivery[]>(\n      `/api/v1/webhooks/${args.id}/deliveries?${params.toString()}`,\n    );\n\n    if (isJsonMode()) {\n      json(deliveries);\n      return;\n    }\n\n    if (deliveries.length === 0) {\n      info(\n        flags[\"failed-only\"]\n          ? \"No failed deliveries found\"\n          : \"No deliveries found\",\n      );\n      return;\n    }\n\n    console.log();\n    console.log(\n      colors.dim(\n        `Showing ${deliveries.length} deliveries for webhook ${args.id}`,\n      ),\n    );\n    console.log();\n\n    // Add computed fields for better display\n    const deliveriesWithComputed = deliveries.map((d) => ({\n      ...d,\n      attemptDisplay: `${d.attempt_number}/${d.max_attempts}`,\n    }));\n\n    table(deliveriesWithComputed, [\n      {\n        header: \"Delivery ID\",\n        key: \"id\",\n        width: 18,\n        formatter: (v) => colors.dim(String(v).slice(0, 15) + \"...\"),\n      },\n      {\n        header: \"Event\",\n        key: \"event_type\",\n        width: 15,\n        formatter: (v) => String(v).replace(\"message.\", \"\"),\n      },\n      {\n        header: \"Status\",\n        key: \"status\",\n        width: 12,\n        formatter: (v) => {\n          switch (v) {\n            case \"delivered\":\n              return colors.success(\"delivered\");\n            case \"failed\":\n              return colors.error(\"failed\");\n            case \"pending\":\n              return colors.warning(\"pending\");\n            case \"cancelled\":\n              return colors.dim(\"cancelled\");\n            default:\n              return String(v);\n          }\n        },\n      },\n      {\n        header: \"Attempt\",\n        key: \"attemptDisplay\",\n        width: 10,\n      },\n      {\n        header: \"Status Code\",\n        key: \"response_status_code\",\n        width: 12,\n        formatter: (v) => {\n          if (!v) return colors.dim(\"-\");\n          const code = Number(v);\n          return code >= 200 && code < 300\n            ? colors.success(String(code))\n            : colors.error(String(code));\n        },\n      },\n      {\n        header: \"Response Time\",\n        key: \"response_time\",\n        width: 14,\n        formatter: (v) => (v ? `${v}ms` : colors.dim(\"-\")),\n      },\n      {\n        header: \"Created\",\n        key: \"created_at\",\n        width: 16,\n        formatter: (v) => formatDate(String(v)),\n      },\n    ]);\n\n    // Show failed delivery details\n    const failed = deliveries.filter(\n      (d) => d.status === \"failed\" && d.error_message,\n    );\n    if (failed.length > 0 && !flags[\"failed-only\"]) {\n      console.log();\n      console.log(colors.error(\"Failed deliveries:\"));\n      failed.forEach((delivery) => {\n        console.log(\n          colors.dim(`  ${delivery.id.slice(0, 15)}...:`),\n          delivery.error_message || \"Unknown error\",\n        );\n      });\n    }\n\n    // Show pending retries\n    const pending = deliveries.filter(\n      (d) => d.status === \"pending\" && d.next_retry_at,\n    );\n    if (pending.length > 0) {\n      console.log();\n      console.log(colors.warning(\"Pending retries:\"));\n      pending.forEach((delivery) => {\n        console.log(\n          colors.dim(`  ${delivery.id.slice(0, 15)}...:`),\n          `Next retry at ${formatDate(delivery.next_retry_at!)}`,\n        );\n      });\n    }\n  }\n}\n"]}
|
|
@@ -27,31 +27,33 @@ export default class WebhooksGet extends AuthenticatedCommand {
|
|
|
27
27
|
header(`Webhook ${webhook.id}`);
|
|
28
28
|
console.log();
|
|
29
29
|
keyValue({
|
|
30
|
-
|
|
31
|
-
|
|
32
|
-
|
|
33
|
-
...(webhook.description ? {
|
|
34
|
-
|
|
35
|
-
|
|
30
|
+
ID: webhook.id,
|
|
31
|
+
URL: webhook.url,
|
|
32
|
+
Events: webhook.events.join(", "),
|
|
33
|
+
...(webhook.description ? { Description: webhook.description } : {}),
|
|
34
|
+
Status: webhook.is_active
|
|
35
|
+
? colors.success("active")
|
|
36
|
+
: colors.warning("inactive"),
|
|
37
|
+
"Circuit State": webhook.circuit_state === "closed"
|
|
36
38
|
? colors.success("closed")
|
|
37
|
-
: webhook.
|
|
39
|
+
: webhook.circuit_state === "open"
|
|
38
40
|
? colors.error("open")
|
|
39
41
|
: colors.warning("half_open"),
|
|
40
|
-
"Failure Count": String(webhook.
|
|
41
|
-
"Secret Version": String(webhook.
|
|
42
|
-
|
|
43
|
-
|
|
42
|
+
"Failure Count": String(webhook.failure_count),
|
|
43
|
+
"Secret Version": String(webhook.secret_version),
|
|
44
|
+
Created: formatDate(webhook.created_at),
|
|
45
|
+
Updated: formatDate(webhook.updated_at),
|
|
44
46
|
});
|
|
45
|
-
if (webhook.
|
|
47
|
+
if (webhook.failure_count > 0) {
|
|
46
48
|
console.log();
|
|
47
|
-
console.log(colors.warning(`⚠ This webhook has failed ${webhook.
|
|
49
|
+
console.log(colors.warning(`⚠ This webhook has failed ${webhook.failure_count} times recently.`));
|
|
48
50
|
console.log(colors.dim("Check delivery history with:"), colors.code(`sendly webhooks deliveries ${webhook.id}`));
|
|
49
51
|
}
|
|
50
|
-
if (webhook.
|
|
52
|
+
if (webhook.circuit_state === "open") {
|
|
51
53
|
console.log();
|
|
52
54
|
console.log(colors.error("⚠ Circuit breaker is OPEN - webhook deliveries are paused."));
|
|
53
55
|
console.log(colors.dim("Test your endpoint and the circuit will auto-recover."));
|
|
54
56
|
}
|
|
55
57
|
}
|
|
56
58
|
}
|
|
57
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
59
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZ2V0LmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vc3JjL2NvbW1hbmRzL3dlYmhvb2tzL2dldC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsSUFBSSxFQUFFLE1BQU0sYUFBYSxDQUFDO0FBQ25DLE9BQU8sRUFBRSxvQkFBb0IsRUFBRSxNQUFNLDJCQUEyQixDQUFDO0FBQ2pFLE9BQU8sRUFBRSxTQUFTLEVBQUUsTUFBTSx5QkFBeUIsQ0FBQztBQUNwRCxPQUFPLEVBQ0wsUUFBUSxFQUNSLElBQUksRUFDSixNQUFNLEVBQ04sVUFBVSxFQUNWLE1BQU0sRUFFTixVQUFVLEdBQ1gsTUFBTSxxQkFBcUIsQ0FBQztBQWU3QixNQUFNLENBQUMsT0FBTyxPQUFPLFdBQVksU0FBUSxvQkFBb0I7SUFDM0QsTUFBTSxDQUFDLFdBQVcsR0FBRyxxQkFBcUIsQ0FBQztJQUUzQyxNQUFNLENBQUMsUUFBUSxHQUFHO1FBQ2hCLDJDQUEyQztRQUMzQyxrREFBa0Q7S0FDbkQsQ0FBQztJQUVGLE1BQU0sQ0FBQyxJQUFJLEdBQUc7UUFDWixFQUFFLEVBQUUsSUFBSSxDQUFDLE1BQU0sQ0FBQztZQUNkLFdBQVcsRUFBRSxZQUFZO1lBQ3pCLFFBQVEsRUFBRSxJQUFJO1NBQ2YsQ0FBQztLQUNILENBQUM7SUFFRixNQUFNLENBQUMsS0FBSyxHQUFHO1FBQ2IsR0FBRyxvQkFBb0IsQ0FBQyxTQUFTO0tBQ2xDLENBQUM7SUFFRixLQUFLLENBQUMsR0FBRztRQUNQLE1BQU0sRUFBRSxJQUFJLEVBQUUsR0FBRyxNQUFNLElBQUksQ0FBQyxLQUFLLENBQUMsV0FBVyxDQUFDLENBQUM7UUFFL0MsTUFBTSxPQUFPLEdBQUcsTUFBTSxTQUFTLENBQUMsR0FBRyxDQUFVLG9CQUFvQixJQUFJLENBQUMsRUFBRSxFQUFFLENBQUMsQ0FBQztRQUU1RSxJQUFJLFVBQVUsRUFBRSxFQUFFLENBQUM7WUFDakIsSUFBSSxDQUFDLE9BQU8sQ0FBQyxDQUFDO1lBQ2QsT0FBTztRQUNULENBQUM7UUFFRCxNQUFNLENBQUMsV0FBVyxPQUFPLENBQUMsRUFBRSxFQUFFLENBQUMsQ0FBQztRQUNoQyxPQUFPLENBQUMsR0FBRyxFQUFFLENBQUM7UUFFZCxRQUFRLENBQUM7WUFDUCxFQUFFLEVBQUUsT0FBTyxDQUFDLEVBQUU7WUFDZCxHQUFHLEVBQUUsT0FBTyxDQUFDLEdBQUc7WUFDaEIsTUFBTSxFQUFFLE9BQU8sQ0FBQyxNQUFNLENBQUMsSUFBSSxDQUFDLElBQUksQ0FBQztZQUNqQyxHQUFHLENBQUMsT0FBTyxDQUFDLFdBQVcsQ0FBQyxDQUFDLENBQUMsRUFBRSxXQUFXLEVBQUUsT0FBTyxDQUFDLFdBQVcsRUFBRSxDQUFDLENBQUMsQ0FBQyxFQUFFLENBQUM7WUFDcEUsTUFBTSxFQUFFLE9BQU8sQ0FBQyxTQUFTO2dCQUN2QixDQUFDLENBQUMsTUFBTSxDQUFDLE9BQU8sQ0FBQyxRQUFRLENBQUM7Z0JBQzFCLENBQUMsQ0FBQyxNQUFNLENBQUMsT0FBTyxDQUFDLFVBQVUsQ0FBQztZQUM5QixlQUFlLEVBQ2IsT0FBTyxDQUFDLGFBQWEsS0FBSyxRQUFRO2dCQUNoQyxDQUFDLENBQUMsTUFBTSxDQUFDLE9BQU8sQ0FBQyxRQUFRLENBQUM7Z0JBQzFCLENBQUMsQ0FBQyxPQUFPLENBQUMsYUFBYSxLQUFLLE1BQU07b0JBQ2hDLENBQUMsQ0FBQyxNQUFNLENBQUMsS0FBSyxDQUFDLE1BQU0sQ0FBQztvQkFDdEIsQ0FBQyxDQUFDLE1BQU0sQ0FBQyxPQUFPLENBQUMsV0FBVyxDQUFDO1lBQ25DLGVBQWUsRUFBRSxNQUFNLENBQUMsT0FBTyxDQUFDLGFBQWEsQ0FBQztZQUM5QyxnQkFBZ0IsRUFBRSxNQUFNLENBQUMsT0FBTyxDQUFDLGNBQWMsQ0FBQztZQUNoRCxPQUFPLEVBQUUsVUFBVSxDQUFDLE9BQU8sQ0FBQyxVQUFVLENBQUM7WUFDdkMsT0FBTyxFQUFFLFVBQVUsQ0FBQyxPQUFPLENBQUMsVUFBVSxDQUFDO1NBQ3hDLENBQUMsQ0FBQztRQUVILElBQUksT0FBTyxDQUFDLGFBQWEsR0FBRyxDQUFDLEVBQUUsQ0FBQztZQUM5QixPQUFPLENBQUMsR0FBRyxFQUFFLENBQUM7WUFDZCxPQUFPLENBQUMsR0FBRyxDQUNULE1BQU0sQ0FBQyxPQUFPLENBQ1osNkJBQTZCLE9BQU8sQ0FBQyxhQUFhLGtCQUFrQixDQUNyRSxDQUNGLENBQUM7WUFDRixPQUFPLENBQUMsR0FBRyxDQUNULE1BQU0sQ0FBQyxHQUFHLENBQUMsOEJBQThCLENBQUMsRUFDMUMsTUFBTSxDQUFDLElBQUksQ0FBQyw4QkFBOEIsT0FBTyxDQUFDLEVBQUUsRUFBRSxDQUFDLENBQ3hELENBQUM7UUFDSixDQUFDO1FBRUQsSUFBSSxPQUFPLENBQUMsYUFBYSxLQUFLLE1BQU0sRUFBRSxDQUFDO1lBQ3JDLE9BQU8sQ0FBQyxHQUFHLEVBQUUsQ0FBQztZQUNkLE9BQU8sQ0FBQyxHQUFHLENBQ1QsTUFBTSxDQUFDLEtBQUssQ0FDViw0REFBNEQsQ0FDN0QsQ0FDRixDQUFDO1lBQ0YsT0FBTyxDQUFDLEdBQUcsQ0FDVCxNQUFNLENBQUMsR0FBRyxDQUFDLHVEQUF1RCxDQUFDLENBQ3BFLENBQUM7UUFDSixDQUFDO0lBQ0gsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IEFyZ3MgfSBmcm9tIFwiQG9jbGlmL2NvcmVcIjtcbmltcG9ydCB7IEF1dGhlbnRpY2F0ZWRDb21tYW5kIH0gZnJvbSBcIi4uLy4uL2xpYi9iYXNlLWNvbW1hbmQuanNcIjtcbmltcG9ydCB7IGFwaUNsaWVudCB9IGZyb20gXCIuLi8uLi9saWIvYXBpLWNsaWVudC5qc1wiO1xuaW1wb3J0IHtcbiAga2V5VmFsdWUsXG4gIGpzb24sXG4gIGhlYWRlcixcbiAgZm9ybWF0RGF0ZSxcbiAgY29sb3JzLFxuICBkaXZpZGVyLFxuICBpc0pzb25Nb2RlLFxufSBmcm9tIFwiLi4vLi4vbGliL291dHB1dC5qc1wiO1xuXG5pbnRlcmZhY2UgV2ViaG9vayB7XG4gIGlkOiBzdHJpbmc7XG4gIHVybDogc3RyaW5nO1xuICBldmVudHM6IHN0cmluZ1tdO1xuICBkZXNjcmlwdGlvbj86IHN0cmluZztcbiAgaXNfYWN0aXZlOiBib29sZWFuO1xuICBmYWlsdXJlX2NvdW50OiBudW1iZXI7XG4gIGNpcmN1aXRfc3RhdGU6IFwiY2xvc2VkXCIgfCBcIm9wZW5cIiB8IFwiaGFsZl9vcGVuXCI7XG4gIHNlY3JldF92ZXJzaW9uOiBudW1iZXI7XG4gIGNyZWF0ZWRfYXQ6IHN0cmluZztcbiAgdXBkYXRlZF9hdDogc3RyaW5nO1xufVxuXG5leHBvcnQgZGVmYXVsdCBjbGFzcyBXZWJob29rc0dldCBleHRlbmRzIEF1dGhlbnRpY2F0ZWRDb21tYW5kIHtcbiAgc3RhdGljIGRlc2NyaXB0aW9uID0gXCJHZXQgd2ViaG9vayBkZXRhaWxzXCI7XG5cbiAgc3RhdGljIGV4YW1wbGVzID0gW1xuICAgIFwiPCU9IGNvbmZpZy5iaW4gJT4gd2ViaG9va3MgZ2V0IHdoa19hYmMxMjNcIixcbiAgICBcIjwlPSBjb25maWcuYmluICU+IHdlYmhvb2tzIGdldCB3aGtfYWJjMTIzIC0tanNvblwiLFxuICBdO1xuXG4gIHN0YXRpYyBhcmdzID0ge1xuICAgIGlkOiBBcmdzLnN0cmluZyh7XG4gICAgICBkZXNjcmlwdGlvbjogXCJXZWJob29rIElEXCIsXG4gICAgICByZXF1aXJlZDogdHJ1ZSxcbiAgICB9KSxcbiAgfTtcblxuICBzdGF0aWMgZmxhZ3MgPSB7XG4gICAgLi4uQXV0aGVudGljYXRlZENvbW1hbmQuYmFzZUZsYWdzLFxuICB9O1xuXG4gIGFzeW5jIHJ1bigpOiBQcm9taXNlPHZvaWQ+IHtcbiAgICBjb25zdCB7IGFyZ3MgfSA9IGF3YWl0IHRoaXMucGFyc2UoV2ViaG9va3NHZXQpO1xuXG4gICAgY29uc3Qgd2ViaG9vayA9IGF3YWl0IGFwaUNsaWVudC5nZXQ8V2ViaG9vaz4oYC9hcGkvdjEvd2ViaG9va3MvJHthcmdzLmlkfWApO1xuXG4gICAgaWYgKGlzSnNvbk1vZGUoKSkge1xuICAgICAganNvbih3ZWJob29rKTtcbiAgICAgIHJldHVybjtcbiAgICB9XG5cbiAgICBoZWFkZXIoYFdlYmhvb2sgJHt3ZWJob29rLmlkfWApO1xuICAgIGNvbnNvbGUubG9nKCk7XG5cbiAgICBrZXlWYWx1ZSh7XG4gICAgICBJRDogd2ViaG9vay5pZCxcbiAgICAgIFVSTDogd2ViaG9vay51cmwsXG4gICAgICBFdmVudHM6IHdlYmhvb2suZXZlbnRzLmpvaW4oXCIsIFwiKSxcbiAgICAgIC4uLih3ZWJob29rLmRlc2NyaXB0aW9uID8geyBEZXNjcmlwdGlvbjogd2ViaG9vay5kZXNjcmlwdGlvbiB9IDoge30pLFxuICAgICAgU3RhdHVzOiB3ZWJob29rLmlzX2FjdGl2ZVxuICAgICAgICA/IGNvbG9ycy5zdWNjZXNzKFwiYWN0aXZlXCIpXG4gICAgICAgIDogY29sb3JzLndhcm5pbmcoXCJpbmFjdGl2ZVwiKSxcbiAgICAgIFwiQ2lyY3VpdCBTdGF0ZVwiOlxuICAgICAgICB3ZWJob29rLmNpcmN1aXRfc3RhdGUgPT09IFwiY2xvc2VkXCJcbiAgICAgICAgICA/IGNvbG9ycy5zdWNjZXNzKFwiY2xvc2VkXCIpXG4gICAgICAgICAgOiB3ZWJob29rLmNpcmN1aXRfc3RhdGUgPT09IFwib3BlblwiXG4gICAgICAgICAgICA/IGNvbG9ycy5lcnJvcihcIm9wZW5cIilcbiAgICAgICAgICAgIDogY29sb3JzLndhcm5pbmcoXCJoYWxmX29wZW5cIiksXG4gICAgICBcIkZhaWx1cmUgQ291bnRcIjogU3RyaW5nKHdlYmhvb2suZmFpbHVyZV9jb3VudCksXG4gICAgICBcIlNlY3JldCBWZXJzaW9uXCI6IFN0cmluZyh3ZWJob29rLnNlY3JldF92ZXJzaW9uKSxcbiAgICAgIENyZWF0ZWQ6IGZvcm1hdERhdGUod2ViaG9vay5jcmVhdGVkX2F0KSxcbiAgICAgIFVwZGF0ZWQ6IGZvcm1hdERhdGUod2ViaG9vay51cGRhdGVkX2F0KSxcbiAgICB9KTtcblxuICAgIGlmICh3ZWJob29rLmZhaWx1cmVfY291bnQgPiAwKSB7XG4gICAgICBjb25zb2xlLmxvZygpO1xuICAgICAgY29uc29sZS5sb2coXG4gICAgICAgIGNvbG9ycy53YXJuaW5nKFxuICAgICAgICAgIGDimqAgVGhpcyB3ZWJob29rIGhhcyBmYWlsZWQgJHt3ZWJob29rLmZhaWx1cmVfY291bnR9IHRpbWVzIHJlY2VudGx5LmAsXG4gICAgICAgICksXG4gICAgICApO1xuICAgICAgY29uc29sZS5sb2coXG4gICAgICAgIGNvbG9ycy5kaW0oXCJDaGVjayBkZWxpdmVyeSBoaXN0b3J5IHdpdGg6XCIpLFxuICAgICAgICBjb2xvcnMuY29kZShgc2VuZGx5IHdlYmhvb2tzIGRlbGl2ZXJpZXMgJHt3ZWJob29rLmlkfWApLFxuICAgICAgKTtcbiAgICB9XG5cbiAgICBpZiAod2ViaG9vay5jaXJjdWl0X3N0YXRlID09PSBcIm9wZW5cIikge1xuICAgICAgY29uc29sZS5sb2coKTtcbiAgICAgIGNvbnNvbGUubG9nKFxuICAgICAgICBjb2xvcnMuZXJyb3IoXG4gICAgICAgICAgXCLimqAgQ2lyY3VpdCBicmVha2VyIGlzIE9QRU4gLSB3ZWJob29rIGRlbGl2ZXJpZXMgYXJlIHBhdXNlZC5cIixcbiAgICAgICAgKSxcbiAgICAgICk7XG4gICAgICBjb25zb2xlLmxvZyhcbiAgICAgICAgY29sb3JzLmRpbShcIlRlc3QgeW91ciBlbmRwb2ludCBhbmQgdGhlIGNpcmN1aXQgd2lsbCBhdXRvLXJlY292ZXIuXCIpLFxuICAgICAgKTtcbiAgICB9XG4gIH1cbn1cbiJdfQ==
|
|
@@ -53,13 +53,13 @@ export default class WebhooksList extends AuthenticatedCommand {
|
|
|
53
53
|
},
|
|
54
54
|
{
|
|
55
55
|
header: "Status",
|
|
56
|
-
key: "
|
|
56
|
+
key: "is_active",
|
|
57
57
|
width: 10,
|
|
58
58
|
formatter: (v) => v ? colors.success("active") : colors.error("disabled"),
|
|
59
59
|
},
|
|
60
60
|
{
|
|
61
61
|
header: "Circuit",
|
|
62
|
-
key: "
|
|
62
|
+
key: "circuit_state",
|
|
63
63
|
width: 10,
|
|
64
64
|
formatter: (v) => {
|
|
65
65
|
switch (v) {
|
|
@@ -77,4 +77,4 @@ export default class WebhooksList extends AuthenticatedCommand {
|
|
|
77
77
|
]);
|
|
78
78
|
}
|
|
79
79
|
}
|
|
80
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
80
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibGlzdC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3NyYy9jb21tYW5kcy93ZWJob29rcy9saXN0LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBRSxvQkFBb0IsRUFBRSxNQUFNLDJCQUEyQixDQUFDO0FBQ2pFLE9BQU8sRUFBRSxTQUFTLEVBQUUsTUFBTSx5QkFBeUIsQ0FBQztBQUNwRCxPQUFPLEVBQ0wsS0FBSyxFQUNMLElBQUksRUFDSixJQUFJLEVBRUosTUFBTSxFQUNOLFVBQVUsR0FDWCxNQUFNLHFCQUFxQixDQUFDO0FBYTdCLE1BQU0sQ0FBQyxPQUFPLE9BQU8sWUFBYSxTQUFRLG9CQUFvQjtJQUM1RCxNQUFNLENBQUMsV0FBVyxHQUFHLDBCQUEwQixDQUFDO0lBRWhELE1BQU0sQ0FBQyxRQUFRLEdBQUc7UUFDaEIsaUNBQWlDO1FBQ2pDLHdDQUF3QztLQUN6QyxDQUFDO0lBRUYsTUFBTSxDQUFDLEtBQUssR0FBRztRQUNiLEdBQUcsb0JBQW9CLENBQUMsU0FBUztLQUNsQyxDQUFDO0lBRUYsS0FBSyxDQUFDLEdBQUc7UUFDUCxNQUFNLFFBQVEsR0FBRyxNQUFNLFNBQVMsQ0FBQyxHQUFHLENBQVksa0JBQWtCLENBQUMsQ0FBQztRQUVwRSxJQUFJLFVBQVUsRUFBRSxFQUFFLENBQUM7WUFDakIsSUFBSSxDQUFDLFFBQVEsQ0FBQyxDQUFDO1lBQ2YsT0FBTztRQUNULENBQUM7UUFFRCxJQUFJLFFBQVEsQ0FBQyxNQUFNLEtBQUssQ0FBQyxFQUFFLENBQUM7WUFDMUIsSUFBSSxDQUFDLHdCQUF3QixDQUFDLENBQUM7WUFDL0IsT0FBTyxDQUFDLEdBQUcsRUFBRSxDQUFDO1lBQ2QsT0FBTyxDQUFDLEdBQUcsQ0FBQyxxQkFBcUIsTUFBTSxDQUFDLElBQUksQ0FBQyx3QkFBd0IsQ0FBQyxFQUFFLENBQUMsQ0FBQztZQUMxRSxPQUFPLENBQUMsR0FBRyxDQUNULDBCQUEwQixNQUFNLENBQUMsSUFBSSxDQUFDLHdCQUF3QixDQUFDLEVBQUUsQ0FDbEUsQ0FBQztZQUNGLE9BQU87UUFDVCxDQUFDO1FBRUQsT0FBTyxDQUFDLEdBQUcsRUFBRSxDQUFDO1FBRWQsS0FBSyxDQUFDLFFBQVEsRUFBRTtZQUNkO2dCQUNFLE1BQU0sRUFBRSxJQUFJO2dCQUNaLEdBQUcsRUFBRSxJQUFJO2dCQUNULEtBQUssRUFBRSxFQUFFO2dCQUNULFNBQVMsRUFBRSxDQUFDLENBQUMsRUFBRSxFQUFFLENBQUMsTUFBTSxDQUFDLEdBQUcsQ0FBQyxNQUFNLENBQUMsQ0FBQyxDQUFDLENBQUMsS0FBSyxDQUFDLENBQUMsRUFBRSxFQUFFLENBQUMsR0FBRyxLQUFLLENBQUM7YUFDN0Q7WUFDRDtnQkFDRSxNQUFNLEVBQUUsS0FBSztnQkFDYixHQUFHLEVBQUUsS0FBSztnQkFDVixLQUFLLEVBQUUsRUFBRTtnQkFDVCxTQUFTLEVBQUUsQ0FBQyxDQUFDLEVBQUUsRUFBRTtvQkFDZixNQUFNLEdBQUcsR0FBRyxNQUFNLENBQUMsQ0FBQyxDQUFDLENBQUM7b0JBQ3RCLE9BQU8sR0FBRyxDQUFDLE1BQU0sR0FBRyxFQUFFLENBQUMsQ0FBQyxDQUFDLEdBQUcsQ0FBQyxLQUFLLENBQUMsQ0FBQyxFQUFFLEVBQUUsQ0FBQyxHQUFHLEtBQUssQ0FBQyxDQUFDLENBQUMsR0FBRyxDQUFDO2dCQUMxRCxDQUFDO2FBQ0Y7WUFDRDtnQkFDRSxNQUFNLEVBQUUsUUFBUTtnQkFDaEIsR0FBRyxFQUFFLFFBQVE7Z0JBQ2IsS0FBSyxFQUFFLEVBQUU7Z0JBQ1QsU0FBUyxFQUFFLENBQUMsQ0FBQyxFQUFFLEVBQUU7b0JBQ2YsTUFBTSxNQUFNLEdBQUcsQ0FBYSxDQUFDO29CQUM3QixPQUFPLE1BQU0sQ0FBQyxNQUFNLEdBQUcsQ0FBQzt3QkFDdEIsQ0FBQyxDQUFDLEdBQUcsTUFBTSxDQUFDLE1BQU0sU0FBUzt3QkFDM0IsQ0FBQyxDQUFDLE1BQU0sQ0FBQyxJQUFJLENBQUMsSUFBSSxDQUFDLENBQUMsT0FBTyxDQUFDLFlBQVksRUFBRSxFQUFFLENBQUMsQ0FBQztnQkFDbEQsQ0FBQzthQUNGO1lBQ0Q7Z0JBQ0UsTUFBTSxFQUFFLFFBQVE7Z0JBQ2hCLEdBQUcsRUFBRSxXQUFXO2dCQUNoQixLQUFLLEVBQUUsRUFBRTtnQkFDVCxTQUFTLEVBQUUsQ0FBQyxDQUFDLEVBQUUsRUFBRSxDQUNmLENBQUMsQ0FBQyxDQUFDLENBQUMsTUFBTSxDQUFDLE9BQU8sQ0FBQyxRQUFRLENBQUMsQ0FBQyxDQUFDLENBQUMsTUFBTSxDQUFDLEtBQUssQ0FBQyxVQUFVLENBQUM7YUFDMUQ7WUFDRDtnQkFDRSxNQUFNLEVBQUUsU0FBUztnQkFDakIsR0FBRyxFQUFFLGVBQWU7Z0JBQ3BCLEtBQUssRUFBRSxFQUFFO2dCQUNULFNBQVMsRUFBRSxDQUFDLENBQUMsRUFBRSxFQUFFO29CQUNmLFFBQVEsQ0FBQyxFQUFFLENBQUM7d0JBQ1YsS0FBSyxRQUFROzRCQUNYLE9BQU8sTUFBTSxDQUFDLE9BQU8sQ0FBQyxRQUFRLENBQUMsQ0FBQzt3QkFDbEMsS0FBSyxNQUFNOzRCQUNULE9BQU8sTUFBTSxDQUFDLEtBQUssQ0FBQyxNQUFNLENBQUMsQ0FBQzt3QkFDOUIsS0FBSyxXQUFXOzRCQUNkLE9BQU8sTUFBTSxDQUFDLE9BQU8sQ0FBQyxXQUFXLENBQUMsQ0FBQzt3QkFDckM7NEJBQ0UsT0FBTyxNQUFNLENBQUMsQ0FBQyxDQUFDLENBQUM7b0JBQ3JCLENBQUM7Z0JBQ0gsQ0FBQzthQUNGO1NBQ0YsQ0FBQyxDQUFDO0lBQ0wsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IEF1dGhlbnRpY2F0ZWRDb21tYW5kIH0gZnJvbSBcIi4uLy4uL2xpYi9iYXNlLWNvbW1hbmQuanNcIjtcbmltcG9ydCB7IGFwaUNsaWVudCB9IGZyb20gXCIuLi8uLi9saWIvYXBpLWNsaWVudC5qc1wiO1xuaW1wb3J0IHtcbiAgdGFibGUsXG4gIGpzb24sXG4gIGluZm8sXG4gIGZvcm1hdFN0YXR1cyxcbiAgY29sb3JzLFxuICBpc0pzb25Nb2RlLFxufSBmcm9tIFwiLi4vLi4vbGliL291dHB1dC5qc1wiO1xuXG5pbnRlcmZhY2UgV2ViaG9vayB7XG4gIGlkOiBzdHJpbmc7XG4gIHVybDogc3RyaW5nO1xuICBkZXNjcmlwdGlvbj86IHN0cmluZztcbiAgZXZlbnRzOiBzdHJpbmdbXTtcbiAgaXNfYWN0aXZlOiBib29sZWFuO1xuICBmYWlsdXJlX2NvdW50OiBudW1iZXI7XG4gIGNpcmN1aXRfc3RhdGU6IFwiY2xvc2VkXCIgfCBcIm9wZW5cIiB8IFwiaGFsZl9vcGVuXCI7XG4gIGNyZWF0ZWRfYXQ6IHN0cmluZztcbn1cblxuZXhwb3J0IGRlZmF1bHQgY2xhc3MgV2ViaG9va3NMaXN0IGV4dGVuZHMgQXV0aGVudGljYXRlZENvbW1hbmQge1xuICBzdGF0aWMgZGVzY3JpcHRpb24gPSBcIkxpc3QgY29uZmlndXJlZCB3ZWJob29rc1wiO1xuXG4gIHN0YXRpYyBleGFtcGxlcyA9IFtcbiAgICBcIjwlPSBjb25maWcuYmluICU+IHdlYmhvb2tzIGxpc3RcIixcbiAgICBcIjwlPSBjb25maWcuYmluICU+IHdlYmhvb2tzIGxpc3QgLS1qc29uXCIsXG4gIF07XG5cbiAgc3RhdGljIGZsYWdzID0ge1xuICAgIC4uLkF1dGhlbnRpY2F0ZWRDb21tYW5kLmJhc2VGbGFncyxcbiAgfTtcblxuICBhc3luYyBydW4oKTogUHJvbWlzZTx2b2lkPiB7XG4gICAgY29uc3Qgd2ViaG9va3MgPSBhd2FpdCBhcGlDbGllbnQuZ2V0PFdlYmhvb2tbXT4oXCIvYXBpL3YxL3dlYmhvb2tzXCIpO1xuXG4gICAgaWYgKGlzSnNvbk1vZGUoKSkge1xuICAgICAganNvbih3ZWJob29rcyk7XG4gICAgICByZXR1cm47XG4gICAgfVxuXG4gICAgaWYgKHdlYmhvb2tzLmxlbmd0aCA9PT0gMCkge1xuICAgICAgaW5mbyhcIk5vIHdlYmhvb2tzIGNvbmZpZ3VyZWRcIik7XG4gICAgICBjb25zb2xlLmxvZygpO1xuICAgICAgY29uc29sZS5sb2coYCAgQ3JlYXRlIG9uZSB3aXRoICR7Y29sb3JzLmNvZGUoXCJzZW5kbHkgd2ViaG9va3MgY3JlYXRlXCIpfWApO1xuICAgICAgY29uc29sZS5sb2coXG4gICAgICAgIGAgIE9yIHRlc3QgbG9jYWxseSB3aXRoICR7Y29sb3JzLmNvZGUoXCJzZW5kbHkgd2ViaG9va3MgbGlzdGVuXCIpfWAsXG4gICAgICApO1xuICAgICAgcmV0dXJuO1xuICAgIH1cblxuICAgIGNvbnNvbGUubG9nKCk7XG5cbiAgICB0YWJsZSh3ZWJob29rcywgW1xuICAgICAge1xuICAgICAgICBoZWFkZXI6IFwiSURcIixcbiAgICAgICAga2V5OiBcImlkXCIsXG4gICAgICAgIHdpZHRoOiAxOCxcbiAgICAgICAgZm9ybWF0dGVyOiAodikgPT4gY29sb3JzLmRpbShTdHJpbmcodikuc2xpY2UoMCwgMTUpICsgXCIuLi5cIiksXG4gICAgICB9LFxuICAgICAge1xuICAgICAgICBoZWFkZXI6IFwiVVJMXCIsXG4gICAgICAgIGtleTogXCJ1cmxcIixcbiAgICAgICAgd2lkdGg6IDM1LFxuICAgICAgICBmb3JtYXR0ZXI6ICh2KSA9PiB7XG4gICAgICAgICAgY29uc3QgdXJsID0gU3RyaW5nKHYpO1xuICAgICAgICAgIHJldHVybiB1cmwubGVuZ3RoID4gMzIgPyB1cmwuc2xpY2UoMCwgMzIpICsgXCIuLi5cIiA6IHVybDtcbiAgICAgICAgfSxcbiAgICAgIH0sXG4gICAgICB7XG4gICAgICAgIGhlYWRlcjogXCJFdmVudHNcIixcbiAgICAgICAga2V5OiBcImV2ZW50c1wiLFxuICAgICAgICB3aWR0aDogMTUsXG4gICAgICAgIGZvcm1hdHRlcjogKHYpID0+IHtcbiAgICAgICAgICBjb25zdCBldmVudHMgPSB2IGFzIHN0cmluZ1tdO1xuICAgICAgICAgIHJldHVybiBldmVudHMubGVuZ3RoID4gMlxuICAgICAgICAgICAgPyBgJHtldmVudHMubGVuZ3RofSBldmVudHNgXG4gICAgICAgICAgICA6IGV2ZW50cy5qb2luKFwiLCBcIikucmVwbGFjZSgvbWVzc2FnZVxcLi9nLCBcIlwiKTtcbiAgICAgICAgfSxcbiAgICAgIH0sXG4gICAgICB7XG4gICAgICAgIGhlYWRlcjogXCJTdGF0dXNcIixcbiAgICAgICAga2V5OiBcImlzX2FjdGl2ZVwiLFxuICAgICAgICB3aWR0aDogMTAsXG4gICAgICAgIGZvcm1hdHRlcjogKHYpID0+XG4gICAgICAgICAgdiA/IGNvbG9ycy5zdWNjZXNzKFwiYWN0aXZlXCIpIDogY29sb3JzLmVycm9yKFwiZGlzYWJsZWRcIiksXG4gICAgICB9LFxuICAgICAge1xuICAgICAgICBoZWFkZXI6IFwiQ2lyY3VpdFwiLFxuICAgICAgICBrZXk6IFwiY2lyY3VpdF9zdGF0ZVwiLFxuICAgICAgICB3aWR0aDogMTAsXG4gICAgICAgIGZvcm1hdHRlcjogKHYpID0+IHtcbiAgICAgICAgICBzd2l0Y2ggKHYpIHtcbiAgICAgICAgICAgIGNhc2UgXCJjbG9zZWRcIjpcbiAgICAgICAgICAgICAgcmV0dXJuIGNvbG9ycy5zdWNjZXNzKFwiY2xvc2VkXCIpO1xuICAgICAgICAgICAgY2FzZSBcIm9wZW5cIjpcbiAgICAgICAgICAgICAgcmV0dXJuIGNvbG9ycy5lcnJvcihcIm9wZW5cIik7XG4gICAgICAgICAgICBjYXNlIFwiaGFsZl9vcGVuXCI6XG4gICAgICAgICAgICAgIHJldHVybiBjb2xvcnMud2FybmluZyhcImhhbGZfb3BlblwiKTtcbiAgICAgICAgICAgIGRlZmF1bHQ6XG4gICAgICAgICAgICAgIHJldHVybiBTdHJpbmcodik7XG4gICAgICAgICAgfVxuICAgICAgICB9LFxuICAgICAgfSxcbiAgICBdKTtcbiAgfVxufVxuIl19
|
|
@@ -2,7 +2,6 @@ import { Flags } from "@oclif/core";
|
|
|
2
2
|
import { AuthenticatedCommand } from "../../lib/base-command.js";
|
|
3
3
|
import { apiClient } from "../../lib/api-client.js";
|
|
4
4
|
import { error, warn, colors, spinner, } from "../../lib/output.js";
|
|
5
|
-
import { getConfigValue } from "../../lib/config.js";
|
|
6
5
|
import localtunnel from "localtunnel";
|
|
7
6
|
export default class WebhooksListen extends AuthenticatedCommand {
|
|
8
7
|
static description = "Listen for webhooks locally (like Stripe CLI). Creates a secure tunnel to forward events to your local server.";
|
|
@@ -91,15 +90,7 @@ export default class WebhooksListen extends AuthenticatedCommand {
|
|
|
91
90
|
}
|
|
92
91
|
}
|
|
93
92
|
async pollEvents(forwardUrl, secret) {
|
|
94
|
-
|
|
95
|
-
// Set up SSE connection for real-time events
|
|
96
|
-
const eventSource = new EventSource(`${baseUrl}/api/cli/webhooks/events?webhookId=${this.webhookId}`, {
|
|
97
|
-
// @ts-ignore - headers not in type but supported
|
|
98
|
-
headers: {
|
|
99
|
-
Authorization: `Bearer ${getConfigValue("apiKey") || getConfigValue("accessToken")}`,
|
|
100
|
-
},
|
|
101
|
-
});
|
|
102
|
-
// Fallback to polling if SSE not available
|
|
93
|
+
// Poll for events at regular intervals
|
|
103
94
|
const pollInterval = setInterval(async () => {
|
|
104
95
|
try {
|
|
105
96
|
const events = await apiClient.get(`/api/cli/webhooks/events?webhookId=${this.webhookId}&since=${Date.now() - 5000}`);
|
|
@@ -174,7 +165,9 @@ export default class WebhooksListen extends AuthenticatedCommand {
|
|
|
174
165
|
const key = await crypto.subtle.importKey("raw", encoder.encode(secret), { name: "HMAC", hash: "SHA-256" }, false, ["sign"]);
|
|
175
166
|
const signature = await crypto.subtle.sign("HMAC", key, encoder.encode(payload));
|
|
176
167
|
const hashArray = Array.from(new Uint8Array(signature));
|
|
177
|
-
const hashHex = hashArray
|
|
168
|
+
const hashHex = hashArray
|
|
169
|
+
.map((b) => b.toString(16).padStart(2, "0"))
|
|
170
|
+
.join("");
|
|
178
171
|
return `v1=${hashHex}`;
|
|
179
172
|
}
|
|
180
173
|
async cleanup() {
|
|
@@ -193,10 +186,4 @@ export default class WebhooksListen extends AuthenticatedCommand {
|
|
|
193
186
|
}
|
|
194
187
|
}
|
|
195
188
|
}
|
|
196
|
-
// Polyfill EventSource for Node.js if needed
|
|
197
|
-
class EventSource {
|
|
198
|
-
constructor(url, options) {
|
|
199
|
-
// Simple implementation - in production use a proper polyfill
|
|
200
|
-
}
|
|
201
|
-
}
|
|
202
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"listen.js","sourceRoot":"","sources":["../../../src/commands/webhooks/listen.ts"],"names":[],"mappings":"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","sourcesContent":["import { Flags } from \"@oclif/core\";\nimport { AuthenticatedCommand } from \"../../lib/base-command.js\";\nimport { apiClient } from \"../../lib/api-client.js\";\nimport {\n  success,\n  error,\n  info,\n  warn,\n  colors,\n  formatStatus,\n  formatDate,\n  spinner,\n  json as jsonOutput,\n  isJsonMode,\n} from \"../../lib/output.js\";\nimport { getConfigValue } from \"../../lib/config.js\";\nimport localtunnel from \"localtunnel\";\n\ninterface WebhookEvent {\n  id: string;\n  type: string;\n  data: Record<string, unknown>;\n  created: number;\n}\n\nexport default class WebhooksListen extends AuthenticatedCommand {\n  static description =\n    \"Listen for webhooks locally (like Stripe CLI). Creates a secure tunnel to forward events to your local server.\";\n\n  static examples = [\n    \"<%= config.bin %> webhooks listen\",\n    \"<%= config.bin %> webhooks listen --forward http://localhost:3000/webhook\",\n    \"<%= config.bin %> webhooks listen --events message.delivered,message.failed\",\n  ];\n\n  static flags = {\n    ...AuthenticatedCommand.baseFlags,\n    forward: Flags.string({\n      char: \"f\",\n      description: \"Local URL to forward events to\",\n      default: \"http://localhost:3000/webhook\",\n    }),\n    events: Flags.string({\n      char: \"e\",\n      description: \"Comma-separated list of events to listen for\",\n      default: \"message.sent,message.delivered,message.failed,message.bounced\",\n    }),\n    port: Flags.integer({\n      char: \"p\",\n      description: \"Local port for the tunnel (auto-detected from forward URL if not specified)\",\n    }),\n  };\n\n  private tunnel: localtunnel.Tunnel | null = null;\n  private webhookId: string | null = null;\n\n  async run(): Promise<void> {\n    const { flags } = await this.parse(WebhooksListen);\n\n    const forwardUrl = new URL(flags.forward);\n    const localPort = flags.port || parseInt(forwardUrl.port) || 3000;\n    const events = flags.events.split(\",\").map((e) => e.trim());\n\n    const spin = spinner(\"Starting webhook listener...\");\n    spin.start();\n\n    try {\n      // Create localtunnel\n      this.tunnel = await localtunnel({\n        port: localPort,\n        subdomain: `sendly-${Date.now().toString(36)}`,\n      });\n\n      const tunnelUrl = this.tunnel.url;\n      spin.succeed(\"Tunnel established\");\n\n      // Register temporary webhook with Sendly\n      const webhookResponse = await apiClient.post<{\n        id: string;\n        secret: string;\n      }>(\"/api/cli/webhooks/listen\", {\n        url: `${tunnelUrl}/cli-webhook`,\n        events,\n        forwardUrl: flags.forward,\n      });\n\n      this.webhookId = webhookResponse.id;\n      const secret = webhookResponse.secret;\n\n      // Display connection info\n      console.log();\n      console.log(colors.bold(colors.primary(\"Webhook listener ready!\")));\n      console.log();\n      console.log(`  ${colors.dim(\"Tunnel URL:\")}     ${colors.code(tunnelUrl)}`);\n      console.log(`  ${colors.dim(\"Forwarding to:\")} ${colors.code(flags.forward)}`);\n      console.log(`  ${colors.dim(\"Events:\")}        ${events.join(\", \")}`);\n      console.log();\n      console.log(`  ${colors.dim(\"Webhook Secret:\")}`);\n      console.log(`  ${colors.primary(secret)}`);\n      console.log();\n      console.log(colors.dim(\"Use this secret to verify webhook signatures in your app.\"));\n      console.log();\n      console.log(colors.bold(\"Waiting for events...\"));\n      console.log(colors.dim(\"─\".repeat(60)));\n      console.log();\n\n      // Set up event forwarding\n      this.tunnel.on(\"request\", (info) => {\n        // This is just for logging - actual forwarding happens server-side\n      });\n\n      // Handle tunnel close\n      this.tunnel.on(\"close\", () => {\n        warn(\"Tunnel closed\");\n        this.cleanup();\n        process.exit(0);\n      });\n\n      this.tunnel.on(\"error\", (err) => {\n        error(`Tunnel error: ${err.message}`);\n      });\n\n      // Poll for events and display them\n      await this.pollEvents(flags.forward, secret);\n    } catch (err) {\n      spin.fail(\"Failed to start listener\");\n      throw err;\n    }\n  }\n\n  private async pollEvents(forwardUrl: string, secret: string): Promise<void> {\n    const baseUrl = getConfigValue(\"baseUrl\") || \"https://sendly.live\";\n\n    // Set up SSE connection for real-time events\n    const eventSource = new EventSource(\n      `${baseUrl}/api/cli/webhooks/events?webhookId=${this.webhookId}`,\n      {\n        // @ts-ignore - headers not in type but supported\n        headers: {\n          Authorization: `Bearer ${getConfigValue(\"apiKey\") || getConfigValue(\"accessToken\")}`,\n        },\n      }\n    );\n\n    // Fallback to polling if SSE not available\n    const pollInterval = setInterval(async () => {\n      try {\n        const events = await apiClient.get<WebhookEvent[]>(\n          `/api/cli/webhooks/events?webhookId=${this.webhookId}&since=${Date.now() - 5000}`\n        );\n\n        for (const event of events) {\n          this.displayEvent(event);\n          await this.forwardEvent(forwardUrl, event, secret);\n        }\n      } catch {\n        // Ignore polling errors\n      }\n    }, 2000);\n\n    // Handle graceful shutdown\n    const cleanup = () => {\n      clearInterval(pollInterval);\n      this.cleanup();\n      process.exit(0);\n    };\n\n    process.on(\"SIGINT\", cleanup);\n    process.on(\"SIGTERM\", cleanup);\n\n    // Keep process alive\n    await new Promise(() => {});\n  }\n\n  private displayEvent(event: WebhookEvent): void {\n    const timestamp = new Date(event.created * 1000).toLocaleTimeString();\n    const eventType = event.type;\n\n    let statusColor = colors.info;\n    if (eventType.includes(\"delivered\")) statusColor = colors.success;\n    if (eventType.includes(\"failed\")) statusColor = colors.error;\n\n    console.log(\n      `${colors.dim(timestamp)} ${statusColor(\"→\")} ${colors.bold(eventType)}`\n    );\n\n    if (event.data) {\n      const messageId = (event.data as any).message_id || (event.data as any).id;\n      const to = (event.data as any).to;\n      if (messageId) {\n        console.log(`  ${colors.dim(\"message_id:\")} ${messageId}`);\n      }\n      if (to) {\n        console.log(`  ${colors.dim(\"to:\")} ${to}`);\n      }\n    }\n    console.log();\n  }\n\n  private async forwardEvent(\n    forwardUrl: string,\n    event: WebhookEvent,\n    secret: string\n  ): Promise<void> {\n    try {\n      const payload = JSON.stringify(event);\n      const signature = await this.generateSignature(payload, secret);\n\n      const response = await fetch(forwardUrl, {\n        method: \"POST\",\n        headers: {\n          \"Content-Type\": \"application/json\",\n          \"X-Sendly-Signature\": signature,\n          \"X-Sendly-Event\": event.type,\n        },\n        body: payload,\n      });\n\n      if (response.ok) {\n        console.log(\n          `  ${colors.success(\"✓\")} Forwarded to ${forwardUrl} (${response.status})`\n        );\n      } else {\n        console.log(\n          `  ${colors.error(\"✗\")} Forward failed (${response.status})`\n        );\n      }\n    } catch (err) {\n      console.log(\n        `  ${colors.error(\"✗\")} Forward error: ${(err as Error).message}`\n      );\n    }\n    console.log();\n  }\n\n  private async generateSignature(\n    payload: string,\n    secret: string\n  ): Promise<string> {\n    const encoder = new TextEncoder();\n    const key = await crypto.subtle.importKey(\n      \"raw\",\n      encoder.encode(secret),\n      { name: \"HMAC\", hash: \"SHA-256\" },\n      false,\n      [\"sign\"]\n    );\n    const signature = await crypto.subtle.sign(\n      \"HMAC\",\n      key,\n      encoder.encode(payload)\n    );\n    const hashArray = Array.from(new Uint8Array(signature));\n    const hashHex = hashArray.map((b) => b.toString(16).padStart(2, \"0\")).join(\"\");\n    return `v1=${hashHex}`;\n  }\n\n  private async cleanup(): Promise<void> {\n    // Clean up tunnel\n    if (this.tunnel) {\n      this.tunnel.close();\n    }\n\n    // Clean up temporary webhook\n    if (this.webhookId) {\n      try {\n        await apiClient.delete(`/api/cli/webhooks/listen/${this.webhookId}`);\n      } catch {\n        // Ignore cleanup errors\n      }\n    }\n  }\n}\n\n// Polyfill EventSource for Node.js if needed\nclass EventSource {\n  constructor(url: string, options?: any) {\n    // Simple implementation - in production use a proper polyfill\n  }\n}\n"]}
|
|
189
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"listen.js","sourceRoot":"","sources":["../../../src/commands/webhooks/listen.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,KAAK,EAAE,MAAM,aAAa,CAAC;AACpC,OAAO,EAAE,oBAAoB,EAAE,MAAM,2BAA2B,CAAC;AACjE,OAAO,EAAE,SAAS,EAAE,MAAM,yBAAyB,CAAC;AACpD,OAAO,EAEL,KAAK,EAEL,IAAI,EACJ,MAAM,EAGN,OAAO,GAGR,MAAM,qBAAqB,CAAC;AAE7B,OAAO,WAAW,MAAM,aAAa,CAAC;AAStC,MAAM,CAAC,OAAO,OAAO,cAAe,SAAQ,oBAAoB;IAC9D,MAAM,CAAC,WAAW,GAChB,gHAAgH,CAAC;IAEnH,MAAM,CAAC,QAAQ,GAAG;QAChB,mCAAmC;QACnC,2EAA2E;QAC3E,6EAA6E;KAC9E,CAAC;IAEF,MAAM,CAAC,KAAK,GAAG;QACb,GAAG,oBAAoB,CAAC,SAAS;QACjC,OAAO,EAAE,KAAK,CAAC,MAAM,CAAC;YACpB,IAAI,EAAE,GAAG;YACT,WAAW,EAAE,gCAAgC;YAC7C,OAAO,EAAE,+BAA+B;SACzC,CAAC;QACF,MAAM,EAAE,KAAK,CAAC,MAAM,CAAC;YACnB,IAAI,EAAE,GAAG;YACT,WAAW,EAAE,8CAA8C;YAC3D,OAAO,EAAE,+DAA+D;SACzE,CAAC;QACF,IAAI,EAAE,KAAK,CAAC,OAAO,CAAC;YAClB,IAAI,EAAE,GAAG;YACT,WAAW,EACT,6EAA6E;SAChF,CAAC;KACH,CAAC;IAEM,MAAM,GAA8B,IAAI,CAAC;IACzC,SAAS,GAAkB,IAAI,CAAC;IAExC,KAAK,CAAC,GAAG;QACP,MAAM,EAAE,KAAK,EAAE,GAAG,MAAM,IAAI,CAAC,KAAK,CAAC,cAAc,CAAC,CAAC;QAEnD,MAAM,UAAU,GAAG,IAAI,GAAG,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC;QAC1C,MAAM,SAAS,GAAG,KAAK,CAAC,IAAI,IAAI,QAAQ,CAAC,UAAU,CAAC,IAAI,CAAC,IAAI,IAAI,CAAC;QAClE,MAAM,MAAM,GAAG,KAAK,CAAC,MAAM,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,IAAI,EAAE,CAAC,CAAC;QAE5D,MAAM,IAAI,GAAG,OAAO,CAAC,8BAA8B,CAAC,CAAC;QACrD,IAAI,CAAC,KAAK,EAAE,CAAC;QAEb,IAAI,CAAC;YACH,qBAAqB;YACrB,IAAI,CAAC,MAAM,GAAG,MAAM,WAAW,CAAC;gBAC9B,IAAI,EAAE,SAAS;gBACf,SAAS,EAAE,UAAU,IAAI,CAAC,GAAG,EAAE,CAAC,QAAQ,CAAC,EAAE,CAAC,EAAE;aAC/C,CAAC,CAAC;YAEH,MAAM,SAAS,GAAG,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC;YAClC,IAAI,CAAC,OAAO,CAAC,oBAAoB,CAAC,CAAC;YAEnC,yCAAyC;YACzC,MAAM,eAAe,GAAG,MAAM,SAAS,CAAC,IAAI,CAGzC,0BAA0B,EAAE;gBAC7B,GAAG,EAAE,GAAG,SAAS,cAAc;gBAC/B,MAAM;gBACN,UAAU,EAAE,KAAK,CAAC,OAAO;aAC1B,CAAC,CAAC;YAEH,IAAI,CAAC,SAAS,GAAG,eAAe,CAAC,EAAE,CAAC;YACpC,MAAM,MAAM,GAAG,eAAe,CAAC,MAAM,CAAC;YAEtC,0BAA0B;YAC1B,OAAO,CAAC,GAAG,EAAE,CAAC;YACd,OAAO,CAAC,GAAG,CAAC,MAAM,CAAC,IAAI,CAAC,MAAM,CAAC,OAAO,CAAC,yBAAyB,CAAC,CAAC,CAAC,CAAC;YACpE,OAAO,CAAC,GAAG,EAAE,CAAC;YACd,OAAO,CAAC,GAAG,CACT,KAAK,MAAM,CAAC,GAAG,CAAC,aAAa,CAAC,QAAQ,MAAM,CAAC,IAAI,CAAC,SAAS,CAAC,EAAE,CAC/D,CAAC;YACF,OAAO,CAAC,GAAG,CACT,KAAK,MAAM,CAAC,GAAG,CAAC,gBAAgB,CAAC,IAAI,MAAM,CAAC,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,EAAE,CAClE,CAAC;YACF,OAAO,CAAC,GAAG,CAAC,KAAK,MAAM,CAAC,GAAG,CAAC,SAAS,CAAC,WAAW,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;YACtE,OAAO,CAAC,GAAG,EAAE,CAAC;YACd,OAAO,CAAC,GAAG,CAAC,KAAK,MAAM,CAAC,GAAG,CAAC,iBAAiB,CAAC,EAAE,CAAC,CAAC;YAClD,OAAO,CAAC,GAAG,CAAC,KAAK,MAAM,CAAC,OAAO,CAAC,MAAM,CAAC,EAAE,CAAC,CAAC;YAC3C,OAAO,CAAC,GAAG,EAAE,CAAC;YACd,OAAO,CAAC,GAAG,CACT,MAAM,CAAC,GAAG,CAAC,2DAA2D,CAAC,CACxE,CAAC;YACF,OAAO,CAAC,GAAG,EAAE,CAAC;YACd,OAAO,CAAC,GAAG,CAAC,MAAM,CAAC,IAAI,CAAC,uBAAuB,CAAC,CAAC,CAAC;YAClD,OAAO,CAAC,GAAG,CAAC,MAAM,CAAC,GAAG,CAAC,GAAG,CAAC,MAAM,CAAC,EAAE,CAAC,CAAC,CAAC,CAAC;YACxC,OAAO,CAAC,GAAG,EAAE,CAAC;YAEd,0BAA0B;YAC1B,IAAI,CAAC,MAAM,CAAC,EAAE,CAAC,SAAS,EAAE,CAAC,IAAI,EAAE,EAAE;gBACjC,mEAAmE;YACrE,CAAC,CAAC,CAAC;YAEH,sBAAsB;YACtB,IAAI,CAAC,MAAM,CAAC,EAAE,CAAC,OAAO,EAAE,GAAG,EAAE;gBAC3B,IAAI,CAAC,eAAe,CAAC,CAAC;gBACtB,IAAI,CAAC,OAAO,EAAE,CAAC;gBACf,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;YAClB,CAAC,CAAC,CAAC;YAEH,IAAI,CAAC,MAAM,CAAC,EAAE,CAAC,OAAO,EAAE,CAAC,GAAG,EAAE,EAAE;gBAC9B,KAAK,CAAC,iBAAiB,GAAG,CAAC,OAAO,EAAE,CAAC,CAAC;YACxC,CAAC,CAAC,CAAC;YAEH,mCAAmC;YACnC,MAAM,IAAI,CAAC,UAAU,CAAC,KAAK,CAAC,OAAO,EAAE,MAAM,CAAC,CAAC;QAC/C,CAAC;QAAC,OAAO,GAAG,EAAE,CAAC;YACb,IAAI,CAAC,IAAI,CAAC,0BAA0B,CAAC,CAAC;YACtC,MAAM,GAAG,CAAC;QACZ,CAAC;IACH,CAAC;IAEO,KAAK,CAAC,UAAU,CAAC,UAAkB,EAAE,MAAc;QACzD,uCAAuC;QACvC,MAAM,YAAY,GAAG,WAAW,CAAC,KAAK,IAAI,EAAE;YAC1C,IAAI,CAAC;gBACH,MAAM,MAAM,GAAG,MAAM,SAAS,CAAC,GAAG,CAChC,sCAAsC,IAAI,CAAC,SAAS,UAAU,IAAI,CAAC,GAAG,EAAE,GAAG,IAAI,EAAE,CAClF,CAAC;gBAEF,KAAK,MAAM,KAAK,IAAI,MAAM,EAAE,CAAC;oBAC3B,IAAI,CAAC,YAAY,CAAC,KAAK,CAAC,CAAC;oBACzB,MAAM,IAAI,CAAC,YAAY,CAAC,UAAU,EAAE,KAAK,EAAE,MAAM,CAAC,CAAC;gBACrD,CAAC;YACH,CAAC;YAAC,MAAM,CAAC;gBACP,wBAAwB;YAC1B,CAAC;QACH,CAAC,EAAE,IAAI,CAAC,CAAC;QAET,2BAA2B;QAC3B,MAAM,OAAO,GAAG,GAAG,EAAE;YACnB,aAAa,CAAC,YAAY,CAAC,CAAC;YAC5B,IAAI,CAAC,OAAO,EAAE,CAAC;YACf,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;QAClB,CAAC,CAAC;QAEF,OAAO,CAAC,EAAE,CAAC,QAAQ,EAAE,OAAO,CAAC,CAAC;QAC9B,OAAO,CAAC,EAAE,CAAC,SAAS,EAAE,OAAO,CAAC,CAAC;QAE/B,qBAAqB;QACrB,MAAM,IAAI,OAAO,CAAC,GAAG,EAAE,GAAE,CAAC,CAAC,CAAC;IAC9B,CAAC;IAEO,YAAY,CAAC,KAAmB;QACtC,MAAM,SAAS,GAAG,IAAI,IAAI,CAAC,KAAK,CAAC,OAAO,GAAG,IAAI,CAAC,CAAC,kBAAkB,EAAE,CAAC;QACtE,MAAM,SAAS,GAAG,KAAK,CAAC,IAAI,CAAC;QAE7B,IAAI,WAAW,GAAG,MAAM,CAAC,IAAI,CAAC;QAC9B,IAAI,SAAS,CAAC,QAAQ,CAAC,WAAW,CAAC;YAAE,WAAW,GAAG,MAAM,CAAC,OAAO,CAAC;QAClE,IAAI,SAAS,CAAC,QAAQ,CAAC,QAAQ,CAAC;YAAE,WAAW,GAAG,MAAM,CAAC,KAAK,CAAC;QAE7D,OAAO,CAAC,GAAG,CACT,GAAG,MAAM,CAAC,GAAG,CAAC,SAAS,CAAC,IAAI,WAAW,CAAC,GAAG,CAAC,IAAI,MAAM,CAAC,IAAI,CAAC,SAAS,CAAC,EAAE,CACzE,CAAC;QAEF,IAAI,KAAK,CAAC,IAAI,EAAE,CAAC;YACf,MAAM,SAAS,GACZ,KAAK,CAAC,IAAY,CAAC,UAAU,IAAK,KAAK,CAAC,IAAY,CAAC,EAAE,CAAC;YAC3D,MAAM,EAAE,GAAI,KAAK,CAAC,IAAY,CAAC,EAAE,CAAC;YAClC,IAAI,SAAS,EAAE,CAAC;gBACd,OAAO,CAAC,GAAG,CAAC,KAAK,MAAM,CAAC,GAAG,CAAC,aAAa,CAAC,IAAI,SAAS,EAAE,CAAC,CAAC;YAC7D,CAAC;YACD,IAAI,EAAE,EAAE,CAAC;gBACP,OAAO,CAAC,GAAG,CAAC,KAAK,MAAM,CAAC,GAAG,CAAC,KAAK,CAAC,IAAI,EAAE,EAAE,CAAC,CAAC;YAC9C,CAAC;QACH,CAAC;QACD,OAAO,CAAC,GAAG,EAAE,CAAC;IAChB,CAAC;IAEO,KAAK,CAAC,YAAY,CACxB,UAAkB,EAClB,KAAmB,EACnB,MAAc;QAEd,IAAI,CAAC;YACH,MAAM,OAAO,GAAG,IAAI,CAAC,SAAS,CAAC,KAAK,CAAC,CAAC;YACtC,MAAM,SAAS,GAAG,MAAM,IAAI,CAAC,iBAAiB,CAAC,OAAO,EAAE,MAAM,CAAC,CAAC;YAEhE,MAAM,QAAQ,GAAG,MAAM,KAAK,CAAC,UAAU,EAAE;gBACvC,MAAM,EAAE,MAAM;gBACd,OAAO,EAAE;oBACP,cAAc,EAAE,kBAAkB;oBAClC,oBAAoB,EAAE,SAAS;oBAC/B,gBAAgB,EAAE,KAAK,CAAC,IAAI;iBAC7B;gBACD,IAAI,EAAE,OAAO;aACd,CAAC,CAAC;YAEH,IAAI,QAAQ,CAAC,EAAE,EAAE,CAAC;gBAChB,OAAO,CAAC,GAAG,CACT,KAAK,MAAM,CAAC,OAAO,CAAC,GAAG,CAAC,iBAAiB,UAAU,KAAK,QAAQ,CAAC,MAAM,GAAG,CAC3E,CAAC;YACJ,CAAC;iBAAM,CAAC;gBACN,OAAO,CAAC,GAAG,CACT,KAAK,MAAM,CAAC,KAAK,CAAC,GAAG,CAAC,oBAAoB,QAAQ,CAAC,MAAM,GAAG,CAC7D,CAAC;YACJ,CAAC;QACH,CAAC;QAAC,OAAO,GAAG,EAAE,CAAC;YACb,OAAO,CAAC,GAAG,CACT,KAAK,MAAM,CAAC,KAAK,CAAC,GAAG,CAAC,mBAAoB,GAAa,CAAC,OAAO,EAAE,CAClE,CAAC;QACJ,CAAC;QACD,OAAO,CAAC,GAAG,EAAE,CAAC;IAChB,CAAC;IAEO,KAAK,CAAC,iBAAiB,CAC7B,OAAe,EACf,MAAc;QAEd,MAAM,OAAO,GAAG,IAAI,WAAW,EAAE,CAAC;QAClC,MAAM,GAAG,GAAG,MAAM,MAAM,CAAC,MAAM,CAAC,SAAS,CACvC,KAAK,EACL,OAAO,CAAC,MAAM,CAAC,MAAM,CAAC,EACtB,EAAE,IAAI,EAAE,MAAM,EAAE,IAAI,EAAE,SAAS,EAAE,EACjC,KAAK,EACL,CAAC,MAAM,CAAC,CACT,CAAC;QACF,MAAM,SAAS,GAAG,MAAM,MAAM,CAAC,MAAM,CAAC,IAAI,CACxC,MAAM,EACN,GAAG,EACH,OAAO,CAAC,MAAM,CAAC,OAAO,CAAC,CACxB,CAAC;QACF,MAAM,SAAS,GAAG,KAAK,CAAC,IAAI,CAAC,IAAI,UAAU,CAAC,SAAS,CAAC,CAAC,CAAC;QACxD,MAAM,OAAO,GAAG,SAAS;aACtB,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,QAAQ,CAAC,EAAE,CAAC,CAAC,QAAQ,CAAC,CAAC,EAAE,GAAG,CAAC,CAAC;aAC3C,IAAI,CAAC,EAAE,CAAC,CAAC;QACZ,OAAO,MAAM,OAAO,EAAE,CAAC;IACzB,CAAC;IAEO,KAAK,CAAC,OAAO;QACnB,kBAAkB;QAClB,IAAI,IAAI,CAAC,MAAM,EAAE,CAAC;YAChB,IAAI,CAAC,MAAM,CAAC,KAAK,EAAE,CAAC;QACtB,CAAC;QAED,6BAA6B;QAC7B,IAAI,IAAI,CAAC,SAAS,EAAE,CAAC;YACnB,IAAI,CAAC;gBACH,MAAM,SAAS,CAAC,MAAM,CAAC,4BAA4B,IAAI,CAAC,SAAS,EAAE,CAAC,CAAC;YACvE,CAAC;YAAC,MAAM,CAAC;gBACP,wBAAwB;YAC1B,CAAC;QACH,CAAC;IACH,CAAC","sourcesContent":["import { Flags } from \"@oclif/core\";\nimport { AuthenticatedCommand } from \"../../lib/base-command.js\";\nimport { apiClient } from \"../../lib/api-client.js\";\nimport {\n  success,\n  error,\n  info,\n  warn,\n  colors,\n  formatStatus,\n  formatDate,\n  spinner,\n  json as jsonOutput,\n  isJsonMode,\n} from \"../../lib/output.js\";\nimport { getConfigValue } from \"../../lib/config.js\";\nimport localtunnel from \"localtunnel\";\n\ninterface WebhookEvent {\n  id: string;\n  type: string;\n  data: Record<string, unknown>;\n  created: number;\n}\n\nexport default class WebhooksListen extends AuthenticatedCommand {\n  static description =\n    \"Listen for webhooks locally (like Stripe CLI). Creates a secure tunnel to forward events to your local server.\";\n\n  static examples = [\n    \"<%= config.bin %> webhooks listen\",\n    \"<%= config.bin %> webhooks listen --forward http://localhost:3000/webhook\",\n    \"<%= config.bin %> webhooks listen --events message.delivered,message.failed\",\n  ];\n\n  static flags = {\n    ...AuthenticatedCommand.baseFlags,\n    forward: Flags.string({\n      char: \"f\",\n      description: \"Local URL to forward events to\",\n      default: \"http://localhost:3000/webhook\",\n    }),\n    events: Flags.string({\n      char: \"e\",\n      description: \"Comma-separated list of events to listen for\",\n      default: \"message.sent,message.delivered,message.failed,message.bounced\",\n    }),\n    port: Flags.integer({\n      char: \"p\",\n      description:\n        \"Local port for the tunnel (auto-detected from forward URL if not specified)\",\n    }),\n  };\n\n  private tunnel: localtunnel.Tunnel | null = null;\n  private webhookId: string | null = null;\n\n  async run(): Promise<void> {\n    const { flags } = await this.parse(WebhooksListen);\n\n    const forwardUrl = new URL(flags.forward);\n    const localPort = flags.port || parseInt(forwardUrl.port) || 3000;\n    const events = flags.events.split(\",\").map((e) => e.trim());\n\n    const spin = spinner(\"Starting webhook listener...\");\n    spin.start();\n\n    try {\n      // Create localtunnel\n      this.tunnel = await localtunnel({\n        port: localPort,\n        subdomain: `sendly-${Date.now().toString(36)}`,\n      });\n\n      const tunnelUrl = this.tunnel.url;\n      spin.succeed(\"Tunnel established\");\n\n      // Register temporary webhook with Sendly\n      const webhookResponse = await apiClient.post<{\n        id: string;\n        secret: string;\n      }>(\"/api/cli/webhooks/listen\", {\n        url: `${tunnelUrl}/cli-webhook`,\n        events,\n        forwardUrl: flags.forward,\n      });\n\n      this.webhookId = webhookResponse.id;\n      const secret = webhookResponse.secret;\n\n      // Display connection info\n      console.log();\n      console.log(colors.bold(colors.primary(\"Webhook listener ready!\")));\n      console.log();\n      console.log(\n        `  ${colors.dim(\"Tunnel URL:\")}     ${colors.code(tunnelUrl)}`,\n      );\n      console.log(\n        `  ${colors.dim(\"Forwarding to:\")} ${colors.code(flags.forward)}`,\n      );\n      console.log(`  ${colors.dim(\"Events:\")}        ${events.join(\", \")}`);\n      console.log();\n      console.log(`  ${colors.dim(\"Webhook Secret:\")}`);\n      console.log(`  ${colors.primary(secret)}`);\n      console.log();\n      console.log(\n        colors.dim(\"Use this secret to verify webhook signatures in your app.\"),\n      );\n      console.log();\n      console.log(colors.bold(\"Waiting for events...\"));\n      console.log(colors.dim(\"─\".repeat(60)));\n      console.log();\n\n      // Set up event forwarding\n      this.tunnel.on(\"request\", (info) => {\n        // This is just for logging - actual forwarding happens server-side\n      });\n\n      // Handle tunnel close\n      this.tunnel.on(\"close\", () => {\n        warn(\"Tunnel closed\");\n        this.cleanup();\n        process.exit(0);\n      });\n\n      this.tunnel.on(\"error\", (err) => {\n        error(`Tunnel error: ${err.message}`);\n      });\n\n      // Poll for events and display them\n      await this.pollEvents(flags.forward, secret);\n    } catch (err) {\n      spin.fail(\"Failed to start listener\");\n      throw err;\n    }\n  }\n\n  private async pollEvents(forwardUrl: string, secret: string): Promise<void> {\n    // Poll for events at regular intervals\n    const pollInterval = setInterval(async () => {\n      try {\n        const events = await apiClient.get<WebhookEvent[]>(\n          `/api/cli/webhooks/events?webhookId=${this.webhookId}&since=${Date.now() - 5000}`,\n        );\n\n        for (const event of events) {\n          this.displayEvent(event);\n          await this.forwardEvent(forwardUrl, event, secret);\n        }\n      } catch {\n        // Ignore polling errors\n      }\n    }, 2000);\n\n    // Handle graceful shutdown\n    const cleanup = () => {\n      clearInterval(pollInterval);\n      this.cleanup();\n      process.exit(0);\n    };\n\n    process.on(\"SIGINT\", cleanup);\n    process.on(\"SIGTERM\", cleanup);\n\n    // Keep process alive\n    await new Promise(() => {});\n  }\n\n  private displayEvent(event: WebhookEvent): void {\n    const timestamp = new Date(event.created * 1000).toLocaleTimeString();\n    const eventType = event.type;\n\n    let statusColor = colors.info;\n    if (eventType.includes(\"delivered\")) statusColor = colors.success;\n    if (eventType.includes(\"failed\")) statusColor = colors.error;\n\n    console.log(\n      `${colors.dim(timestamp)} ${statusColor(\"→\")} ${colors.bold(eventType)}`,\n    );\n\n    if (event.data) {\n      const messageId =\n        (event.data as any).message_id || (event.data as any).id;\n      const to = (event.data as any).to;\n      if (messageId) {\n        console.log(`  ${colors.dim(\"message_id:\")} ${messageId}`);\n      }\n      if (to) {\n        console.log(`  ${colors.dim(\"to:\")} ${to}`);\n      }\n    }\n    console.log();\n  }\n\n  private async forwardEvent(\n    forwardUrl: string,\n    event: WebhookEvent,\n    secret: string,\n  ): Promise<void> {\n    try {\n      const payload = JSON.stringify(event);\n      const signature = await this.generateSignature(payload, secret);\n\n      const response = await fetch(forwardUrl, {\n        method: \"POST\",\n        headers: {\n          \"Content-Type\": \"application/json\",\n          \"X-Sendly-Signature\": signature,\n          \"X-Sendly-Event\": event.type,\n        },\n        body: payload,\n      });\n\n      if (response.ok) {\n        console.log(\n          `  ${colors.success(\"✓\")} Forwarded to ${forwardUrl} (${response.status})`,\n        );\n      } else {\n        console.log(\n          `  ${colors.error(\"✗\")} Forward failed (${response.status})`,\n        );\n      }\n    } catch (err) {\n      console.log(\n        `  ${colors.error(\"✗\")} Forward error: ${(err as Error).message}`,\n      );\n    }\n    console.log();\n  }\n\n  private async generateSignature(\n    payload: string,\n    secret: string,\n  ): Promise<string> {\n    const encoder = new TextEncoder();\n    const key = await crypto.subtle.importKey(\n      \"raw\",\n      encoder.encode(secret),\n      { name: \"HMAC\", hash: \"SHA-256\" },\n      false,\n      [\"sign\"],\n    );\n    const signature = await crypto.subtle.sign(\n      \"HMAC\",\n      key,\n      encoder.encode(payload),\n    );\n    const hashArray = Array.from(new Uint8Array(signature));\n    const hashHex = hashArray\n      .map((b) => b.toString(16).padStart(2, \"0\"))\n      .join(\"\");\n    return `v1=${hashHex}`;\n  }\n\n  private async cleanup(): Promise<void> {\n    // Clean up tunnel\n    if (this.tunnel) {\n      this.tunnel.close();\n    }\n\n    // Clean up temporary webhook\n    if (this.webhookId) {\n      try {\n        await apiClient.delete(`/api/cli/webhooks/listen/${this.webhookId}`);\n      } catch {\n        // Ignore cleanup errors\n      }\n    }\n  }\n}\n"]}
|
|
@@ -62,16 +62,16 @@ export default class WebhooksRotateSecret extends AuthenticatedCommand {
|
|
|
62
62
|
}
|
|
63
63
|
success("Webhook secret rotated", {
|
|
64
64
|
"Webhook ID": result.id,
|
|
65
|
-
"Secret Version": `${webhook.
|
|
66
|
-
"Grace Period": `${result.
|
|
67
|
-
"Rotated At": result.
|
|
65
|
+
"Secret Version": `${webhook.secret_version} → ${result.new_secret_version}`,
|
|
66
|
+
"Grace Period": `${result.grace_period_hours} hours`,
|
|
67
|
+
"Rotated At": result.rotated_at,
|
|
68
68
|
});
|
|
69
69
|
console.log();
|
|
70
70
|
warn("Copy your new webhook secret now. The old secret will expire in 24 hours!");
|
|
71
|
-
codeBlock(result.
|
|
71
|
+
codeBlock(result.new_secret);
|
|
72
72
|
console.log();
|
|
73
73
|
console.log(colors.dim("Update your application with this new secret for webhook signature verification."));
|
|
74
|
-
console.log(colors.dim(`The old secret will remain valid for ${result.
|
|
74
|
+
console.log(colors.dim(`The old secret will remain valid for ${result.grace_period_hours} hours to allow for graceful migration.`));
|
|
75
75
|
}
|
|
76
76
|
catch (err) {
|
|
77
77
|
if (err instanceof Error) {
|
|
@@ -84,4 +84,4 @@ export default class WebhooksRotateSecret extends AuthenticatedCommand {
|
|
|
84
84
|
}
|
|
85
85
|
}
|
|
86
86
|
}
|
|
87
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
87
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"rotate-secret.js","sourceRoot":"","sources":["../../../src/commands/webhooks/rotate-secret.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,IAAI,EAAE,KAAK,EAAE,MAAM,aAAa,CAAC;AAC1C,OAAO,EAAE,oBAAoB,EAAE,MAAM,2BAA2B,CAAC;AACjE,OAAO,EAAE,SAAS,EAAE,MAAM,yBAAyB,CAAC;AACpD,OAAO,EACL,OAAO,EACP,IAAI,EACJ,KAAK,EACL,IAAI,EACJ,MAAM,EACN,SAAS,EACT,UAAU,GACX,MAAM,qBAAqB,CAAC;AAC7B,OAAO,QAAQ,MAAM,UAAU,CAAC;AAUhC,MAAM,CAAC,OAAO,OAAO,oBAAqB,SAAQ,oBAAoB;IACpE,MAAM,CAAC,WAAW,GAAG,uBAAuB,CAAC;IAE7C,MAAM,CAAC,QAAQ,GAAG;QAChB,qDAAqD;QACrD,2DAA2D;QAC3D,4DAA4D;KAC7D,CAAC;IAEF,MAAM,CAAC,IAAI,GAAG;QACZ,EAAE,EAAE,IAAI,CAAC,MAAM,CAAC;YACd,WAAW,EAAE,YAAY;YACzB,QAAQ,EAAE,IAAI;SACf,CAAC;KACH,CAAC;IAEF,MAAM,CAAC,KAAK,GAAG;QACb,GAAG,oBAAoB,CAAC,SAAS;QACjC,GAAG,EAAE,KAAK,CAAC,OAAO,CAAC;YACjB,IAAI,EAAE,GAAG;YACT,WAAW,EAAE,0BAA0B;YACvC,OAAO,EAAE,KAAK;SACf,CAAC;KACH,CAAC;IAEF,KAAK,CAAC,GAAG;QACP,MAAM,EAAE,IAAI,EAAE,KAAK,EAAE,GAAG,MAAM,IAAI,CAAC,KAAK,CAAC,oBAAoB,CAAC,CAAC;QAE/D,uCAAuC;QACvC,IAAI,OAAO,CAAC;QACZ,IAAI,CAAC;YACH,OAAO,GAAG,MAAM,SAAS,CAAC,GAAG,CAI1B,oBAAoB,IAAI,CAAC,EAAE,EAAE,CAAC,CAAC;QACpC,CAAC;QAAC,OAAO,GAAG,EAAE,CAAC;YACb,KAAK,CAAC,sBAAsB,IAAI,CAAC,EAAE,EAAE,CAAC,CAAC;YACvC,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;QACf,CAAC;QAED,mBAAmB;QACnB,IAAI,CAAC,KAAK,CAAC,GAAG,IAAI,CAAC,UAAU,EAAE,EAAE,CAAC;YAChC,OAAO,CAAC,GAAG,EAAE,CAAC;YACd,OAAO,CAAC,GAAG,CACT,MAAM,CAAC,OAAO,CACZ,kFAAkF,CACnF,CACF,CAAC;YACF,OAAO,CAAC,GAAG,CACT,MAAM,CAAC,GAAG,CAAC,2DAA2D,CAAC,CACxE,CAAC;YACF,OAAO,CAAC,GAAG,EAAE,CAAC;YAEd,MAAM,EAAE,OAAO,EAAE,GAAG,MAAM,QAAQ,CAAC,MAAM,CAAC;gBACxC;oBACE,IAAI,EAAE,SAAS;oBACf,IAAI,EAAE,SAAS;oBACf,OAAO,EAAE,6BAA6B,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,EAAE,CAAC,KAAK,MAAM,CAAC,GAAG,CAAC,OAAO,CAAC,GAAG,CAAC,IAAI;oBAC1F,OAAO,EAAE,KAAK;iBACf;aACF,CAAC,CAAC;YAEH,IAAI,CAAC,OAAO,EAAE,CAAC;gBACb,KAAK,CAAC,2BAA2B,CAAC,CAAC;gBACnC,OAAO;YACT,CAAC;QACH,CAAC;QAED,IAAI,CAAC;YACH,MAAM,MAAM,GAAG,MAAM,SAAS,CAAC,IAAI,CACjC,oBAAoB,IAAI,CAAC,EAAE,gBAAgB,CAC5C,CAAC;YAEF,IAAI,UAAU,EAAE,EAAE,CAAC;gBACjB,IAAI,CAAC,MAAM,CAAC,CAAC;gBACb,OAAO;YACT,CAAC;YAED,OAAO,CAAC,wBAAwB,EAAE;gBAChC,YAAY,EAAE,MAAM,CAAC,EAAE;gBACvB,gBAAgB,EAAE,GAAG,OAAO,CAAC,cAAc,MAAM,MAAM,CAAC,kBAAkB,EAAE;gBAC5E,cAAc,EAAE,GAAG,MAAM,CAAC,kBAAkB,QAAQ;gBACpD,YAAY,EAAE,MAAM,CAAC,UAAU;aAChC,CAAC,CAAC;YAEH,OAAO,CAAC,GAAG,EAAE,CAAC;YACd,IAAI,CACF,2EAA2E,CAC5E,CAAC;YACF,SAAS,CAAC,MAAM,CAAC,UAAU,CAAC,CAAC;YAE7B,OAAO,CAAC,GAAG,EAAE,CAAC;YACd,OAAO,CAAC,GAAG,CACT,MAAM,CAAC,GAAG,CACR,kFAAkF,CACnF,CACF,CAAC;YACF,OAAO,CAAC,GAAG,CACT,MAAM,CAAC,GAAG,CACR,wCAAwC,MAAM,CAAC,kBAAkB,yCAAyC,CAC3G,CACF,CAAC;QACJ,CAAC;QAAC,OAAO,GAAG,EAAE,CAAC;YACb,IAAI,GAAG,YAAY,KAAK,EAAE,CAAC;gBACzB,KAAK,CAAC,4BAA4B,GAAG,CAAC,OAAO,EAAE,CAAC,CAAC;YACnD,CAAC;iBAAM,CAAC;gBACN,KAAK,CAAC,4BAA4B,MAAM,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC;YACnD,CAAC;YACD,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;QACf,CAAC;IACH,CAAC","sourcesContent":["import { Args, Flags } from \"@oclif/core\";\nimport { AuthenticatedCommand } from \"../../lib/base-command.js\";\nimport { apiClient } from \"../../lib/api-client.js\";\nimport {\n  success,\n  warn,\n  error,\n  json,\n  colors,\n  codeBlock,\n  isJsonMode,\n} from \"../../lib/output.js\";\nimport inquirer from \"inquirer\";\n\ninterface RotateSecretResponse {\n  id: string;\n  new_secret: string;\n  new_secret_version: number;\n  grace_period_hours: number;\n  rotated_at: string;\n}\n\nexport default class WebhooksRotateSecret extends AuthenticatedCommand {\n  static description = \"Rotate webhook secret\";\n\n  static examples = [\n    \"<%= config.bin %> webhooks rotate-secret whk_abc123\",\n    \"<%= config.bin %> webhooks rotate-secret whk_abc123 --yes\",\n    \"<%= config.bin %> webhooks rotate-secret whk_abc123 --json\",\n  ];\n\n  static args = {\n    id: Args.string({\n      description: \"Webhook ID\",\n      required: true,\n    }),\n  };\n\n  static flags = {\n    ...AuthenticatedCommand.baseFlags,\n    yes: Flags.boolean({\n      char: \"y\",\n      description: \"Skip confirmation prompt\",\n      default: false,\n    }),\n  };\n\n  async run(): Promise<void> {\n    const { args, flags } = await this.parse(WebhooksRotateSecret);\n\n    // Get webhook details for confirmation\n    let webhook;\n    try {\n      webhook = await apiClient.get<{\n        id: string;\n        url: string;\n        secret_version: number;\n      }>(`/api/v1/webhooks/${args.id}`);\n    } catch (err) {\n      error(`Webhook not found: ${args.id}`);\n      this.exit(1);\n    }\n\n    // Confirm rotation\n    if (!flags.yes && !isJsonMode()) {\n      console.log();\n      console.log(\n        colors.warning(\n          \"⚠ This will rotate the webhook secret and invalidate the old one after 24 hours.\",\n        ),\n      );\n      console.log(\n        colors.dim(\"Make sure to update your application with the new secret.\"),\n      );\n      console.log();\n\n      const { confirm } = await inquirer.prompt([\n        {\n          type: \"confirm\",\n          name: \"confirm\",\n          message: `Rotate secret for webhook ${colors.code(args.id)} (${colors.dim(webhook.url)})?`,\n          default: false,\n        },\n      ]);\n\n      if (!confirm) {\n        error(\"Secret rotation cancelled\");\n        return;\n      }\n    }\n\n    try {\n      const result = await apiClient.post<RotateSecretResponse>(\n        `/api/v1/webhooks/${args.id}/rotate-secret`,\n      );\n\n      if (isJsonMode()) {\n        json(result);\n        return;\n      }\n\n      success(\"Webhook secret rotated\", {\n        \"Webhook ID\": result.id,\n        \"Secret Version\": `${webhook.secret_version} → ${result.new_secret_version}`,\n        \"Grace Period\": `${result.grace_period_hours} hours`,\n        \"Rotated At\": result.rotated_at,\n      });\n\n      console.log();\n      warn(\n        \"Copy your new webhook secret now. The old secret will expire in 24 hours!\",\n      );\n      codeBlock(result.new_secret);\n\n      console.log();\n      console.log(\n        colors.dim(\n          \"Update your application with this new secret for webhook signature verification.\",\n        ),\n      );\n      console.log(\n        colors.dim(\n          `The old secret will remain valid for ${result.grace_period_hours} hours to allow for graceful migration.`,\n        ),\n      );\n    } catch (err) {\n      if (err instanceof Error) {\n        error(`Failed to rotate secret: ${err.message}`);\n      } else {\n        error(`Failed to rotate secret: ${String(err)}`);\n      }\n      this.exit(1);\n    }\n  }\n}\n"]}
|
|
@@ -30,26 +30,29 @@ export default class WebhooksTest extends AuthenticatedCommand {
|
|
|
30
30
|
}
|
|
31
31
|
if (result.status === "delivered") {
|
|
32
32
|
success("Test event delivered", {
|
|
33
|
-
"Delivery ID": result.
|
|
34
|
-
"Webhook URL": result.
|
|
35
|
-
"Event Type": result.
|
|
36
|
-
"Response Time": `${result.
|
|
37
|
-
"Status Code": String(result.
|
|
38
|
-
"Delivered At": result.
|
|
33
|
+
"Delivery ID": result.delivery_id,
|
|
34
|
+
"Webhook URL": result.webhook_url,
|
|
35
|
+
"Event Type": result.event_type,
|
|
36
|
+
"Response Time": `${result.response_time}ms`,
|
|
37
|
+
"Status Code": String(result.status_code),
|
|
38
|
+
"Delivered At": result.delivered_at,
|
|
39
39
|
});
|
|
40
|
-
if (result.
|
|
40
|
+
if (result.response_body) {
|
|
41
41
|
console.log();
|
|
42
42
|
console.log(colors.dim("Response Body:"));
|
|
43
|
-
console.log(result.
|
|
43
|
+
console.log(result.response_body.substring(0, 200) +
|
|
44
|
+
(result.response_body.length > 200 ? "..." : ""));
|
|
44
45
|
}
|
|
45
46
|
}
|
|
46
47
|
else {
|
|
47
48
|
error("Test event failed", {
|
|
48
|
-
"Delivery ID": result.
|
|
49
|
-
"Webhook URL": result.
|
|
50
|
-
|
|
51
|
-
|
|
52
|
-
...(result.
|
|
49
|
+
"Delivery ID": result.delivery_id,
|
|
50
|
+
"Webhook URL": result.webhook_url,
|
|
51
|
+
Status: result.status,
|
|
52
|
+
Error: result.error || "Unknown error",
|
|
53
|
+
...(result.status_code && {
|
|
54
|
+
"Status Code": String(result.status_code),
|
|
55
|
+
}),
|
|
53
56
|
});
|
|
54
57
|
}
|
|
55
58
|
}
|
|
@@ -70,4 +73,4 @@ export default class WebhooksTest extends AuthenticatedCommand {
|
|
|
70
73
|
}
|
|
71
74
|
}
|
|
72
75
|
}
|
|
73
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
76
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidGVzdC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3NyYy9jb21tYW5kcy93ZWJob29rcy90ZXN0LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBRSxJQUFJLEVBQUUsTUFBTSxhQUFhLENBQUM7QUFDbkMsT0FBTyxFQUFFLG9CQUFvQixFQUFFLE1BQU0sMkJBQTJCLENBQUM7QUFDakUsT0FBTyxFQUFFLFNBQVMsRUFBRSxNQUFNLHlCQUF5QixDQUFDO0FBQ3BELE9BQU8sRUFDTCxPQUFPLEVBQ1AsS0FBSyxFQUNMLE9BQU8sRUFDUCxJQUFJLEVBQ0osTUFBTSxFQUVOLFVBQVUsR0FDWCxNQUFNLHFCQUFxQixDQUFDO0FBZTdCLE1BQU0sQ0FBQyxPQUFPLE9BQU8sWUFBYSxTQUFRLG9CQUFvQjtJQUM1RCxNQUFNLENBQUMsV0FBVyxHQUFHLGdDQUFnQyxDQUFDO0lBRXRELE1BQU0sQ0FBQyxRQUFRLEdBQUc7UUFDaEIsNENBQTRDO1FBQzVDLG1EQUFtRDtLQUNwRCxDQUFDO0lBRUYsTUFBTSxDQUFDLElBQUksR0FBRztRQUNaLEVBQUUsRUFBRSxJQUFJLENBQUMsTUFBTSxDQUFDO1lBQ2QsV0FBVyxFQUFFLG9CQUFvQjtZQUNqQyxRQUFRLEVBQUUsSUFBSTtTQUNmLENBQUM7S0FDSCxDQUFDO0lBRUYsTUFBTSxDQUFDLEtBQUssR0FBRztRQUNiLEdBQUcsb0JBQW9CLENBQUMsU0FBUztLQUNsQyxDQUFDO0lBRUYsS0FBSyxDQUFDLEdBQUc7UUFDUCxNQUFNLEVBQUUsSUFBSSxFQUFFLEdBQUcsTUFBTSxJQUFJLENBQUMsS0FBSyxDQUFDLFlBQVksQ0FBQyxDQUFDO1FBRWhELE1BQU0sV0FBVyxHQUFHLE9BQU8sQ0FBQyx1QkFBdUIsQ0FBQyxDQUFDO1FBQ3JELFdBQVcsQ0FBQyxLQUFLLEVBQUUsQ0FBQztRQUVwQixJQUFJLENBQUM7WUFDSCxNQUFNLE1BQU0sR0FBRyxNQUFNLFNBQVMsQ0FBQyxJQUFJLENBQ2pDLG9CQUFvQixJQUFJLENBQUMsRUFBRSxPQUFPLENBQ25DLENBQUM7WUFFRixXQUFXLENBQUMsSUFBSSxFQUFFLENBQUM7WUFFbkIsSUFBSSxVQUFVLEVBQUUsRUFBRSxDQUFDO2dCQUNqQixJQUFJLENBQUMsTUFBTSxDQUFDLENBQUM7Z0JBQ2IsT0FBTztZQUNULENBQUM7WUFFRCxJQUFJLE1BQU0sQ0FBQyxNQUFNLEtBQUssV0FBVyxFQUFFLENBQUM7Z0JBQ2xDLE9BQU8sQ0FBQyxzQkFBc0IsRUFBRTtvQkFDOUIsYUFBYSxFQUFFLE1BQU0sQ0FBQyxXQUFXO29CQUNqQyxhQUFhLEVBQUUsTUFBTSxDQUFDLFdBQVc7b0JBQ2pDLFlBQVksRUFBRSxNQUFNLENBQUMsVUFBVTtvQkFDL0IsZUFBZSxFQUFFLEdBQUcsTUFBTSxDQUFDLGFBQWEsSUFBSTtvQkFDNUMsYUFBYSxFQUFFLE1BQU0sQ0FBQyxNQUFNLENBQUMsV0FBVyxDQUFDO29CQUN6QyxjQUFjLEVBQUUsTUFBTSxDQUFDLFlBQVk7aUJBQ3BDLENBQUMsQ0FBQztnQkFFSCxJQUFJLE1BQU0sQ0FBQyxhQUFhLEVBQUUsQ0FBQztvQkFDekIsT0FBTyxDQUFDLEdBQUcsRUFBRSxDQUFDO29CQUNkLE9BQU8sQ0FBQyxHQUFHLENBQUMsTUFBTSxDQUFDLEdBQUcsQ0FBQyxnQkFBZ0IsQ0FBQyxDQUFDLENBQUM7b0JBQzFDLE9BQU8sQ0FBQyxHQUFHLENBQ1QsTUFBTSxDQUFDLGFBQWEsQ0FBQyxTQUFTLENBQUMsQ0FBQyxFQUFFLEdBQUcsQ0FBQzt3QkFDcEMsQ0FBQyxNQUFNLENBQUMsYUFBYSxDQUFDLE1BQU0sR0FBRyxHQUFHLENBQUMsQ0FBQyxDQUFDLEtBQUssQ0FBQyxDQUFDLENBQUMsRUFBRSxDQUFDLENBQ25ELENBQUM7Z0JBQ0osQ0FBQztZQUNILENBQUM7aUJBQU0sQ0FBQztnQkFDTixLQUFLLENBQUMsbUJBQW1CLEVBQUU7b0JBQ3pCLGFBQWEsRUFBRSxNQUFNLENBQUMsV0FBVztvQkFDakMsYUFBYSxFQUFFLE1BQU0sQ0FBQyxXQUFXO29CQUNqQyxNQUFNLEVBQUUsTUFBTSxDQUFDLE1BQU07b0JBQ3JCLEtBQUssRUFBRSxNQUFNLENBQUMsS0FBSyxJQUFJLGVBQWU7b0JBQ3RDLEdBQUcsQ0FBQyxNQUFNLENBQUMsV0FBVyxJQUFJO3dCQUN4QixhQUFhLEVBQUUsTUFBTSxDQUFDLE1BQU0sQ0FBQyxXQUFXLENBQUM7cUJBQzFDLENBQUM7aUJBQ0gsQ0FBQyxDQUFDO1lBQ0wsQ0FBQztRQUNILENBQUM7UUFBQyxPQUFPLEdBQUcsRUFBRSxDQUFDO1lBQ2IsV0FBVyxDQUFDLElBQUksRUFBRSxDQUFDO1lBRW5CLElBQUksR0FBRyxZQUFZLEtBQUssSUFBSSxHQUFHLENBQUMsT0FBTyxDQUFDLFFBQVEsQ0FBQyxLQUFLLENBQUMsRUFBRSxDQUFDO2dCQUN4RCxLQUFLLENBQUMsc0JBQXNCLElBQUksQ0FBQyxFQUFFLEVBQUUsQ0FBQyxDQUFDO1lBQ3pDLENBQUM7aUJBQU0sQ0FBQztnQkFDTixJQUFJLEdBQUcsWUFBWSxLQUFLLEVBQUUsQ0FBQztvQkFDekIsS0FBSyxDQUFDLDhCQUE4QixHQUFHLENBQUMsT0FBTyxFQUFFLENBQUMsQ0FBQztnQkFDckQsQ0FBQztxQkFBTSxDQUFDO29CQUNOLEtBQUssQ0FBQyw4QkFBOEIsTUFBTSxDQUFDLEdBQUcsQ0FBQyxFQUFFLENBQUMsQ0FBQztnQkFDckQsQ0FBQztZQUNILENBQUM7WUFDRCxJQUFJLENBQUMsSUFBSSxDQUFDLENBQUMsQ0FBQyxDQUFDO1FBQ2YsQ0FBQztJQUNILENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBBcmdzIH0gZnJvbSBcIkBvY2xpZi9jb3JlXCI7XG5pbXBvcnQgeyBBdXRoZW50aWNhdGVkQ29tbWFuZCB9IGZyb20gXCIuLi8uLi9saWIvYmFzZS1jb21tYW5kLmpzXCI7XG5pbXBvcnQgeyBhcGlDbGllbnQgfSBmcm9tIFwiLi4vLi4vbGliL2FwaS1jbGllbnQuanNcIjtcbmltcG9ydCB7XG4gIHN1Y2Nlc3MsXG4gIGVycm9yLFxuICBzcGlubmVyLFxuICBqc29uLFxuICBjb2xvcnMsXG4gIGtleVZhbHVlLFxuICBpc0pzb25Nb2RlLFxufSBmcm9tIFwiLi4vLi4vbGliL291dHB1dC5qc1wiO1xuXG5pbnRlcmZhY2UgVGVzdFdlYmhvb2tSZXNwb25zZSB7XG4gIGlkOiBzdHJpbmc7XG4gIGRlbGl2ZXJ5X2lkOiBzdHJpbmc7XG4gIHdlYmhvb2tfdXJsOiBzdHJpbmc7XG4gIGV2ZW50X3R5cGU6IHN0cmluZztcbiAgc3RhdHVzOiBzdHJpbmc7XG4gIHJlc3BvbnNlX3RpbWU6IG51bWJlcjtcbiAgc3RhdHVzX2NvZGU/OiBudW1iZXI7XG4gIHJlc3BvbnNlX2JvZHk/OiBzdHJpbmc7XG4gIGVycm9yPzogc3RyaW5nO1xuICBkZWxpdmVyZWRfYXQ6IHN0cmluZztcbn1cblxuZXhwb3J0IGRlZmF1bHQgY2xhc3MgV2ViaG9va3NUZXN0IGV4dGVuZHMgQXV0aGVudGljYXRlZENvbW1hbmQge1xuICBzdGF0aWMgZGVzY3JpcHRpb24gPSBcIlNlbmQgYSB0ZXN0IGV2ZW50IHRvIGEgd2ViaG9va1wiO1xuXG4gIHN0YXRpYyBleGFtcGxlcyA9IFtcbiAgICBcIjwlPSBjb25maWcuYmluICU+IHdlYmhvb2tzIHRlc3Qgd2hrX2FiYzEyM1wiLFxuICAgIFwiPCU9IGNvbmZpZy5iaW4gJT4gd2ViaG9va3MgdGVzdCB3aGtfYWJjMTIzIC0tanNvblwiLFxuICBdO1xuXG4gIHN0YXRpYyBhcmdzID0ge1xuICAgIGlkOiBBcmdzLnN0cmluZyh7XG4gICAgICBkZXNjcmlwdGlvbjogXCJXZWJob29rIElEIHRvIHRlc3RcIixcbiAgICAgIHJlcXVpcmVkOiB0cnVlLFxuICAgIH0pLFxuICB9O1xuXG4gIHN0YXRpYyBmbGFncyA9IHtcbiAgICAuLi5BdXRoZW50aWNhdGVkQ29tbWFuZC5iYXNlRmxhZ3MsXG4gIH07XG5cbiAgYXN5bmMgcnVuKCk6IFByb21pc2U8dm9pZD4ge1xuICAgIGNvbnN0IHsgYXJncyB9ID0gYXdhaXQgdGhpcy5wYXJzZShXZWJob29rc1Rlc3QpO1xuXG4gICAgY29uc3QgdGVzdFNwaW5uZXIgPSBzcGlubmVyKFwiU2VuZGluZyB0ZXN0IGV2ZW50Li4uXCIpO1xuICAgIHRlc3RTcGlubmVyLnN0YXJ0KCk7XG5cbiAgICB0cnkge1xuICAgICAgY29uc3QgcmVzdWx0ID0gYXdhaXQgYXBpQ2xpZW50LnBvc3Q8VGVzdFdlYmhvb2tSZXNwb25zZT4oXG4gICAgICAgIGAvYXBpL3YxL3dlYmhvb2tzLyR7YXJncy5pZH0vdGVzdGAsXG4gICAgICApO1xuXG4gICAgICB0ZXN0U3Bpbm5lci5zdG9wKCk7XG5cbiAgICAgIGlmIChpc0pzb25Nb2RlKCkpIHtcbiAgICAgICAganNvbihyZXN1bHQpO1xuICAgICAgICByZXR1cm47XG4gICAgICB9XG5cbiAgICAgIGlmIChyZXN1bHQuc3RhdHVzID09PSBcImRlbGl2ZXJlZFwiKSB7XG4gICAgICAgIHN1Y2Nlc3MoXCJUZXN0IGV2ZW50IGRlbGl2ZXJlZFwiLCB7XG4gICAgICAgICAgXCJEZWxpdmVyeSBJRFwiOiByZXN1bHQuZGVsaXZlcnlfaWQsXG4gICAgICAgICAgXCJXZWJob29rIFVSTFwiOiByZXN1bHQud2ViaG9va191cmwsXG4gICAgICAgICAgXCJFdmVudCBUeXBlXCI6IHJlc3VsdC5ldmVudF90eXBlLFxuICAgICAgICAgIFwiUmVzcG9uc2UgVGltZVwiOiBgJHtyZXN1bHQucmVzcG9uc2VfdGltZX1tc2AsXG4gICAgICAgICAgXCJTdGF0dXMgQ29kZVwiOiBTdHJpbmcocmVzdWx0LnN0YXR1c19jb2RlKSxcbiAgICAgICAgICBcIkRlbGl2ZXJlZCBBdFwiOiByZXN1bHQuZGVsaXZlcmVkX2F0LFxuICAgICAgICB9KTtcblxuICAgICAgICBpZiAocmVzdWx0LnJlc3BvbnNlX2JvZHkpIHtcbiAgICAgICAgICBjb25zb2xlLmxvZygpO1xuICAgICAgICAgIGNvbnNvbGUubG9nKGNvbG9ycy5kaW0oXCJSZXNwb25zZSBCb2R5OlwiKSk7XG4gICAgICAgICAgY29uc29sZS5sb2coXG4gICAgICAgICAgICByZXN1bHQucmVzcG9uc2VfYm9keS5zdWJzdHJpbmcoMCwgMjAwKSArXG4gICAgICAgICAgICAgIChyZXN1bHQucmVzcG9uc2VfYm9keS5sZW5ndGggPiAyMDAgPyBcIi4uLlwiIDogXCJcIiksXG4gICAgICAgICAgKTtcbiAgICAgICAgfVxuICAgICAgfSBlbHNlIHtcbiAgICAgICAgZXJyb3IoXCJUZXN0IGV2ZW50IGZhaWxlZFwiLCB7XG4gICAgICAgICAgXCJEZWxpdmVyeSBJRFwiOiByZXN1bHQuZGVsaXZlcnlfaWQsXG4gICAgICAgICAgXCJXZWJob29rIFVSTFwiOiByZXN1bHQud2ViaG9va191cmwsXG4gICAgICAgICAgU3RhdHVzOiByZXN1bHQuc3RhdHVzLFxuICAgICAgICAgIEVycm9yOiByZXN1bHQuZXJyb3IgfHwgXCJVbmtub3duIGVycm9yXCIsXG4gICAgICAgICAgLi4uKHJlc3VsdC5zdGF0dXNfY29kZSAmJiB7XG4gICAgICAgICAgICBcIlN0YXR1cyBDb2RlXCI6IFN0cmluZyhyZXN1bHQuc3RhdHVzX2NvZGUpLFxuICAgICAgICAgIH0pLFxuICAgICAgICB9KTtcbiAgICAgIH1cbiAgICB9IGNhdGNoIChlcnIpIHtcbiAgICAgIHRlc3RTcGlubmVyLnN0b3AoKTtcblxuICAgICAgaWYgKGVyciBpbnN0YW5jZW9mIEVycm9yICYmIGVyci5tZXNzYWdlLmluY2x1ZGVzKFwiNDA0XCIpKSB7XG4gICAgICAgIGVycm9yKGBXZWJob29rIG5vdCBmb3VuZDogJHthcmdzLmlkfWApO1xuICAgICAgfSBlbHNlIHtcbiAgICAgICAgaWYgKGVyciBpbnN0YW5jZW9mIEVycm9yKSB7XG4gICAgICAgICAgZXJyb3IoYEZhaWxlZCB0byBzZW5kIHRlc3QgZXZlbnQ6ICR7ZXJyLm1lc3NhZ2V9YCk7XG4gICAgICAgIH0gZWxzZSB7XG4gICAgICAgICAgZXJyb3IoYEZhaWxlZCB0byBzZW5kIHRlc3QgZXZlbnQ6ICR7U3RyaW5nKGVycil9YCk7XG4gICAgICAgIH1cbiAgICAgIH1cbiAgICAgIHRoaXMuZXhpdCgxKTtcbiAgICB9XG4gIH1cbn1cbiJdfQ==
|