@jimiford/channel-webex 1.0.0

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.
@@ -0,0 +1,33 @@
1
+ /**
2
+ * OpenClaw Webex Channel Plugin
3
+ *
4
+ * A channel plugin for integrating Cisco Webex messaging with OpenClaw.
5
+ *
6
+ * @packageDocumentation
7
+ */
8
+ export { WebexChannel, createWebexChannel, createAndInitialize } from './channel';
9
+ export { WebexSender, WebexApiRequestError } from './send';
10
+ export { WebexWebhookHandler, WebhookValidationError } from './webhook';
11
+ export type { WebexChannelConfig, DmPolicy, WebexPerson, WebexRoom, WebexMessage, WebexAttachment, AdaptiveCard, WebexWebhook, WebexWebhookResource, WebexWebhookEvent, WebexWebhookPayload, WebexWebhookData, CreateMessageRequest, CreateWebhookRequest, WebexApiError, PaginatedResponse, OpenClawEnvelope, OpenClawAttachment, OpenClawOutboundMessage, WebexChannelPlugin, WebhookHandler, RetryOptions, RequestOptions, } from './types';
12
+ import { createWebexChannel } from './channel';
13
+ export default createWebexChannel;
14
+ /**
15
+ * Plugin metadata for OpenClaw discovery
16
+ */
17
+ export declare const pluginMetadata: {
18
+ name: string;
19
+ version: string;
20
+ type: string;
21
+ channel: string;
22
+ description: string;
23
+ author: string;
24
+ repository: string;
25
+ capabilities: {
26
+ dm: boolean;
27
+ groups: boolean;
28
+ threads: boolean;
29
+ attachments: boolean;
30
+ cards: boolean;
31
+ webhooks: boolean;
32
+ };
33
+ };
package/dist/index.js ADDED
@@ -0,0 +1,47 @@
1
+ "use strict";
2
+ /**
3
+ * OpenClaw Webex Channel Plugin
4
+ *
5
+ * A channel plugin for integrating Cisco Webex messaging with OpenClaw.
6
+ *
7
+ * @packageDocumentation
8
+ */
9
+ Object.defineProperty(exports, "__esModule", { value: true });
10
+ exports.pluginMetadata = exports.WebhookValidationError = exports.WebexWebhookHandler = exports.WebexApiRequestError = exports.WebexSender = exports.createAndInitialize = exports.createWebexChannel = exports.WebexChannel = void 0;
11
+ // Main channel export
12
+ var channel_1 = require("./channel");
13
+ Object.defineProperty(exports, "WebexChannel", { enumerable: true, get: function () { return channel_1.WebexChannel; } });
14
+ Object.defineProperty(exports, "createWebexChannel", { enumerable: true, get: function () { return channel_1.createWebexChannel; } });
15
+ Object.defineProperty(exports, "createAndInitialize", { enumerable: true, get: function () { return channel_1.createAndInitialize; } });
16
+ // Sender export
17
+ var send_1 = require("./send");
18
+ Object.defineProperty(exports, "WebexSender", { enumerable: true, get: function () { return send_1.WebexSender; } });
19
+ Object.defineProperty(exports, "WebexApiRequestError", { enumerable: true, get: function () { return send_1.WebexApiRequestError; } });
20
+ // Webhook handler export
21
+ var webhook_1 = require("./webhook");
22
+ Object.defineProperty(exports, "WebexWebhookHandler", { enumerable: true, get: function () { return webhook_1.WebexWebhookHandler; } });
23
+ Object.defineProperty(exports, "WebhookValidationError", { enumerable: true, get: function () { return webhook_1.WebhookValidationError; } });
24
+ // Default export: factory function
25
+ const channel_2 = require("./channel");
26
+ exports.default = channel_2.createWebexChannel;
27
+ /**
28
+ * Plugin metadata for OpenClaw discovery
29
+ */
30
+ exports.pluginMetadata = {
31
+ name: '@openclaw/channel-webex',
32
+ version: '1.0.0',
33
+ type: 'channel',
34
+ channel: 'webex',
35
+ description: 'Cisco Webex messaging channel plugin for OpenClaw',
36
+ author: 'OpenClaw Contributors',
37
+ repository: 'https://github.com/openclaw/channel-webex',
38
+ capabilities: {
39
+ dm: true,
40
+ groups: true,
41
+ threads: true,
42
+ attachments: true,
43
+ cards: true,
44
+ webhooks: true,
45
+ },
46
+ };
47
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi9zcmMvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IjtBQUFBOzs7Ozs7R0FNRzs7O0FBRUgsc0JBQXNCO0FBQ3RCLHFDQUFrRjtBQUF6RSx1R0FBQSxZQUFZLE9BQUE7QUFBRSw2R0FBQSxrQkFBa0IsT0FBQTtBQUFFLDhHQUFBLG1CQUFtQixPQUFBO0FBRTlELGdCQUFnQjtBQUNoQiwrQkFBMkQ7QUFBbEQsbUdBQUEsV0FBVyxPQUFBO0FBQUUsNEdBQUEsb0JBQW9CLE9BQUE7QUFFMUMseUJBQXlCO0FBQ3pCLHFDQUF3RTtBQUEvRCw4R0FBQSxtQkFBbUIsT0FBQTtBQUFFLGlIQUFBLHNCQUFzQixPQUFBO0FBd0NwRCxtQ0FBbUM7QUFDbkMsdUNBQStDO0FBQy9DLGtCQUFlLDRCQUFrQixDQUFDO0FBRWxDOztHQUVHO0FBQ1UsUUFBQSxjQUFjLEdBQUc7SUFDNUIsSUFBSSxFQUFFLHlCQUF5QjtJQUMvQixPQUFPLEVBQUUsT0FBTztJQUNoQixJQUFJLEVBQUUsU0FBUztJQUNmLE9BQU8sRUFBRSxPQUFPO0lBQ2hCLFdBQVcsRUFBRSxtREFBbUQ7SUFDaEUsTUFBTSxFQUFFLHVCQUF1QjtJQUMvQixVQUFVLEVBQUUsMkNBQTJDO0lBQ3ZELFlBQVksRUFBRTtRQUNaLEVBQUUsRUFBRSxJQUFJO1FBQ1IsTUFBTSxFQUFFLElBQUk7UUFDWixPQUFPLEVBQUUsSUFBSTtRQUNiLFdBQVcsRUFBRSxJQUFJO1FBQ2pCLEtBQUssRUFBRSxJQUFJO1FBQ1gsUUFBUSxFQUFFLElBQUk7S0FDZjtDQUNGLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqIE9wZW5DbGF3IFdlYmV4IENoYW5uZWwgUGx1Z2luXG4gKlxuICogQSBjaGFubmVsIHBsdWdpbiBmb3IgaW50ZWdyYXRpbmcgQ2lzY28gV2ViZXggbWVzc2FnaW5nIHdpdGggT3BlbkNsYXcuXG4gKlxuICogQHBhY2thZ2VEb2N1bWVudGF0aW9uXG4gKi9cblxuLy8gTWFpbiBjaGFubmVsIGV4cG9ydFxuZXhwb3J0IHsgV2ViZXhDaGFubmVsLCBjcmVhdGVXZWJleENoYW5uZWwsIGNyZWF0ZUFuZEluaXRpYWxpemUgfSBmcm9tICcuL2NoYW5uZWwnO1xuXG4vLyBTZW5kZXIgZXhwb3J0XG5leHBvcnQgeyBXZWJleFNlbmRlciwgV2ViZXhBcGlSZXF1ZXN0RXJyb3IgfSBmcm9tICcuL3NlbmQnO1xuXG4vLyBXZWJob29rIGhhbmRsZXIgZXhwb3J0XG5leHBvcnQgeyBXZWJleFdlYmhvb2tIYW5kbGVyLCBXZWJob29rVmFsaWRhdGlvbkVycm9yIH0gZnJvbSAnLi93ZWJob29rJztcblxuLy8gVHlwZSBleHBvcnRzXG5leHBvcnQgdHlwZSB7XG4gIC8vIENvbmZpZ3VyYXRpb25cbiAgV2ViZXhDaGFubmVsQ29uZmlnLFxuICBEbVBvbGljeSxcblxuICAvLyBXZWJleCBBUEkgdHlwZXNcbiAgV2ViZXhQZXJzb24sXG4gIFdlYmV4Um9vbSxcbiAgV2ViZXhNZXNzYWdlLFxuICBXZWJleEF0dGFjaG1lbnQsXG4gIEFkYXB0aXZlQ2FyZCxcbiAgV2ViZXhXZWJob29rLFxuICBXZWJleFdlYmhvb2tSZXNvdXJjZSxcbiAgV2ViZXhXZWJob29rRXZlbnQsXG4gIFdlYmV4V2ViaG9va1BheWxvYWQsXG4gIFdlYmV4V2ViaG9va0RhdGEsXG5cbiAgLy8gUmVxdWVzdC9SZXNwb25zZSB0eXBlc1xuICBDcmVhdGVNZXNzYWdlUmVxdWVzdCxcbiAgQ3JlYXRlV2ViaG9va1JlcXVlc3QsXG4gIFdlYmV4QXBpRXJyb3IsXG4gIFBhZ2luYXRlZFJlc3BvbnNlLFxuXG4gIC8vIE9wZW5DbGF3IGVudmVsb3BlIHR5cGVzXG4gIE9wZW5DbGF3RW52ZWxvcGUsXG4gIE9wZW5DbGF3QXR0YWNobWVudCxcbiAgT3BlbkNsYXdPdXRib3VuZE1lc3NhZ2UsXG5cbiAgLy8gUGx1Z2luIHR5cGVzXG4gIFdlYmV4Q2hhbm5lbFBsdWdpbixcbiAgV2ViaG9va0hhbmRsZXIsXG5cbiAgLy8gSW50ZXJuYWwgdHlwZXMgKGV4cG9ydGVkIGZvciBhZHZhbmNlZCB1c2UpXG4gIFJldHJ5T3B0aW9ucyxcbiAgUmVxdWVzdE9wdGlvbnMsXG59IGZyb20gJy4vdHlwZXMnO1xuXG4vLyBEZWZhdWx0IGV4cG9ydDogZmFjdG9yeSBmdW5jdGlvblxuaW1wb3J0IHsgY3JlYXRlV2ViZXhDaGFubmVsIH0gZnJvbSAnLi9jaGFubmVsJztcbmV4cG9ydCBkZWZhdWx0IGNyZWF0ZVdlYmV4Q2hhbm5lbDtcblxuLyoqXG4gKiBQbHVnaW4gbWV0YWRhdGEgZm9yIE9wZW5DbGF3IGRpc2NvdmVyeVxuICovXG5leHBvcnQgY29uc3QgcGx1Z2luTWV0YWRhdGEgPSB7XG4gIG5hbWU6ICdAb3BlbmNsYXcvY2hhbm5lbC13ZWJleCcsXG4gIHZlcnNpb246ICcxLjAuMCcsXG4gIHR5cGU6ICdjaGFubmVsJyxcbiAgY2hhbm5lbDogJ3dlYmV4JyxcbiAgZGVzY3JpcHRpb246ICdDaXNjbyBXZWJleCBtZXNzYWdpbmcgY2hhbm5lbCBwbHVnaW4gZm9yIE9wZW5DbGF3JyxcbiAgYXV0aG9yOiAnT3BlbkNsYXcgQ29udHJpYnV0b3JzJyxcbiAgcmVwb3NpdG9yeTogJ2h0dHBzOi8vZ2l0aHViLmNvbS9vcGVuY2xhdy9jaGFubmVsLXdlYmV4JyxcbiAgY2FwYWJpbGl0aWVzOiB7XG4gICAgZG06IHRydWUsXG4gICAgZ3JvdXBzOiB0cnVlLFxuICAgIHRocmVhZHM6IHRydWUsXG4gICAgYXR0YWNobWVudHM6IHRydWUsXG4gICAgY2FyZHM6IHRydWUsXG4gICAgd2ViaG9va3M6IHRydWUsXG4gIH0sXG59O1xuIl19
package/dist/send.d.ts ADDED
@@ -0,0 +1,92 @@
1
+ /**
2
+ * Webex Message Sending Module
3
+ */
4
+ import type { WebexChannelConfig, WebexMessage, OpenClawOutboundMessage } from './types';
5
+ export declare class WebexSender {
6
+ private config;
7
+ private apiBaseUrl;
8
+ private retryOptions;
9
+ constructor(config: WebexChannelConfig);
10
+ /**
11
+ * Send a message to Webex
12
+ */
13
+ send(message: OpenClawOutboundMessage): Promise<WebexMessage>;
14
+ /**
15
+ * Send a text message to a room
16
+ */
17
+ sendToRoom(roomId: string, text: string, markdown?: string): Promise<WebexMessage>;
18
+ /**
19
+ * Send a direct message to a person by ID
20
+ */
21
+ sendDirectById(personId: string, text: string, markdown?: string): Promise<WebexMessage>;
22
+ /**
23
+ * Send a direct message to a person by email
24
+ */
25
+ sendDirectByEmail(email: string, text: string, markdown?: string): Promise<WebexMessage>;
26
+ /**
27
+ * Send a message with file attachment
28
+ */
29
+ sendWithFile(roomId: string, text: string, fileUrl: string): Promise<WebexMessage>;
30
+ /**
31
+ * Send a threaded reply
32
+ */
33
+ sendReply(roomId: string, parentId: string, text: string, markdown?: string): Promise<WebexMessage>;
34
+ /**
35
+ * Get a message by ID
36
+ */
37
+ getMessage(messageId: string): Promise<WebexMessage>;
38
+ /**
39
+ * Delete a message by ID
40
+ */
41
+ deleteMessage(messageId: string): Promise<void>;
42
+ /**
43
+ * Build a Webex message request from an OpenClaw outbound message
44
+ */
45
+ private buildMessageRequest;
46
+ /**
47
+ * Create a message via the Webex API
48
+ */
49
+ private createMessage;
50
+ /**
51
+ * Validate a message request before sending
52
+ */
53
+ private validateMessageRequest;
54
+ /**
55
+ * Make an API request with retry logic
56
+ */
57
+ private request;
58
+ /**
59
+ * Execute a single API request
60
+ */
61
+ private executeRequest;
62
+ /**
63
+ * Parse error response from Webex API
64
+ */
65
+ private parseErrorResponse;
66
+ /**
67
+ * Determine if a request should be retried
68
+ */
69
+ private shouldRetry;
70
+ /**
71
+ * Calculate backoff delay with exponential backoff and jitter
72
+ */
73
+ private calculateBackoff;
74
+ /**
75
+ * Sleep for a given number of milliseconds
76
+ */
77
+ private sleep;
78
+ }
79
+ /**
80
+ * Custom error class for Webex API errors
81
+ */
82
+ export declare class WebexApiRequestError extends Error {
83
+ readonly statusCode: number;
84
+ readonly trackingId?: string;
85
+ readonly details?: Array<{
86
+ description: string;
87
+ }>;
88
+ constructor(message: string, statusCode: number, trackingId?: string, details?: Array<{
89
+ description: string;
90
+ }>);
91
+ toJSON(): object;
92
+ }
package/dist/send.js ADDED
@@ -0,0 +1,304 @@
1
+ "use strict";
2
+ /**
3
+ * Webex Message Sending Module
4
+ */
5
+ var __importDefault = (this && this.__importDefault) || function (mod) {
6
+ return (mod && mod.__esModule) ? mod : { "default": mod };
7
+ };
8
+ Object.defineProperty(exports, "__esModule", { value: true });
9
+ exports.WebexApiRequestError = exports.WebexSender = void 0;
10
+ const node_fetch_1 = __importDefault(require("node-fetch"));
11
+ const DEFAULT_API_BASE_URL = 'https://webexapis.com/v1';
12
+ const DEFAULT_MAX_RETRIES = 3;
13
+ const DEFAULT_RETRY_DELAY_MS = 1000;
14
+ // Rate limit status codes that should trigger retry
15
+ const RETRY_STATUS_CODES = [429, 502, 503, 504];
16
+ class WebexSender {
17
+ config;
18
+ apiBaseUrl;
19
+ retryOptions;
20
+ constructor(config) {
21
+ this.config = config;
22
+ this.apiBaseUrl = config.apiBaseUrl || DEFAULT_API_BASE_URL;
23
+ this.retryOptions = {
24
+ maxRetries: config.maxRetries ?? DEFAULT_MAX_RETRIES,
25
+ retryDelayMs: config.retryDelayMs ?? DEFAULT_RETRY_DELAY_MS,
26
+ };
27
+ }
28
+ /**
29
+ * Send a message to Webex
30
+ */
31
+ async send(message) {
32
+ const request = this.buildMessageRequest(message);
33
+ return this.createMessage(request);
34
+ }
35
+ /**
36
+ * Send a text message to a room
37
+ */
38
+ async sendToRoom(roomId, text, markdown) {
39
+ return this.createMessage({
40
+ roomId,
41
+ text,
42
+ markdown,
43
+ });
44
+ }
45
+ /**
46
+ * Send a direct message to a person by ID
47
+ */
48
+ async sendDirectById(personId, text, markdown) {
49
+ return this.createMessage({
50
+ toPersonId: personId,
51
+ text,
52
+ markdown,
53
+ });
54
+ }
55
+ /**
56
+ * Send a direct message to a person by email
57
+ */
58
+ async sendDirectByEmail(email, text, markdown) {
59
+ return this.createMessage({
60
+ toPersonEmail: email,
61
+ text,
62
+ markdown,
63
+ });
64
+ }
65
+ /**
66
+ * Send a message with file attachment
67
+ */
68
+ async sendWithFile(roomId, text, fileUrl) {
69
+ return this.createMessage({
70
+ roomId,
71
+ text,
72
+ files: [fileUrl],
73
+ });
74
+ }
75
+ /**
76
+ * Send a threaded reply
77
+ */
78
+ async sendReply(roomId, parentId, text, markdown) {
79
+ return this.createMessage({
80
+ roomId,
81
+ parentId,
82
+ text,
83
+ markdown,
84
+ });
85
+ }
86
+ /**
87
+ * Get a message by ID
88
+ */
89
+ async getMessage(messageId) {
90
+ return this.request({
91
+ method: 'GET',
92
+ path: `/messages/${messageId}`,
93
+ });
94
+ }
95
+ /**
96
+ * Delete a message by ID
97
+ */
98
+ async deleteMessage(messageId) {
99
+ await this.request({
100
+ method: 'DELETE',
101
+ path: `/messages/${messageId}`,
102
+ });
103
+ }
104
+ /**
105
+ * Build a Webex message request from an OpenClaw outbound message
106
+ */
107
+ buildMessageRequest(message) {
108
+ const request = {};
109
+ // Determine target: roomId, personId, or email
110
+ const to = message.to;
111
+ if (to.includes('@')) {
112
+ request.toPersonEmail = to;
113
+ }
114
+ else if (to.startsWith('Y2lzY29zcGFyazovL3')) {
115
+ // Base64-encoded Webex IDs start with this prefix
116
+ if (to.includes('ROOM')) {
117
+ request.roomId = to;
118
+ }
119
+ else {
120
+ request.toPersonId = to;
121
+ }
122
+ }
123
+ else {
124
+ // Assume it's a roomId if not an email
125
+ request.roomId = to;
126
+ }
127
+ // Set content
128
+ if (message.content.text) {
129
+ request.text = message.content.text;
130
+ }
131
+ if (message.content.markdown) {
132
+ request.markdown = message.content.markdown;
133
+ }
134
+ if (message.content.files && message.content.files.length > 0) {
135
+ // Webex only allows one file per message
136
+ request.files = [message.content.files[0]];
137
+ }
138
+ if (message.content.card) {
139
+ request.attachments = [
140
+ {
141
+ contentType: 'application/vnd.microsoft.card.adaptive',
142
+ content: message.content.card,
143
+ },
144
+ ];
145
+ }
146
+ // Set threading
147
+ if (message.parentId) {
148
+ request.parentId = message.parentId;
149
+ }
150
+ return request;
151
+ }
152
+ /**
153
+ * Create a message via the Webex API
154
+ */
155
+ async createMessage(request) {
156
+ this.validateMessageRequest(request);
157
+ return this.request({
158
+ method: 'POST',
159
+ path: '/messages',
160
+ body: request,
161
+ });
162
+ }
163
+ /**
164
+ * Validate a message request before sending
165
+ */
166
+ validateMessageRequest(request) {
167
+ // Must have a target
168
+ if (!request.roomId && !request.toPersonId && !request.toPersonEmail) {
169
+ throw new Error('Message must have a target: roomId, toPersonId, or toPersonEmail');
170
+ }
171
+ // Must have content
172
+ if (!request.text && !request.markdown && !request.files?.length && !request.attachments?.length) {
173
+ throw new Error('Message must have content: text, markdown, files, or attachments');
174
+ }
175
+ // Text has a max size of 7439 bytes
176
+ if (request.text && Buffer.byteLength(request.text, 'utf8') > 7439) {
177
+ throw new Error('Message text exceeds maximum size of 7439 bytes');
178
+ }
179
+ }
180
+ /**
181
+ * Make an API request with retry logic
182
+ */
183
+ async request(options) {
184
+ let lastError = null;
185
+ let attempt = 0;
186
+ while (attempt <= this.retryOptions.maxRetries) {
187
+ try {
188
+ return await this.executeRequest(options);
189
+ }
190
+ catch (error) {
191
+ lastError = error;
192
+ attempt++;
193
+ if (attempt > this.retryOptions.maxRetries) {
194
+ break;
195
+ }
196
+ if (!this.shouldRetry(error, attempt)) {
197
+ break;
198
+ }
199
+ // Exponential backoff with jitter
200
+ const delay = this.calculateBackoff(attempt);
201
+ await this.sleep(delay);
202
+ }
203
+ }
204
+ throw lastError || new Error('Request failed after retries');
205
+ }
206
+ /**
207
+ * Execute a single API request
208
+ */
209
+ async executeRequest(options) {
210
+ const url = `${this.apiBaseUrl}${options.path}`;
211
+ const headers = {
212
+ 'Authorization': `Bearer ${this.config.token}`,
213
+ 'Content-Type': 'application/json',
214
+ ...options.headers,
215
+ };
216
+ const response = await (0, node_fetch_1.default)(url, {
217
+ method: options.method,
218
+ headers,
219
+ body: options.body ? JSON.stringify(options.body) : undefined,
220
+ });
221
+ if (!response.ok) {
222
+ const error = await this.parseErrorResponse(response);
223
+ throw error;
224
+ }
225
+ // DELETE requests return 204 No Content
226
+ if (response.status === 204) {
227
+ return undefined;
228
+ }
229
+ return response.json();
230
+ }
231
+ /**
232
+ * Parse error response from Webex API
233
+ */
234
+ async parseErrorResponse(response) {
235
+ let errorData = null;
236
+ try {
237
+ errorData = await response.json();
238
+ }
239
+ catch {
240
+ // Response body might not be JSON
241
+ }
242
+ const message = errorData?.message || `HTTP ${response.status}: ${response.statusText}`;
243
+ const error = new WebexApiRequestError(message, response.status, errorData?.trackingId, errorData?.errors);
244
+ return error;
245
+ }
246
+ /**
247
+ * Determine if a request should be retried
248
+ */
249
+ shouldRetry(error, attempt) {
250
+ if (error instanceof WebexApiRequestError) {
251
+ return RETRY_STATUS_CODES.includes(error.statusCode);
252
+ }
253
+ // Retry network errors
254
+ return error.message.includes('ECONNRESET') ||
255
+ error.message.includes('ETIMEDOUT') ||
256
+ error.message.includes('ENOTFOUND');
257
+ }
258
+ /**
259
+ * Calculate backoff delay with exponential backoff and jitter
260
+ */
261
+ calculateBackoff(attempt) {
262
+ const baseDelay = this.retryOptions.retryDelayMs;
263
+ const exponentialDelay = baseDelay * Math.pow(2, attempt - 1);
264
+ const jitter = Math.random() * 0.3 * exponentialDelay;
265
+ return Math.min(exponentialDelay + jitter, 30000); // Cap at 30 seconds
266
+ }
267
+ /**
268
+ * Sleep for a given number of milliseconds
269
+ */
270
+ sleep(ms) {
271
+ return new Promise(resolve => setTimeout(resolve, ms));
272
+ }
273
+ }
274
+ exports.WebexSender = WebexSender;
275
+ /**
276
+ * Custom error class for Webex API errors
277
+ */
278
+ class WebexApiRequestError extends Error {
279
+ statusCode;
280
+ trackingId;
281
+ details;
282
+ constructor(message, statusCode, trackingId, details) {
283
+ super(message);
284
+ this.name = 'WebexApiRequestError';
285
+ this.statusCode = statusCode;
286
+ this.trackingId = trackingId;
287
+ this.details = details;
288
+ // Maintains proper stack trace for where error was thrown
289
+ if (Error.captureStackTrace) {
290
+ Error.captureStackTrace(this, WebexApiRequestError);
291
+ }
292
+ }
293
+ toJSON() {
294
+ return {
295
+ name: this.name,
296
+ message: this.message,
297
+ statusCode: this.statusCode,
298
+ trackingId: this.trackingId,
299
+ details: this.details,
300
+ };
301
+ }
302
+ }
303
+ exports.WebexApiRequestError = WebexApiRequestError;
304
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"send.js","sourceRoot":"","sources":["../src/send.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Webex Message Sending Module\n */\n\nimport fetch, { Response } from 'node-fetch';\nimport type {\n  WebexChannelConfig,\n  WebexMessage,\n  CreateMessageRequest,\n  OpenClawOutboundMessage,\n  WebexApiError,\n  RetryOptions,\n  RequestOptions,\n} from './types';\n\nconst DEFAULT_API_BASE_URL = 'https://webexapis.com/v1';\nconst DEFAULT_MAX_RETRIES = 3;\nconst DEFAULT_RETRY_DELAY_MS = 1000;\n\n// Rate limit status codes that should trigger retry\nconst RETRY_STATUS_CODES = [429, 502, 503, 504];\n\nexport class WebexSender {\n  private config: WebexChannelConfig;\n  private apiBaseUrl: string;\n  private retryOptions: RetryOptions;\n\n  constructor(config: WebexChannelConfig) {\n    this.config = config;\n    this.apiBaseUrl = config.apiBaseUrl || DEFAULT_API_BASE_URL;\n    this.retryOptions = {\n      maxRetries: config.maxRetries ?? DEFAULT_MAX_RETRIES,\n      retryDelayMs: config.retryDelayMs ?? DEFAULT_RETRY_DELAY_MS,\n    };\n  }\n\n  /**\n   * Send a message to Webex\n   */\n  async send(message: OpenClawOutboundMessage): Promise<WebexMessage> {\n    const request = this.buildMessageRequest(message);\n    return this.createMessage(request);\n  }\n\n  /**\n   * Send a text message to a room\n   */\n  async sendToRoom(roomId: string, text: string, markdown?: string): Promise<WebexMessage> {\n    return this.createMessage({\n      roomId,\n      text,\n      markdown,\n    });\n  }\n\n  /**\n   * Send a direct message to a person by ID\n   */\n  async sendDirectById(personId: string, text: string, markdown?: string): Promise<WebexMessage> {\n    return this.createMessage({\n      toPersonId: personId,\n      text,\n      markdown,\n    });\n  }\n\n  /**\n   * Send a direct message to a person by email\n   */\n  async sendDirectByEmail(email: string, text: string, markdown?: string): Promise<WebexMessage> {\n    return this.createMessage({\n      toPersonEmail: email,\n      text,\n      markdown,\n    });\n  }\n\n  /**\n   * Send a message with file attachment\n   */\n  async sendWithFile(\n    roomId: string,\n    text: string,\n    fileUrl: string\n  ): Promise<WebexMessage> {\n    return this.createMessage({\n      roomId,\n      text,\n      files: [fileUrl],\n    });\n  }\n\n  /**\n   * Send a threaded reply\n   */\n  async sendReply(\n    roomId: string,\n    parentId: string,\n    text: string,\n    markdown?: string\n  ): Promise<WebexMessage> {\n    return this.createMessage({\n      roomId,\n      parentId,\n      text,\n      markdown,\n    });\n  }\n\n  /**\n   * Get a message by ID\n   */\n  async getMessage(messageId: string): Promise<WebexMessage> {\n    return this.request<WebexMessage>({\n      method: 'GET',\n      path: `/messages/${messageId}`,\n    });\n  }\n\n  /**\n   * Delete a message by ID\n   */\n  async deleteMessage(messageId: string): Promise<void> {\n    await this.request<void>({\n      method: 'DELETE',\n      path: `/messages/${messageId}`,\n    });\n  }\n\n  /**\n   * Build a Webex message request from an OpenClaw outbound message\n   */\n  private buildMessageRequest(message: OpenClawOutboundMessage): CreateMessageRequest {\n    const request: CreateMessageRequest = {};\n\n    // Determine target: roomId, personId, or email\n    const to = message.to;\n    if (to.includes('@')) {\n      request.toPersonEmail = to;\n    } else if (to.startsWith('Y2lzY29zcGFyazovL3')) {\n      // Base64-encoded Webex IDs start with this prefix\n      if (to.includes('ROOM')) {\n        request.roomId = to;\n      } else {\n        request.toPersonId = to;\n      }\n    } else {\n      // Assume it's a roomId if not an email\n      request.roomId = to;\n    }\n\n    // Set content\n    if (message.content.text) {\n      request.text = message.content.text;\n    }\n    if (message.content.markdown) {\n      request.markdown = message.content.markdown;\n    }\n    if (message.content.files && message.content.files.length > 0) {\n      // Webex only allows one file per message\n      request.files = [message.content.files[0]];\n    }\n    if (message.content.card) {\n      request.attachments = [\n        {\n          contentType: 'application/vnd.microsoft.card.adaptive',\n          content: message.content.card,\n        },\n      ];\n    }\n\n    // Set threading\n    if (message.parentId) {\n      request.parentId = message.parentId;\n    }\n\n    return request;\n  }\n\n  /**\n   * Create a message via the Webex API\n   */\n  private async createMessage(request: CreateMessageRequest): Promise<WebexMessage> {\n    this.validateMessageRequest(request);\n\n    return this.request<WebexMessage>({\n      method: 'POST',\n      path: '/messages',\n      body: request,\n    });\n  }\n\n  /**\n   * Validate a message request before sending\n   */\n  private validateMessageRequest(request: CreateMessageRequest): void {\n    // Must have a target\n    if (!request.roomId && !request.toPersonId && !request.toPersonEmail) {\n      throw new Error('Message must have a target: roomId, toPersonId, or toPersonEmail');\n    }\n\n    // Must have content\n    if (!request.text && !request.markdown && !request.files?.length && !request.attachments?.length) {\n      throw new Error('Message must have content: text, markdown, files, or attachments');\n    }\n\n    // Text has a max size of 7439 bytes\n    if (request.text && Buffer.byteLength(request.text, 'utf8') > 7439) {\n      throw new Error('Message text exceeds maximum size of 7439 bytes');\n    }\n  }\n\n  /**\n   * Make an API request with retry logic\n   */\n  private async request<T>(options: RequestOptions): Promise<T> {\n    let lastError: Error | null = null;\n    let attempt = 0;\n\n    while (attempt <= this.retryOptions.maxRetries) {\n      try {\n        return await this.executeRequest<T>(options);\n      } catch (error) {\n        lastError = error as Error;\n        attempt++;\n\n        if (attempt > this.retryOptions.maxRetries) {\n          break;\n        }\n\n        if (!this.shouldRetry(error as Error, attempt)) {\n          break;\n        }\n\n        // Exponential backoff with jitter\n        const delay = this.calculateBackoff(attempt);\n        await this.sleep(delay);\n      }\n    }\n\n    throw lastError || new Error('Request failed after retries');\n  }\n\n  /**\n   * Execute a single API request\n   */\n  private async executeRequest<T>(options: RequestOptions): Promise<T> {\n    const url = `${this.apiBaseUrl}${options.path}`;\n    const headers: Record<string, string> = {\n      'Authorization': `Bearer ${this.config.token}`,\n      'Content-Type': 'application/json',\n      ...options.headers,\n    };\n\n    const response = await fetch(url, {\n      method: options.method,\n      headers,\n      body: options.body ? JSON.stringify(options.body) : undefined,\n    });\n\n    if (!response.ok) {\n      const error = await this.parseErrorResponse(response);\n      throw error;\n    }\n\n    // DELETE requests return 204 No Content\n    if (response.status === 204) {\n      return undefined as T;\n    }\n\n    return response.json() as Promise<T>;\n  }\n\n  /**\n   * Parse error response from Webex API\n   */\n  private async parseErrorResponse(response: Response): Promise<Error> {\n    let errorData: WebexApiError | null = null;\n\n    try {\n      errorData = await response.json() as WebexApiError;\n    } catch {\n      // Response body might not be JSON\n    }\n\n    const message = errorData?.message || `HTTP ${response.status}: ${response.statusText}`;\n    const error = new WebexApiRequestError(\n      message,\n      response.status,\n      errorData?.trackingId,\n      errorData?.errors\n    );\n\n    return error;\n  }\n\n  /**\n   * Determine if a request should be retried\n   */\n  private shouldRetry(error: Error, attempt: number): boolean {\n    if (error instanceof WebexApiRequestError) {\n      return RETRY_STATUS_CODES.includes(error.statusCode);\n    }\n    // Retry network errors\n    return error.message.includes('ECONNRESET') ||\n           error.message.includes('ETIMEDOUT') ||\n           error.message.includes('ENOTFOUND');\n  }\n\n  /**\n   * Calculate backoff delay with exponential backoff and jitter\n   */\n  private calculateBackoff(attempt: number): number {\n    const baseDelay = this.retryOptions.retryDelayMs;\n    const exponentialDelay = baseDelay * Math.pow(2, attempt - 1);\n    const jitter = Math.random() * 0.3 * exponentialDelay;\n    return Math.min(exponentialDelay + jitter, 30000); // Cap at 30 seconds\n  }\n\n  /**\n   * Sleep for a given number of milliseconds\n   */\n  private sleep(ms: number): Promise<void> {\n    return new Promise(resolve => setTimeout(resolve, ms));\n  }\n}\n\n/**\n * Custom error class for Webex API errors\n */\nexport class WebexApiRequestError extends Error {\n  readonly statusCode: number;\n  readonly trackingId?: string;\n  readonly details?: Array<{ description: string }>;\n\n  constructor(\n    message: string,\n    statusCode: number,\n    trackingId?: string,\n    details?: Array<{ description: string }>\n  ) {\n    super(message);\n    this.name = 'WebexApiRequestError';\n    this.statusCode = statusCode;\n    this.trackingId = trackingId;\n    this.details = details;\n\n    // Maintains proper stack trace for where error was thrown\n    if (Error.captureStackTrace) {\n      Error.captureStackTrace(this, WebexApiRequestError);\n    }\n  }\n\n  toJSON(): object {\n    return {\n      name: this.name,\n      message: this.message,\n      statusCode: this.statusCode,\n      trackingId: this.trackingId,\n      details: this.details,\n    };\n  }\n}\n"]}