@gakr-gakr/whatsapp 0.1.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.
- package/action-runtime-api.ts +1 -0
- package/action-runtime.runtime.ts +1 -0
- package/api.ts +67 -0
- package/auth-presence.ts +80 -0
- package/autobot.plugin.json +23 -0
- package/channel-config-api.ts +1 -0
- package/channel-plugin-api.ts +3 -0
- package/config-api.ts +4 -0
- package/constants.ts +1 -0
- package/contract-api.ts +29 -0
- package/directory-contract-api.ts +4 -0
- package/doctor-contract-api.ts +8 -0
- package/index.ts +16 -0
- package/legacy-session-surface-api.ts +6 -0
- package/legacy-state-migrations-api.ts +1 -0
- package/light-runtime-api.ts +12 -0
- package/login-qr-api.ts +1 -0
- package/login-qr-runtime.ts +23 -0
- package/outbound-payload-test-api.ts +1 -0
- package/package.json +76 -0
- package/runtime-api.ts +84 -0
- package/secret-contract-api.ts +4 -0
- package/security-contract-api.ts +4 -0
- package/setup-entry.ts +21 -0
- package/setup-plugin-api.ts +3 -0
- package/src/account-config.ts +77 -0
- package/src/account-ids.ts +17 -0
- package/src/account-types.ts +5 -0
- package/src/accounts.ts +176 -0
- package/src/action-runtime-target-auth.ts +27 -0
- package/src/action-runtime.ts +76 -0
- package/src/active-listener.ts +17 -0
- package/src/agent-tools-login.ts +113 -0
- package/src/approval-auth.ts +27 -0
- package/src/auth-store.runtime.ts +1 -0
- package/src/auth-store.ts +494 -0
- package/src/auto-reply/config.runtime.ts +16 -0
- package/src/auto-reply/constants.ts +1 -0
- package/src/auto-reply/deliver-reply.ts +332 -0
- package/src/auto-reply/loggers.ts +6 -0
- package/src/auto-reply/mentions.ts +131 -0
- package/src/auto-reply/monitor/ack-reaction.ts +99 -0
- package/src/auto-reply/monitor/audio-preflight.runtime.ts +9 -0
- package/src/auto-reply/monitor/broadcast.ts +153 -0
- package/src/auto-reply/monitor/commands.ts +19 -0
- package/src/auto-reply/monitor/echo.ts +64 -0
- package/src/auto-reply/monitor/group-activation.runtime.ts +1 -0
- package/src/auto-reply/monitor/group-activation.ts +73 -0
- package/src/auto-reply/monitor/group-gating.runtime.ts +8 -0
- package/src/auto-reply/monitor/group-gating.ts +218 -0
- package/src/auto-reply/monitor/group-members.ts +65 -0
- package/src/auto-reply/monitor/inbound-context.ts +92 -0
- package/src/auto-reply/monitor/inbound-dispatch.runtime.ts +22 -0
- package/src/auto-reply/monitor/inbound-dispatch.ts +749 -0
- package/src/auto-reply/monitor/last-route.ts +61 -0
- package/src/auto-reply/monitor/listener-log.ts +28 -0
- package/src/auto-reply/monitor/message-line.runtime.ts +38 -0
- package/src/auto-reply/monitor/message-line.ts +54 -0
- package/src/auto-reply/monitor/on-message.ts +333 -0
- package/src/auto-reply/monitor/peer.ts +17 -0
- package/src/auto-reply/monitor/process-message.ts +584 -0
- package/src/auto-reply/monitor/runtime-api.ts +36 -0
- package/src/auto-reply/monitor/status-reaction.ts +108 -0
- package/src/auto-reply/monitor-state.ts +114 -0
- package/src/auto-reply/monitor.ts +720 -0
- package/src/auto-reply/reply-resolver.runtime.ts +1 -0
- package/src/auto-reply/types.ts +48 -0
- package/src/auto-reply/util.ts +62 -0
- package/src/auto-reply.impl.ts +6 -0
- package/src/auto-reply.ts +1 -0
- package/src/channel-actions.runtime.ts +7 -0
- package/src/channel-actions.ts +85 -0
- package/src/channel-outbound.ts +87 -0
- package/src/channel-react-action.runtime.ts +10 -0
- package/src/channel-react-action.ts +247 -0
- package/src/channel.runtime.ts +117 -0
- package/src/channel.setup.ts +32 -0
- package/src/channel.ts +356 -0
- package/src/command-policy.ts +7 -0
- package/src/config-accessors.ts +22 -0
- package/src/config-schema.ts +6 -0
- package/src/config-ui-hints.ts +24 -0
- package/src/connection-controller-registry.ts +49 -0
- package/src/connection-controller.ts +680 -0
- package/src/creds-files.ts +19 -0
- package/src/creds-persistence.ts +71 -0
- package/src/directory-config.ts +40 -0
- package/src/doctor-contract.ts +11 -0
- package/src/doctor.ts +56 -0
- package/src/document-filename.ts +17 -0
- package/src/group-intro.ts +15 -0
- package/src/group-policy.ts +40 -0
- package/src/group-session-contract.ts +20 -0
- package/src/group-session-key.ts +42 -0
- package/src/heartbeat.ts +34 -0
- package/src/identity.ts +164 -0
- package/src/inbound/access-control.ts +187 -0
- package/src/inbound/dedupe.ts +132 -0
- package/src/inbound/extract.ts +484 -0
- package/src/inbound/lifecycle.ts +39 -0
- package/src/inbound/media.ts +128 -0
- package/src/inbound/monitor.ts +1042 -0
- package/src/inbound/outbound-mentions.ts +260 -0
- package/src/inbound/runtime-api.ts +7 -0
- package/src/inbound/save-media.runtime.ts +1 -0
- package/src/inbound/send-api.ts +203 -0
- package/src/inbound/send-result.ts +109 -0
- package/src/inbound/types.ts +107 -0
- package/src/inbound-policy.ts +215 -0
- package/src/inbound.ts +9 -0
- package/src/login-qr.ts +542 -0
- package/src/login.ts +83 -0
- package/src/media.ts +10 -0
- package/src/monitor-inbox.allows-messages-from-senders-allowfrom-list.test-support.ts +417 -0
- package/src/monitor-inbox.append-upsert.test-support.ts +133 -0
- package/src/monitor-inbox.blocks-messages-from-unauthorized-senders-not-allowfrom.test-support.ts +418 -0
- package/src/monitor-inbox.captures-media-path-image-messages.test-support.ts +308 -0
- package/src/monitor-inbox.streams-inbound-messages.test-support.ts +824 -0
- package/src/normalize-target.ts +148 -0
- package/src/normalize.ts +8 -0
- package/src/outbound-adapter.ts +36 -0
- package/src/outbound-base.ts +256 -0
- package/src/outbound-media-contract.ts +307 -0
- package/src/outbound-media.runtime.ts +41 -0
- package/src/outbound-send-deps.ts +1 -0
- package/src/outbound-test-support.ts +16 -0
- package/src/qa-driver.runtime.ts +189 -0
- package/src/qr-image.ts +1 -0
- package/src/qr-terminal.ts +1 -0
- package/src/quoted-message.ts +184 -0
- package/src/reaction-level.ts +24 -0
- package/src/reconnect.ts +55 -0
- package/src/resolve-outbound-target.ts +58 -0
- package/src/runtime-api.ts +59 -0
- package/src/runtime-group-policy.ts +16 -0
- package/src/runtime.ts +9 -0
- package/src/security-contract.ts +47 -0
- package/src/security-fix.ts +71 -0
- package/src/send.ts +342 -0
- package/src/session-contract.ts +43 -0
- package/src/session-errors.ts +125 -0
- package/src/session-route.ts +32 -0
- package/src/session.runtime.ts +8 -0
- package/src/session.ts +327 -0
- package/src/setup-core.ts +52 -0
- package/src/setup-finalize.ts +450 -0
- package/src/setup-surface.ts +71 -0
- package/src/setup-test-helpers.ts +217 -0
- package/src/shared.ts +291 -0
- package/src/socket-timing.ts +38 -0
- package/src/state-migrations.ts +55 -0
- package/src/status-issues.ts +185 -0
- package/src/system-prompt.ts +31 -0
- package/src/targets-runtime.ts +221 -0
- package/src/text-runtime.ts +18 -0
- package/src/vcard.ts +84 -0
- package/targets.ts +5 -0
- package/test-api.ts +2 -0
- package/tsconfig.json +16 -0
|
@@ -0,0 +1,215 @@
|
|
|
1
|
+
import { resolveStableChannelMessageIngress } from "autobot/plugin-sdk/channel-ingress-runtime";
|
|
2
|
+
import {
|
|
3
|
+
resolveChannelGroupPolicy,
|
|
4
|
+
resolveChannelGroupRequireMention,
|
|
5
|
+
} from "autobot/plugin-sdk/channel-policy";
|
|
6
|
+
import type {
|
|
7
|
+
ChannelGroupPolicy,
|
|
8
|
+
DmPolicy,
|
|
9
|
+
GroupPolicy,
|
|
10
|
+
AutoBotConfig,
|
|
11
|
+
} from "autobot/plugin-sdk/config-contracts";
|
|
12
|
+
import { resolveDefaultGroupPolicy } from "autobot/plugin-sdk/runtime-group-policy";
|
|
13
|
+
import { resolveGroupSessionKey } from "autobot/plugin-sdk/session-store-runtime";
|
|
14
|
+
import { resolveWhatsAppAccount, type ResolvedWhatsAppAccount } from "./accounts.js";
|
|
15
|
+
import { getSelfIdentity, getSenderIdentity } from "./identity.js";
|
|
16
|
+
import type { WebInboundMessage } from "./inbound/types.js";
|
|
17
|
+
import { resolveWhatsAppRuntimeGroupPolicy } from "./runtime-group-policy.js";
|
|
18
|
+
import { isSelfChatMode, normalizeE164 } from "./text-runtime.js";
|
|
19
|
+
|
|
20
|
+
export type ResolvedWhatsAppInboundPolicy = {
|
|
21
|
+
account: ResolvedWhatsAppAccount;
|
|
22
|
+
dmPolicy: DmPolicy;
|
|
23
|
+
groupPolicy: GroupPolicy;
|
|
24
|
+
configuredAllowFrom: string[];
|
|
25
|
+
dmAllowFrom: string[];
|
|
26
|
+
groupAllowFrom: string[];
|
|
27
|
+
isSelfChat: boolean;
|
|
28
|
+
providerMissingFallbackApplied: boolean;
|
|
29
|
+
isSamePhone: (value?: string | null) => boolean;
|
|
30
|
+
resolveConversationGroupPolicy: (conversationId: string) => ChannelGroupPolicy;
|
|
31
|
+
resolveConversationRequireMention: (conversationId: string) => boolean;
|
|
32
|
+
};
|
|
33
|
+
|
|
34
|
+
function normalizeWhatsAppIngressPhone(value: string): string | null {
|
|
35
|
+
const trimmed = value.trim();
|
|
36
|
+
if (!trimmed) {
|
|
37
|
+
return null;
|
|
38
|
+
}
|
|
39
|
+
return normalizeE164(trimmed);
|
|
40
|
+
}
|
|
41
|
+
|
|
42
|
+
function resolveGroupConversationId(conversationId: string): string {
|
|
43
|
+
return (
|
|
44
|
+
resolveGroupSessionKey({
|
|
45
|
+
From: conversationId,
|
|
46
|
+
ChatType: "group",
|
|
47
|
+
Provider: "whatsapp",
|
|
48
|
+
})?.id ?? conversationId
|
|
49
|
+
);
|
|
50
|
+
}
|
|
51
|
+
|
|
52
|
+
function maybeSamePhoneDmAllowFrom(params: {
|
|
53
|
+
isGroup: boolean;
|
|
54
|
+
policy: ResolvedWhatsAppInboundPolicy;
|
|
55
|
+
dmSenderId?: string | null;
|
|
56
|
+
}): string[] {
|
|
57
|
+
if (params.isGroup || !params.dmSenderId || !params.policy.isSamePhone(params.dmSenderId)) {
|
|
58
|
+
return [];
|
|
59
|
+
}
|
|
60
|
+
return [params.dmSenderId];
|
|
61
|
+
}
|
|
62
|
+
|
|
63
|
+
function buildResolvedWhatsAppGroupConfig(params: {
|
|
64
|
+
groupPolicy: GroupPolicy;
|
|
65
|
+
groups: ResolvedWhatsAppAccount["groups"];
|
|
66
|
+
}): AutoBotConfig {
|
|
67
|
+
return {
|
|
68
|
+
channels: {
|
|
69
|
+
whatsapp: {
|
|
70
|
+
groupPolicy: params.groupPolicy,
|
|
71
|
+
groups: params.groups,
|
|
72
|
+
},
|
|
73
|
+
},
|
|
74
|
+
} as AutoBotConfig;
|
|
75
|
+
}
|
|
76
|
+
|
|
77
|
+
export function resolveWhatsAppInboundPolicy(params: {
|
|
78
|
+
cfg: AutoBotConfig;
|
|
79
|
+
accountId?: string | null;
|
|
80
|
+
selfE164?: string | null;
|
|
81
|
+
}): ResolvedWhatsAppInboundPolicy {
|
|
82
|
+
const account = resolveWhatsAppAccount({
|
|
83
|
+
cfg: params.cfg,
|
|
84
|
+
accountId: params.accountId,
|
|
85
|
+
});
|
|
86
|
+
const configuredAllowFrom = account.allowFrom ?? [];
|
|
87
|
+
const dmPolicy = account.dmPolicy ?? "pairing";
|
|
88
|
+
const dmAllowFrom =
|
|
89
|
+
configuredAllowFrom.length > 0 ? configuredAllowFrom : params.selfE164 ? [params.selfE164] : [];
|
|
90
|
+
const configuredGroupAllowFrom =
|
|
91
|
+
Array.isArray(account.groupAllowFrom) && account.groupAllowFrom.length > 0
|
|
92
|
+
? account.groupAllowFrom
|
|
93
|
+
: undefined;
|
|
94
|
+
const groupAllowFrom =
|
|
95
|
+
configuredGroupAllowFrom ??
|
|
96
|
+
(configuredAllowFrom.length > 0 ? configuredAllowFrom : undefined) ??
|
|
97
|
+
[];
|
|
98
|
+
const defaultGroupPolicy = resolveDefaultGroupPolicy(params.cfg);
|
|
99
|
+
const { groupPolicy, providerMissingFallbackApplied } = resolveWhatsAppRuntimeGroupPolicy({
|
|
100
|
+
providerConfigPresent: params.cfg.channels?.whatsapp !== undefined,
|
|
101
|
+
groupPolicy: account.groupPolicy,
|
|
102
|
+
defaultGroupPolicy,
|
|
103
|
+
});
|
|
104
|
+
const resolvedGroupCfg = buildResolvedWhatsAppGroupConfig({
|
|
105
|
+
groupPolicy,
|
|
106
|
+
groups: account.groups,
|
|
107
|
+
});
|
|
108
|
+
const isSamePhone = (value?: string | null) =>
|
|
109
|
+
typeof value === "string" && typeof params.selfE164 === "string" && value === params.selfE164;
|
|
110
|
+
return {
|
|
111
|
+
account,
|
|
112
|
+
dmPolicy,
|
|
113
|
+
groupPolicy,
|
|
114
|
+
configuredAllowFrom,
|
|
115
|
+
dmAllowFrom,
|
|
116
|
+
groupAllowFrom,
|
|
117
|
+
isSelfChat: account.selfChatMode ?? isSelfChatMode(params.selfE164, configuredAllowFrom),
|
|
118
|
+
providerMissingFallbackApplied,
|
|
119
|
+
isSamePhone,
|
|
120
|
+
resolveConversationGroupPolicy: (conversationId) =>
|
|
121
|
+
resolveChannelGroupPolicy({
|
|
122
|
+
cfg: resolvedGroupCfg,
|
|
123
|
+
channel: "whatsapp",
|
|
124
|
+
groupId: resolveGroupConversationId(conversationId),
|
|
125
|
+
hasGroupAllowFrom: groupAllowFrom.length > 0,
|
|
126
|
+
}),
|
|
127
|
+
resolveConversationRequireMention: (conversationId) =>
|
|
128
|
+
resolveChannelGroupRequireMention({
|
|
129
|
+
cfg: resolvedGroupCfg,
|
|
130
|
+
channel: "whatsapp",
|
|
131
|
+
groupId: resolveGroupConversationId(conversationId),
|
|
132
|
+
}),
|
|
133
|
+
};
|
|
134
|
+
}
|
|
135
|
+
|
|
136
|
+
export async function resolveWhatsAppIngressAccess(params: {
|
|
137
|
+
cfg: AutoBotConfig;
|
|
138
|
+
policy: ResolvedWhatsAppInboundPolicy;
|
|
139
|
+
isGroup: boolean;
|
|
140
|
+
conversationId: string;
|
|
141
|
+
senderId?: string | null;
|
|
142
|
+
dmSenderId?: string | null;
|
|
143
|
+
includeCommand?: boolean;
|
|
144
|
+
}) {
|
|
145
|
+
const samePhoneDmAllowFrom = maybeSamePhoneDmAllowFrom({
|
|
146
|
+
isGroup: params.isGroup,
|
|
147
|
+
policy: params.policy,
|
|
148
|
+
dmSenderId: params.dmSenderId,
|
|
149
|
+
});
|
|
150
|
+
const dmAllowFrom = [...params.policy.dmAllowFrom, ...samePhoneDmAllowFrom];
|
|
151
|
+
return await resolveStableChannelMessageIngress({
|
|
152
|
+
channelId: "whatsapp",
|
|
153
|
+
accountId: params.policy.account.accountId,
|
|
154
|
+
identity: {
|
|
155
|
+
key: "whatsapp-sender-phone",
|
|
156
|
+
kind: "phone",
|
|
157
|
+
normalize: normalizeWhatsAppIngressPhone,
|
|
158
|
+
sensitivity: "pii",
|
|
159
|
+
entryIdPrefix: "whatsapp-entry",
|
|
160
|
+
},
|
|
161
|
+
cfg: params.cfg,
|
|
162
|
+
useDefaultPairingStore: true,
|
|
163
|
+
subject: { stableId: params.senderId ?? "" },
|
|
164
|
+
conversation: {
|
|
165
|
+
kind: params.isGroup ? "group" : "direct",
|
|
166
|
+
id: params.conversationId,
|
|
167
|
+
},
|
|
168
|
+
dmPolicy: params.policy.dmPolicy,
|
|
169
|
+
groupPolicy: params.policy.groupPolicy,
|
|
170
|
+
policy: {
|
|
171
|
+
groupAllowFromFallbackToAllowFrom: false,
|
|
172
|
+
},
|
|
173
|
+
allowFrom: dmAllowFrom,
|
|
174
|
+
groupAllowFrom: params.policy.groupAllowFrom,
|
|
175
|
+
command: params.includeCommand === true ? {} : undefined,
|
|
176
|
+
});
|
|
177
|
+
}
|
|
178
|
+
|
|
179
|
+
export async function resolveWhatsAppCommandAuthorized(params: {
|
|
180
|
+
cfg: AutoBotConfig;
|
|
181
|
+
msg: WebInboundMessage;
|
|
182
|
+
policy?: ResolvedWhatsAppInboundPolicy;
|
|
183
|
+
}): Promise<boolean> {
|
|
184
|
+
const useAccessGroups = params.cfg.commands?.useAccessGroups !== false;
|
|
185
|
+
if (!useAccessGroups) {
|
|
186
|
+
return true;
|
|
187
|
+
}
|
|
188
|
+
|
|
189
|
+
const self = getSelfIdentity(params.msg);
|
|
190
|
+
const policy =
|
|
191
|
+
params.policy ??
|
|
192
|
+
resolveWhatsAppInboundPolicy({
|
|
193
|
+
cfg: params.cfg,
|
|
194
|
+
accountId: params.msg.accountId,
|
|
195
|
+
selfE164: self.e164 ?? null,
|
|
196
|
+
});
|
|
197
|
+
const isGroup = params.msg.chatType === "group";
|
|
198
|
+
const sender = getSenderIdentity(params.msg);
|
|
199
|
+
const dmSender = sender.e164 ?? params.msg.from ?? "";
|
|
200
|
+
const groupSender = sender.e164 ?? "";
|
|
201
|
+
if (!normalizeE164(isGroup ? groupSender : dmSender)) {
|
|
202
|
+
return false;
|
|
203
|
+
}
|
|
204
|
+
|
|
205
|
+
const access = await resolveWhatsAppIngressAccess({
|
|
206
|
+
cfg: params.cfg,
|
|
207
|
+
policy,
|
|
208
|
+
isGroup,
|
|
209
|
+
conversationId: params.msg.conversationId ?? params.msg.chatId ?? params.msg.from,
|
|
210
|
+
senderId: isGroup ? groupSender : dmSender,
|
|
211
|
+
dmSenderId: dmSender,
|
|
212
|
+
includeCommand: true,
|
|
213
|
+
});
|
|
214
|
+
return access.commandAccess.authorized;
|
|
215
|
+
}
|
package/src/inbound.ts
ADDED
|
@@ -0,0 +1,9 @@
|
|
|
1
|
+
export { resetWebInboundDedupe } from "./inbound/dedupe.js";
|
|
2
|
+
export {
|
|
3
|
+
extractContactContext,
|
|
4
|
+
extractLocationData,
|
|
5
|
+
extractMediaPlaceholder,
|
|
6
|
+
extractText,
|
|
7
|
+
} from "./inbound/extract.js";
|
|
8
|
+
export { monitorWebInbox } from "./inbound/monitor.js";
|
|
9
|
+
export type { WebInboundMessage, WebListenerCloseReason } from "./inbound/types.js";
|