Mainstream messaging

iMessage

Trạng thái: tích hợp CLI bên ngoài dạng native. Gateway sinh tiến trình imsg rpc và giao tiếp qua JSON-RPC trên stdio (không có daemon/cổng riêng). Các hành động nâng cao yêu cầu imsg launch và một lần thăm dò API riêng tư thành công.

Thiết lập nhanh

Local Mac (fast path)

  • Install and verify imsg

    bash
    brew install steipete/tap/imsgimsg rpc --helpimsg launchopenclaw channels status --probe
  • Configure OpenClaw

    json5
    {channels: {imessage: {enabled: true,cliPath: "/usr/local/bin/imsg",dbPath: "/Users/user/Library/Messages/chat.db",},},}
  • Start gateway

    bash
    openclaw gateway
  • Approve first DM pairing (default dmPolicy)

    bash
    openclaw pairing list imessageopenclaw pairing approve imessage <CODE>

    Yêu cầu ghép đôi hết hạn sau 1 giờ.

  • Remote Mac over SSH

    OpenClaw chỉ cần một cliPath tương thích với stdio, nên bạn có thể trỏ cliPath đến một script trình bao bọc SSH vào máy Mac từ xa và chạy imsg.

    bash
    #!/usr/bin/env bashexec ssh -T gateway-host imsg "$@"

    Cấu hình được khuyến nghị khi bật tệp đính kèm:

    json5
    {channels: {imessage: {  enabled: true,  cliPath: "~/.openclaw/scripts/imsg-ssh",  remoteHost: "user@gateway-host", // used for SCP attachment fetches  includeAttachments: true,  // Optional: override allowed attachment roots.  // Defaults include /Users/*/Library/Messages/Attachments  attachmentRoots: ["/Users/*/Library/Messages/Attachments"],  remoteAttachmentRoots: ["/Users/*/Library/Messages/Attachments"],},},}

    Nếu remoteHost chưa được đặt, OpenClaw sẽ cố tự động phát hiện bằng cách phân tích script trình bao bọc SSH. remoteHost phải là host hoặc user@host (không có khoảng trắng hoặc tùy chọn SSH). OpenClaw dùng kiểm tra khóa máy chủ nghiêm ngặt cho SCP, nên khóa máy chủ chuyển tiếp phải đã tồn tại trong ~/.ssh/known_hosts. Đường dẫn tệp đính kèm được xác thực theo các gốc được phép (attachmentRoots / remoteAttachmentRoots).

    Yêu cầu và quyền (macOS)

    • Messages phải được đăng nhập trên máy Mac chạy imsg.
    • Full Disk Access là bắt buộc cho ngữ cảnh tiến trình đang chạy OpenClaw/imsg (truy cập CSDL Messages).
    • Quyền Automation là bắt buộc để gửi tin nhắn qua Messages.app.
    • Đối với các hành động nâng cao (react / edit / unsend / threaded reply / effects / group ops), System Integrity Protection phải bị tắt — xem Bật API riêng tư imsg bên dưới. Gửi/nhận văn bản và phương tiện cơ bản hoạt động mà không cần tắt.

    Bật API riêng tư imsg

    imsg có hai chế độ vận hành:

    • Chế độ cơ bản (mặc định, không cần thay đổi SIP): gửi văn bản và phương tiện đi qua send, theo dõi/lịch sử đến, danh sách cuộc trò chuyện. Đây là những gì bạn có ngay từ đầu sau khi brew install steipete/tap/imsg mới cùng các quyền macOS tiêu chuẩn ở trên.
    • Chế độ API riêng tư: imsg tiêm một dylib trợ giúp vào Messages.app để gọi các hàm IMCore nội bộ. Đây là chế độ mở khóa react, edit, unsend, reply (theo luồng), sendWithEffect, renameGroup, setGroupIcon, addParticipant, removeParticipant, leaveGroup, cùng chỉ báo đang nhập và biên nhận đã đọc.

    Để dùng bề mặt hành động nâng cao mà trang kênh này mô tả, bạn cần chế độ API riêng tư. README của imsg nêu rõ yêu cầu này:

    Các tính năng nâng cao như read, typing, launch, gửi nội dung phong phú dựa trên bridge, sửa đổi tin nhắn và quản lý cuộc trò chuyện là tùy chọn bật. Chúng yêu cầu tắt SIP và tiêm một dylib trợ giúp vào Messages.app. imsg launch từ chối tiêm khi SIP đang bật.

    Kỹ thuật tiêm trợ giúp dùng dylib riêng của imsg để truy cập các API riêng tư của Messages. Không có máy chủ bên thứ ba hoặc runtime BlueBubbles trong đường dẫn OpenClaw iMessage.

    Thiết lập

    1. Cài đặt (hoặc nâng cấp) imsg trên máy Mac chạy Messages.app:

      bash
      brew install steipete/tap/imsgimsg --versionimsg status --json

      Đầu ra imsg status --json báo cáo bridge_version, rpc_methodsselectors theo từng phương thức để bạn có thể xem bản dựng hiện tại hỗ trợ những gì trước khi bắt đầu.

    2. Tắt System Integrity Protection. Việc này phụ thuộc vào phiên bản macOS vì yêu cầu nền tảng của Apple phụ thuộc vào hệ điều hành và phần cứng:

      • macOS 10.13–10.15 (Sierra–Catalina): tắt Library Validation qua Terminal, khởi động lại vào Recovery Mode, chạy csrutil disable, khởi động lại.
      • macOS 11+ (Big Sur trở lên), Intel: Recovery Mode (hoặc Internet Recovery), csrutil disable, khởi động lại.
      • macOS 11+, Apple Silicon: dùng chuỗi khởi động bằng nút nguồn để vào Recovery; trên các phiên bản macOS gần đây, giữ phím Left Shift khi bạn bấm Continue, rồi csrutil disable. Các thiết lập máy ảo theo một luồng riêng — hãy chụp snapshot VM trước.
      • macOS 26 / Tahoe: các chính sách library-validation và kiểm tra private-entitlement của imagent đã siết chặt hơn nữa; imsg có thể cần một bản dựng cập nhật để theo kịp. Nếu việc tiêm bằng imsg launch hoặc các selectors cụ thể bắt đầu trả về false sau một bản nâng cấp macOS lớn, hãy kiểm tra ghi chú phát hành của imsg trước khi giả định bước SIP đã thành công.

      Làm theo luồng Recovery-mode của Apple cho máy Mac của bạn để tắt SIP trước khi chạy imsg launch.

    3. Tiêm trình trợ giúp. Khi SIP đã tắt và Messages.app đã đăng nhập:

      bash
      imsg launch

      imsg launch từ chối tiêm khi SIP vẫn đang bật, nên lệnh này cũng đồng thời xác nhận rằng bước 2 đã có hiệu lực.

    4. Xác minh bridge từ OpenClaw:

      bash
      openclaw channels status --probe

      Mục iMessage phải báo cáo works, và imsg status --json | jq '.selectors' phải hiển thị retractMessagePart: true cùng bất kỳ selector edit / typing / read nào mà bản dựng macOS của bạn cung cấp. Cơ chế gating theo từng phương thức của Plugin OpenClaw trong actions.ts chỉ quảng bá các hành động có selector nền tảng là true, nên bề mặt hành động bạn thấy trong danh sách công cụ của agent phản ánh những gì bridge thực sự có thể làm trên máy chủ này.

    Nếu openclaw channels status --probe báo cáo kênh là works nhưng các hành động cụ thể ném lỗi "iMessage <action> requires the imsg private API bridge" vào lúc dispatch, hãy chạy lại imsg launch — trình trợ giúp có thể bị rơi ra (Messages.app khởi động lại, cập nhật hệ điều hành, v.v.) và trạng thái available: true được lưu trong bộ nhớ đệm sẽ tiếp tục quảng bá hành động cho đến khi lần thăm dò tiếp theo làm mới.

    Khi bạn không thể tắt SIP

    Nếu việc tắt SIP không chấp nhận được với mô hình đe dọa của bạn:

    • imsg quay về chế độ cơ bản — chỉ văn bản + phương tiện + nhận.
    • Plugin OpenClaw vẫn quảng bá gửi văn bản/phương tiện và giám sát tin nhắn đến; nó chỉ ẩn react, edit, unsend, reply, sendWithEffect và group ops khỏi bề mặt hành động (theo cổng năng lực theo từng phương thức).
    • Bạn có thể chạy một máy Mac không phải Apple Silicon riêng (hoặc một máy Mac bot chuyên dụng) đã tắt SIP cho khối lượng công việc iMessage, trong khi vẫn bật SIP trên các thiết bị chính của mình. Xem Người dùng macOS bot chuyên dụng (danh tính iMessage riêng) bên dưới.

    Kiểm soát truy cập và định tuyến

    DM policy

    channels.imessage.dmPolicy kiểm soát tin nhắn trực tiếp:

    • pairing (mặc định)
    • allowlist
    • open (yêu cầu allowFrom bao gồm "*")
    • disabled

    Trường danh sách cho phép: channels.imessage.allowFrom.

    Các mục danh sách cho phép phải xác định người gửi: handle hoặc nhóm truy cập người gửi tĩnh (accessGroup:<name>). Dùng channels.imessage.groupAllowFrom cho các mục tiêu cuộc trò chuyện như chat_id:*, chat_guid:*, hoặc chat_identifier:*; dùng channels.imessage.groups cho các khóa registry chat_id dạng số.

    Group policy + mentions

    channels.imessage.groupPolicy kiểm soát việc xử lý nhóm:

    • allowlist (mặc định khi được cấu hình)
    • open
    • disabled

    Danh sách cho phép người gửi nhóm: channels.imessage.groupAllowFrom.

    Các mục groupAllowFrom cũng có thể tham chiếu nhóm truy cập người gửi tĩnh (accessGroup:<name>).

    Dự phòng runtime: nếu groupAllowFrom chưa được đặt, kiểm tra người gửi nhóm iMessage sẽ dùng allowFrom; đặt groupAllowFrom khi việc tiếp nhận DM và nhóm nên khác nhau. Ghi chú runtime: nếu hoàn toàn thiếu channels.imessage, runtime sẽ quay về groupPolicy="allowlist" và ghi một cảnh báo (ngay cả khi channels.defaults.groupPolicy đã được đặt).

    Kiểm soát lượt nhắc đến cho nhóm:

    • iMessage không có siêu dữ liệu lượt nhắc đến gốc
    • phát hiện lượt nhắc đến dùng các mẫu regex (agents.list[].groupChat.mentionPatterns, dự phòng messages.groupChat.mentionPatterns)
    • khi không có mẫu nào được cấu hình, không thể thực thi kiểm soát lượt nhắc đến

    Lệnh điều khiển từ người gửi được ủy quyền có thể bỏ qua kiểm soát lượt nhắc đến trong nhóm.

    systemPrompt theo từng nhóm:

    Mỗi mục trong channels.imessage.groups.* chấp nhận một chuỗi systemPrompt tùy chọn. Giá trị này được chèn vào system prompt của tác tử ở mọi lượt xử lý tin nhắn trong nhóm đó. Cách phân giải phản chiếu cách phân giải prompt theo từng nhóm được dùng bởi channels.whatsapp.groups:

    1. System prompt riêng cho nhóm (groups["<chat_id>"].systemPrompt): được dùng khi mục nhóm cụ thể tồn tại trong ánh xạ khóa systemPrompt của mục đó được định nghĩa. Nếu systemPrompt là chuỗi rỗng ("") thì ký tự đại diện bị chặn và không áp dụng system prompt nào cho nhóm đó.
    2. System prompt ký tự đại diện của nhóm (groups["*"].systemPrompt): được dùng khi mục nhóm cụ thể hoàn toàn không có trong ánh xạ, hoặc khi mục đó tồn tại nhưng không định nghĩa khóa systemPrompt.
    json5
    {  channels: {    imessage: {      groupPolicy: "allowlist",      groupAllowFrom: ["+15555550123"],      groups: {        "*": { systemPrompt: "Use British spelling." },        "8421": {          requireMention: true,          systemPrompt: "This is the on-call rotation chat. Keep replies under 3 sentences.",        },        "9907": {          // explicit suppression: the wildcard "Use British spelling." does not apply here          systemPrompt: "",        },      },    },  },}

    Prompt theo từng nhóm chỉ áp dụng cho tin nhắn nhóm — tin nhắn trực tiếp trong kênh này không bị ảnh hưởng.

    Phiên và phản hồi xác định

    • DM dùng định tuyến trực tiếp; nhóm dùng định tuyến nhóm.
    • Với session.dmScope=main mặc định, DM iMessage được gộp vào phiên chính của tác tử.
    • Phiên nhóm được tách biệt (agent:<agentId>:imessage:group:<chat_id>).
    • Phản hồi được định tuyến ngược lại iMessage bằng siêu dữ liệu kênh/đích gốc.

    Hành vi luồng dạng nhóm:

    Một số luồng iMessage có nhiều người tham gia có thể đến với is_group=false. Nếu chat_id đó được cấu hình rõ ràng trong channels.imessage.groups, OpenClaw xử lý nó như lưu lượng nhóm (kiểm soát nhóm + tách biệt phiên nhóm).

    Liên kết cuộc hội thoại ACP

    Các cuộc trò chuyện iMessage cũ cũng có thể được liên kết với phiên ACP.

    Luồng thao tác nhanh:

    • Chạy /acp spawn codex --bind here bên trong DM hoặc cuộc trò chuyện nhóm được phép.
    • Các tin nhắn tương lai trong cùng cuộc hội thoại iMessage đó định tuyến đến phiên ACP đã được tạo.
    • /new/reset đặt lại chính phiên ACP đã liên kết tại chỗ.
    • /acp close đóng phiên ACP và xóa liên kết.

    Các liên kết cố định được cấu hình được hỗ trợ thông qua các mục bindings[] cấp cao nhất với type: "acp"match.channel: "imessage".

    match.peer.id có thể dùng:

    • định danh DM đã chuẩn hóa như +15555550123 hoặc [email protected]
    • chat_id:<id> (được khuyến nghị cho liên kết nhóm ổn định)
    • chat_guid:<guid>
    • chat_identifier:<identifier>

    Ví dụ:

    json5
    {  agents: {    list: [      {        id: "codex",        runtime: {          type: "acp",          acp: { agent: "codex", backend: "acpx", mode: "persistent" },        },      },    ],  },  bindings: [    {      type: "acp",      agentId: "codex",      match: {        channel: "imessage",        accountId: "default",        peer: { kind: "group", id: "chat_id:123" },      },      acp: { label: "codex-group" },    },  ],}

    Xem Tác tử ACP để biết hành vi liên kết ACP dùng chung.

    Mẫu triển khai

    Người dùng macOS bot chuyên dụng (danh tính iMessage riêng)

    Dùng một Apple ID và người dùng macOS chuyên dụng để lưu lượng bot được tách biệt khỏi hồ sơ Messages cá nhân của bạn.

    Luồng điển hình:

    1. Tạo/đăng nhập một người dùng macOS chuyên dụng.
    2. Đăng nhập vào Messages bằng Apple ID của bot trong người dùng đó.
    3. Cài đặt imsg trong người dùng đó.
    4. Tạo trình bao bọc SSH để OpenClaw có thể chạy imsg trong ngữ cảnh người dùng đó.
    5. Trỏ channels.imessage.accounts.<id>.cliPath.dbPath đến hồ sơ người dùng đó.

    Lần chạy đầu tiên có thể yêu cầu phê duyệt GUI (Automation + Full Disk Access) trong phiên người dùng bot đó.

    Mac từ xa qua Tailscale (ví dụ)

    Tô pô thường gặp:

    • Gateway chạy trên Linux/VM
    • iMessage + imsg chạy trên một máy Mac trong tailnet của bạn
    • trình bao bọc cliPath dùng SSH để chạy imsg
    • remoteHost bật lấy tệp đính kèm qua SCP

    Ví dụ:

    json5
    {  channels: {    imessage: {      enabled: true,      cliPath: "~/.openclaw/scripts/imsg-ssh",      remoteHost: "[email protected]",      includeAttachments: true,      dbPath: "/Users/bot/Library/Messages/chat.db",    },  },}
    bash
    #!/usr/bin/env bashexec ssh -T [email protected] imsg "$@"

    Dùng khóa SSH để cả SSH và SCP đều không tương tác. Đảm bảo khóa máy chủ đã được tin cậy trước (ví dụ ssh [email protected]) để known_hosts được điền.

    Mẫu nhiều tài khoản

    iMessage hỗ trợ cấu hình theo từng tài khoản trong channels.imessage.accounts.

    Mỗi tài khoản có thể ghi đè các trường như cliPath, dbPath, allowFrom, groupPolicy, mediaMaxMb, cài đặt lịch sử và danh sách cho phép gốc tệp đính kèm.

    Phương tiện, chia đoạn và đích gửi

    Tệp đính kèm và phương tiện
    • tiếp nhận tệp đính kèm đến tắt theo mặc định — đặt channels.imessage.includeAttachments: true để chuyển tiếp ảnh, bản ghi âm, video và các tệp đính kèm khác đến tác tử. Khi tắt, iMessage chỉ có tệp đính kèm sẽ bị loại bỏ trước khi đến tác tử và có thể hoàn toàn không tạo dòng nhật ký Inbound message.
    • đường dẫn tệp đính kèm từ xa có thể được lấy qua SCP khi remoteHost được đặt
    • đường dẫn tệp đính kèm phải khớp với các gốc được phép:
      • channels.imessage.attachmentRoots (cục bộ)
      • channels.imessage.remoteAttachmentRoots (chế độ SCP từ xa)
      • mẫu gốc mặc định: /Users/*/Library/Messages/Attachments
    • SCP dùng kiểm tra khóa máy chủ nghiêm ngặt (StrictHostKeyChecking=yes)
    • kích thước phương tiện gửi đi dùng channels.imessage.mediaMaxMb (mặc định 16 MB)
    Chia đoạn gửi đi
    • giới hạn đoạn văn bản: channels.imessage.textChunkLimit (mặc định 4000)
    • chế độ chia đoạn: channels.imessage.chunkMode
      • length (mặc định)
      • newline (tách ưu tiên theo đoạn văn)
    Định dạng định địa chỉ

    Đích rõ ràng được ưu tiên:

    • chat_id:123 (được khuyến nghị để định tuyến ổn định)
    • chat_guid:...
    • chat_identifier:...

    Đích theo định danh liên hệ cũng được hỗ trợ:

    bash
    imsg chats --limit 20

    Hành động API riêng tư

    Khi imsg launch đang chạy và openclaw channels status --probe báo cáo privateApi.available: true, công cụ tin nhắn có thể dùng các hành động gốc iMessage ngoài việc gửi văn bản thông thường.

    json5
    {  channels: {    imessage: {      actions: {        reactions: true,        edit: true,        unsend: true,        reply: true,        sendWithEffect: true,        sendAttachment: true,        renameGroup: true,        setGroupIcon: true,        addParticipant: true,        removeParticipant: true,        leaveGroup: true,      },    },  },}
    Hành động có sẵn
    • react: Thêm/xóa tapback iMessage (messageId, emoji, remove). Các tapback được hỗ trợ ánh xạ đến love, like, dislike, laugh, emphasize và question.
    • reply: Gửi phản hồi theo luồng tới một tin nhắn hiện có (messageId, text hoặc message, cộng với chatGuid, chatId, chatIdentifier, hoặc to).
    • sendWithEffect: Gửi văn bản với hiệu ứng iMessage (text hoặc message, effect hoặc effectId).
    • edit: Sửa một tin nhắn đã gửi trên các phiên bản macOS/API riêng tư được hỗ trợ (messageId, text hoặc newText).
    • unsend: Thu hồi một tin nhắn đã gửi trên các phiên bản macOS/API riêng tư được hỗ trợ (messageId).
    • upload-file: Gửi phương tiện/tệp (buffer dưới dạng base64 hoặc một media/path/filePath đã được nạp đầy đủ, filename, tùy chọn asVoice). Bí danh cũ: sendAttachment.
    • renameGroup, setGroupIcon, addParticipant, removeParticipant, leaveGroup: Quản lý cuộc trò chuyện nhóm khi đích hiện tại là một cuộc hội thoại nhóm.
    ID tin nhắn

    Ngữ cảnh iMessage đến bao gồm cả các giá trị MessageSid ngắn và GUID tin nhắn đầy đủ khi có. ID ngắn được giới hạn trong bộ nhớ đệm phản hồi gần đây trong bộ nhớ và được kiểm tra với cuộc trò chuyện hiện tại trước khi dùng. Nếu một ID ngắn đã hết hạn hoặc thuộc về cuộc trò chuyện khác, hãy thử lại với MessageSidFull đầy đủ.

    Phát hiện năng lực

    OpenClaw chỉ ẩn các hành động API riêng tư khi trạng thái thăm dò được lưu trong bộ nhớ đệm cho biết cầu nối không khả dụng. Nếu trạng thái chưa biết, hành động vẫn hiển thị và quá trình gửi sẽ thăm dò một cách lười để hành động đầu tiên có thể thành công sau imsg launch mà không cần làm mới trạng thái thủ công riêng.

    Biên nhận đã đọc và đang nhập

    Khi cầu nối API riêng tư hoạt động, các cuộc trò chuyện đến được chấp nhận được đánh dấu đã đọc trước khi gửi xử lý và bong bóng đang nhập được hiển thị cho người gửi trong khi tác tử tạo phản hồi. Tắt đánh dấu đã đọc bằng:

    json5
    {  channels: {    imessage: {      sendReadReceipts: false,    },  },}

    Các bản dựng imsg cũ hơn, có trước danh sách năng lực theo từng phương thức, sẽ tắt âm thầm tính năng đang nhập/đã đọc; OpenClaw ghi một cảnh báo một lần cho mỗi lần khởi động lại để có thể quy nguyên nhân cho biên nhận bị thiếu.

    Tapback đến

    OpenClaw đăng ký tapback iMessage và định tuyến các phản ứng được chấp nhận dưới dạng sự kiện hệ thống thay vì văn bản tin nhắn thông thường, vì vậy tapback của người dùng không kích hoạt vòng lặp phản hồi thông thường.

    Chế độ thông báo được điều khiển bởi channels.imessage.reactionNotifications:

    • "own" (mặc định): chỉ thông báo khi người dùng phản ứng với tin nhắn do bot tạo.
    • "all": thông báo cho tất cả tapback đến từ người gửi được ủy quyền.
    • "off": bỏ qua tapback đến.

    Ghi đè theo từng tài khoản dùng channels.imessage.accounts.<id>.reactionNotifications.

    Ghi cấu hình

    iMessage cho phép ghi cấu hình do kênh khởi tạo theo mặc định (cho /config set|unset khi commands.config: true).

    Tắt:

    json5
    {  channels: {    imessage: {      configWrites: false,    },  },}

    Gộp DM bị tách khi gửi (lệnh + URL trong một lần soạn)

    Khi người dùng nhập một lệnh cùng với URL — ví dụ Dump https://example.com/article — ứng dụng Messages của Apple tách lần gửi đó thành hai hàng chat.db riêng biệt:

    1. Một tin nhắn văn bản ("Dump").
    2. Một bong bóng xem trước URL ("https://...") với ảnh xem trước OG dưới dạng tệp đính kèm.

    Hai hàng này đến OpenClaw cách nhau khoảng 0,8-2,0 giây trên hầu hết thiết lập. Nếu không gộp, tác nhân chỉ nhận lệnh ở lượt 1, trả lời (thường là "hãy gửi URL cho tôi"), và chỉ thấy URL ở lượt 2 — lúc đó ngữ cảnh lệnh đã bị mất. Đây là pipeline gửi của Apple, không phải thứ do OpenClaw hay imsg đưa vào.

    channels.imessage.coalesceSameSenderDms chọn cho DM gộp các hàng liên tiếp từ cùng người gửi vào một lượt tác nhân duy nhất. Trò chuyện nhóm tiếp tục gửi theo từng tin nhắn để giữ nguyên cấu trúc lượt của nhiều người dùng.

    Khi nào bật

    Bật khi:

    • Bạn cung cấp Skills yêu cầu command + payload trong một tin nhắn (kết xuất, dán, lưu, đưa vào hàng đợi, v.v.).
    • Người dùng của bạn dán URL, hình ảnh, hoặc nội dung dài cùng với lệnh.
    • Bạn có thể chấp nhận độ trễ lượt DM tăng thêm (xem bên dưới).

    Để tắt khi:

    • Bạn cần độ trễ lệnh tối thiểu cho các trigger DM một từ.
    • Tất cả luồng của bạn là lệnh một lần, không có payload theo sau.

    Bật

    json5
    {  channels: {    imessage: {      coalesceSameSenderDms: true, // opt in (default: false)    },  },}

    Khi bật cờ này và không có messages.inbound.byChannel.imessage rõ ràng, cửa sổ debounce mở rộng thành 2500 ms (mặc định cũ là 0 ms — không debounce). Cửa sổ rộng hơn là bắt buộc vì nhịp gửi tách của Apple trong 0,8-2,0 giây không phù hợp với mặc định chặt hơn.

    Để tự tinh chỉnh cửa sổ:

    json5
    {  messages: {    inbound: {      byChannel: {        // 2500 ms works for most setups; raise to 4000 ms if your Mac is        // slow or under memory pressure (observed gap can stretch past 2 s        // then).        imessage: 2500,      },    },  },}

    Đánh đổi

    • Tăng độ trễ cho tin nhắn DM. Khi bật cờ này, mọi DM (bao gồm lệnh điều khiển độc lập và phản hồi tiếp theo chỉ có văn bản) sẽ chờ tối đa bằng cửa sổ debounce trước khi gửi đi, phòng trường hợp một hàng payload đang đến. Tin nhắn trò chuyện nhóm vẫn gửi tức thì.
    • Đầu ra đã gộp có giới hạn. Văn bản đã gộp giới hạn ở 4000 ký tự với marker …[truncated] rõ ràng; tệp đính kèm giới hạn ở 20; mục nguồn giới hạn ở 10 (giữ mục đầu tiên cộng với các mục mới nhất khi vượt quá). Mọi GUID nguồn được theo dõi trong coalescedMessageGuids cho telemetry hạ nguồn.
    • Chỉ DM. Trò chuyện nhóm chuyển sang gửi theo từng tin nhắn để bot vẫn phản hồi nhanh khi nhiều người đang nhập.
    • Opt-in, theo từng kênh. Các kênh khác (Telegram, WhatsApp, Slack, …) không bị ảnh hưởng. Cấu hình BlueBubbles cũ đặt channels.bluebubbles.coalesceSameSenderDms nên chuyển giá trị đó sang channels.imessage.coalesceSameSenderDms.

    Kịch bản và những gì tác nhân thấy

    Người dùng soạn chat.db tạo ra Tắt cờ (mặc định) Bật cờ + cửa sổ 2500 ms
    Dump https://example.com (một lần gửi) 2 hàng cách nhau ~1 giây Hai lượt tác nhân: chỉ "Dump", rồi URL Một lượt: văn bản đã gộp Dump https://example.com
    Save this 📎image.jpg caption (tệp đính kèm + văn bản) 2 hàng Hai lượt (tệp đính kèm bị bỏ khi gộp) Một lượt: giữ nguyên văn bản + hình ảnh
    /status (lệnh độc lập) 1 hàng Gửi tức thì Chờ tối đa bằng cửa sổ, rồi gửi
    URL được dán riêng 1 hàng Gửi tức thì Gửi tức thì (chỉ một mục trong bucket)
    Văn bản + URL được gửi dưới dạng hai tin nhắn riêng có chủ ý, cách nhau vài phút 2 hàng ngoài cửa sổ Hai lượt Hai lượt (cửa sổ hết hạn giữa chúng)
    Luồng nhanh (>10 DM nhỏ trong cửa sổ) N hàng N lượt Một lượt, đầu ra có giới hạn (áp dụng giới hạn mục đầu tiên + mới nhất, văn bản/tệp đính kèm)
    Hai người nhập trong một trò chuyện nhóm N hàng từ M người gửi M+ lượt (mỗi bucket người gửi một lượt) M+ lượt — trò chuyện nhóm không được gộp

    Bắt kịp sau khi Gateway ngừng hoạt động

    Khi Gateway ngoại tuyến (crash, khởi động lại, Mac ngủ, máy tắt), imsg watch tiếp tục từ trạng thái chat.db hiện tại sau khi Gateway hoạt động trở lại — theo mặc định, bất kỳ thứ gì đến trong khoảng gián đoạn sẽ không bao giờ được thấy. Catchup phát lại các tin nhắn đó trong lần khởi động tiếp theo để tác nhân không âm thầm bỏ lỡ lưu lượng vào.

    Catchup bị tắt theo mặc định. Bật theo từng kênh:

    ts
    channels: {  imessage: {    catchup: {      enabled: true,             // master switch (default: false)      maxAgeMinutes: 120,        // skip rows older than now - 2h (default: 120, clamp 1..720)      perRunLimit: 50,           // max rows replayed per startup (default: 50, clamp 1..500)      firstRunLookbackMinutes: 30, // first run with no cursor: look back 30 min (default: 30)      maxFailureRetries: 10,     // give up on a wedged guid after 10 dispatch failures (default: 10)    },  },}

    Cách chạy

    Một lượt cho mỗi lần khởi động monitorIMessageProvider, được sắp xếp theo thứ tự imsg launch sẵn sàng → watch.subscribeperformIMessageCatchup → vòng lặp gửi trực tiếp. Bản thân catchup dùng chats.list + messages.history theo từng cuộc trò chuyện trên cùng client JSON-RPC mà imsg watch dùng. Bất cứ thứ gì đến trong lượt catchup sẽ đi qua cơ chế gửi trực tiếp như bình thường; cache chống trùng lặp đầu vào hiện có hấp thụ mọi phần chồng lấn với các hàng được phát lại.

    Mỗi hàng được phát lại được đưa qua đường gửi trực tiếp (evaluateIMessageInbound + dispatchInboundMessage), nên allowlist, chính sách nhóm, debouncer, cache echo, và biên nhận đã đọc hoạt động giống hệt nhau trên tin nhắn được phát lại và tin nhắn trực tiếp.

    Ngữ nghĩa cursor và thử lại

    Catchup giữ một cursor theo từng tài khoản tại <openclawStateDir>/imessage/catchup/<account>__<hash>.json (thư mục trạng thái OpenClaw mặc định là ~/.openclaw, có thể ghi đè bằng OPENCLAW_STATE_DIR):

    json
    {  "lastSeenMs": 1717900800000,  "lastSeenRowid": 482910,  "updatedAt": 1717900801234,  "failureRetries": { "<guid>": 1 }}
    • Cursor tiến lên sau mỗi lần gửi thành công và được giữ nguyên khi việc gửi của một hàng ném lỗi — lần khởi động tiếp theo thử lại cùng hàng đó từ cursor đã giữ.
    • Sau maxFailureRetries lần ném lỗi liên tiếp với cùng guid, catchup ghi log warn và buộc cursor tiến qua tin nhắn bị kẹt để các lần khởi động tiếp theo có thể tiếp tục.
    • Các guid đã bị bỏ cuộc được bỏ qua ngay khi thấy (không thử gửi) trong các lần chạy sau và được tính vào skippedGivenUp trong tóm tắt lượt chạy.

    Tín hiệu hiển thị cho người vận hành

    Code
    imessage catchup: replayed=N skippedFromMe=… skippedGivenUp=… failed=… givenUp=… fetchedCount=…imessage catchup: giving up on guid=<guid> after &lt;N&gt; failures; advancing cursor past itimessage catchup: fetched &lt;X&gt; rows across chats, capped to perRunLimit=&lt;Y&gt;

    Một dòng WARN ... capped to perRunLimit nghĩa là một lần khởi động chưa xả hết toàn bộ backlog. Tăng perRunLimit (tối đa 500) nếu các khoảng gián đoạn của bạn thường xuyên vượt quá lượt mặc định 50 hàng.

    Khi nào nên để tắt

    • Gateway chạy liên tục với watchdog tự khởi động lại và các khoảng gián đoạn luôn < vài giây — mặc định tắt là ổn.
    • Lưu lượng DM thấp và tin nhắn bị bỏ lỡ sẽ không thay đổi hành vi tác nhân — cửa sổ ban đầu firstRunLookbackMinutes có thể gửi ngữ cảnh cũ bất ngờ trong lần bật đầu tiên.

    Khi bạn bật catchup, lần khởi động đầu tiên không có cursor chỉ nhìn lại firstRunLookbackMinutes (mặc định 30 phút), không phải toàn bộ cửa sổ maxAgeMinutes — điều này tránh phát lại một lịch sử dài các tin nhắn trước khi bật.

    Khắc phục sự cố

    Không tìm thấy imsg hoặc RPC không được hỗ trợ

    Xác thực binary và hỗ trợ RPC:

    bash
    imsg rpc --helpimsg status --jsonopenclaw channels status --probe

    Nếu probe báo RPC không được hỗ trợ, hãy cập nhật imsg. Nếu các hành động API riêng không khả dụng, hãy chạy imsg launch trong phiên người dùng macOS đã đăng nhập và probe lại. Nếu Gateway không chạy trên macOS, hãy dùng thiết lập Mac từ xa qua SSH ở trên thay vì đường dẫn imsg local mặc định.

    Gateway không chạy trên macOS

    cliPath: "imsg" mặc định phải chạy trên Mac đã đăng nhập vào Messages. Trên Linux hoặc Windows, đặt channels.imessage.cliPath thành script wrapper SSH đến Mac đó và chạy imsg "$@".

    bash
    #!/usr/bin/env bashexec ssh -T messages-mac imsg "$@"

    Sau đó chạy:

    bash
    openclaw channels status --probe --channel imessage
    DM bị bỏ qua

    Kiểm tra:

    • channels.imessage.dmPolicy
    • channels.imessage.allowFrom
    • phê duyệt ghép đôi (openclaw pairing list imessage)
    Tin nhắn nhóm bị bỏ qua

    Kiểm tra:

    • channels.imessage.groupPolicy
    • channels.imessage.groupAllowFrom
    • hành vi allowlist của channels.imessage.groups
    • cấu hình mẫu nhắc đến (agents.list[].groupChat.mentionPatterns)
    Tệp đính kèm từ xa thất bại

    Kiểm tra:

    • channels.imessage.remoteHost
    • channels.imessage.remoteAttachmentRoots
    • xác thực khóa SSH/SCP từ máy chủ Gateway
    • khóa máy chủ tồn tại trong ~/.ssh/known_hosts trên máy chủ Gateway
    • khả năng đọc đường dẫn từ xa trên Mac đang chạy Messages
    Đã bỏ lỡ lời nhắc quyền macOS

    Chạy lại trong terminal GUI tương tác trong cùng ngữ cảnh người dùng/phiên và phê duyệt lời nhắc:

    bash
    imsg chats --limit 1imsg send <handle> "test"

    Xác nhận Full Disk Access + Automation đã được cấp cho ngữ cảnh tiến trình chạy OpenClaw/imsg.

    Con trỏ tham chiếu cấu hình

    Liên quan

    Was this useful?