@jimiford/webex 0.1.1
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/LICENSE +21 -0
- package/README.md +314 -0
- package/dist/channel-plugin.d.ts +18 -0
- package/dist/channel-plugin.js +410 -0
- package/dist/channel.d.ts +98 -0
- package/dist/channel.js +224 -0
- package/dist/index.d.ts +15 -0
- package/dist/index.js +32 -0
- package/dist/plugin.d.ts +15 -0
- package/dist/plugin.js +23 -0
- package/dist/send.d.ts +92 -0
- package/dist/send.js +304 -0
- package/dist/types.d.ts +223 -0
- package/dist/types.js +6 -0
- package/dist/webhook.d.ts +64 -0
- package/dist/webhook.js +297 -0
- package/openclaw.plugin.json +33 -0
- package/package.json +71 -0
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"]}
|
package/dist/types.d.ts
ADDED
|
@@ -0,0 +1,223 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Webex Channel Plugin Types
|
|
3
|
+
*/
|
|
4
|
+
export type DmPolicy = 'allow' | 'deny' | 'allowlisted' | 'allowlist' | 'pairing';
|
|
5
|
+
export interface WebexChannelConfig {
|
|
6
|
+
/** Webex Bot access token */
|
|
7
|
+
token: string;
|
|
8
|
+
/** Public URL where webhooks will be received */
|
|
9
|
+
webhookUrl: string;
|
|
10
|
+
/** Policy for handling direct messages */
|
|
11
|
+
dmPolicy: DmPolicy;
|
|
12
|
+
/** List of allowed person IDs or emails (used when dmPolicy is 'allowlisted') */
|
|
13
|
+
allowFrom?: string[];
|
|
14
|
+
/** Webhook secret for payload verification */
|
|
15
|
+
webhookSecret?: string;
|
|
16
|
+
/** Base URL for Webex API (defaults to https://webexapis.com/v1) */
|
|
17
|
+
apiBaseUrl?: string;
|
|
18
|
+
/** Maximum retry attempts for failed requests */
|
|
19
|
+
maxRetries?: number;
|
|
20
|
+
/** Retry delay in milliseconds */
|
|
21
|
+
retryDelayMs?: number;
|
|
22
|
+
}
|
|
23
|
+
export interface WebexPerson {
|
|
24
|
+
id: string;
|
|
25
|
+
emails: string[];
|
|
26
|
+
displayName: string;
|
|
27
|
+
nickName?: string;
|
|
28
|
+
firstName?: string;
|
|
29
|
+
lastName?: string;
|
|
30
|
+
avatar?: string;
|
|
31
|
+
orgId: string;
|
|
32
|
+
created: string;
|
|
33
|
+
lastModified?: string;
|
|
34
|
+
type: 'person' | 'bot';
|
|
35
|
+
}
|
|
36
|
+
export interface WebexRoom {
|
|
37
|
+
id: string;
|
|
38
|
+
title: string;
|
|
39
|
+
type: 'direct' | 'group';
|
|
40
|
+
isLocked: boolean;
|
|
41
|
+
teamId?: string;
|
|
42
|
+
lastActivity: string;
|
|
43
|
+
creatorId: string;
|
|
44
|
+
created: string;
|
|
45
|
+
ownerId?: string;
|
|
46
|
+
}
|
|
47
|
+
export interface WebexMessage {
|
|
48
|
+
id: string;
|
|
49
|
+
roomId: string;
|
|
50
|
+
roomType: 'direct' | 'group';
|
|
51
|
+
toPersonId?: string;
|
|
52
|
+
toPersonEmail?: string;
|
|
53
|
+
text?: string;
|
|
54
|
+
markdown?: string;
|
|
55
|
+
html?: string;
|
|
56
|
+
files?: string[];
|
|
57
|
+
personId: string;
|
|
58
|
+
personEmail: string;
|
|
59
|
+
mentionedPeople?: string[];
|
|
60
|
+
mentionedGroups?: string[];
|
|
61
|
+
attachments?: WebexAttachment[];
|
|
62
|
+
created: string;
|
|
63
|
+
updated?: string;
|
|
64
|
+
parentId?: string;
|
|
65
|
+
}
|
|
66
|
+
export interface WebexAttachment {
|
|
67
|
+
contentType: 'application/vnd.microsoft.card.adaptive';
|
|
68
|
+
content: AdaptiveCard;
|
|
69
|
+
}
|
|
70
|
+
export interface AdaptiveCard {
|
|
71
|
+
type: 'AdaptiveCard';
|
|
72
|
+
version: string;
|
|
73
|
+
body: unknown[];
|
|
74
|
+
actions?: unknown[];
|
|
75
|
+
}
|
|
76
|
+
export interface WebexWebhook {
|
|
77
|
+
id: string;
|
|
78
|
+
name: string;
|
|
79
|
+
targetUrl: string;
|
|
80
|
+
resource: WebexWebhookResource;
|
|
81
|
+
event: WebexWebhookEvent;
|
|
82
|
+
filter?: string;
|
|
83
|
+
secret?: string;
|
|
84
|
+
status: 'active' | 'inactive';
|
|
85
|
+
created: string;
|
|
86
|
+
orgId: string;
|
|
87
|
+
createdBy: string;
|
|
88
|
+
appId: string;
|
|
89
|
+
ownedBy: 'creator' | 'org';
|
|
90
|
+
}
|
|
91
|
+
export type WebexWebhookResource = 'messages' | 'memberships' | 'rooms' | 'attachmentActions' | 'meetings' | 'recordings';
|
|
92
|
+
export type WebexWebhookEvent = 'created' | 'updated' | 'deleted' | 'started' | 'ended';
|
|
93
|
+
export interface WebexWebhookPayload {
|
|
94
|
+
id: string;
|
|
95
|
+
name: string;
|
|
96
|
+
targetUrl: string;
|
|
97
|
+
resource: WebexWebhookResource;
|
|
98
|
+
event: WebexWebhookEvent;
|
|
99
|
+
filter?: string;
|
|
100
|
+
orgId: string;
|
|
101
|
+
createdBy: string;
|
|
102
|
+
appId: string;
|
|
103
|
+
ownedBy: string;
|
|
104
|
+
status: string;
|
|
105
|
+
created: string;
|
|
106
|
+
actorId: string;
|
|
107
|
+
data: WebexWebhookData;
|
|
108
|
+
}
|
|
109
|
+
export interface WebexWebhookData {
|
|
110
|
+
id: string;
|
|
111
|
+
roomId: string;
|
|
112
|
+
roomType: 'direct' | 'group';
|
|
113
|
+
personId: string;
|
|
114
|
+
personEmail: string;
|
|
115
|
+
created: string;
|
|
116
|
+
mentionedPeople?: string[];
|
|
117
|
+
mentionedGroups?: string[];
|
|
118
|
+
files?: string[];
|
|
119
|
+
}
|
|
120
|
+
export interface CreateMessageRequest {
|
|
121
|
+
roomId?: string;
|
|
122
|
+
toPersonId?: string;
|
|
123
|
+
toPersonEmail?: string;
|
|
124
|
+
text?: string;
|
|
125
|
+
markdown?: string;
|
|
126
|
+
files?: string[];
|
|
127
|
+
attachments?: WebexAttachment[];
|
|
128
|
+
parentId?: string;
|
|
129
|
+
}
|
|
130
|
+
export interface CreateWebhookRequest {
|
|
131
|
+
name: string;
|
|
132
|
+
targetUrl: string;
|
|
133
|
+
resource: WebexWebhookResource;
|
|
134
|
+
event: WebexWebhookEvent;
|
|
135
|
+
filter?: string;
|
|
136
|
+
secret?: string;
|
|
137
|
+
}
|
|
138
|
+
export interface WebexApiError {
|
|
139
|
+
message: string;
|
|
140
|
+
errors?: Array<{
|
|
141
|
+
description: string;
|
|
142
|
+
}>;
|
|
143
|
+
trackingId: string;
|
|
144
|
+
}
|
|
145
|
+
export interface PaginatedResponse<T> {
|
|
146
|
+
items: T[];
|
|
147
|
+
}
|
|
148
|
+
export interface OpenClawEnvelope {
|
|
149
|
+
/** Unique message identifier */
|
|
150
|
+
id: string;
|
|
151
|
+
/** Channel identifier */
|
|
152
|
+
channel: 'webex';
|
|
153
|
+
/** Conversation/thread identifier */
|
|
154
|
+
conversationId: string;
|
|
155
|
+
/** Message author information */
|
|
156
|
+
author: {
|
|
157
|
+
id: string;
|
|
158
|
+
email?: string;
|
|
159
|
+
displayName?: string;
|
|
160
|
+
isBot: boolean;
|
|
161
|
+
};
|
|
162
|
+
/** Message content */
|
|
163
|
+
content: {
|
|
164
|
+
text?: string;
|
|
165
|
+
markdown?: string;
|
|
166
|
+
attachments?: OpenClawAttachment[];
|
|
167
|
+
};
|
|
168
|
+
/** Message metadata */
|
|
169
|
+
metadata: {
|
|
170
|
+
roomType: 'direct' | 'group';
|
|
171
|
+
roomId: string;
|
|
172
|
+
timestamp: string;
|
|
173
|
+
mentions?: string[];
|
|
174
|
+
parentId?: string;
|
|
175
|
+
raw: WebexMessage;
|
|
176
|
+
};
|
|
177
|
+
}
|
|
178
|
+
export interface OpenClawAttachment {
|
|
179
|
+
type: 'file' | 'card';
|
|
180
|
+
url?: string;
|
|
181
|
+
content?: unknown;
|
|
182
|
+
}
|
|
183
|
+
export interface OpenClawOutboundMessage {
|
|
184
|
+
/** Target conversation ID (roomId) or person ID/email for DMs */
|
|
185
|
+
to: string;
|
|
186
|
+
/** Message content */
|
|
187
|
+
content: {
|
|
188
|
+
text?: string;
|
|
189
|
+
markdown?: string;
|
|
190
|
+
files?: string[];
|
|
191
|
+
card?: AdaptiveCard;
|
|
192
|
+
};
|
|
193
|
+
/** Optional parent message ID for threading */
|
|
194
|
+
parentId?: string;
|
|
195
|
+
}
|
|
196
|
+
export interface WebexChannelPlugin {
|
|
197
|
+
name: string;
|
|
198
|
+
version: string;
|
|
199
|
+
/** Initialize the channel with configuration */
|
|
200
|
+
initialize(config: WebexChannelConfig): Promise<void>;
|
|
201
|
+
/** Send a message */
|
|
202
|
+
send(message: OpenClawOutboundMessage): Promise<WebexMessage>;
|
|
203
|
+
/** Handle incoming webhook */
|
|
204
|
+
handleWebhook(payload: WebexWebhookPayload, signature?: string): Promise<OpenClawEnvelope | null>;
|
|
205
|
+
/** Register webhooks with Webex */
|
|
206
|
+
registerWebhooks(): Promise<WebexWebhook[]>;
|
|
207
|
+
/** Cleanup and shutdown */
|
|
208
|
+
shutdown(): Promise<void>;
|
|
209
|
+
}
|
|
210
|
+
export interface WebhookHandler {
|
|
211
|
+
(envelope: OpenClawEnvelope): Promise<void> | void;
|
|
212
|
+
}
|
|
213
|
+
export interface RetryOptions {
|
|
214
|
+
maxRetries: number;
|
|
215
|
+
retryDelayMs: number;
|
|
216
|
+
shouldRetry?: (error: Error, attempt: number) => boolean;
|
|
217
|
+
}
|
|
218
|
+
export interface RequestOptions {
|
|
219
|
+
method: 'GET' | 'POST' | 'PUT' | 'DELETE';
|
|
220
|
+
path: string;
|
|
221
|
+
body?: unknown;
|
|
222
|
+
headers?: Record<string, string>;
|
|
223
|
+
}
|
package/dist/types.js
ADDED
|
@@ -0,0 +1,6 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
/**
|
|
3
|
+
* Webex Channel Plugin Types
|
|
4
|
+
*/
|
|
5
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
6
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"types.js","sourceRoot":"","sources":["../src/types.ts"],"names":[],"mappings":";AAAA;;GAEG","sourcesContent":["/**\n * Webex Channel Plugin Types\n */\n\n// ============================================================================\n// Configuration Types\n// ============================================================================\n\nexport type DmPolicy = 'allow' | 'deny' | 'allowlisted' | 'allowlist' | 'pairing';\n\nexport interface WebexChannelConfig {\n  /** Webex Bot access token */\n  token: string;\n\n  /** Public URL where webhooks will be received */\n  webhookUrl: string;\n\n  /** Policy for handling direct messages */\n  dmPolicy: DmPolicy;\n\n  /** List of allowed person IDs or emails (used when dmPolicy is 'allowlisted') */\n  allowFrom?: string[];\n\n  /** Webhook secret for payload verification */\n  webhookSecret?: string;\n\n  /** Base URL for Webex API (defaults to https://webexapis.com/v1) */\n  apiBaseUrl?: string;\n\n  /** Maximum retry attempts for failed requests */\n  maxRetries?: number;\n\n  /** Retry delay in milliseconds */\n  retryDelayMs?: number;\n}\n\n// ============================================================================\n// Webex API Types\n// ============================================================================\n\nexport interface WebexPerson {\n  id: string;\n  emails: string[];\n  displayName: string;\n  nickName?: string;\n  firstName?: string;\n  lastName?: string;\n  avatar?: string;\n  orgId: string;\n  created: string;\n  lastModified?: string;\n  type: 'person' | 'bot';\n}\n\nexport interface WebexRoom {\n  id: string;\n  title: string;\n  type: 'direct' | 'group';\n  isLocked: boolean;\n  teamId?: string;\n  lastActivity: string;\n  creatorId: string;\n  created: string;\n  ownerId?: string;\n}\n\nexport interface WebexMessage {\n  id: string;\n  roomId: string;\n  roomType: 'direct' | 'group';\n  toPersonId?: string;\n  toPersonEmail?: string;\n  text?: string;\n  markdown?: string;\n  html?: string;\n  files?: string[];\n  personId: string;\n  personEmail: string;\n  mentionedPeople?: string[];\n  mentionedGroups?: string[];\n  attachments?: WebexAttachment[];\n  created: string;\n  updated?: string;\n  parentId?: string;\n}\n\nexport interface WebexAttachment {\n  contentType: 'application/vnd.microsoft.card.adaptive';\n  content: AdaptiveCard;\n}\n\nexport interface AdaptiveCard {\n  type: 'AdaptiveCard';\n  version: string;\n  body: unknown[];\n  actions?: unknown[];\n}\n\nexport interface WebexWebhook {\n  id: string;\n  name: string;\n  targetUrl: string;\n  resource: WebexWebhookResource;\n  event: WebexWebhookEvent;\n  filter?: string;\n  secret?: string;\n  status: 'active' | 'inactive';\n  created: string;\n  orgId: string;\n  createdBy: string;\n  appId: string;\n  ownedBy: 'creator' | 'org';\n}\n\nexport type WebexWebhookResource =\n  | 'messages'\n  | 'memberships'\n  | 'rooms'\n  | 'attachmentActions'\n  | 'meetings'\n  | 'recordings';\n\nexport type WebexWebhookEvent =\n  | 'created'\n  | 'updated'\n  | 'deleted'\n  | 'started'\n  | 'ended';\n\nexport interface WebexWebhookPayload {\n  id: string;\n  name: string;\n  targetUrl: string;\n  resource: WebexWebhookResource;\n  event: WebexWebhookEvent;\n  filter?: string;\n  orgId: string;\n  createdBy: string;\n  appId: string;\n  ownedBy: string;\n  status: string;\n  created: string;\n  actorId: string;\n  data: WebexWebhookData;\n}\n\nexport interface WebexWebhookData {\n  id: string;\n  roomId: string;\n  roomType: 'direct' | 'group';\n  personId: string;\n  personEmail: string;\n  created: string;\n  mentionedPeople?: string[];\n  mentionedGroups?: string[];\n  files?: string[];\n}\n\n// ============================================================================\n// API Request/Response Types\n// ============================================================================\n\nexport interface CreateMessageRequest {\n  roomId?: string;\n  toPersonId?: string;\n  toPersonEmail?: string;\n  text?: string;\n  markdown?: string;\n  files?: string[];\n  attachments?: WebexAttachment[];\n  parentId?: string;\n}\n\nexport interface CreateWebhookRequest {\n  name: string;\n  targetUrl: string;\n  resource: WebexWebhookResource;\n  event: WebexWebhookEvent;\n  filter?: string;\n  secret?: string;\n}\n\nexport interface WebexApiError {\n  message: string;\n  errors?: Array<{\n    description: string;\n  }>;\n  trackingId: string;\n}\n\nexport interface PaginatedResponse<T> {\n  items: T[];\n}\n\n// ============================================================================\n// OpenClaw Envelope Types\n// ============================================================================\n\nexport interface OpenClawEnvelope {\n  /** Unique message identifier */\n  id: string;\n\n  /** Channel identifier */\n  channel: 'webex';\n\n  /** Conversation/thread identifier */\n  conversationId: string;\n\n  /** Message author information */\n  author: {\n    id: string;\n    email?: string;\n    displayName?: string;\n    isBot: boolean;\n  };\n\n  /** Message content */\n  content: {\n    text?: string;\n    markdown?: string;\n    attachments?: OpenClawAttachment[];\n  };\n\n  /** Message metadata */\n  metadata: {\n    roomType: 'direct' | 'group';\n    roomId: string;\n    timestamp: string;\n    mentions?: string[];\n    parentId?: string;\n    raw: WebexMessage;\n  };\n}\n\nexport interface OpenClawAttachment {\n  type: 'file' | 'card';\n  url?: string;\n  content?: unknown;\n}\n\nexport interface OpenClawOutboundMessage {\n  /** Target conversation ID (roomId) or person ID/email for DMs */\n  to: string;\n\n  /** Message content */\n  content: {\n    text?: string;\n    markdown?: string;\n    files?: string[];\n    card?: AdaptiveCard;\n  };\n\n  /** Optional parent message ID for threading */\n  parentId?: string;\n}\n\n// ============================================================================\n// Plugin Types\n// ============================================================================\n\nexport interface WebexChannelPlugin {\n  name: string;\n  version: string;\n\n  /** Initialize the channel with configuration */\n  initialize(config: WebexChannelConfig): Promise<void>;\n\n  /** Send a message */\n  send(message: OpenClawOutboundMessage): Promise<WebexMessage>;\n\n  /** Handle incoming webhook */\n  handleWebhook(payload: WebexWebhookPayload, signature?: string): Promise<OpenClawEnvelope | null>;\n\n  /** Register webhooks with Webex */\n  registerWebhooks(): Promise<WebexWebhook[]>;\n\n  /** Cleanup and shutdown */\n  shutdown(): Promise<void>;\n}\n\nexport interface WebhookHandler {\n  (envelope: OpenClawEnvelope): Promise<void> | void;\n}\n\n// ============================================================================\n// Internal Types\n// ============================================================================\n\nexport interface RetryOptions {\n  maxRetries: number;\n  retryDelayMs: number;\n  shouldRetry?: (error: Error, attempt: number) => boolean;\n}\n\nexport interface RequestOptions {\n  method: 'GET' | 'POST' | 'PUT' | 'DELETE';\n  path: string;\n  body?: unknown;\n  headers?: Record<string, string>;\n}\n"]}
|
|
@@ -0,0 +1,64 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Webex Webhook Handler Module
|
|
3
|
+
*/
|
|
4
|
+
import type { WebexChannelConfig, WebexWebhookPayload, WebexWebhook, CreateWebhookRequest, OpenClawEnvelope } from './types';
|
|
5
|
+
export declare class WebexWebhookHandler {
|
|
6
|
+
private config;
|
|
7
|
+
private apiBaseUrl;
|
|
8
|
+
private botId;
|
|
9
|
+
constructor(config: WebexChannelConfig);
|
|
10
|
+
/**
|
|
11
|
+
* Initialize the webhook handler (fetch bot info)
|
|
12
|
+
*/
|
|
13
|
+
initialize(): Promise<void>;
|
|
14
|
+
/**
|
|
15
|
+
* Handle an incoming webhook request
|
|
16
|
+
*/
|
|
17
|
+
handleWebhook(payload: WebexWebhookPayload, signature?: string): Promise<OpenClawEnvelope | null>;
|
|
18
|
+
/**
|
|
19
|
+
* Verify webhook signature using HMAC-SHA1
|
|
20
|
+
*/
|
|
21
|
+
verifySignature(payload: WebexWebhookPayload, signature: string): boolean;
|
|
22
|
+
/**
|
|
23
|
+
* Check if the sender is allowed based on DM policy
|
|
24
|
+
*/
|
|
25
|
+
private isAllowedSender;
|
|
26
|
+
/**
|
|
27
|
+
* Fetch full message details from Webex API
|
|
28
|
+
*/
|
|
29
|
+
private fetchMessage;
|
|
30
|
+
/**
|
|
31
|
+
* Normalize a Webex message to OpenClaw envelope format
|
|
32
|
+
*/
|
|
33
|
+
private normalizeMessage;
|
|
34
|
+
/**
|
|
35
|
+
* Register webhooks with Webex
|
|
36
|
+
*/
|
|
37
|
+
registerWebhooks(): Promise<WebexWebhook[]>;
|
|
38
|
+
/**
|
|
39
|
+
* List all webhooks
|
|
40
|
+
*/
|
|
41
|
+
listWebhooks(): Promise<WebexWebhook[]>;
|
|
42
|
+
/**
|
|
43
|
+
* Create a webhook
|
|
44
|
+
*/
|
|
45
|
+
createWebhook(request: CreateWebhookRequest): Promise<WebexWebhook>;
|
|
46
|
+
/**
|
|
47
|
+
* Delete a webhook
|
|
48
|
+
*/
|
|
49
|
+
deleteWebhook(webhookId: string): Promise<void>;
|
|
50
|
+
/**
|
|
51
|
+
* Get bot information
|
|
52
|
+
*/
|
|
53
|
+
private getBotInfo;
|
|
54
|
+
/**
|
|
55
|
+
* Get the bot ID (after initialization)
|
|
56
|
+
*/
|
|
57
|
+
getBotId(): string | null;
|
|
58
|
+
}
|
|
59
|
+
/**
|
|
60
|
+
* Custom error for webhook validation failures
|
|
61
|
+
*/
|
|
62
|
+
export declare class WebhookValidationError extends Error {
|
|
63
|
+
constructor(message: string);
|
|
64
|
+
}
|