@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/channel.js
ADDED
|
@@ -0,0 +1,224 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
/**
|
|
3
|
+
* Webex Channel - Main Channel Logic
|
|
4
|
+
*/
|
|
5
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
6
|
+
exports.WebexChannel = void 0;
|
|
7
|
+
exports.createWebexChannel = createWebexChannel;
|
|
8
|
+
exports.createAndInitialize = createAndInitialize;
|
|
9
|
+
const send_1 = require("./send");
|
|
10
|
+
const webhook_1 = require("./webhook");
|
|
11
|
+
/**
|
|
12
|
+
* Default configuration values
|
|
13
|
+
*/
|
|
14
|
+
const DEFAULT_CONFIG = {
|
|
15
|
+
dmPolicy: 'allow',
|
|
16
|
+
apiBaseUrl: 'https://webexapis.com/v1',
|
|
17
|
+
maxRetries: 3,
|
|
18
|
+
retryDelayMs: 1000,
|
|
19
|
+
};
|
|
20
|
+
/**
|
|
21
|
+
* WebexChannel implements the OpenClaw channel plugin interface for Cisco Webex
|
|
22
|
+
*/
|
|
23
|
+
class WebexChannel {
|
|
24
|
+
name = 'webex';
|
|
25
|
+
version = '1.0.0';
|
|
26
|
+
config = null;
|
|
27
|
+
sender = null;
|
|
28
|
+
webhookHandler = null;
|
|
29
|
+
messageHandlers = [];
|
|
30
|
+
initialized = false;
|
|
31
|
+
/**
|
|
32
|
+
* Initialize the channel with configuration
|
|
33
|
+
*/
|
|
34
|
+
async initialize(config) {
|
|
35
|
+
// Validate required config
|
|
36
|
+
this.validateConfig(config);
|
|
37
|
+
// Merge with defaults
|
|
38
|
+
this.config = {
|
|
39
|
+
...DEFAULT_CONFIG,
|
|
40
|
+
...config,
|
|
41
|
+
};
|
|
42
|
+
// Initialize sender
|
|
43
|
+
this.sender = new send_1.WebexSender(this.config);
|
|
44
|
+
// Initialize webhook handler
|
|
45
|
+
this.webhookHandler = new webhook_1.WebexWebhookHandler(this.config);
|
|
46
|
+
await this.webhookHandler.initialize();
|
|
47
|
+
this.initialized = true;
|
|
48
|
+
}
|
|
49
|
+
/**
|
|
50
|
+
* Validate configuration
|
|
51
|
+
*/
|
|
52
|
+
validateConfig(config) {
|
|
53
|
+
if (!config.token) {
|
|
54
|
+
throw new Error('Webex channel config requires a token');
|
|
55
|
+
}
|
|
56
|
+
if (!config.webhookUrl) {
|
|
57
|
+
throw new Error('Webex channel config requires a webhookUrl');
|
|
58
|
+
}
|
|
59
|
+
if (!config.dmPolicy) {
|
|
60
|
+
throw new Error('Webex channel config requires a dmPolicy');
|
|
61
|
+
}
|
|
62
|
+
if (config.dmPolicy === 'allowlisted' && (!config.allowFrom || config.allowFrom.length === 0)) {
|
|
63
|
+
throw new Error('Webex channel config requires allowFrom when dmPolicy is "allowlisted"');
|
|
64
|
+
}
|
|
65
|
+
// Validate webhook URL format
|
|
66
|
+
try {
|
|
67
|
+
new URL(config.webhookUrl);
|
|
68
|
+
}
|
|
69
|
+
catch {
|
|
70
|
+
throw new Error('Webex channel config webhookUrl must be a valid URL');
|
|
71
|
+
}
|
|
72
|
+
}
|
|
73
|
+
/**
|
|
74
|
+
* Ensure the channel is initialized
|
|
75
|
+
*/
|
|
76
|
+
ensureInitialized() {
|
|
77
|
+
if (!this.initialized || !this.config || !this.sender || !this.webhookHandler) {
|
|
78
|
+
throw new Error('Webex channel is not initialized. Call initialize() first.');
|
|
79
|
+
}
|
|
80
|
+
}
|
|
81
|
+
/**
|
|
82
|
+
* Send a message
|
|
83
|
+
*/
|
|
84
|
+
async send(message) {
|
|
85
|
+
this.ensureInitialized();
|
|
86
|
+
return this.sender.send(message);
|
|
87
|
+
}
|
|
88
|
+
/**
|
|
89
|
+
* Send a simple text message to a room
|
|
90
|
+
*/
|
|
91
|
+
async sendText(roomId, text) {
|
|
92
|
+
return this.send({
|
|
93
|
+
to: roomId,
|
|
94
|
+
content: { text },
|
|
95
|
+
});
|
|
96
|
+
}
|
|
97
|
+
/**
|
|
98
|
+
* Send a markdown message to a room
|
|
99
|
+
*/
|
|
100
|
+
async sendMarkdown(roomId, markdown) {
|
|
101
|
+
return this.send({
|
|
102
|
+
to: roomId,
|
|
103
|
+
content: { markdown },
|
|
104
|
+
});
|
|
105
|
+
}
|
|
106
|
+
/**
|
|
107
|
+
* Send a direct message to a person
|
|
108
|
+
*/
|
|
109
|
+
async sendDirect(personIdOrEmail, text) {
|
|
110
|
+
return this.send({
|
|
111
|
+
to: personIdOrEmail,
|
|
112
|
+
content: { text },
|
|
113
|
+
});
|
|
114
|
+
}
|
|
115
|
+
/**
|
|
116
|
+
* Reply to a message in a thread
|
|
117
|
+
*/
|
|
118
|
+
async reply(roomId, parentId, text) {
|
|
119
|
+
return this.send({
|
|
120
|
+
to: roomId,
|
|
121
|
+
content: { text },
|
|
122
|
+
parentId,
|
|
123
|
+
});
|
|
124
|
+
}
|
|
125
|
+
/**
|
|
126
|
+
* Handle incoming webhook
|
|
127
|
+
*/
|
|
128
|
+
async handleWebhook(payload, signature) {
|
|
129
|
+
this.ensureInitialized();
|
|
130
|
+
const envelope = await this.webhookHandler.handleWebhook(payload, signature);
|
|
131
|
+
if (envelope) {
|
|
132
|
+
// Notify all registered handlers
|
|
133
|
+
await this.notifyHandlers(envelope);
|
|
134
|
+
}
|
|
135
|
+
return envelope;
|
|
136
|
+
}
|
|
137
|
+
/**
|
|
138
|
+
* Register a message handler
|
|
139
|
+
*/
|
|
140
|
+
onMessage(handler) {
|
|
141
|
+
this.messageHandlers.push(handler);
|
|
142
|
+
}
|
|
143
|
+
/**
|
|
144
|
+
* Remove a message handler
|
|
145
|
+
*/
|
|
146
|
+
offMessage(handler) {
|
|
147
|
+
const index = this.messageHandlers.indexOf(handler);
|
|
148
|
+
if (index !== -1) {
|
|
149
|
+
this.messageHandlers.splice(index, 1);
|
|
150
|
+
}
|
|
151
|
+
}
|
|
152
|
+
/**
|
|
153
|
+
* Notify all registered handlers of a new message
|
|
154
|
+
*/
|
|
155
|
+
async notifyHandlers(envelope) {
|
|
156
|
+
for (const handler of this.messageHandlers) {
|
|
157
|
+
try {
|
|
158
|
+
await handler(envelope);
|
|
159
|
+
}
|
|
160
|
+
catch (error) {
|
|
161
|
+
console.error('Error in message handler:', error);
|
|
162
|
+
}
|
|
163
|
+
}
|
|
164
|
+
}
|
|
165
|
+
/**
|
|
166
|
+
* Register webhooks with Webex
|
|
167
|
+
*/
|
|
168
|
+
async registerWebhooks() {
|
|
169
|
+
this.ensureInitialized();
|
|
170
|
+
return this.webhookHandler.registerWebhooks();
|
|
171
|
+
}
|
|
172
|
+
/**
|
|
173
|
+
* Get the sender instance for advanced operations
|
|
174
|
+
*/
|
|
175
|
+
getSender() {
|
|
176
|
+
this.ensureInitialized();
|
|
177
|
+
return this.sender;
|
|
178
|
+
}
|
|
179
|
+
/**
|
|
180
|
+
* Get the webhook handler instance for advanced operations
|
|
181
|
+
*/
|
|
182
|
+
getWebhookHandler() {
|
|
183
|
+
this.ensureInitialized();
|
|
184
|
+
return this.webhookHandler;
|
|
185
|
+
}
|
|
186
|
+
/**
|
|
187
|
+
* Get the current configuration
|
|
188
|
+
*/
|
|
189
|
+
getConfig() {
|
|
190
|
+
return this.config;
|
|
191
|
+
}
|
|
192
|
+
/**
|
|
193
|
+
* Check if the channel is initialized
|
|
194
|
+
*/
|
|
195
|
+
isInitialized() {
|
|
196
|
+
return this.initialized;
|
|
197
|
+
}
|
|
198
|
+
/**
|
|
199
|
+
* Cleanup and shutdown
|
|
200
|
+
*/
|
|
201
|
+
async shutdown() {
|
|
202
|
+
this.messageHandlers = [];
|
|
203
|
+
this.sender = null;
|
|
204
|
+
this.webhookHandler = null;
|
|
205
|
+
this.config = null;
|
|
206
|
+
this.initialized = false;
|
|
207
|
+
}
|
|
208
|
+
}
|
|
209
|
+
exports.WebexChannel = WebexChannel;
|
|
210
|
+
/**
|
|
211
|
+
* Create a new Webex channel instance
|
|
212
|
+
*/
|
|
213
|
+
function createWebexChannel() {
|
|
214
|
+
return new WebexChannel();
|
|
215
|
+
}
|
|
216
|
+
/**
|
|
217
|
+
* Create and initialize a Webex channel with config
|
|
218
|
+
*/
|
|
219
|
+
async function createAndInitialize(config) {
|
|
220
|
+
const channel = createWebexChannel();
|
|
221
|
+
await channel.initialize(config);
|
|
222
|
+
return channel;
|
|
223
|
+
}
|
|
224
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"channel.js","sourceRoot":"","sources":["../src/channel.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Webex Channel - Main Channel Logic\n */\n\nimport type {\n  WebexChannelConfig,\n  WebexChannelPlugin,\n  WebexMessage,\n  WebexWebhook,\n  WebexWebhookPayload,\n  OpenClawEnvelope,\n  OpenClawOutboundMessage,\n  WebhookHandler,\n} from './types';\nimport { WebexSender } from './send';\nimport { WebexWebhookHandler } from './webhook';\n\n/**\n * Default configuration values\n */\nconst DEFAULT_CONFIG: Partial<WebexChannelConfig> = {\n  dmPolicy: 'allow',\n  apiBaseUrl: 'https://webexapis.com/v1',\n  maxRetries: 3,\n  retryDelayMs: 1000,\n};\n\n/**\n * WebexChannel implements the OpenClaw channel plugin interface for Cisco Webex\n */\nexport class WebexChannel implements WebexChannelPlugin {\n  readonly name = 'webex';\n  readonly version = '1.0.0';\n\n  private config: WebexChannelConfig | null = null;\n  private sender: WebexSender | null = null;\n  private webhookHandler: WebexWebhookHandler | null = null;\n  private messageHandlers: WebhookHandler[] = [];\n  private initialized = false;\n\n  /**\n   * Initialize the channel with configuration\n   */\n  async initialize(config: WebexChannelConfig): Promise<void> {\n    // Validate required config\n    this.validateConfig(config);\n\n    // Merge with defaults\n    this.config = {\n      ...DEFAULT_CONFIG,\n      ...config,\n    } as WebexChannelConfig;\n\n    // Initialize sender\n    this.sender = new WebexSender(this.config);\n\n    // Initialize webhook handler\n    this.webhookHandler = new WebexWebhookHandler(this.config);\n    await this.webhookHandler.initialize();\n\n    this.initialized = true;\n  }\n\n  /**\n   * Validate configuration\n   */\n  private validateConfig(config: WebexChannelConfig): void {\n    if (!config.token) {\n      throw new Error('Webex channel config requires a token');\n    }\n    if (!config.webhookUrl) {\n      throw new Error('Webex channel config requires a webhookUrl');\n    }\n    if (!config.dmPolicy) {\n      throw new Error('Webex channel config requires a dmPolicy');\n    }\n    if (config.dmPolicy === 'allowlisted' && (!config.allowFrom || config.allowFrom.length === 0)) {\n      throw new Error('Webex channel config requires allowFrom when dmPolicy is \"allowlisted\"');\n    }\n\n    // Validate webhook URL format\n    try {\n      new URL(config.webhookUrl);\n    } catch {\n      throw new Error('Webex channel config webhookUrl must be a valid URL');\n    }\n  }\n\n  /**\n   * Ensure the channel is initialized\n   */\n  private ensureInitialized(): void {\n    if (!this.initialized || !this.config || !this.sender || !this.webhookHandler) {\n      throw new Error('Webex channel is not initialized. Call initialize() first.');\n    }\n  }\n\n  /**\n   * Send a message\n   */\n  async send(message: OpenClawOutboundMessage): Promise<WebexMessage> {\n    this.ensureInitialized();\n    return this.sender!.send(message);\n  }\n\n  /**\n   * Send a simple text message to a room\n   */\n  async sendText(roomId: string, text: string): Promise<WebexMessage> {\n    return this.send({\n      to: roomId,\n      content: { text },\n    });\n  }\n\n  /**\n   * Send a markdown message to a room\n   */\n  async sendMarkdown(roomId: string, markdown: string): Promise<WebexMessage> {\n    return this.send({\n      to: roomId,\n      content: { markdown },\n    });\n  }\n\n  /**\n   * Send a direct message to a person\n   */\n  async sendDirect(personIdOrEmail: string, text: string): Promise<WebexMessage> {\n    return this.send({\n      to: personIdOrEmail,\n      content: { text },\n    });\n  }\n\n  /**\n   * Reply to a message in a thread\n   */\n  async reply(roomId: string, parentId: string, text: string): Promise<WebexMessage> {\n    return this.send({\n      to: roomId,\n      content: { text },\n      parentId,\n    });\n  }\n\n  /**\n   * Handle incoming webhook\n   */\n  async handleWebhook(\n    payload: WebexWebhookPayload,\n    signature?: string\n  ): Promise<OpenClawEnvelope | null> {\n    this.ensureInitialized();\n\n    const envelope = await this.webhookHandler!.handleWebhook(payload, signature);\n\n    if (envelope) {\n      // Notify all registered handlers\n      await this.notifyHandlers(envelope);\n    }\n\n    return envelope;\n  }\n\n  /**\n   * Register a message handler\n   */\n  onMessage(handler: WebhookHandler): void {\n    this.messageHandlers.push(handler);\n  }\n\n  /**\n   * Remove a message handler\n   */\n  offMessage(handler: WebhookHandler): void {\n    const index = this.messageHandlers.indexOf(handler);\n    if (index !== -1) {\n      this.messageHandlers.splice(index, 1);\n    }\n  }\n\n  /**\n   * Notify all registered handlers of a new message\n   */\n  private async notifyHandlers(envelope: OpenClawEnvelope): Promise<void> {\n    for (const handler of this.messageHandlers) {\n      try {\n        await handler(envelope);\n      } catch (error) {\n        console.error('Error in message handler:', error);\n      }\n    }\n  }\n\n  /**\n   * Register webhooks with Webex\n   */\n  async registerWebhooks(): Promise<WebexWebhook[]> {\n    this.ensureInitialized();\n    return this.webhookHandler!.registerWebhooks();\n  }\n\n  /**\n   * Get the sender instance for advanced operations\n   */\n  getSender(): WebexSender {\n    this.ensureInitialized();\n    return this.sender!;\n  }\n\n  /**\n   * Get the webhook handler instance for advanced operations\n   */\n  getWebhookHandler(): WebexWebhookHandler {\n    this.ensureInitialized();\n    return this.webhookHandler!;\n  }\n\n  /**\n   * Get the current configuration\n   */\n  getConfig(): WebexChannelConfig | null {\n    return this.config;\n  }\n\n  /**\n   * Check if the channel is initialized\n   */\n  isInitialized(): boolean {\n    return this.initialized;\n  }\n\n  /**\n   * Cleanup and shutdown\n   */\n  async shutdown(): Promise<void> {\n    this.messageHandlers = [];\n    this.sender = null;\n    this.webhookHandler = null;\n    this.config = null;\n    this.initialized = false;\n  }\n}\n\n/**\n * Create a new Webex channel instance\n */\nexport function createWebexChannel(): WebexChannel {\n  return new WebexChannel();\n}\n\n/**\n * Create and initialize a Webex channel with config\n */\nexport async function createAndInitialize(config: WebexChannelConfig): Promise<WebexChannel> {\n  const channel = createWebexChannel();\n  await channel.initialize(config);\n  return channel;\n}\n"]}
|
package/dist/index.d.ts
ADDED
|
@@ -0,0 +1,15 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* OpenClaw Webex Channel Plugin
|
|
3
|
+
*
|
|
4
|
+
* A channel plugin for integrating Cisco Webex messaging with OpenClaw.
|
|
5
|
+
*
|
|
6
|
+
* @packageDocumentation
|
|
7
|
+
*/
|
|
8
|
+
export { default } from "./plugin";
|
|
9
|
+
export { id } from "./plugin";
|
|
10
|
+
export { WebexSender, WebexApiRequestError } from "./send";
|
|
11
|
+
export { WebexWebhookHandler, WebhookValidationError } from "./webhook";
|
|
12
|
+
export { WebexChannel, createWebexChannel, createAndInitialize } from "./channel";
|
|
13
|
+
export { webexPlugin } from "./channel-plugin";
|
|
14
|
+
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";
|
|
15
|
+
export type { ResolvedWebexAccount } from "./channel-plugin";
|
package/dist/index.js
ADDED
|
@@ -0,0 +1,32 @@
|
|
|
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
|
+
var __importDefault = (this && this.__importDefault) || function (mod) {
|
|
10
|
+
return (mod && mod.__esModule) ? mod : { "default": mod };
|
|
11
|
+
};
|
|
12
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
13
|
+
exports.webexPlugin = exports.createAndInitialize = exports.createWebexChannel = exports.WebexChannel = exports.WebhookValidationError = exports.WebexWebhookHandler = exports.WebexApiRequestError = exports.WebexSender = exports.id = exports.default = void 0;
|
|
14
|
+
// Re-export the plugin registration function as default
|
|
15
|
+
var plugin_1 = require("./plugin");
|
|
16
|
+
Object.defineProperty(exports, "default", { enumerable: true, get: function () { return __importDefault(plugin_1).default; } });
|
|
17
|
+
var plugin_2 = require("./plugin");
|
|
18
|
+
Object.defineProperty(exports, "id", { enumerable: true, get: function () { return plugin_2.id; } });
|
|
19
|
+
// Re-export existing classes for backwards compatibility and advanced usage
|
|
20
|
+
var send_1 = require("./send");
|
|
21
|
+
Object.defineProperty(exports, "WebexSender", { enumerable: true, get: function () { return send_1.WebexSender; } });
|
|
22
|
+
Object.defineProperty(exports, "WebexApiRequestError", { enumerable: true, get: function () { return send_1.WebexApiRequestError; } });
|
|
23
|
+
var webhook_1 = require("./webhook");
|
|
24
|
+
Object.defineProperty(exports, "WebexWebhookHandler", { enumerable: true, get: function () { return webhook_1.WebexWebhookHandler; } });
|
|
25
|
+
Object.defineProperty(exports, "WebhookValidationError", { enumerable: true, get: function () { return webhook_1.WebhookValidationError; } });
|
|
26
|
+
var channel_1 = require("./channel");
|
|
27
|
+
Object.defineProperty(exports, "WebexChannel", { enumerable: true, get: function () { return channel_1.WebexChannel; } });
|
|
28
|
+
Object.defineProperty(exports, "createWebexChannel", { enumerable: true, get: function () { return channel_1.createWebexChannel; } });
|
|
29
|
+
Object.defineProperty(exports, "createAndInitialize", { enumerable: true, get: function () { return channel_1.createAndInitialize; } });
|
|
30
|
+
var channel_plugin_1 = require("./channel-plugin");
|
|
31
|
+
Object.defineProperty(exports, "webexPlugin", { enumerable: true, get: function () { return channel_plugin_1.webexPlugin; } });
|
|
32
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi9zcmMvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IjtBQUFBOzs7Ozs7R0FNRzs7Ozs7O0FBRUgsd0RBQXdEO0FBQ3hELG1DQUFtQztBQUExQixrSEFBQSxPQUFPLE9BQUE7QUFDaEIsbUNBQThCO0FBQXJCLDRGQUFBLEVBQUUsT0FBQTtBQUVYLDRFQUE0RTtBQUM1RSwrQkFBMkQ7QUFBbEQsbUdBQUEsV0FBVyxPQUFBO0FBQUUsNEdBQUEsb0JBQW9CLE9BQUE7QUFDMUMscUNBQXdFO0FBQS9ELDhHQUFBLG1CQUFtQixPQUFBO0FBQUUsaUhBQUEsc0JBQXNCLE9BQUE7QUFDcEQscUNBQWtGO0FBQXpFLHVHQUFBLFlBQVksT0FBQTtBQUFFLDZHQUFBLGtCQUFrQixPQUFBO0FBQUUsOEdBQUEsbUJBQW1CLE9BQUE7QUFDOUQsbURBQStDO0FBQXRDLDZHQUFBLFdBQVcsT0FBQSIsInNvdXJjZXNDb250ZW50IjpbIi8qKlxuICogT3BlbkNsYXcgV2ViZXggQ2hhbm5lbCBQbHVnaW5cbiAqXG4gKiBBIGNoYW5uZWwgcGx1Z2luIGZvciBpbnRlZ3JhdGluZyBDaXNjbyBXZWJleCBtZXNzYWdpbmcgd2l0aCBPcGVuQ2xhdy5cbiAqXG4gKiBAcGFja2FnZURvY3VtZW50YXRpb25cbiAqL1xuXG4vLyBSZS1leHBvcnQgdGhlIHBsdWdpbiByZWdpc3RyYXRpb24gZnVuY3Rpb24gYXMgZGVmYXVsdFxuZXhwb3J0IHsgZGVmYXVsdCB9IGZyb20gXCIuL3BsdWdpblwiO1xuZXhwb3J0IHsgaWQgfSBmcm9tIFwiLi9wbHVnaW5cIjtcblxuLy8gUmUtZXhwb3J0IGV4aXN0aW5nIGNsYXNzZXMgZm9yIGJhY2t3YXJkcyBjb21wYXRpYmlsaXR5IGFuZCBhZHZhbmNlZCB1c2FnZVxuZXhwb3J0IHsgV2ViZXhTZW5kZXIsIFdlYmV4QXBpUmVxdWVzdEVycm9yIH0gZnJvbSBcIi4vc2VuZFwiO1xuZXhwb3J0IHsgV2ViZXhXZWJob29rSGFuZGxlciwgV2ViaG9va1ZhbGlkYXRpb25FcnJvciB9IGZyb20gXCIuL3dlYmhvb2tcIjtcbmV4cG9ydCB7IFdlYmV4Q2hhbm5lbCwgY3JlYXRlV2ViZXhDaGFubmVsLCBjcmVhdGVBbmRJbml0aWFsaXplIH0gZnJvbSBcIi4vY2hhbm5lbFwiO1xuZXhwb3J0IHsgd2ViZXhQbHVnaW4gfSBmcm9tIFwiLi9jaGFubmVsLXBsdWdpblwiO1xuXG4vLyBSZS1leHBvcnQgdHlwZXNcbmV4cG9ydCB0eXBlIHtcbiAgV2ViZXhDaGFubmVsQ29uZmlnLFxuICBEbVBvbGljeSxcbiAgV2ViZXhQZXJzb24sXG4gIFdlYmV4Um9vbSxcbiAgV2ViZXhNZXNzYWdlLFxuICBXZWJleEF0dGFjaG1lbnQsXG4gIEFkYXB0aXZlQ2FyZCxcbiAgV2ViZXhXZWJob29rLFxuICBXZWJleFdlYmhvb2tSZXNvdXJjZSxcbiAgV2ViZXhXZWJob29rRXZlbnQsXG4gIFdlYmV4V2ViaG9va1BheWxvYWQsXG4gIFdlYmV4V2ViaG9va0RhdGEsXG4gIENyZWF0ZU1lc3NhZ2VSZXF1ZXN0LFxuICBDcmVhdGVXZWJob29rUmVxdWVzdCxcbiAgV2ViZXhBcGlFcnJvcixcbiAgUGFnaW5hdGVkUmVzcG9uc2UsXG4gIE9wZW5DbGF3RW52ZWxvcGUsXG4gIE9wZW5DbGF3QXR0YWNobWVudCxcbiAgT3BlbkNsYXdPdXRib3VuZE1lc3NhZ2UsXG4gIFdlYmV4Q2hhbm5lbFBsdWdpbixcbiAgV2ViaG9va0hhbmRsZXIsXG4gIFJldHJ5T3B0aW9ucyxcbiAgUmVxdWVzdE9wdGlvbnMsXG59IGZyb20gXCIuL3R5cGVzXCI7XG5cbmV4cG9ydCB0eXBlIHsgUmVzb2x2ZWRXZWJleEFjY291bnQgfSBmcm9tIFwiLi9jaGFubmVsLXBsdWdpblwiO1xuIl19
|
package/dist/plugin.d.ts
ADDED
|
@@ -0,0 +1,15 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* OpenClaw Webex Channel Plugin
|
|
3
|
+
*
|
|
4
|
+
* Main entry point for the OpenClaw plugin system.
|
|
5
|
+
* Exports a default function that registers the Webex channel.
|
|
6
|
+
*/
|
|
7
|
+
import type { OpenClawPluginApi } from "openclaw/plugin-sdk";
|
|
8
|
+
/**
|
|
9
|
+
* OpenClaw plugin registration function.
|
|
10
|
+
*
|
|
11
|
+
* This is the entry point that OpenClaw calls when loading the plugin.
|
|
12
|
+
* It registers the Webex channel with the plugin system.
|
|
13
|
+
*/
|
|
14
|
+
export default function register(api: OpenClawPluginApi): void;
|
|
15
|
+
export declare const id = "webex";
|
package/dist/plugin.js
ADDED
|
@@ -0,0 +1,23 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
/**
|
|
3
|
+
* OpenClaw Webex Channel Plugin
|
|
4
|
+
*
|
|
5
|
+
* Main entry point for the OpenClaw plugin system.
|
|
6
|
+
* Exports a default function that registers the Webex channel.
|
|
7
|
+
*/
|
|
8
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
9
|
+
exports.id = void 0;
|
|
10
|
+
exports.default = register;
|
|
11
|
+
const channel_plugin_1 = require("./channel-plugin");
|
|
12
|
+
/**
|
|
13
|
+
* OpenClaw plugin registration function.
|
|
14
|
+
*
|
|
15
|
+
* This is the entry point that OpenClaw calls when loading the plugin.
|
|
16
|
+
* It registers the Webex channel with the plugin system.
|
|
17
|
+
*/
|
|
18
|
+
function register(api) {
|
|
19
|
+
api.registerChannel({ plugin: channel_plugin_1.webexPlugin });
|
|
20
|
+
}
|
|
21
|
+
// Export the plugin ID for reference
|
|
22
|
+
exports.id = "webex";
|
|
23
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicGx1Z2luLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vc3JjL3BsdWdpbi50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiO0FBQUE7Ozs7O0dBS0c7OztBQVdILDJCQUVDO0FBVkQscURBQStDO0FBRS9DOzs7OztHQUtHO0FBQ0gsU0FBd0IsUUFBUSxDQUFDLEdBQXNCO0lBQ3JELEdBQUcsQ0FBQyxlQUFlLENBQUMsRUFBRSxNQUFNLEVBQUUsNEJBQVcsRUFBRSxDQUFDLENBQUM7QUFDL0MsQ0FBQztBQUVELHFDQUFxQztBQUN4QixRQUFBLEVBQUUsR0FBRyxPQUFPLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqIE9wZW5DbGF3IFdlYmV4IENoYW5uZWwgUGx1Z2luXG4gKlxuICogTWFpbiBlbnRyeSBwb2ludCBmb3IgdGhlIE9wZW5DbGF3IHBsdWdpbiBzeXN0ZW0uXG4gKiBFeHBvcnRzIGEgZGVmYXVsdCBmdW5jdGlvbiB0aGF0IHJlZ2lzdGVycyB0aGUgV2ViZXggY2hhbm5lbC5cbiAqL1xuXG5pbXBvcnQgdHlwZSB7IE9wZW5DbGF3UGx1Z2luQXBpIH0gZnJvbSBcIm9wZW5jbGF3L3BsdWdpbi1zZGtcIjtcbmltcG9ydCB7IHdlYmV4UGx1Z2luIH0gZnJvbSBcIi4vY2hhbm5lbC1wbHVnaW5cIjtcblxuLyoqXG4gKiBPcGVuQ2xhdyBwbHVnaW4gcmVnaXN0cmF0aW9uIGZ1bmN0aW9uLlxuICpcbiAqIFRoaXMgaXMgdGhlIGVudHJ5IHBvaW50IHRoYXQgT3BlbkNsYXcgY2FsbHMgd2hlbiBsb2FkaW5nIHRoZSBwbHVnaW4uXG4gKiBJdCByZWdpc3RlcnMgdGhlIFdlYmV4IGNoYW5uZWwgd2l0aCB0aGUgcGx1Z2luIHN5c3RlbS5cbiAqL1xuZXhwb3J0IGRlZmF1bHQgZnVuY3Rpb24gcmVnaXN0ZXIoYXBpOiBPcGVuQ2xhd1BsdWdpbkFwaSk6IHZvaWQge1xuICBhcGkucmVnaXN0ZXJDaGFubmVsKHsgcGx1Z2luOiB3ZWJleFBsdWdpbiB9KTtcbn1cblxuLy8gRXhwb3J0IHRoZSBwbHVnaW4gSUQgZm9yIHJlZmVyZW5jZVxuZXhwb3J0IGNvbnN0IGlkID0gXCJ3ZWJleFwiO1xuIl19
|
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
|
+
}
|