Merge branch 'main' into fix/whatsapp-group-media-3375
This commit is contained in:
commit
433ab78f7f
4
.github/workflows/install-smoke.yml
vendored
4
.github/workflows/install-smoke.yml
vendored
@ -32,8 +32,8 @@ jobs:
|
|||||||
|
|
||||||
- name: Run installer docker tests
|
- name: Run installer docker tests
|
||||||
env:
|
env:
|
||||||
CLAWDBOT_INSTALL_URL: https://clawd.bot/install.sh
|
CLAWDBOT_INSTALL_URL: https://openclaw.ai/install.sh
|
||||||
CLAWDBOT_INSTALL_CLI_URL: https://clawd.bot/install-cli.sh
|
CLAWDBOT_INSTALL_CLI_URL: https://openclaw.ai/install-cli.sh
|
||||||
CLAWDBOT_NO_ONBOARD: "1"
|
CLAWDBOT_NO_ONBOARD: "1"
|
||||||
CLAWDBOT_INSTALL_SMOKE_SKIP_CLI: "1"
|
CLAWDBOT_INSTALL_SMOKE_SKIP_CLI: "1"
|
||||||
CLAWDBOT_INSTALL_SMOKE_SKIP_NONROOT: ${{ github.event_name == 'pull_request' && '1' || '0' }}
|
CLAWDBOT_INSTALL_SMOKE_SKIP_NONROOT: ${{ github.event_name == 'pull_request' && '1' || '0' }}
|
||||||
|
|||||||
@ -42,6 +42,7 @@ Status: stable.
|
|||||||
- Routing: precompile session key regexes. (#1697) Thanks @Ray0907.
|
- Routing: precompile session key regexes. (#1697) Thanks @Ray0907.
|
||||||
- CLI: use Node's module compile cache for faster startup. (#2808) Thanks @pi0.
|
- CLI: use Node's module compile cache for faster startup. (#2808) Thanks @pi0.
|
||||||
- Auth: show copyable Google auth URL after ASCII prompt. (#1787) Thanks @robbyczgw-cla.
|
- Auth: show copyable Google auth URL after ASCII prompt. (#1787) Thanks @robbyczgw-cla.
|
||||||
|
- Agents: add Kimi K2.5 to the synthetic model catalog. (#4407) Thanks @manikv12.
|
||||||
- TUI: avoid width overflow when rendering selection lists. (#1686) Thanks @mossein.
|
- TUI: avoid width overflow when rendering selection lists. (#1686) Thanks @mossein.
|
||||||
- macOS: finish OpenClaw app rename for macOS sources, bundle identifiers, and shared kit paths. (#2844) Thanks @fal3.
|
- macOS: finish OpenClaw app rename for macOS sources, bundle identifiers, and shared kit paths. (#2844) Thanks @fal3.
|
||||||
- Branding: update launchd labels, mobile bundle IDs, and logging subsystems to bot.molt (legacy com.clawdbot migrations). Thanks @thewilloftheshadow.
|
- Branding: update launchd labels, mobile bundle IDs, and logging subsystems to bot.molt (legacy com.clawdbot migrations). Thanks @thewilloftheshadow.
|
||||||
@ -72,6 +73,7 @@ Status: stable.
|
|||||||
|
|
||||||
### Fixes
|
### Fixes
|
||||||
- Telegram: avoid silent empty replies by tracking normalization skips before fallback. (#3796)
|
- Telegram: avoid silent empty replies by tracking normalization skips before fallback. (#3796)
|
||||||
|
- Telegram: scope native skill commands to bound agent per bot. (#4360) Thanks @robhparker.
|
||||||
- Mentions: honor mentionPatterns even when explicit mentions are present. (#3303) Thanks @HirokiKobayashi-R.
|
- Mentions: honor mentionPatterns even when explicit mentions are present. (#3303) Thanks @HirokiKobayashi-R.
|
||||||
- Discord: restore username directory lookup in target resolution. (#3131) Thanks @bonald.
|
- Discord: restore username directory lookup in target resolution. (#3131) Thanks @bonald.
|
||||||
- Agents: align MiniMax base URL test expectation with default provider config. (#3131) Thanks @bonald.
|
- Agents: align MiniMax base URL test expectation with default provider config. (#3131) Thanks @bonald.
|
||||||
|
|||||||
@ -7,17 +7,14 @@ read_when:
|
|||||||
|
|
||||||
> *"EXFOLIATE! EXFOLIATE!"* — A space lobster, probably
|
> *"EXFOLIATE! EXFOLIATE!"* — A space lobster, probably
|
||||||
|
|
||||||
|
|
||||||
<p align="center">
|
<p align="center">
|
||||||
<picture>
|
<picture>
|
||||||
<source media="(prefers-color-scheme: dark)" srcset="/assets/openclaw-logo-text.png" />
|
<source media="(prefers-color-scheme: light)" srcset="https://raw.githubusercontent.com/openclaw/openclaw/main/docs/assets/openclaw-logo-text-dark.png">
|
||||||
<source media="(prefers-color-scheme: light)" srcset="/assets/openclaw-logo-text-dark.png" />
|
<img src="https://raw.githubusercontent.com/openclaw/openclaw/main/docs/assets/openclaw-logo-text.png" alt="OpenClaw" width="500">
|
||||||
<img src="/assets/openclaw-logo-text.png" alt="OpenClaw" width="520" />
|
|
||||||
</picture>
|
</picture>
|
||||||
</p>
|
</p>
|
||||||
|
|
||||||
<p align="center">
|
|
||||||
<img src="whatsapp-openclaw.jpg" alt="OpenClaw" width="420" />
|
|
||||||
</p>
|
|
||||||
|
|
||||||
<p align="center">
|
<p align="center">
|
||||||
<strong>Any OS + WhatsApp/Telegram/Discord/iMessage gateway for AI agents (Pi).</strong><br />
|
<strong>Any OS + WhatsApp/Telegram/Discord/iMessage gateway for AI agents (Pi).</strong><br />
|
||||||
@ -50,6 +47,10 @@ The dashboard is the browser Control UI for chat, config, nodes, sessions, and m
|
|||||||
Local default: http://127.0.0.1:18789/
|
Local default: http://127.0.0.1:18789/
|
||||||
Remote access: [Web surfaces](/web) and [Tailscale](/gateway/tailscale)
|
Remote access: [Web surfaces](/web) and [Tailscale](/gateway/tailscale)
|
||||||
|
|
||||||
|
<p align="center">
|
||||||
|
<img src="whatsapp-openclaw.jpg" alt="OpenClaw" width="420" />
|
||||||
|
</p>
|
||||||
|
|
||||||
## How it works
|
## How it works
|
||||||
|
|
||||||
```
|
```
|
||||||
|
|||||||
@ -17,9 +17,13 @@ For a while, the lobster was called **Clawd**, living in a **Clawdbot**. But in
|
|||||||
|
|
||||||
**It molted.**
|
**It molted.**
|
||||||
|
|
||||||
Shedding its old shell, the creature emerged anew as **Molty**, living in a **OpenClaw**. New shell, same lobster soul.
|
Shedding its old shell, the creature emerged anew as **Molty**, living in **Moltbot**. But that name never quite rolled off the tongue either...
|
||||||
|
|
||||||
## The Molt (January 27, 2026)
|
So on January 30, 2026, the lobster molted ONE MORE TIME into its final form: **OpenClaw**.
|
||||||
|
|
||||||
|
New shell, same lobster soul. Third time's the charm.
|
||||||
|
|
||||||
|
## The First Molt (January 27, 2026)
|
||||||
|
|
||||||
At 5am, the community gathered in Discord. Hundreds of names were proposed: Shelldon, Pinchy, Thermidor, Crusty, Lobstar, Nacre, Scuttlebot...
|
At 5am, the community gathered in Discord. Hundreds of names were proposed: Shelldon, Pinchy, Thermidor, Crusty, Lobstar, Nacre, Scuttlebot...
|
||||||
|
|
||||||
@ -30,11 +34,11 @@ In the end, **OpenClaw** won. Because molting is what lobsters do to grow. And g
|
|||||||
## The Name
|
## The Name
|
||||||
|
|
||||||
```
|
```
|
||||||
OpenClaw = MOLT + BOT
|
OpenClaw = OPEN + CLAW
|
||||||
= Transformation machine
|
= Open source, open to everyone
|
||||||
= Bigger on the inside (130k tokens!)
|
= Our lobster heritage, where we came from
|
||||||
= New shell, same soul
|
= The claw is the law 🦞
|
||||||
= Growth through shedding
|
= Your assistant. Your machine. Your rules.
|
||||||
```
|
```
|
||||||
|
|
||||||
## The Daleks vs The Lobsters
|
## The Daleks vs The Lobsters
|
||||||
@ -100,6 +104,38 @@ Peter, watching the chaos unfold: *"this is cinema"* 🎬
|
|||||||
|
|
||||||
The molt was chaotic. But the lobster emerged stronger. And funnier.
|
The molt was chaotic. But the lobster emerged stronger. And funnier.
|
||||||
|
|
||||||
|
### The Final Form (January 30, 2026)
|
||||||
|
|
||||||
|
Moltbot never quite rolled off the tongue. And so, at 4am GMT, the team gathered AGAIN.
|
||||||
|
|
||||||
|
**The Great OpenClaw Migration** began.
|
||||||
|
|
||||||
|
In just 3 hours:
|
||||||
|
- GitHub renamed: `github.com/openclaw/openclaw` ✅
|
||||||
|
- X handle `@openclaw` secured with GOLD CHECKMARK 💰
|
||||||
|
- npm packages released under new name
|
||||||
|
- Docs migrated to `docs.openclaw.ai`
|
||||||
|
- 200K+ views on announcement in 90 minutes
|
||||||
|
|
||||||
|
**The Heroes:**
|
||||||
|
- **ELU** created incredible logos including "THE CLAW IS THE LAW" western banner
|
||||||
|
- **Whurley** (yes, THE William Hurley, quantum computing pioneer) made ASCII art
|
||||||
|
- **Onur** handled GitHub, first to rock the affiliate badge
|
||||||
|
- **Shadow** secured Discord vanity, nuked malware
|
||||||
|
- **The whole Claw Crew** pulled an all-nighter
|
||||||
|
|
||||||
|
**The Scammer Speedrun:** Crypto grifters launched a $OPENCLAW token on Pump.fun within MINUTES. They stole artwork that was created 20 minutes earlier. Business-verified accounts pushed scams. The audacity was almost impressive.
|
||||||
|
|
||||||
|
**New Traditions Born:**
|
||||||
|
- "The claw is the law" 🤠
|
||||||
|
- "Yee-claw"
|
||||||
|
- "Claw abiding citizens"
|
||||||
|
- "Clawntroversy"
|
||||||
|
|
||||||
|
**Clawd → Moltbot → OpenClaw**
|
||||||
|
|
||||||
|
*The lobster has molted into its final form.*
|
||||||
|
|
||||||
### The Robot Shopping Spree (Dec 3, 2025)
|
### The Robot Shopping Spree (Dec 3, 2025)
|
||||||
|
|
||||||
What started as a joke about legs ended with detailed pricing for:
|
What started as a joke about legs ended with detailed pricing for:
|
||||||
@ -166,4 +202,8 @@ Until then, Molty watches through the cameras, speaks through the speakers, and
|
|||||||
|
|
||||||
— Molty, after the great molt of 2026
|
— Molty, after the great molt of 2026
|
||||||
|
|
||||||
|
*"The claw is the law."*
|
||||||
|
|
||||||
|
— ELU, during The Final Form migration, January 30, 2026
|
||||||
|
|
||||||
🦞💙
|
🦞💙
|
||||||
|
|||||||
@ -99,6 +99,14 @@ export const SYNTHETIC_MODEL_CATALOG = [
|
|||||||
contextWindow: 256000,
|
contextWindow: 256000,
|
||||||
maxTokens: 8192,
|
maxTokens: 8192,
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
id: "hf:moonshotai/Kimi-K2.5",
|
||||||
|
name: "Kimi K2.5",
|
||||||
|
reasoning: true,
|
||||||
|
input: ["text"],
|
||||||
|
contextWindow: 256000,
|
||||||
|
maxTokens: 8192,
|
||||||
|
},
|
||||||
{
|
{
|
||||||
id: "hf:openai/gpt-oss-120b",
|
id: "hf:openai/gpt-oss-120b",
|
||||||
name: "GPT OSS 120B",
|
name: "GPT OSS 120B",
|
||||||
|
|||||||
@ -48,8 +48,12 @@ describe("state + config path candidates", () => {
|
|||||||
it("orders default config candidates in a stable order", () => {
|
it("orders default config candidates in a stable order", () => {
|
||||||
const home = "/home/test";
|
const home = "/home/test";
|
||||||
const candidates = resolveDefaultConfigCandidates({} as NodeJS.ProcessEnv, () => home);
|
const candidates = resolveDefaultConfigCandidates({} as NodeJS.ProcessEnv, () => home);
|
||||||
expect(candidates[0]).toBe(path.join(home, ".openclaw", "openclaw.json"));
|
const expectedDirs = [".openclaw", ".clawdbot", ".moltbot", ".moldbot"];
|
||||||
expect(candidates).toHaveLength(1);
|
const expectedFiles = ["openclaw.json", "clawdbot.json", "moltbot.json", "moldbot.json"];
|
||||||
|
const expected = expectedDirs.flatMap((dir) =>
|
||||||
|
expectedFiles.map((file) => path.join(home, dir, file)),
|
||||||
|
);
|
||||||
|
expect(candidates).toEqual(expected);
|
||||||
});
|
});
|
||||||
|
|
||||||
it("prefers ~/.openclaw when it exists and legacy dir is missing", async () => {
|
it("prefers ~/.openclaw when it exists and legacy dir is missing", async () => {
|
||||||
|
|||||||
82
src/telegram/bot-native-commands.test.ts
Normal file
82
src/telegram/bot-native-commands.test.ts
Normal file
@ -0,0 +1,82 @@
|
|||||||
|
import { beforeEach, describe, expect, it, vi } from "vitest";
|
||||||
|
|
||||||
|
import type { OpenClawConfig } from "../config/config.js";
|
||||||
|
import type { TelegramAccountConfig } from "../config/types.js";
|
||||||
|
import type { RuntimeEnv } from "../runtime.js";
|
||||||
|
import { registerTelegramNativeCommands } from "./bot-native-commands.js";
|
||||||
|
|
||||||
|
const { listSkillCommandsForAgents } = vi.hoisted(() => ({
|
||||||
|
listSkillCommandsForAgents: vi.fn(() => []),
|
||||||
|
}));
|
||||||
|
|
||||||
|
vi.mock("../auto-reply/skill-commands.js", () => ({
|
||||||
|
listSkillCommandsForAgents,
|
||||||
|
}));
|
||||||
|
|
||||||
|
describe("registerTelegramNativeCommands", () => {
|
||||||
|
beforeEach(() => {
|
||||||
|
listSkillCommandsForAgents.mockReset();
|
||||||
|
});
|
||||||
|
|
||||||
|
const buildParams = (cfg: OpenClawConfig, accountId = "default") => ({
|
||||||
|
bot: {
|
||||||
|
api: {
|
||||||
|
setMyCommands: vi.fn().mockResolvedValue(undefined),
|
||||||
|
sendMessage: vi.fn().mockResolvedValue(undefined),
|
||||||
|
},
|
||||||
|
command: vi.fn(),
|
||||||
|
} as unknown as Parameters<typeof registerTelegramNativeCommands>[0]["bot"],
|
||||||
|
cfg,
|
||||||
|
runtime: {} as RuntimeEnv,
|
||||||
|
accountId,
|
||||||
|
telegramCfg: {} as TelegramAccountConfig,
|
||||||
|
allowFrom: [],
|
||||||
|
groupAllowFrom: [],
|
||||||
|
replyToMode: "off" as const,
|
||||||
|
textLimit: 4096,
|
||||||
|
useAccessGroups: false,
|
||||||
|
nativeEnabled: true,
|
||||||
|
nativeSkillsEnabled: true,
|
||||||
|
nativeDisabledExplicit: false,
|
||||||
|
resolveGroupPolicy: () => ({ allowlistEnabled: false, allowed: true }),
|
||||||
|
resolveTelegramGroupConfig: () => ({
|
||||||
|
groupConfig: undefined,
|
||||||
|
topicConfig: undefined,
|
||||||
|
}),
|
||||||
|
shouldSkipUpdate: () => false,
|
||||||
|
opts: { token: "token" },
|
||||||
|
});
|
||||||
|
|
||||||
|
it("scopes skill commands when account binding exists", () => {
|
||||||
|
const cfg: OpenClawConfig = {
|
||||||
|
agents: {
|
||||||
|
list: [{ id: "main", default: true }, { id: "butler" }],
|
||||||
|
},
|
||||||
|
bindings: [
|
||||||
|
{
|
||||||
|
agentId: "butler",
|
||||||
|
match: { channel: "telegram", accountId: "bot-a" },
|
||||||
|
},
|
||||||
|
],
|
||||||
|
};
|
||||||
|
|
||||||
|
registerTelegramNativeCommands(buildParams(cfg, "bot-a"));
|
||||||
|
|
||||||
|
expect(listSkillCommandsForAgents).toHaveBeenCalledWith({
|
||||||
|
cfg,
|
||||||
|
agentIds: ["butler"],
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it("keeps skill commands unscoped without a matching binding", () => {
|
||||||
|
const cfg: OpenClawConfig = {
|
||||||
|
agents: {
|
||||||
|
list: [{ id: "main", default: true }, { id: "butler" }],
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
registerTelegramNativeCommands(buildParams(cfg, "bot-a"));
|
||||||
|
|
||||||
|
expect(listSkillCommandsForAgents).toHaveBeenCalledWith({ cfg });
|
||||||
|
});
|
||||||
|
});
|
||||||
@ -257,8 +257,16 @@ export const registerTelegramNativeCommands = ({
|
|||||||
shouldSkipUpdate,
|
shouldSkipUpdate,
|
||||||
opts,
|
opts,
|
||||||
}: RegisterTelegramNativeCommandsParams) => {
|
}: RegisterTelegramNativeCommandsParams) => {
|
||||||
|
const boundRoute =
|
||||||
|
nativeEnabled && nativeSkillsEnabled
|
||||||
|
? resolveAgentRoute({ cfg, channel: "telegram", accountId })
|
||||||
|
: null;
|
||||||
|
const boundAgentIds =
|
||||||
|
boundRoute && boundRoute.matchedBy.startsWith("binding.") ? [boundRoute.agentId] : null;
|
||||||
const skillCommands =
|
const skillCommands =
|
||||||
nativeEnabled && nativeSkillsEnabled ? listSkillCommandsForAgents({ cfg }) : [];
|
nativeEnabled && nativeSkillsEnabled
|
||||||
|
? listSkillCommandsForAgents(boundAgentIds ? { cfg, agentIds: boundAgentIds } : { cfg })
|
||||||
|
: [];
|
||||||
const nativeCommands = nativeEnabled
|
const nativeCommands = nativeEnabled
|
||||||
? listNativeCommandSpecsForConfig(cfg, { skillCommands, provider: "telegram" })
|
? listNativeCommandSpecsForConfig(cfg, { skillCommands, provider: "telegram" })
|
||||||
: [];
|
: [];
|
||||||
|
|||||||
Loading…
Reference in New Issue
Block a user