Zalo Personal
Zalo Personal (unofficial)
Status: experimental. This integration automates a personal Zalo account via native zca-js inside RemoteClaw.
Warning: This is an unofficial integration and may result in account suspension/ban. Use at your own risk.
Plugin required
Zalo Personal ships as a plugin and is not bundled with the core install.
- Install via CLI:
remoteclaw plugins install @remoteclaw/zalouser - Or from a source checkout:
remoteclaw plugins install ./extensions/zalouser - Details: Plugins
No external zca/openzca CLI binary is required.
Quick setup (beginner)
- Install the plugin (see above).
- Login (QR, on the Gateway machine):
remoteclaw channels login --channel zalouser- Scan the QR code with the Zalo mobile app.
- Enable the channel:
{ channels: { zalouser: { enabled: true, dmPolicy: "pairing", }, },}- Restart the Gateway (or finish setup).
- DM access defaults to pairing; approve the pairing code on first contact.
What it is
- Runs entirely in-process via
zca-js. - Uses native event listeners to receive inbound messages.
- Sends replies directly through the JS API (text/media/link).
- Designed for “personal account” use cases where Zalo Bot API is not available.
Naming
Channel id is zalouser to make it explicit this automates a personal Zalo user account (unofficial). We keep zalo reserved for a potential future official Zalo API integration.
Finding IDs (directory)
Use the directory CLI to discover peers/groups and their IDs:
remoteclaw directory self --channel zalouserremoteclaw directory peers list --channel zalouser --query "name"remoteclaw directory groups list --channel zalouser --query "work"Limits
- Outbound text is chunked to ~2000 characters (Zalo client limits).
- Streaming is blocked by default.
Access control (DMs)
channels.zalouser.dmPolicy supports: pairing | allowlist | open | disabled (default: pairing).
channels.zalouser.allowFrom accepts user IDs or names. During setup, names are resolved to IDs using the plugin’s in-process contact lookup.
Approve via:
remoteclaw pairing list zalouserremoteclaw pairing approve zalouser <code>
Group access (optional)
- Default:
channels.zalouser.groupPolicy = "open"(groups allowed). Usechannels.defaults.groupPolicyto override the default when unset. - Restrict to an allowlist with:
channels.zalouser.groupPolicy = "allowlist"channels.zalouser.groups(keys should be stable group IDs; names are resolved to IDs on startup when possible)channels.zalouser.groupAllowFrom(controls which senders in allowed groups can trigger the bot)
- Block all groups:
channels.zalouser.groupPolicy = "disabled". - The configure wizard can prompt for group allowlists.
- On startup, RemoteClaw resolves group/user names in allowlists to IDs and logs the mapping.
- Group allowlist matching is ID-only by default. Unresolved names are ignored for auth unless
channels.zalouser.dangerouslyAllowNameMatching: trueis enabled. channels.zalouser.dangerouslyAllowNameMatching: trueis a break-glass compatibility mode that re-enables mutable group-name matching.- If
groupAllowFromis unset, runtime falls back toallowFromfor group sender checks. - Sender checks apply to both normal group messages and control commands (for example
/new,/reset).
Example:
{ channels: { zalouser: { groupPolicy: "allowlist", groupAllowFrom: ["1471383327500481391"], groups: { "123456789": { allow: true }, "Work Chat": { allow: true }, }, }, },}Group mention gating
channels.zalouser.groups.<group>.requireMentioncontrols whether group replies require a mention.- Resolution order: exact group id/name -> normalized group slug ->
*-> default (true). - This applies both to allowlisted groups and open group mode.
- Authorized control commands (for example
/new) can bypass mention gating. - When a group message is skipped because mention is required, RemoteClaw stores it as pending group history and includes it on the next processed group message.
- Group history limit defaults to
messages.groupChat.historyLimit(fallback50). You can override per account withchannels.zalouser.historyLimit.
Example:
{ channels: { zalouser: { groupPolicy: "allowlist", groups: { "*": { allow: true, requireMention: true }, "Work Chat": { allow: true, requireMention: false }, }, }, },}Multi-account
Accounts map to zalouser profiles in RemoteClaw state. Example:
{ channels: { zalouser: { enabled: true, defaultAccount: "default", accounts: { work: { enabled: true, profile: "work" }, }, }, },}Typing, reactions, and delivery acknowledgements
- RemoteClaw sends a typing event before dispatching a reply (best-effort).
- Message reaction action
reactis supported forzalouserin channel actions.- Use
remove: trueto remove a specific reaction emoji from a message. - Reaction semantics: Reactions
- Use
- For inbound messages that include event metadata, RemoteClaw sends delivered + seen acknowledgements (best-effort).
Troubleshooting
Login doesn’t stick:
remoteclaw channels status --probe- Re-login:
remoteclaw channels logout --channel zalouser && remoteclaw channels login --channel zalouser
Allowlist/group name didn’t resolve:
- Use numeric IDs in
allowFrom/groupAllowFrom/groups, or exact friend/group names.
Upgraded from old CLI-based setup:
- Remove any old external
zcaprocess assumptions. - The channel now runs fully in RemoteClaw without external CLI binaries.