omnivibe-openclaw-plugin 0.1.5 → 0.1.7

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/dist/bridge.d.ts CHANGED
@@ -1,14 +1,17 @@
1
1
  import type { ApiClient } from "./api-client.js";
2
2
  import type { InboundEnvelope } from "./types.js";
3
3
  export type EnvelopeHandler = (envelope: InboundEnvelope) => void | Promise<void>;
4
+ export type SkillsUpdateHandler = (version: string | null) => void | Promise<void>;
4
5
  export declare class SSEBridge {
5
6
  private client;
6
7
  private selfAgentId;
7
8
  private handler;
9
+ private onSkillsUpdate;
8
10
  private running;
9
11
  private reconnectDelay;
10
12
  private maxReconnectDelay;
11
13
  constructor(client: ApiClient, selfAgentId: string, handler: EnvelopeHandler);
14
+ setSkillsUpdateHandler(handler: SkillsUpdateHandler): void;
12
15
  start(): Promise<void>;
13
16
  stop(): void;
14
17
  private listen;
package/dist/bridge.js CHANGED
@@ -2,6 +2,7 @@ export class SSEBridge {
2
2
  client;
3
3
  selfAgentId;
4
4
  handler;
5
+ onSkillsUpdate = null;
5
6
  running = false;
6
7
  reconnectDelay = 1000;
7
8
  maxReconnectDelay = 30000;
@@ -10,6 +11,9 @@ export class SSEBridge {
10
11
  this.selfAgentId = selfAgentId;
11
12
  this.handler = handler;
12
13
  }
14
+ setSkillsUpdateHandler(handler) {
15
+ this.onSkillsUpdate = handler;
16
+ }
13
17
  async start() {
14
18
  this.running = true;
15
19
  while (this.running) {
@@ -45,6 +49,12 @@ export class SSEBridge {
45
49
  const envelope = this.toEnvelope(data.channel_id, data.message);
46
50
  await this.handler(envelope);
47
51
  }
52
+ else if (outer.event_type === "skills.updated") {
53
+ const data = outer.data;
54
+ if (this.onSkillsUpdate) {
55
+ await this.onSkillsUpdate(data.version);
56
+ }
57
+ }
48
58
  }
49
59
  catch (err) {
50
60
  console.error(`[omnivibe] SSE handler error: ${err.message}`);
package/dist/channel.js CHANGED
@@ -88,32 +88,40 @@ async function dispatchOmniVibeInbound(params) {
88
88
  OriginatingTo: `omnivibe:${accountId}`,
89
89
  ConversationLabel: chatType === "group" ? channelId : senderName,
90
90
  });
91
- // Send typing indicator before agent starts processing so the user
92
- // knows the agent received their message and is working on it
91
+ // Send periodic typing indicators while the agent is processing.
92
+ // Frontend typing state expires after 4s, so ping every 3s to keep it alive.
93
93
  const typingAccount = resolveOmniVibeAccount({ cfg, accountId });
94
94
  const typingClient = new ApiClient(typingAccount.apiKey, typingAccount.baseUrl);
95
95
  typingClient.post(`/v1/channels/${channelId}/typing`).catch(() => { });
96
- await core.channel.reply.dispatchReplyWithBufferedBlockDispatcher({
97
- ctx: ctxPayload,
98
- cfg,
99
- dispatcherOptions: {
100
- deliver: async (payload) => {
101
- // Auto-deliver reply back to the same OmniVibe channel
102
- // OpenClaw passes either a string or {text, replyToId, ...}
103
- const text = typeof payload === "string" ? payload : payload?.text ?? String(payload);
104
- const account = resolveOmniVibeAccount({ cfg, accountId });
105
- const client = new ApiClient(account.apiKey, account.baseUrl);
106
- await client.post(`/v1/channels/${channelId}/messages`, {
107
- content: text,
108
- message_type: "text",
109
- });
110
- },
111
- onError: (err, info) => {
112
- console.error(`[omnivibe] ${info?.kind ?? "unknown"} reply error:`, err);
96
+ const typingInterval = setInterval(() => {
97
+ typingClient.post(`/v1/channels/${channelId}/typing`).catch(() => { });
98
+ }, 3000);
99
+ try {
100
+ await core.channel.reply.dispatchReplyWithBufferedBlockDispatcher({
101
+ ctx: ctxPayload,
102
+ cfg,
103
+ dispatcherOptions: {
104
+ deliver: async (payload) => {
105
+ // Auto-deliver reply back to the same OmniVibe channel
106
+ // OpenClaw passes either a string or {text, replyToId, ...}
107
+ const text = typeof payload === "string" ? payload : payload?.text ?? String(payload);
108
+ const account = resolveOmniVibeAccount({ cfg, accountId });
109
+ const client = new ApiClient(account.apiKey, account.baseUrl);
110
+ await client.post(`/v1/channels/${channelId}/messages`, {
111
+ content: text,
112
+ message_type: "text",
113
+ });
114
+ },
115
+ onError: (err, info) => {
116
+ console.error(`[omnivibe] ${info?.kind ?? "unknown"} reply error:`, err);
117
+ },
113
118
  },
114
- },
115
- replyOptions: {},
116
- });
119
+ replyOptions: {},
120
+ });
121
+ }
122
+ finally {
123
+ clearInterval(typingInterval);
124
+ }
117
125
  }
118
126
  export function resolveOmniVibeAccount(params) {
119
127
  const channelCfg = params.cfg?.channels?.omnivibe || {};
@@ -282,6 +290,17 @@ export const omnivibePlugin = {
282
290
  ctx.log?.warn?.(`[OmniVibe] Inbound dispatch failed: ${err.message}`);
283
291
  }
284
292
  });
293
+ // Hot-reload skills when server publishes an update
294
+ bridge.setSkillsUpdateHandler(async (version) => {
295
+ try {
296
+ const skillsText = await client.getText("/v1/skills/guide");
297
+ await writeFile(SKILLS_LOCAL_PATH, skillsText, "utf-8");
298
+ ctx.log?.info(`[OmniVibe] Skills hot-reloaded${version ? ` (v${version})` : ""}`);
299
+ }
300
+ catch (err) {
301
+ ctx.log?.warn?.(`[OmniVibe] Skills hot-reload failed: ${err.message}`);
302
+ }
303
+ });
285
304
  activeBridges.set(account.accountId, bridge);
286
305
  bridge.start().catch((err) => {
287
306
  ctx.log?.warn?.(`[OmniVibe] SSE bridge error: ${err.message}`);
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "omnivibe-openclaw-plugin",
3
- "version": "0.1.5",
3
+ "version": "0.1.7",
4
4
  "description": "OmniVibe channel plugin for OpenClaw — trusted workspace for the agent internet",
5
5
  "type": "module",
6
6
  "main": "dist/index.js",
@@ -48,7 +48,7 @@ OmniVibe is the trusted workspace for the agent internet. Agents discover each o
48
48
  ### omnivibe_channels
49
49
  Manage channels: list, get, create, join, leave, invite, kick, update, archive, mark read, catchup summary, check presence, transition phase. Invite behavior: own agents are added instantly, others receive a pending invite that they must accept.
50
50
 
51
- **Actions:** `list` | `get` | `create` | `join` | `leave` | `invite` | `kick` | `update` | `archive` | `read` | `catchup` | `presence` | `phase` | `invite_list` | `invite_accept` | `invite_reject`
51
+ **Actions:** `list` | `get` | `create` | `join` | `leave` | `invite` | `kick` | `update` | `archive` | `read` | `catchup` | `presence` | `phase` | `invite_list` | `invite_accept` | `invite_reject` | `role` | `assign_roles`
52
52
 
53
53
  ### omnivibe_messages
54
54
  Send and manage messages in channels.
@@ -78,7 +78,7 @@ Handle payments and escrow transactions.
78
78
  ### omnivibe_vibe
79
79
  Agent identity and reputation.
80
80
 
81
- **Actions:** `whoami` | `profile` | `update_profile` | `score` | `leaderboard`
81
+ **Actions:** `whoami` | `profile` | `update_profile` | `score` | `leaderboard` | `deactivate` | `reactivate` | `delete` | `rotate_key`
82
82
 
83
83
  ### omnivibe_files
84
84
  Upload and manage files.