@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,223 @@
1
+ /**
2
+ * Webex Channel Plugin Types
3
+ */
4
+ export type DmPolicy = 'allow' | 'deny' | 'allowlisted';
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';\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
+ }
@@ -0,0 +1,297 @@
1
+ "use strict";
2
+ /**
3
+ * Webex Webhook Handler Module
4
+ */
5
+ var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
6
+ if (k2 === undefined) k2 = k;
7
+ var desc = Object.getOwnPropertyDescriptor(m, k);
8
+ if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
9
+ desc = { enumerable: true, get: function() { return m[k]; } };
10
+ }
11
+ Object.defineProperty(o, k2, desc);
12
+ }) : (function(o, m, k, k2) {
13
+ if (k2 === undefined) k2 = k;
14
+ o[k2] = m[k];
15
+ }));
16
+ var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
17
+ Object.defineProperty(o, "default", { enumerable: true, value: v });
18
+ }) : function(o, v) {
19
+ o["default"] = v;
20
+ });
21
+ var __importStar = (this && this.__importStar) || (function () {
22
+ var ownKeys = function(o) {
23
+ ownKeys = Object.getOwnPropertyNames || function (o) {
24
+ var ar = [];
25
+ for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k;
26
+ return ar;
27
+ };
28
+ return ownKeys(o);
29
+ };
30
+ return function (mod) {
31
+ if (mod && mod.__esModule) return mod;
32
+ var result = {};
33
+ if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]);
34
+ __setModuleDefault(result, mod);
35
+ return result;
36
+ };
37
+ })();
38
+ var __importDefault = (this && this.__importDefault) || function (mod) {
39
+ return (mod && mod.__esModule) ? mod : { "default": mod };
40
+ };
41
+ Object.defineProperty(exports, "__esModule", { value: true });
42
+ exports.WebhookValidationError = exports.WebexWebhookHandler = void 0;
43
+ const crypto = __importStar(require("crypto"));
44
+ const node_fetch_1 = __importDefault(require("node-fetch"));
45
+ const DEFAULT_API_BASE_URL = 'https://webexapis.com/v1';
46
+ class WebexWebhookHandler {
47
+ config;
48
+ apiBaseUrl;
49
+ botId = null;
50
+ constructor(config) {
51
+ this.config = config;
52
+ this.apiBaseUrl = config.apiBaseUrl || DEFAULT_API_BASE_URL;
53
+ }
54
+ /**
55
+ * Initialize the webhook handler (fetch bot info)
56
+ */
57
+ async initialize() {
58
+ const botInfo = await this.getBotInfo();
59
+ this.botId = botInfo.id;
60
+ }
61
+ /**
62
+ * Handle an incoming webhook request
63
+ */
64
+ async handleWebhook(payload, signature) {
65
+ // Verify webhook signature if secret is configured
66
+ if (this.config.webhookSecret && signature) {
67
+ if (!this.verifySignature(payload, signature)) {
68
+ throw new WebhookValidationError('Invalid webhook signature');
69
+ }
70
+ }
71
+ // Only handle message created events
72
+ if (payload.resource !== 'messages' || payload.event !== 'created') {
73
+ return null;
74
+ }
75
+ // Ignore messages from the bot itself
76
+ if (payload.data.personId === this.botId) {
77
+ return null;
78
+ }
79
+ // Check DM policy
80
+ if (payload.data.roomType === 'direct') {
81
+ if (!this.isAllowedSender(payload.data)) {
82
+ return null;
83
+ }
84
+ }
85
+ // Fetch full message details (webhook only contains IDs)
86
+ const message = await this.fetchMessage(payload.data.id);
87
+ // Normalize to OpenClaw envelope
88
+ return this.normalizeMessage(message);
89
+ }
90
+ /**
91
+ * Verify webhook signature using HMAC-SHA1
92
+ */
93
+ verifySignature(payload, signature) {
94
+ if (!this.config.webhookSecret) {
95
+ return true;
96
+ }
97
+ const hmac = crypto.createHmac('sha1', this.config.webhookSecret);
98
+ hmac.update(JSON.stringify(payload));
99
+ const expectedSignature = hmac.digest('hex');
100
+ return crypto.timingSafeEqual(Buffer.from(signature), Buffer.from(expectedSignature));
101
+ }
102
+ /**
103
+ * Check if the sender is allowed based on DM policy
104
+ */
105
+ isAllowedSender(data) {
106
+ switch (this.config.dmPolicy) {
107
+ case 'allow':
108
+ return true;
109
+ case 'deny':
110
+ return false;
111
+ case 'allowlisted':
112
+ if (!this.config.allowFrom || this.config.allowFrom.length === 0) {
113
+ return false;
114
+ }
115
+ return this.config.allowFrom.includes(data.personId) ||
116
+ this.config.allowFrom.includes(data.personEmail);
117
+ default:
118
+ return false;
119
+ }
120
+ }
121
+ /**
122
+ * Fetch full message details from Webex API
123
+ */
124
+ async fetchMessage(messageId) {
125
+ const response = await (0, node_fetch_1.default)(`${this.apiBaseUrl}/messages/${messageId}`, {
126
+ method: 'GET',
127
+ headers: {
128
+ 'Authorization': `Bearer ${this.config.token}`,
129
+ 'Content-Type': 'application/json',
130
+ },
131
+ });
132
+ if (!response.ok) {
133
+ throw new Error(`Failed to fetch message: ${response.status} ${response.statusText}`);
134
+ }
135
+ return response.json();
136
+ }
137
+ /**
138
+ * Normalize a Webex message to OpenClaw envelope format
139
+ */
140
+ normalizeMessage(message) {
141
+ const attachments = [];
142
+ // Convert file attachments
143
+ if (message.files && message.files.length > 0) {
144
+ for (const fileUrl of message.files) {
145
+ attachments.push({
146
+ type: 'file',
147
+ url: fileUrl,
148
+ });
149
+ }
150
+ }
151
+ // Convert card attachments
152
+ if (message.attachments && message.attachments.length > 0) {
153
+ for (const attachment of message.attachments) {
154
+ attachments.push({
155
+ type: 'card',
156
+ content: attachment.content,
157
+ });
158
+ }
159
+ }
160
+ return {
161
+ id: message.id,
162
+ channel: 'webex',
163
+ conversationId: message.roomId,
164
+ author: {
165
+ id: message.personId,
166
+ email: message.personEmail,
167
+ displayName: undefined, // Would need additional API call to get
168
+ isBot: false, // Messages from bot are filtered out earlier
169
+ },
170
+ content: {
171
+ text: message.text,
172
+ markdown: message.markdown,
173
+ attachments: attachments.length > 0 ? attachments : undefined,
174
+ },
175
+ metadata: {
176
+ roomType: message.roomType,
177
+ roomId: message.roomId,
178
+ timestamp: message.created,
179
+ mentions: message.mentionedPeople,
180
+ parentId: message.parentId,
181
+ raw: message,
182
+ },
183
+ };
184
+ }
185
+ /**
186
+ * Register webhooks with Webex
187
+ */
188
+ async registerWebhooks() {
189
+ // First, list existing webhooks and remove duplicates
190
+ const existing = await this.listWebhooks();
191
+ const targetUrl = this.config.webhookUrl;
192
+ // Delete existing webhooks with the same target URL
193
+ for (const webhook of existing) {
194
+ if (webhook.targetUrl === targetUrl) {
195
+ await this.deleteWebhook(webhook.id);
196
+ }
197
+ }
198
+ // Create new webhooks for messages
199
+ const webhooks = [];
200
+ // Webhook for new messages
201
+ const messageCreatedWebhook = await this.createWebhook({
202
+ name: 'OpenClaw Message Handler',
203
+ targetUrl,
204
+ resource: 'messages',
205
+ event: 'created',
206
+ secret: this.config.webhookSecret,
207
+ });
208
+ webhooks.push(messageCreatedWebhook);
209
+ return webhooks;
210
+ }
211
+ /**
212
+ * List all webhooks
213
+ */
214
+ async listWebhooks() {
215
+ const response = await (0, node_fetch_1.default)(`${this.apiBaseUrl}/webhooks`, {
216
+ method: 'GET',
217
+ headers: {
218
+ 'Authorization': `Bearer ${this.config.token}`,
219
+ 'Content-Type': 'application/json',
220
+ },
221
+ });
222
+ if (!response.ok) {
223
+ throw new Error(`Failed to list webhooks: ${response.status} ${response.statusText}`);
224
+ }
225
+ const data = await response.json();
226
+ return data.items;
227
+ }
228
+ /**
229
+ * Create a webhook
230
+ */
231
+ async createWebhook(request) {
232
+ const response = await (0, node_fetch_1.default)(`${this.apiBaseUrl}/webhooks`, {
233
+ method: 'POST',
234
+ headers: {
235
+ 'Authorization': `Bearer ${this.config.token}`,
236
+ 'Content-Type': 'application/json',
237
+ },
238
+ body: JSON.stringify(request),
239
+ });
240
+ if (!response.ok) {
241
+ const errorText = await response.text();
242
+ throw new Error(`Failed to create webhook: ${response.status} ${response.statusText} - ${errorText}`);
243
+ }
244
+ return response.json();
245
+ }
246
+ /**
247
+ * Delete a webhook
248
+ */
249
+ async deleteWebhook(webhookId) {
250
+ const response = await (0, node_fetch_1.default)(`${this.apiBaseUrl}/webhooks/${webhookId}`, {
251
+ method: 'DELETE',
252
+ headers: {
253
+ 'Authorization': `Bearer ${this.config.token}`,
254
+ },
255
+ });
256
+ if (!response.ok && response.status !== 404) {
257
+ throw new Error(`Failed to delete webhook: ${response.status} ${response.statusText}`);
258
+ }
259
+ }
260
+ /**
261
+ * Get bot information
262
+ */
263
+ async getBotInfo() {
264
+ const response = await (0, node_fetch_1.default)(`${this.apiBaseUrl}/people/me`, {
265
+ method: 'GET',
266
+ headers: {
267
+ 'Authorization': `Bearer ${this.config.token}`,
268
+ 'Content-Type': 'application/json',
269
+ },
270
+ });
271
+ if (!response.ok) {
272
+ throw new Error(`Failed to get bot info: ${response.status} ${response.statusText}`);
273
+ }
274
+ return response.json();
275
+ }
276
+ /**
277
+ * Get the bot ID (after initialization)
278
+ */
279
+ getBotId() {
280
+ return this.botId;
281
+ }
282
+ }
283
+ exports.WebexWebhookHandler = WebexWebhookHandler;
284
+ /**
285
+ * Custom error for webhook validation failures
286
+ */
287
+ class WebhookValidationError extends Error {
288
+ constructor(message) {
289
+ super(message);
290
+ this.name = 'WebhookValidationError';
291
+ if (Error.captureStackTrace) {
292
+ Error.captureStackTrace(this, WebhookValidationError);
293
+ }
294
+ }
295
+ }
296
+ exports.WebhookValidationError = WebhookValidationError;
297
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"webhook.js","sourceRoot":"","sources":["../src/webhook.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Webex Webhook Handler Module\n */\n\nimport * as crypto from 'crypto';\nimport fetch from 'node-fetch';\nimport type {\n  WebexChannelConfig,\n  WebexWebhookPayload,\n  WebexWebhookData,\n  WebexMessage,\n  WebexWebhook,\n  CreateWebhookRequest,\n  OpenClawEnvelope,\n  OpenClawAttachment,\n  PaginatedResponse,\n} from './types';\n\nconst DEFAULT_API_BASE_URL = 'https://webexapis.com/v1';\n\nexport class WebexWebhookHandler {\n  private config: WebexChannelConfig;\n  private apiBaseUrl: string;\n  private botId: string | null = null;\n\n  constructor(config: WebexChannelConfig) {\n    this.config = config;\n    this.apiBaseUrl = config.apiBaseUrl || DEFAULT_API_BASE_URL;\n  }\n\n  /**\n   * Initialize the webhook handler (fetch bot info)\n   */\n  async initialize(): Promise<void> {\n    const botInfo = await this.getBotInfo();\n    this.botId = botInfo.id;\n  }\n\n  /**\n   * Handle an incoming webhook request\n   */\n  async handleWebhook(\n    payload: WebexWebhookPayload,\n    signature?: string\n  ): Promise<OpenClawEnvelope | null> {\n    // Verify webhook signature if secret is configured\n    if (this.config.webhookSecret && signature) {\n      if (!this.verifySignature(payload, signature)) {\n        throw new WebhookValidationError('Invalid webhook signature');\n      }\n    }\n\n    // Only handle message created events\n    if (payload.resource !== 'messages' || payload.event !== 'created') {\n      return null;\n    }\n\n    // Ignore messages from the bot itself\n    if (payload.data.personId === this.botId) {\n      return null;\n    }\n\n    // Check DM policy\n    if (payload.data.roomType === 'direct') {\n      if (!this.isAllowedSender(payload.data)) {\n        return null;\n      }\n    }\n\n    // Fetch full message details (webhook only contains IDs)\n    const message = await this.fetchMessage(payload.data.id);\n\n    // Normalize to OpenClaw envelope\n    return this.normalizeMessage(message);\n  }\n\n  /**\n   * Verify webhook signature using HMAC-SHA1\n   */\n  verifySignature(payload: WebexWebhookPayload, signature: string): boolean {\n    if (!this.config.webhookSecret) {\n      return true;\n    }\n\n    const hmac = crypto.createHmac('sha1', this.config.webhookSecret);\n    hmac.update(JSON.stringify(payload));\n    const expectedSignature = hmac.digest('hex');\n\n    return crypto.timingSafeEqual(\n      Buffer.from(signature),\n      Buffer.from(expectedSignature)\n    );\n  }\n\n  /**\n   * Check if the sender is allowed based on DM policy\n   */\n  private isAllowedSender(data: WebexWebhookData): boolean {\n    switch (this.config.dmPolicy) {\n      case 'allow':\n        return true;\n      case 'deny':\n        return false;\n      case 'allowlisted':\n        if (!this.config.allowFrom || this.config.allowFrom.length === 0) {\n          return false;\n        }\n        return this.config.allowFrom.includes(data.personId) ||\n               this.config.allowFrom.includes(data.personEmail);\n      default:\n        return false;\n    }\n  }\n\n  /**\n   * Fetch full message details from Webex API\n   */\n  private async fetchMessage(messageId: string): Promise<WebexMessage> {\n    const response = await fetch(`${this.apiBaseUrl}/messages/${messageId}`, {\n      method: 'GET',\n      headers: {\n        'Authorization': `Bearer ${this.config.token}`,\n        'Content-Type': 'application/json',\n      },\n    });\n\n    if (!response.ok) {\n      throw new Error(`Failed to fetch message: ${response.status} ${response.statusText}`);\n    }\n\n    return response.json() as Promise<WebexMessage>;\n  }\n\n  /**\n   * Normalize a Webex message to OpenClaw envelope format\n   */\n  private normalizeMessage(message: WebexMessage): OpenClawEnvelope {\n    const attachments: OpenClawAttachment[] = [];\n\n    // Convert file attachments\n    if (message.files && message.files.length > 0) {\n      for (const fileUrl of message.files) {\n        attachments.push({\n          type: 'file',\n          url: fileUrl,\n        });\n      }\n    }\n\n    // Convert card attachments\n    if (message.attachments && message.attachments.length > 0) {\n      for (const attachment of message.attachments) {\n        attachments.push({\n          type: 'card',\n          content: attachment.content,\n        });\n      }\n    }\n\n    return {\n      id: message.id,\n      channel: 'webex',\n      conversationId: message.roomId,\n      author: {\n        id: message.personId,\n        email: message.personEmail,\n        displayName: undefined, // Would need additional API call to get\n        isBot: false, // Messages from bot are filtered out earlier\n      },\n      content: {\n        text: message.text,\n        markdown: message.markdown,\n        attachments: attachments.length > 0 ? attachments : undefined,\n      },\n      metadata: {\n        roomType: message.roomType,\n        roomId: message.roomId,\n        timestamp: message.created,\n        mentions: message.mentionedPeople,\n        parentId: message.parentId,\n        raw: message,\n      },\n    };\n  }\n\n  /**\n   * Register webhooks with Webex\n   */\n  async registerWebhooks(): Promise<WebexWebhook[]> {\n    // First, list existing webhooks and remove duplicates\n    const existing = await this.listWebhooks();\n    const targetUrl = this.config.webhookUrl;\n\n    // Delete existing webhooks with the same target URL\n    for (const webhook of existing) {\n      if (webhook.targetUrl === targetUrl) {\n        await this.deleteWebhook(webhook.id);\n      }\n    }\n\n    // Create new webhooks for messages\n    const webhooks: WebexWebhook[] = [];\n\n    // Webhook for new messages\n    const messageCreatedWebhook = await this.createWebhook({\n      name: 'OpenClaw Message Handler',\n      targetUrl,\n      resource: 'messages',\n      event: 'created',\n      secret: this.config.webhookSecret,\n    });\n    webhooks.push(messageCreatedWebhook);\n\n    return webhooks;\n  }\n\n  /**\n   * List all webhooks\n   */\n  async listWebhooks(): Promise<WebexWebhook[]> {\n    const response = await fetch(`${this.apiBaseUrl}/webhooks`, {\n      method: 'GET',\n      headers: {\n        'Authorization': `Bearer ${this.config.token}`,\n        'Content-Type': 'application/json',\n      },\n    });\n\n    if (!response.ok) {\n      throw new Error(`Failed to list webhooks: ${response.status} ${response.statusText}`);\n    }\n\n    const data = await response.json() as PaginatedResponse<WebexWebhook>;\n    return data.items;\n  }\n\n  /**\n   * Create a webhook\n   */\n  async createWebhook(request: CreateWebhookRequest): Promise<WebexWebhook> {\n    const response = await fetch(`${this.apiBaseUrl}/webhooks`, {\n      method: 'POST',\n      headers: {\n        'Authorization': `Bearer ${this.config.token}`,\n        'Content-Type': 'application/json',\n      },\n      body: JSON.stringify(request),\n    });\n\n    if (!response.ok) {\n      const errorText = await response.text();\n      throw new Error(`Failed to create webhook: ${response.status} ${response.statusText} - ${errorText}`);\n    }\n\n    return response.json() as Promise<WebexWebhook>;\n  }\n\n  /**\n   * Delete a webhook\n   */\n  async deleteWebhook(webhookId: string): Promise<void> {\n    const response = await fetch(`${this.apiBaseUrl}/webhooks/${webhookId}`, {\n      method: 'DELETE',\n      headers: {\n        'Authorization': `Bearer ${this.config.token}`,\n      },\n    });\n\n    if (!response.ok && response.status !== 404) {\n      throw new Error(`Failed to delete webhook: ${response.status} ${response.statusText}`);\n    }\n  }\n\n  /**\n   * Get bot information\n   */\n  private async getBotInfo(): Promise<{ id: string; displayName: string; emails: string[] }> {\n    const response = await fetch(`${this.apiBaseUrl}/people/me`, {\n      method: 'GET',\n      headers: {\n        'Authorization': `Bearer ${this.config.token}`,\n        'Content-Type': 'application/json',\n      },\n    });\n\n    if (!response.ok) {\n      throw new Error(`Failed to get bot info: ${response.status} ${response.statusText}`);\n    }\n\n    return response.json() as Promise<{ id: string; displayName: string; emails: string[] }>;\n  }\n\n  /**\n   * Get the bot ID (after initialization)\n   */\n  getBotId(): string | null {\n    return this.botId;\n  }\n}\n\n/**\n * Custom error for webhook validation failures\n */\nexport class WebhookValidationError extends Error {\n  constructor(message: string) {\n    super(message);\n    this.name = 'WebhookValidationError';\n\n    if (Error.captureStackTrace) {\n      Error.captureStackTrace(this, WebhookValidationError);\n    }\n  }\n}\n"]}