react-native-mcp-kit 4.2.1 → 4.4.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/dist/bin/ios-hid CHANGED
Binary file
@@ -52,6 +52,49 @@ export interface ServerContext {
52
52
  export declare const jsonError: (msg: string) => {
53
53
  content: TextContent[];
54
54
  };
55
+ export type ClientIdsParse = {
56
+ ids: string[];
57
+ mode: 'broadcast';
58
+ ok: true;
59
+ } | {
60
+ clientId: string | undefined;
61
+ mode: 'single';
62
+ ok: true;
63
+ } | {
64
+ error: string;
65
+ ok: false;
66
+ };
67
+ /**
68
+ * Normalises the `clientId` arg accepted by every public tool.
69
+ *
70
+ * Forms:
71
+ * undefined / null — auto-resolve to the single connected client.
72
+ * "ios-1" — literal single-client target (image content passes
73
+ * through, single-shape response).
74
+ * "/^ios/" — regex against connected client IDs; always returns
75
+ * the broadcast envelope (even if matched count is 1).
76
+ * ["ios-1","..."] — array form: literals and regex strings can be mixed.
77
+ * Regex entries are expanded against `bridge.listClients()`
78
+ * and unioned with literals before dedup.
79
+ *
80
+ * Errors when an array entry is non-string / array is empty / a regex matches
81
+ * zero connected clients. Literals that aren't connected are NOT pre-validated
82
+ * here — they fall through to dispatchTool which reports a per-client error,
83
+ * matching the broadcast "not fail-fast" semantics.
84
+ */
85
+ export declare const parseClientIds: (raw: unknown, bridge: Bridge) => ClientIdsParse;
86
+ export interface BroadcastDispatch {
87
+ clientId: string;
88
+ result: DispatchResult;
89
+ }
90
+ /**
91
+ * Build MCP content blocks from per-client dispatch results. When every result
92
+ * is text-only, the function returns a single JSON envelope so the agent can
93
+ * parse one blob; when any result carries image content (e.g. screenshots),
94
+ * results are emitted as per-client blocks prefixed with `## <clientId>`
95
+ * headers so each image stays paired with its source.
96
+ */
97
+ export declare const buildBroadcastContent: (results: BroadcastDispatch[], formatResult: (result: unknown) => Array<TextContent | ImageContent>) => Array<TextContent | ImageContent>;
55
98
  /**
56
99
  * Parse a `call`-style args argument that may arrive as a JSON string (older
57
100
  * clients) or a plain object (new form). Returns { ok, args } or { ok: false,
@@ -1 +1 @@
1
- {"version":3,"file":"helpers.d.ts","sourceRoot":"","sources":["../../src/server/helpers.ts"],"names":[],"mappings":"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"}
1
+ {"version":3,"file":"helpers.d.ts","sourceRoot":"","sources":["../../src/server/helpers.ts"],"names":[],"mappings":"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"}
@@ -1,6 +1,6 @@
1
1
  "use strict";
2
2
  Object.defineProperty(exports, "__esModule", { value: true });
3
- exports.formatResult = exports.buildToolGroups = exports.findToolInClient = exports.canonicalizeGroup = exports.canonicalize = exports.parseCallArgs = exports.jsonError = void 0;
3
+ exports.formatResult = exports.buildToolGroups = exports.findToolInClient = exports.canonicalizeGroup = exports.canonicalize = exports.parseCallArgs = exports.buildBroadcastContent = exports.parseClientIds = exports.jsonError = void 0;
4
4
  const protocol_1 = require("../shared/protocol");
5
5
  const jsonError = (msg) => {
6
6
  return {
@@ -8,6 +8,166 @@ const jsonError = (msg) => {
8
8
  };
9
9
  };
10
10
  exports.jsonError = jsonError;
11
+ // Same `/body/flags` literal recogniser used by logBox.parsePattern — keep the
12
+ // vocabulary in sync so an agent that knows the form from fiber_tree hook
13
+ // filters can use it here too.
14
+ const REGEX_LITERAL = /^\/(.+)\/([gimsuy]*)$/;
15
+ const tryParseRegex = (raw) => {
16
+ const m = raw.match(REGEX_LITERAL);
17
+ if (!m)
18
+ return null;
19
+ try {
20
+ return new RegExp(m[1], m[2]);
21
+ }
22
+ catch (err) {
23
+ return { error: `Invalid regex in clientId "${raw}": ${err.message}` };
24
+ }
25
+ };
26
+ /**
27
+ * Normalises the `clientId` arg accepted by every public tool.
28
+ *
29
+ * Forms:
30
+ * undefined / null — auto-resolve to the single connected client.
31
+ * "ios-1" — literal single-client target (image content passes
32
+ * through, single-shape response).
33
+ * "/^ios/" — regex against connected client IDs; always returns
34
+ * the broadcast envelope (even if matched count is 1).
35
+ * ["ios-1","..."] — array form: literals and regex strings can be mixed.
36
+ * Regex entries are expanded against `bridge.listClients()`
37
+ * and unioned with literals before dedup.
38
+ *
39
+ * Errors when an array entry is non-string / array is empty / a regex matches
40
+ * zero connected clients. Literals that aren't connected are NOT pre-validated
41
+ * here — they fall through to dispatchTool which reports a per-client error,
42
+ * matching the broadcast "not fail-fast" semantics.
43
+ */
44
+ const parseClientIds = (raw, bridge) => {
45
+ if (raw === undefined || raw === null)
46
+ return { clientId: undefined, mode: 'single', ok: true };
47
+ if (typeof raw === 'string') {
48
+ const re = tryParseRegex(raw);
49
+ if (re === null)
50
+ return { clientId: raw, mode: 'single', ok: true };
51
+ if ('error' in re)
52
+ return { error: re.error, ok: false };
53
+ const ids = expandRegex(re, bridge);
54
+ if (ids.length === 0) {
55
+ return { error: `Pattern "${raw}" matched no connected clients.`, ok: false };
56
+ }
57
+ return { ids, mode: 'broadcast', ok: true };
58
+ }
59
+ if (Array.isArray(raw)) {
60
+ if (raw.length === 0) {
61
+ return { error: 'clientId array must contain at least one entry.', ok: false };
62
+ }
63
+ const seen = new Set();
64
+ const ids = [];
65
+ for (const entry of raw) {
66
+ if (typeof entry !== 'string') {
67
+ return { error: 'clientId array must contain strings only.', ok: false };
68
+ }
69
+ const re = tryParseRegex(entry);
70
+ if (re && 'error' in re)
71
+ return { error: re.error, ok: false };
72
+ if (re instanceof RegExp) {
73
+ const matched = expandRegex(re, bridge);
74
+ if (matched.length === 0) {
75
+ return { error: `Pattern "${entry}" matched no connected clients.`, ok: false };
76
+ }
77
+ for (const id of matched) {
78
+ if (!seen.has(id)) {
79
+ seen.add(id);
80
+ ids.push(id);
81
+ }
82
+ }
83
+ }
84
+ else if (!seen.has(entry)) {
85
+ seen.add(entry);
86
+ ids.push(entry);
87
+ }
88
+ }
89
+ return { ids, mode: 'broadcast', ok: true };
90
+ }
91
+ return { error: 'clientId must be a string or an array of strings.', ok: false };
92
+ };
93
+ exports.parseClientIds = parseClientIds;
94
+ const expandRegex = (re, bridge) => {
95
+ return bridge
96
+ .listClients()
97
+ .filter((c) => {
98
+ return re.test(c.id);
99
+ })
100
+ .map((c) => {
101
+ return c.id;
102
+ });
103
+ };
104
+ /**
105
+ * Detects an MCP image-content payload (the shape host__screenshot returns)
106
+ * without re-running formatResult. Used to pick the broadcast aggregation
107
+ * strategy — text-only results collapse into a single JSON envelope, image
108
+ * results keep per-client blocks so each image is anchored to its source.
109
+ */
110
+ const isImageResult = (result) => {
111
+ if (!Array.isArray(result) || result.length === 0)
112
+ return false;
113
+ const first = result[0];
114
+ return (typeof first === 'object' &&
115
+ first !== null &&
116
+ 'type' in first &&
117
+ first.type === 'image');
118
+ };
119
+ /**
120
+ * Build MCP content blocks from per-client dispatch results. When every result
121
+ * is text-only, the function returns a single JSON envelope so the agent can
122
+ * parse one blob; when any result carries image content (e.g. screenshots),
123
+ * results are emitted as per-client blocks prefixed with `## <clientId>`
124
+ * headers so each image stays paired with its source.
125
+ */
126
+ const buildBroadcastContent = (results, formatResult) => {
127
+ const hasImage = results.some(({ result }) => {
128
+ return result.ok && isImageResult(result.result);
129
+ });
130
+ const okCount = results.reduce((n, { result }) => {
131
+ return n + (result.ok ? 1 : 0);
132
+ }, 0);
133
+ const failedCount = results.length - okCount;
134
+ if (!hasImage) {
135
+ return [
136
+ {
137
+ text: JSON.stringify({
138
+ failedCount,
139
+ okCount,
140
+ results: results.map(({ clientId, result }) => {
141
+ if (result.ok)
142
+ return { clientId, ok: true, result: result.result };
143
+ return { clientId, error: result.error, ok: false };
144
+ }),
145
+ }, null, 2),
146
+ type: 'text',
147
+ },
148
+ ];
149
+ }
150
+ const blocks = [
151
+ {
152
+ text: `Broadcast: ${okCount} ok, ${failedCount} failed (${results.length} clients).`,
153
+ type: 'text',
154
+ },
155
+ ];
156
+ for (const { clientId, result } of results) {
157
+ blocks.push({ text: `## ${clientId}`, type: 'text' });
158
+ if (result.ok) {
159
+ blocks.push(...formatResult(result.result));
160
+ }
161
+ else {
162
+ blocks.push({
163
+ text: JSON.stringify({ error: result.error }, null, 2),
164
+ type: 'text',
165
+ });
166
+ }
167
+ }
168
+ return blocks;
169
+ };
170
+ exports.buildBroadcastContent = buildBroadcastContent;
11
171
  /**
12
172
  * Parse a `call`-style args argument that may arrive as a JSON string (older
13
173
  * clients) or a plain object (new form). Returns { ok, args } or { ok: false,
@@ -1 +1 @@
1
- {"version":3,"file":"helpers.js","sourceRoot":"","sources":["../../src/server/helpers.ts"],"names":[],"mappings":"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"}
1
+ {"version":3,"file":"helpers.js","sourceRoot":"","sources":["../../src/server/helpers.ts"],"names":[],"mappings":";;;AAAA,gDAAqD;AAmD9C,MAAM,SAAS,GAAG,CAAC,GAAW,EAA8B,EAAE;IACnE,OAAO;QACL,OAAO,EAAE,CAAC,EAAE,IAAI,EAAE,IAAI,CAAC,SAAS,CAAC,EAAE,KAAK,EAAE,GAAG,EAAE,CAAC,EAAE,IAAI,EAAE,MAAe,EAAE,CAAC;KAC3E,CAAC;AACJ,CAAC,CAAC;AAJW,QAAA,SAAS,aAIpB;AAOF,+EAA+E;AAC/E,0EAA0E;AAC1E,+BAA+B;AAC/B,MAAM,aAAa,GAAG,uBAAuB,CAAC;AAE9C,MAAM,aAAa,GAAG,CAAC,GAAW,EAAqC,EAAE;IACvE,MAAM,CAAC,GAAG,GAAG,CAAC,KAAK,CAAC,aAAa,CAAC,CAAC;IACnC,IAAI,CAAC,CAAC;QAAE,OAAO,IAAI,CAAC;IACpB,IAAI,CAAC;QACH,OAAO,IAAI,MAAM,CAAC,CAAC,CAAC,CAAC,CAAE,EAAE,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;IACjC,CAAC;IAAC,OAAO,GAAG,EAAE,CAAC;QACb,OAAO,EAAE,KAAK,EAAE,8BAA8B,GAAG,MAAO,GAAa,CAAC,OAAO,EAAE,EAAE,CAAC;IACpF,CAAC;AACH,CAAC,CAAC;AAEF;;;;;;;;;;;;;;;;;GAiBG;AACI,MAAM,cAAc,GAAG,CAAC,GAAY,EAAE,MAAc,EAAkB,EAAE;IAC7E,IAAI,GAAG,KAAK,SAAS,IAAI,GAAG,KAAK,IAAI;QAAE,OAAO,EAAE,QAAQ,EAAE,SAAS,EAAE,IAAI,EAAE,QAAQ,EAAE,EAAE,EAAE,IAAI,EAAE,CAAC;IAEhG,IAAI,OAAO,GAAG,KAAK,QAAQ,EAAE,CAAC;QAC5B,MAAM,EAAE,GAAG,aAAa,CAAC,GAAG,CAAC,CAAC;QAC9B,IAAI,EAAE,KAAK,IAAI;YAAE,OAAO,EAAE,QAAQ,EAAE,GAAG,EAAE,IAAI,EAAE,QAAQ,EAAE,EAAE,EAAE,IAAI,EAAE,CAAC;QACpE,IAAI,OAAO,IAAI,EAAE;YAAE,OAAO,EAAE,KAAK,EAAE,EAAE,CAAC,KAAK,EAAE,EAAE,EAAE,KAAK,EAAE,CAAC;QACzD,MAAM,GAAG,GAAG,WAAW,CAAC,EAAE,EAAE,MAAM,CAAC,CAAC;QACpC,IAAI,GAAG,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;YACrB,OAAO,EAAE,KAAK,EAAE,YAAY,GAAG,iCAAiC,EAAE,EAAE,EAAE,KAAK,EAAE,CAAC;QAChF,CAAC;QACD,OAAO,EAAE,GAAG,EAAE,IAAI,EAAE,WAAW,EAAE,EAAE,EAAE,IAAI,EAAE,CAAC;IAC9C,CAAC;IAED,IAAI,KAAK,CAAC,OAAO,CAAC,GAAG,CAAC,EAAE,CAAC;QACvB,IAAI,GAAG,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;YACrB,OAAO,EAAE,KAAK,EAAE,iDAAiD,EAAE,EAAE,EAAE,KAAK,EAAE,CAAC;QACjF,CAAC;QACD,MAAM,IAAI,GAAG,IAAI,GAAG,EAAU,CAAC;QAC/B,MAAM,GAAG,GAAa,EAAE,CAAC;QACzB,KAAK,MAAM,KAAK,IAAI,GAAG,EAAE,CAAC;YACxB,IAAI,OAAO,KAAK,KAAK,QAAQ,EAAE,CAAC;gBAC9B,OAAO,EAAE,KAAK,EAAE,2CAA2C,EAAE,EAAE,EAAE,KAAK,EAAE,CAAC;YAC3E,CAAC;YACD,MAAM,EAAE,GAAG,aAAa,CAAC,KAAK,CAAC,CAAC;YAChC,IAAI,EAAE,IAAI,OAAO,IAAI,EAAE;gBAAE,OAAO,EAAE,KAAK,EAAE,EAAE,CAAC,KAAK,EAAE,EAAE,EAAE,KAAK,EAAE,CAAC;YAC/D,IAAI,EAAE,YAAY,MAAM,EAAE,CAAC;gBACzB,MAAM,OAAO,GAAG,WAAW,CAAC,EAAE,EAAE,MAAM,CAAC,CAAC;gBACxC,IAAI,OAAO,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;oBACzB,OAAO,EAAE,KAAK,EAAE,YAAY,KAAK,iCAAiC,EAAE,EAAE,EAAE,KAAK,EAAE,CAAC;gBAClF,CAAC;gBACD,KAAK,MAAM,EAAE,IAAI,OAAO,EAAE,CAAC;oBACzB,IAAI,CAAC,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,EAAE,CAAC;wBAClB,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC;wBACb,GAAG,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;oBACf,CAAC;gBACH,CAAC;YACH,CAAC;iBAAM,IAAI,CAAC,IAAI,CAAC,GAAG,CAAC,KAAK,CAAC,EAAE,CAAC;gBAC5B,IAAI,CAAC,GAAG,CAAC,KAAK,CAAC,CAAC;gBAChB,GAAG,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;YAClB,CAAC;QACH,CAAC;QACD,OAAO,EAAE,GAAG,EAAE,IAAI,EAAE,WAAW,EAAE,EAAE,EAAE,IAAI,EAAE,CAAC;IAC9C,CAAC;IAED,OAAO,EAAE,KAAK,EAAE,mDAAmD,EAAE,EAAE,EAAE,KAAK,EAAE,CAAC;AACnF,CAAC,CAAC;AA9CW,QAAA,cAAc,kBA8CzB;AAEF,MAAM,WAAW,GAAG,CAAC,EAAU,EAAE,MAAc,EAAY,EAAE;IAC3D,OAAO,MAAM;SACV,WAAW,EAAE;SACb,MAAM,CAAC,CAAC,CAAC,EAAE,EAAE;QACZ,OAAO,EAAE,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC;IACvB,CAAC,CAAC;SACD,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE;QACT,OAAO,CAAC,CAAC,EAAE,CAAC;IACd,CAAC,CAAC,CAAC;AACP,CAAC,CAAC;AAEF;;;;;GAKG;AACH,MAAM,aAAa,GAAG,CAAC,MAAe,EAAW,EAAE;IACjD,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,MAAM,CAAC,IAAI,MAAM,CAAC,MAAM,KAAK,CAAC;QAAE,OAAO,KAAK,CAAC;IAChE,MAAM,KAAK,GAAG,MAAM,CAAC,CAAC,CAAC,CAAC;IACxB,OAAO,CACL,OAAO,KAAK,KAAK,QAAQ;QACzB,KAAK,KAAK,IAAI;QACd,MAAM,IAAI,KAAK;QACd,KAA2B,CAAC,IAAI,KAAK,OAAO,CAC9C,CAAC;AACJ,CAAC,CAAC;AAOF;;;;;;GAMG;AACI,MAAM,qBAAqB,GAAG,CACnC,OAA4B,EAC5B,YAAoE,EACjC,EAAE;IACrC,MAAM,QAAQ,GAAG,OAAO,CAAC,IAAI,CAAC,CAAC,EAAE,MAAM,EAAE,EAAE,EAAE;QAC3C,OAAO,MAAM,CAAC,EAAE,IAAI,aAAa,CAAC,MAAM,CAAC,MAAM,CAAC,CAAC;IACnD,CAAC,CAAC,CAAC;IAEH,MAAM,OAAO,GAAG,OAAO,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,EAAE,MAAM,EAAE,EAAE,EAAE;QAC/C,OAAO,CAAC,GAAG,CAAC,MAAM,CAAC,EAAE,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;IACjC,CAAC,EAAE,CAAC,CAAC,CAAC;IACN,MAAM,WAAW,GAAG,OAAO,CAAC,MAAM,GAAG,OAAO,CAAC;IAE7C,IAAI,CAAC,QAAQ,EAAE,CAAC;QACd,OAAO;YACL;gBACE,IAAI,EAAE,IAAI,CAAC,SAAS,CAClB;oBACE,WAAW;oBACX,OAAO;oBACP,OAAO,EAAE,OAAO,CAAC,GAAG,CAAC,CAAC,EAAE,QAAQ,EAAE,MAAM,EAAE,EAAE,EAAE;wBAC5C,IAAI,MAAM,CAAC,EAAE;4BAAE,OAAO,EAAE,QAAQ,EAAE,EAAE,EAAE,IAAI,EAAE,MAAM,EAAE,MAAM,CAAC,MAAM,EAAE,CAAC;wBACpE,OAAO,EAAE,QAAQ,EAAE,KAAK,EAAE,MAAM,CAAC,KAAK,EAAE,EAAE,EAAE,KAAK,EAAE,CAAC;oBACtD,CAAC,CAAC;iBACH,EACD,IAAI,EACJ,CAAC,CACF;gBACD,IAAI,EAAE,MAAe;aACtB;SACF,CAAC;IACJ,CAAC;IAED,MAAM,MAAM,GAAsC;QAChD;YACE,IAAI,EAAE,cAAc,OAAO,QAAQ,WAAW,YAAY,OAAO,CAAC,MAAM,YAAY;YACpF,IAAI,EAAE,MAAe;SACtB;KACF,CAAC;IACF,KAAK,MAAM,EAAE,QAAQ,EAAE,MAAM,EAAE,IAAI,OAAO,EAAE,CAAC;QAC3C,MAAM,CAAC,IAAI,CAAC,EAAE,IAAI,EAAE,MAAM,QAAQ,EAAE,EAAE,IAAI,EAAE,MAAe,EAAE,CAAC,CAAC;QAC/D,IAAI,MAAM,CAAC,EAAE,EAAE,CAAC;YACd,MAAM,CAAC,IAAI,CAAC,GAAG,YAAY,CAAC,MAAM,CAAC,MAAM,CAAC,CAAC,CAAC;QAC9C,CAAC;aAAM,CAAC;YACN,MAAM,CAAC,IAAI,CAAC;gBACV,IAAI,EAAE,IAAI,CAAC,SAAS,CAAC,EAAE,KAAK,EAAE,MAAM,CAAC,KAAK,EAAE,EAAE,IAAI,EAAE,CAAC,CAAC;gBACtD,IAAI,EAAE,MAAe;aACtB,CAAC,CAAC;QACL,CAAC;IACH,CAAC;IACD,OAAO,MAAM,CAAC;AAChB,CAAC,CAAC;AAnDW,QAAA,qBAAqB,yBAmDhC;AAEF;;;;GAIG;AACI,MAAM,aAAa,GAAG,CAC3B,GAAY,EACgE,EAAE;IAC9E,IAAI,GAAG,KAAK,SAAS,IAAI,GAAG,KAAK,IAAI;QAAE,OAAO,EAAE,IAAI,EAAE,EAAE,EAAE,EAAE,EAAE,IAAI,EAAE,CAAC;IACrE,IAAI,OAAO,GAAG,KAAK,QAAQ,EAAE,CAAC;QAC5B,IAAI,GAAG,CAAC,MAAM,KAAK,CAAC;YAAE,OAAO,EAAE,IAAI,EAAE,EAAE,EAAE,EAAE,EAAE,IAAI,EAAE,CAAC;QACpD,IAAI,CAAC;YACH,MAAM,MAAM,GAAG,IAAI,CAAC,KAAK,CAAC,GAAG,CAAY,CAAC;YAC1C,IAAI,MAAM,IAAI,OAAO,MAAM,KAAK,QAAQ,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,MAAM,CAAC,EAAE,CAAC;gBACnE,OAAO,EAAE,IAAI,EAAE,MAAiC,EAAE,EAAE,EAAE,IAAI,EAAE,CAAC;YAC/D,CAAC;YACD,OAAO,EAAE,KAAK,EAAE,gCAAgC,EAAE,EAAE,EAAE,KAAK,EAAE,CAAC;QAChE,CAAC;QAAC,MAAM,CAAC;YACP,OAAO,EAAE,KAAK,EAAE,uBAAuB,EAAE,EAAE,EAAE,KAAK,EAAE,CAAC;QACvD,CAAC;IACH,CAAC;IACD,IAAI,OAAO,GAAG,KAAK,QAAQ,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,GAAG,CAAC,EAAE,CAAC;QACnD,OAAO,EAAE,IAAI,EAAE,GAA8B,EAAE,EAAE,EAAE,IAAI,EAAE,CAAC;IAC5D,CAAC;IACD,OAAO,EAAE,KAAK,EAAE,0CAA0C,EAAE,EAAE,EAAE,KAAK,EAAE,CAAC;AAC1E,CAAC,CAAC;AApBW,QAAA,aAAa,iBAoBxB;AAEF;;;;;GAKG;AACI,MAAM,YAAY,GAAG,CAAC,KAAc,EAAU,EAAE;IACrD,OAAO,IAAI,CAAC,SAAS,CAAC,KAAK,EAAE,CAAC,IAAI,EAAE,CAAU,EAAE,EAAE;QAChD,IAAI,CAAC,IAAI,OAAO,CAAC,KAAK,QAAQ,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC,CAAC,EAAE,CAAC;YACpD,MAAM,MAAM,GAA4B,EAAE,CAAC;YAC3C,KAAK,MAAM,CAAC,IAAI,MAAM,CAAC,IAAI,CAAC,CAA4B,CAAC,CAAC,IAAI,EAAE,EAAE,CAAC;gBACjE,MAAM,CAAC,CAAC,CAAC,GAAI,CAA6B,CAAC,CAAC,CAAC,CAAC;YAChD,CAAC;YACD,OAAO,MAAM,CAAC;QAChB,CAAC;QACD,OAAO,CAAC,CAAC;IACX,CAAC,CAAC,CAAC;AACL,CAAC,CAAC;AAXW,QAAA,YAAY,gBAWvB;AAEF;;;;GAIG;AACI,MAAM,iBAAiB,GAAG,CAAC,KAAgB,EAAU,EAAE;IAC5D,MAAM,UAAU,GAAG;QACjB,WAAW,EAAE,KAAK,CAAC,WAAW;QAC9B,MAAM,EAAE,KAAK,CAAC,MAAM;QACpB,KAAK,EAAE,CAAC,GAAG,KAAK,CAAC,KAAK,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,EAAE,EAAE;YACpC,OAAO,CAAC,CAAC,IAAI,CAAC,aAAa,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC;QACtC,CAAC,CAAC;KACH,CAAC;IACF,OAAO,IAAA,oBAAY,EAAC,UAAU,CAAC,CAAC;AAClC,CAAC,CAAC;AATW,QAAA,iBAAiB,qBAS5B;AAEF;;;;GAIG;AACI,MAAM,gBAAgB,GAAG,CAC9B,MAAmB,EACnB,YAAoB,EACQ,EAAE;IAC9B,KAAK,MAAM,GAAG,IAAI,MAAM,CAAC,OAAO,EAAE,CAAC;QACjC,MAAM,MAAM,GAAG,GAAG,GAAG,CAAC,IAAI,GAAG,2BAAgB,EAAE,CAAC;QAChD,IAAI,YAAY,CAAC,UAAU,CAAC,MAAM,CAAC,EAAE,CAAC;YACpC,MAAM,UAAU,GAAG,YAAY,CAAC,KAAK,CAAC,MAAM,CAAC,MAAM,CAAC,CAAC;YACrD,MAAM,OAAO,GAAG,GAAG,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,EAAE;gBACnC,OAAO,CAAC,CAAC,IAAI,KAAK,UAAU,CAAC;YAC/B,CAAC,CAAC,CAAC;YACH,IAAI,OAAO,EAAE,CAAC;gBACZ,OAAO;oBACL,WAAW,EAAE,OAAO,CAAC,WAAW;oBAChC,WAAW,EAAE,OAAO,CAAC,WAAW;oBAChC,IAAI,EAAE,YAAY;iBACnB,CAAC;YACJ,CAAC;QACH,CAAC;IACH,CAAC;IAED,MAAM,YAAY,GAAG,MAAM,CAAC,YAAY,CAAC,GAAG,CAAC,YAAY,CAAC,CAAC;IAC3D,IAAI,YAAY,EAAE,CAAC;QACjB,OAAO;YACL,WAAW,EAAE,YAAY,CAAC,WAAW;YACrC,WAAW,EAAE,YAAY,CAAC,WAAW;YACrC,IAAI,EAAE,YAAY;SACnB,CAAC;IACJ,CAAC;IAED,OAAO,IAAI,CAAC;AACd,CAAC,CAAC;AA/BW,QAAA,gBAAgB,oBA+B3B;AAEF;;;;GAIG;AACI,MAAM,eAAe,GAAG,CAAC,MAAmB,EAAe,EAAE;IAClE,MAAM,MAAM,GAAgB,MAAM,CAAC,OAAO,CAAC,GAAG,CAAC,CAAC,GAAG,EAAE,EAAE;QACrD,OAAO;YACL,WAAW,EAAE,GAAG,CAAC,WAAW;YAC5B,MAAM,EAAE,GAAG,CAAC,IAAI;YAChB,KAAK,EAAE,GAAG,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE;gBACzB,OAAO;oBACL,WAAW,EAAE,CAAC,CAAC,WAAW;oBAC1B,WAAW,EAAE,CAAC,CAAC,WAAW;oBAC1B,IAAI,EAAE,GAAG,GAAG,CAAC,IAAI,GAAG,2BAAgB,GAAG,CAAC,CAAC,IAAI,EAAE;iBAChD,CAAC;YACJ,CAAC,CAAC;SACH,CAAC;IACJ,CAAC,CAAC,CAAC;IAEH,IAAI,MAAM,CAAC,YAAY,CAAC,IAAI,GAAG,CAAC,EAAE,CAAC;QACjC,MAAM,eAAe,GAAG,IAAI,GAAG,EAG5B,CAAC;QACJ,KAAK,MAAM,CAAC,QAAQ,EAAE,IAAI,CAAC,IAAI,MAAM,CAAC,YAAY,EAAE,CAAC;YACnD,MAAM,QAAQ,GAAG,eAAe,CAAC,GAAG,CAAC,IAAI,CAAC,MAAM,CAAC,IAAI,EAAE,CAAC;YACxD,QAAQ,CAAC,IAAI,CAAC;gBACZ,WAAW,EAAE,IAAI,CAAC,WAAW;gBAC7B,WAAW,EAAE,IAAI,CAAC,WAAW;gBAC7B,IAAI,EAAE,QAAQ;aACf,CAAC,CAAC;YACH,eAAe,CAAC,GAAG,CAAC,IAAI,CAAC,MAAM,EAAE,QAAQ,CAAC,CAAC;QAC7C,CAAC;QACD,KAAK,MAAM,CAAC,MAAM,EAAE,QAAQ,CAAC,IAAI,eAAe,EAAE,CAAC;YACjD,MAAM,CAAC,IAAI,CAAC;gBACV,WAAW,EAAE,oDAAoD;gBACjE,MAAM,EAAE,GAAG,MAAM,YAAY;gBAC7B,KAAK,EAAE,QAAQ;aAChB,CAAC,CAAC;QACL,CAAC;IACH,CAAC;IAED,OAAO,MAAM,CAAC;AAChB,CAAC,CAAC;AAvCW,QAAA,eAAe,mBAuC1B;AAEF;;;GAGG;AACI,MAAM,YAAY,GAAG,CAAC,MAAe,EAAqC,EAAE;IACjF,IAAI,KAAK,CAAC,OAAO,CAAC,MAAM,CAAC,IAAI,MAAM,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;QAC/C,MAAM,KAAK,GAAG,MAAM,CAAC,CAAC,CAAC,CAAC;QACxB,IAAI,OAAO,KAAK,KAAK,QAAQ,IAAI,KAAK,KAAK,IAAI,IAAI,MAAM,IAAI,KAAK,IAAI,KAAK,CAAC,IAAI,KAAK,OAAO,EAAE,CAAC;YAC7F,OAAO,MAA2C,CAAC;QACrD,CAAC;IACH,CAAC;IACD,OAAO,CAAC,EAAE,IAAI,EAAE,IAAI,CAAC,SAAS,CAAC,MAAM,EAAE,IAAI,EAAE,CAAC,CAAC,EAAE,IAAI,EAAE,MAAe,EAAE,CAAC,CAAC;AAC5E,CAAC,CAAC;AARW,QAAA,YAAY,gBAQvB"}
@@ -1,2 +1,2 @@
1
- export declare const BASE_INSTRUCTIONS = "You are connected to a running React Native app via the react-native-mcp-kit bridge.\n\nMultiple React Native apps can connect simultaneously \u2014 each is identified by a short ID like \"ios-1\", \"android-1\", or \"client-1\". Use `connection_status` or `list_tools` to see which clients are connected and their IDs, platforms, and labels.\n\n## How to interact\n\n1. Use `connection_status` to check which clients are connected.\n2. Use `list_tools` to browse all available tool names and short descriptions. The response is compact \u2014 modules that are structurally identical across multiple clients are deduplicated into a single entry with a `clientIds` array, and input schemas are omitted. Narrow the listing with `{ module }` or `{ clientId }`, or pass `{ compact: true }` to drop module-level descriptions.\n3. Use `describe_tool` with `{ tool, clientId? }` to fetch the full input schema of a specific tool before calling it. Required when you need to know the argument shape. Host tools are resolved directly (no clientId needed). For in-app tools, omit `clientId` to auto-pick; specify it only when multiple clients have the same tool with different schemas.\n4. Use `call` to invoke any tool with format: module__method (e.g. navigation__navigate). When more than one client is connected, specify `clientId`. When exactly one client is connected, `clientId` is optional \u2014 it's auto-picked. `args` accepts either a plain object or a JSON string \u2014 prefer objects to avoid quote escaping.\n5. Use `wait_until` to poll any tool until a predicate over its result holds (or timeout). Replaces \"screenshot in a loop + sleep\" for state-level waits (\"wait for network to idle\", \"wait for state key X to become Y\"). Predicate supports compound forms: { all: [...] } (AND), { any: [...] } (OR), { not: predicate }.\n6. For UI-level waits (\"wait for a screen to appear\", \"wait for a spinner to disappear\") use `fiber_tree__query` with `waitFor: { until: \"appear\" | \"disappear\", timeout?, interval?, stable? }` \u2014 it polls the same query with cache bypassed until the target state holds. `stable: <ms>` requires continuous presence/absence for that many ms to ignore transient matches during screen transitions.\n7. Use `assert` for a single-shot checkpoint after actions \u2014 same predicate vocabulary as wait_until, returns { pass, actual, expected?, result? }. Natural pair: do action \u2192 wait_until / fiber_tree waitFor \u2192 assert.\n8. Use `host__tap_fiber` to collapse \"fiber_tree__query \u2192 host__tap at bounds\" into one call. Pass fiber_tree steps; if exactly one fiber matches, its center is tapped. Ambiguous match returns the candidate list so you can add `index` or narrow the chain.\n\nSome tools run inline on the MCP server host (e.g. `host__screenshot`, `host__list_devices`, `host__launch_app`, `host__terminate_app`, `host__restart_app`, `metro__reload`, `metro__symbolicate`) and work even when no React Native client is connected. They use xcrun simctl / adb on the dev machine. When `clientId` is provided, host tools use that client's platform/label/deviceId as hints to resolve the target device; otherwise they prefer the device of the single connected client, falling back to the single booted sim / online device. `launch_app`, `terminate_app`, and `restart_app` accept an `appId` arg (iOS bundle ID / Android package name); omit it to reuse the target client's registered `bundleId` from its connection metadata.\n\n## Driving the UI \u2014 pick the right tool\n1. **`host__tap_fiber` with `steps: [...]`** \u2014 the canonical way to simulate a user tap. One call locates the fiber via fiber_tree__query and taps its center through the real OS gesture pipeline, so Pressable feedback, gesture responders, and hit-test logic all run. Ambiguous matches return a candidate list so you can add `index` or narrow `steps`. This is what you want whenever the user asks to simulate a tap / press / button click.\n2. **`fiber_tree__query` with `select: [\"mcpId\",\"name\",\"bounds\"]` + `host__tap`** when you want to inspect a match set before committing \u2014 e.g. verify bounds, or skim candidates before picking one. `props` is opt-in on `select` to keep responses small.\n3. **`fiber_tree__call`** for non-gesture callbacks or imperative ref methods. Pass `prop` to call a callback prop (`{ prop: 'onPress' }`) or `method` to call a native-ref method (`{ method: 'focus' }`). Good when the component is off-screen / virtualised, when a scroll-handler parent swallows taps, or when you're driving focus / blur / measure / scrollTo via the native ref. For simulating a user tap, prefer tap_fiber (above).\n4. **`host__screenshot` + manual coordinate estimation + `host__tap`** ONLY for non-React surfaces: system permission dialogs, native alerts, the on-screen keyboard, WebView content, native splash. These have no fiber and no bounds. Pair with `region: { x, y, width, height }` to screenshot just the area you're inspecting \u2014 vision-token cheap.\n\nGesture tools: `host__tap` / `host__long_press` / `host__swipe` / `host__drag` / `host__type_text` / `host__type_text_batch` / `host__press_key` work on both platforms with no external daemons: Android via `adb shell input`, iOS via a bundled `ios-hid` binary that injects HID events directly into iOS Simulator through SimulatorKit.\n\nStack traces: `errors__get_errors` and `log_box__get_logs` return parsed `stackFrames` you can pass straight into `metro__symbolicate` to resolve bundled frames back to source paths via Metro.\n\n## Consolidated tools \u2014 verb + arg, not verb-per-action\n\nEvery module exposes one verb per concept rather than one tool per variant. Reach for the listing tool with a filter arg, not a tool named after the filter:\n\n- **console** \u2014 `get_logs({ level? })` for all levels (no per-level shortcuts).\n- **errors** \u2014 `get_errors({ fatal? })` covers fatal-only too.\n- **network** \u2014 `get_requests({ status?, method?, url? })` covers errors / pending / URL filter.\n- **log_box** \u2014 `clear({ level? })` (warn / error / syntax / all); `set_installed({ enabled })` toggles install/uninstall.\n- **navigation** \u2014 `navigate({ mode: 'reuse' | 'push' | 'replace' })` covers navigate/push/replace; `pop({ to: <number> | <screenName> | 'top' })` covers pop / pop_to / pop_to_top; `get_current_route({ withState? })` covers the route-state read too.\n- **query** (reactQuery) \u2014 `mutate({ action: 'invalidate' | 'refetch' | 'remove' | 'reset', key? })` covers all four cache mutations.\n- **device** \u2014 `info({ select?: [...] })` aggregates platform / dimensions / pixelRatio / appearance / appState / accessibility / keyboard / initialUrl / dev / identity / app / battery / memoryStorage. `open_url({ dryRun? })` covers the canOpenURL check too.\n- **fiber_tree** \u2014 `query` (all inspection: mcpId / name / testID / props / hooks / bounds / refMethods / children) and `call({ prop? | method? })` (callback prop OR native-ref method) \u2014 two tools cover everything fibers offer.\n\n## Path-based drill into heavy responses\n\nAll listing tools that return heavy JSON (console / network / errors / storage / reactQuery / log_box / navigation / metro events / fiber_tree) accept the standard `path` / `depth` / `maxBytes` projection args. Heavy nested values collapse to compact `${...}`-keyed markers; primitives stay raw. Drill into a specific subtree via the same tool with `path` \u2014 no separate by-id fetcher.\n\n `network__get_requests({ path: '[-1:][0].response.body' })` \u2014 last request's body\n `network__get_requests({ path: '[-1:][0].response.body', depth: 8 })` \u2014 fully expanded\n `console__get_logs({ path: '[-3:][0].args[1]' })` \u2014 second arg of the third-from-last log\n `storage__get_item({ key: 'session', path: 'value.user.email' })` \u2014 drill into a stored value\n `query__get_data({ key: '[\"users\"]', path: 'data.email' })` \u2014 drill into cached data\n\nPath syntax (JS-style):\n `.key` or `[\"key.with.dots\"]` \u2014 object key access\n `[N]` \u2014 index (numeric for arrays, Nth char for strings, Nth key in insertion order for objects)\n `[start:end]` / `[start:]` / `[:end]` \u2014 slice (Python/jq-style; negative indices count from end). Works on arrays, strings, and objects (key slice). After array slice, chained `.key` maps over each element; `[N]` picks one.\n\nPath drilling to a string scalar applies `previewCap` as usual \u2014 long leaves still wrap in a `${str}` marker. To get a raw substring, end the path with a slice: `stack[0:500]` returns the first 500 chars verbatim (slice = explicit truncation request, previewCap bypassed). Default `previewCap` is 250; override per call via `previewCap: <N>`.\n\n`depth` (default per tool, max 8) controls how many container levels are walked before collapsing to a marker. Bump `depth` for an exploratory survey, use `path` for a targeted drill. `maxBytes` (default 50KB) caps the total response size \u2014 overflow is replaced with a single `${str}` marker carrying the original byte count + a preview.\n\nFor `fiber_tree__query`, heavy fields (`props`, `hooks`) take projection knobs per-field via `select` so the rest of the response stays raw:\n `select: [{ props: { path: \"style\", depth: 2 } }]` \u2014 drill into props.style 2 levels deep\n `select: [{ props: { path: \"data[0:5]\" } }]` \u2014 slice path: take first 5 items of props.data\n `select: [{ hooks: { kinds: [\"State\"], names: [\"isLoading\"], withValues: true, depth: 2 } }]` \u2014 filter + project hook values\n `select: [{ children: 5 }]` \u2014 light-only recursive tree walker (5 levels deep); use `scope: \"root\"` as the first step to dump the whole tree. props/hooks not supported inside; sub-children past treeDepth surface as `${arr}: N`.\n\nHook filters: `kinds` (State/Effect/Memo/Ref/Custom/...), `names` (exact or `/regex/flags`), `withValues` (adds resolved values), `expansionDepth` (caps custom-hook recursion), `format: \"tree\"` (nested children vs flat `via` chains). Each hook entry carries `{ kind, name, hook?, via?, expanded? }`. Sensitive names (password, token, jwt, secret, credential, apiKey, authorization, Pin suffix) are auto-redacted; configure via `fiberTreeModule({ redactHookNames, additionalRedactHookNames })`.\n\nMarker format: `{ \"${obj}\": N }` for collapsed objects (N keys), `{ \"${arr}\": N }` for arrays (N items), `{ \"${fun}\": \"name\" }` for functions, `{ \"${str}\": { len, preview } }` for long strings (>`previewCap`, default 250), `{ \"${Date}\": \"iso\" }` for Date, `{ \"${Err}\": { name, msg } }` for Error, `{ \"${RegExp}\": \"/.../i\" }` for RegExp, `{ \"${map}\": N }` / `{ \"${set}\": N }` for Map/Set sizes, `{ \"${cyc}\": true }` for cycles, `{ \"${ref}\": { mcpId, name } }` for fiber/native refs, `{ \"${cls}\": { name, len } }` for class instances, `{ \"${truncated}\": { total, slice } }` first key/item when a container is wider than the cap (30 keys for objects, 50 items for arrays).\n";
1
+ export declare const BASE_INSTRUCTIONS = "You are connected to a running React Native app via the react-native-mcp-kit bridge.\n\nMultiple React Native apps can connect simultaneously \u2014 each is identified by a short ID like \"ios-1\", \"android-1\", or \"client-1\". Use `connection_status` or `list_tools` to see which clients are connected and their IDs, platforms, and labels.\n\n## How to interact\n\n1. Use `connection_status` to check which clients are connected.\n2. Use `list_tools` to browse all available tool names and short descriptions. The response is compact \u2014 modules that are structurally identical across multiple clients are deduplicated into a single entry with a `clientIds` array, and input schemas are omitted. Narrow the listing with `{ module }` or `{ clientId }`, or pass `{ compact: true }` to drop module-level descriptions.\n3. Use `describe_tool` with `{ tool, clientId? }` to fetch the full input schema of a specific tool before calling it. Required when you need to know the argument shape. Host tools are resolved directly (no clientId needed). For in-app tools, omit `clientId` to auto-pick; specify it only when multiple clients have the same tool with different schemas.\n4. Use `call` to invoke any tool with format: module__method (e.g. navigation__navigate). When more than one client is connected, specify `clientId`. When exactly one client is connected, `clientId` is optional \u2014 it's auto-picked. `args` accepts either a plain object or a JSON string \u2014 prefer objects to avoid quote escaping.\n5. Use `wait_until` to poll any tool until a predicate over its result holds (or timeout). Replaces \"screenshot in a loop + sleep\" for state-level waits (\"wait for network to idle\", \"wait for state key X to become Y\"). Predicate supports compound forms: { all: [...] } (AND), { any: [...] } (OR), { not: predicate }.\n6. For UI-level waits (\"wait for a screen to appear\", \"wait for a spinner to disappear\") use `fiber_tree__query` with `waitFor: { until: \"appear\" | \"disappear\", timeout?, interval?, stable? }` \u2014 it polls the same query with cache bypassed until the target state holds. `stable: <ms>` requires continuous presence/absence for that many ms to ignore transient matches during screen transitions.\n7. Use `assert` for a single-shot checkpoint after actions \u2014 same predicate vocabulary as wait_until, returns { pass, actual, expected?, result? }. Natural pair: do action \u2192 wait_until / fiber_tree waitFor \u2192 assert.\n\n### Broadcast \u2014 same call on several clients\n\n`call`, `wait_until` and `assert` accept `clientId` as a string, a `/body/flags` regex literal, or an array (literals and regex mixed). A plain string keeps the single-client shape (image content passes through). Regex and array forms switch into broadcast mode \u2014 every matching connected client is dispatched in parallel. Same regex slash form as fiber_tree hook filters and log_box ignore patterns.\n\n `call({ clientId: [\"ios-1\", \"android-1\"], tool: \"host__screenshot\" })` \u2014 two screenshots in one response\n `call({ clientId: \"/^ios/\", tool: \"host__screenshot\" })` \u2014 broadcast to every connected iOS client\n `wait_until({ clientId: [\"ios-1\", \"android-1\"], tool: \"navigation__get_current_route\", predicate: { op: \"equals\", path: \"name\", value: \"CART\" } })` \u2014 wait until both clients land on CART\n `assert({ clientId: \"/./\", tool: \"fiber_tree__query\", args: { mcpId: \"checkout:button:submit\" }, predicate: { op: \"exists\" } })` \u2014 the same fiber exists on every connected client\n\nBroadcast result shapes (each carries top-level counters so you can pick out failures without scanning per-client):\n `call` \u2014 text-only: `{ okCount, failedCount, results: [{ clientId, ok, result | error }, ...] }`. Image results: a summary text block `Broadcast: N ok, M failed (T clients).` precedes per-client `## <clientId>` headers + blocks.\n `wait_until` \u2014 `{ ok, okCount, failedCount, perClient: [{ clientId, ok, attempts, elapsedMs, matched? | lastResult, lastError?, reason? }, ...] }`; overall `ok` is true only if every client matched within the shared timeout.\n `assert` \u2014 `{ pass, passedCount, failedCount, perClient: [{ clientId, pass, actual?, expected?, op?, path?, message?, result?, error? }, ...] }`; overall `pass` is true only when every client passed.\n\n`list_tools` and `describe_tool` accept the same regex / array forms \u2014 there they narrow the considered client set (filter / canonicalisation pool) rather than triggering a broadcast.\n\nRegex form details:\n \u2022 A leading `/` plus a trailing `/<flags>` switches the string into regex mode. Flags from `[gimsuy]`. Anything else is treated as a literal client ID.\n \u2022 Pattern is matched against connected client IDs (`ios-1`, `android-2`, \u2026). `\"/./\"` matches every connected client; `\"/^ios/\"` only iOS; `\"/-1$/\"` only the first client per platform.\n \u2022 Pattern that matches zero connected clients returns an error up front \u2014 broadcasting to nobody is almost always a mistake. Literals that are unconnected still fall through to the per-client error in the broadcast envelope (matches the not-fail-fast contract).\n8. Use `host__tap_fiber` to collapse \"fiber_tree__query \u2192 host__tap at bounds\" into one call. Pass fiber_tree steps; if exactly one fiber matches, its center is tapped. Ambiguous match returns the candidate list so you can add `index` or narrow the chain.\n\nSome tools run inline on the MCP server host (e.g. `host__screenshot`, `host__list_devices`, `host__launch_app`, `host__terminate_app`, `host__restart_app`, `metro__reload`, `metro__symbolicate`) and work even when no React Native client is connected. They use xcrun simctl / adb on the dev machine. When `clientId` is provided, host tools use that client's platform/label/deviceId as hints to resolve the target device; otherwise they prefer the device of the single connected client, falling back to the single booted sim / online device. `launch_app`, `terminate_app`, and `restart_app` accept an `appId` arg (iOS bundle ID / Android package name); omit it to reuse the target client's registered `bundleId` from its connection metadata.\n\n## Driving the UI \u2014 pick the right tool\n1. **`host__tap_fiber` with `steps: [...]`** \u2014 the canonical way to simulate a user tap. One call locates the fiber via fiber_tree__query and taps its center through the real OS gesture pipeline, so Pressable feedback, gesture responders, and hit-test logic all run. Ambiguous matches return a candidate list so you can add `index` or narrow `steps`. This is what you want whenever the user asks to simulate a tap / press / button click.\n2. **`fiber_tree__query` with `select: [\"mcpId\",\"name\",\"bounds\"]` + `host__tap`** when you want to inspect a match set before committing \u2014 e.g. verify bounds, or skim candidates before picking one. `props` is opt-in on `select` to keep responses small.\n3. **`fiber_tree__call`** for non-gesture callbacks or imperative ref methods. Pass `prop` to call a callback prop (`{ prop: 'onPress' }`) or `method` to call a native-ref method (`{ method: 'focus' }`). Good when the component is off-screen / virtualised, when a scroll-handler parent swallows taps, or when you're driving focus / blur / measure / scrollTo via the native ref. For simulating a user tap, prefer tap_fiber (above).\n4. **`host__screenshot` + manual coordinate estimation + `host__tap`** ONLY for non-React surfaces: system permission dialogs, native alerts, the on-screen keyboard, WebView content, native splash. These have no fiber and no bounds. Pair with `region: { x, y, width, height }` to screenshot just the area you're inspecting \u2014 vision-token cheap.\n\nGesture tools: `host__tap` / `host__long_press` / `host__swipe` / `host__drag` / `host__type_text` / `host__type_text_batch` / `host__press_key` work on both platforms with no external daemons: Android via `adb shell input`, iOS via a bundled `ios-hid` binary that injects HID events directly into iOS Simulator through SimulatorKit.\n\nStack traces: `errors__get_errors` and `log_box__get_logs` return parsed `stackFrames` you can pass straight into `metro__symbolicate` to resolve bundled frames back to source paths via Metro.\n\n## Consolidated tools \u2014 verb + arg, not verb-per-action\n\nEvery module exposes one verb per concept rather than one tool per variant. Reach for the listing tool with a filter arg, not a tool named after the filter:\n\n- **console** \u2014 `get_logs({ level? })` for all levels (no per-level shortcuts).\n- **errors** \u2014 `get_errors({ fatal? })` covers fatal-only too.\n- **network** \u2014 `get_requests({ status?, method?, url? })` covers errors / pending / URL filter.\n- **log_box** \u2014 `clear({ level? })` (warn / error / syntax / all); `set_installed({ enabled })` toggles install/uninstall.\n- **navigation** \u2014 `navigate({ mode: 'reuse' | 'push' | 'replace' })` covers navigate/push/replace; `pop({ to: <number> | <screenName> | 'top' })` covers pop / pop_to / pop_to_top; `get_current_route({ withState? })` covers the route-state read too.\n- **query** (reactQuery) \u2014 `mutate({ action: 'invalidate' | 'refetch' | 'remove' | 'reset', key? })` covers all four cache mutations.\n- **device** \u2014 `info({ select?: [...] })` aggregates platform / dimensions / pixelRatio / appearance / appState / accessibility / keyboard / initialUrl / dev / identity / app / battery / memoryStorage. `open_url({ dryRun? })` covers the canOpenURL check too.\n- **fiber_tree** \u2014 `query` (all inspection: mcpId / name / testID / props / hooks / bounds / refMethods / children) and `call({ prop? | method? })` (callback prop OR native-ref method) \u2014 two tools cover everything fibers offer.\n\n## Path-based drill into heavy responses\n\nAll listing tools that return heavy JSON (console / network / errors / storage / reactQuery / log_box / navigation / metro events / fiber_tree) accept the standard `path` / `depth` / `maxBytes` projection args. Heavy nested values collapse to compact `${...}`-keyed markers; primitives stay raw. Drill into a specific subtree via the same tool with `path` \u2014 no separate by-id fetcher.\n\n `network__get_requests({ path: '[-1:][0].response.body' })` \u2014 last request's body\n `network__get_requests({ path: '[-1:][0].response.body', depth: 8 })` \u2014 fully expanded\n `console__get_logs({ path: '[-3:][0].args[1]' })` \u2014 second arg of the third-from-last log\n `storage__get_item({ key: 'session', path: 'value.user.email' })` \u2014 drill into a stored value\n `query__get_data({ key: '[\"users\"]', path: 'data.email' })` \u2014 drill into cached data\n\nPath syntax (JS-style):\n `.key` or `[\"key.with.dots\"]` \u2014 object key access\n `[N]` \u2014 index (numeric for arrays, Nth char for strings, Nth key in insertion order for objects)\n `[start:end]` / `[start:]` / `[:end]` \u2014 slice (Python/jq-style; negative indices count from end). Works on arrays, strings, and objects (key slice). After array slice, chained `.key` maps over each element; `[N]` picks one.\n\nPath drilling to a string scalar applies `previewCap` as usual \u2014 long leaves still wrap in a `${str}` marker. To get a raw substring, end the path with a slice: `stack[0:500]` returns the first 500 chars verbatim (slice = explicit truncation request, previewCap bypassed). Default `previewCap` is 250; override per call via `previewCap: <N>`.\n\n`depth` (default per tool, max 8) controls how many container levels are walked before collapsing to a marker. Bump `depth` for an exploratory survey, use `path` for a targeted drill. `maxBytes` (default 50KB) caps the total response size \u2014 overflow is replaced with a single `${str}` marker carrying the original byte count + a preview.\n\nFor `fiber_tree__query`, heavy fields (`props`, `hooks`) take projection knobs per-field via `select` so the rest of the response stays raw:\n `select: [{ props: { path: \"style\", depth: 2 } }]` \u2014 drill into props.style 2 levels deep\n `select: [{ props: { path: \"data[0:5]\" } }]` \u2014 slice path: take first 5 items of props.data\n `select: [{ hooks: { kinds: [\"State\"], names: [\"isLoading\"], withValues: true, depth: 2 } }]` \u2014 filter + project hook values\n `select: [{ children: 5 }]` \u2014 light-only recursive tree walker (5 levels deep); use `scope: \"root\"` as the first step to dump the whole tree. props/hooks not supported inside; sub-children past treeDepth surface as `${arr}: N`.\n\nHook filters: `kinds` (State/Effect/Memo/Ref/Custom/...), `names` (exact or `/regex/flags`), `withValues` (adds resolved values), `expansionDepth` (caps custom-hook recursion), `format: \"tree\"` (nested children vs flat `via` chains). Each hook entry carries `{ kind, name, hook?, via?, expanded? }`. Sensitive names (password, token, jwt, secret, credential, apiKey, authorization, Pin suffix) are auto-redacted; configure via `fiberTreeModule({ redactHookNames, additionalRedactHookNames })`.\n\nMarker format: `{ \"${obj}\": N }` for collapsed objects (N keys), `{ \"${arr}\": N }` for arrays (N items), `{ \"${fun}\": \"name\" }` for functions, `{ \"${str}\": { len, preview } }` for long strings (>`previewCap`, default 250), `{ \"${Date}\": \"iso\" }` for Date, `{ \"${Err}\": { name, msg } }` for Error, `{ \"${RegExp}\": \"/.../i\" }` for RegExp, `{ \"${map}\": N }` / `{ \"${set}\": N }` for Map/Set sizes, `{ \"${cyc}\": true }` for cycles, `{ \"${ref}\": { mcpId, name } }` for fiber/native refs, `{ \"${cls}\": { name, len } }` for class instances, `{ \"${truncated}\": { total, slice } }` first key/item when a container is wider than the cap (30 keys for objects, 50 items for arrays).\n";
2
2
  //# sourceMappingURL=instructions.d.ts.map
@@ -1 +1 @@
1
- {"version":3,"file":"instructions.d.ts","sourceRoot":"","sources":["../../src/server/instructions.ts"],"names":[],"mappings":"AAEA,eAAO,MAAM,iBAAiB,22VAoE7B,CAAC"}
1
+ {"version":3,"file":"instructions.d.ts","sourceRoot":"","sources":["../../src/server/instructions.ts"],"names":[],"mappings":"AAEA,eAAO,MAAM,iBAAiB,ylbAyF7B,CAAC"}
@@ -15,6 +15,27 @@ Multiple React Native apps can connect simultaneously — each is identified by
15
15
  5. Use \`wait_until\` to poll any tool until a predicate over its result holds (or timeout). Replaces "screenshot in a loop + sleep" for state-level waits ("wait for network to idle", "wait for state key X to become Y"). Predicate supports compound forms: { all: [...] } (AND), { any: [...] } (OR), { not: predicate }.
16
16
  6. For UI-level waits ("wait for a screen to appear", "wait for a spinner to disappear") use \`fiber_tree${protocol_1.MODULE_SEPARATOR}query\` with \`waitFor: { until: "appear" | "disappear", timeout?, interval?, stable? }\` — it polls the same query with cache bypassed until the target state holds. \`stable: <ms>\` requires continuous presence/absence for that many ms to ignore transient matches during screen transitions.
17
17
  7. Use \`assert\` for a single-shot checkpoint after actions — same predicate vocabulary as wait_until, returns { pass, actual, expected?, result? }. Natural pair: do action → wait_until / fiber_tree waitFor → assert.
18
+
19
+ ### Broadcast — same call on several clients
20
+
21
+ \`call\`, \`wait_until\` and \`assert\` accept \`clientId\` as a string, a \`/body/flags\` regex literal, or an array (literals and regex mixed). A plain string keeps the single-client shape (image content passes through). Regex and array forms switch into broadcast mode — every matching connected client is dispatched in parallel. Same regex slash form as fiber_tree hook filters and log_box ignore patterns.
22
+
23
+ \`call({ clientId: ["ios-1", "android-1"], tool: "host${protocol_1.MODULE_SEPARATOR}screenshot" })\` — two screenshots in one response
24
+ \`call({ clientId: "/^ios/", tool: "host${protocol_1.MODULE_SEPARATOR}screenshot" })\` — broadcast to every connected iOS client
25
+ \`wait_until({ clientId: ["ios-1", "android-1"], tool: "navigation${protocol_1.MODULE_SEPARATOR}get_current_route", predicate: { op: "equals", path: "name", value: "CART" } })\` — wait until both clients land on CART
26
+ \`assert({ clientId: "/./", tool: "fiber_tree${protocol_1.MODULE_SEPARATOR}query", args: { mcpId: "checkout:button:submit" }, predicate: { op: "exists" } })\` — the same fiber exists on every connected client
27
+
28
+ Broadcast result shapes (each carries top-level counters so you can pick out failures without scanning per-client):
29
+ \`call\` — text-only: \`{ okCount, failedCount, results: [{ clientId, ok, result | error }, ...] }\`. Image results: a summary text block \`Broadcast: N ok, M failed (T clients).\` precedes per-client \`## <clientId>\` headers + blocks.
30
+ \`wait_until\` — \`{ ok, okCount, failedCount, perClient: [{ clientId, ok, attempts, elapsedMs, matched? | lastResult, lastError?, reason? }, ...] }\`; overall \`ok\` is true only if every client matched within the shared timeout.
31
+ \`assert\` — \`{ pass, passedCount, failedCount, perClient: [{ clientId, pass, actual?, expected?, op?, path?, message?, result?, error? }, ...] }\`; overall \`pass\` is true only when every client passed.
32
+
33
+ \`list_tools\` and \`describe_tool\` accept the same regex / array forms — there they narrow the considered client set (filter / canonicalisation pool) rather than triggering a broadcast.
34
+
35
+ Regex form details:
36
+ • A leading \`/\` plus a trailing \`/<flags>\` switches the string into regex mode. Flags from \`[gimsuy]\`. Anything else is treated as a literal client ID.
37
+ • Pattern is matched against connected client IDs (\`ios-1\`, \`android-2\`, …). \`"/./"\` matches every connected client; \`"/^ios/"\` only iOS; \`"/-1$/"\` only the first client per platform.
38
+ • Pattern that matches zero connected clients returns an error up front — broadcasting to nobody is almost always a mistake. Literals that are unconnected still fall through to the per-client error in the broadcast envelope (matches the not-fail-fast contract).
18
39
  8. Use \`host${protocol_1.MODULE_SEPARATOR}tap_fiber\` to collapse "fiber_tree__query → host__tap at bounds" into one call. Pass fiber_tree steps; if exactly one fiber matches, its center is tapped. Ambiguous match returns the candidate list so you can add \`index\` or narrow the chain.
19
40
 
20
41
  Some tools run inline on the MCP server host (e.g. \`host${protocol_1.MODULE_SEPARATOR}screenshot\`, \`host${protocol_1.MODULE_SEPARATOR}list_devices\`, \`host${protocol_1.MODULE_SEPARATOR}launch_app\`, \`host${protocol_1.MODULE_SEPARATOR}terminate_app\`, \`host${protocol_1.MODULE_SEPARATOR}restart_app\`, \`metro${protocol_1.MODULE_SEPARATOR}reload\`, \`metro${protocol_1.MODULE_SEPARATOR}symbolicate\`) and work even when no React Native client is connected. They use xcrun simctl / adb on the dev machine. When \`clientId\` is provided, host tools use that client's platform/label/deviceId as hints to resolve the target device; otherwise they prefer the device of the single connected client, falling back to the single booted sim / online device. \`launch_app\`, \`terminate_app\`, and \`restart_app\` accept an \`appId\` arg (iOS bundle ID / Android package name); omit it to reuse the target client's registered \`bundleId\` from its connection metadata.
@@ -1 +1 @@
1
- {"version":3,"file":"instructions.js","sourceRoot":"","sources":["../../src/server/instructions.ts"],"names":[],"mappings":";;;AAAA,gDAAqD;AAExC,QAAA,iBAAiB,GAAG;;;;;;;;;wDASuB,2BAAgB,0BAA0B,2BAAgB;;2GAEP,2BAAgB;;eAE5G,2BAAgB;;2DAE4B,2BAAgB,uBAAuB,2BAAgB,yBAAyB,2BAAgB,uBAAuB,2BAAgB,0BAA0B,2BAAgB,yBAAyB,2BAAgB,oBAAoB,2BAAgB;;;aAG5S,2BAAgB;mBACV,2BAAgB,8DAA8D,2BAAgB;mBAC9F,2BAAgB;aACtB,2BAAgB,uDAAuD,2BAAgB;;uBAE7E,2BAAgB,iBAAiB,2BAAgB,wBAAwB,2BAAgB,mBAAmB,2BAAgB,kBAAkB,2BAAgB,uBAAuB,2BAAgB,6BAA6B,2BAAgB;;wBAEjP,2BAAgB,6BAA6B,2BAAgB,8EAA8E,2BAAgB;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;CA2ClL,CAAC"}
1
+ {"version":3,"file":"instructions.js","sourceRoot":"","sources":["../../src/server/instructions.ts"],"names":[],"mappings":";;;AAAA,gDAAqD;AAExC,QAAA,iBAAiB,GAAG;;;;;;;;;wDASuB,2BAAgB,0BAA0B,2BAAgB;;2GAEP,2BAAgB;;;;;;;0DAOjE,2BAAgB;4CAC9B,2BAAgB;sEACU,2BAAgB;iDACrC,2BAAgB;;;;;;;;;;;;;eAalD,2BAAgB;;2DAE4B,2BAAgB,uBAAuB,2BAAgB,yBAAyB,2BAAgB,uBAAuB,2BAAgB,0BAA0B,2BAAgB,yBAAyB,2BAAgB,oBAAoB,2BAAgB;;;aAG5S,2BAAgB;mBACV,2BAAgB,8DAA8D,2BAAgB;mBAC9F,2BAAgB;aACtB,2BAAgB,uDAAuD,2BAAgB;;uBAE7E,2BAAgB,iBAAiB,2BAAgB,wBAAwB,2BAAgB,mBAAmB,2BAAgB,kBAAkB,2BAAgB,uBAAuB,2BAAgB,6BAA6B,2BAAgB;;wBAEjP,2BAAgB,6BAA6B,2BAAgB,8EAA8E,2BAAgB;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;CA2ClL,CAAC"}
@@ -1 +1 @@
1
- {"version":3,"file":"assert.d.ts","sourceRoot":"","sources":["../../../src/server/tools/assert.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,KAAK,SAAS,EAAE,MAAM,yCAAyC,CAAC;AAGzE,OAAO,EAA4B,KAAK,aAAa,EAAE,MAAM,kBAAkB,CAAC;AAUhF,eAAO,MAAM,kBAAkB,QAAS,SAAS,OAAO,aAAa,KAAG,IAyEvE,CAAC"}
1
+ {"version":3,"file":"assert.d.ts","sourceRoot":"","sources":["../../../src/server/tools/assert.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,KAAK,SAAS,EAAE,MAAM,yCAAyC,CAAC;AAGzE,OAAO,EAKL,KAAK,aAAa,EACnB,MAAM,kBAAkB,CAAC;AA4C1B,eAAO,MAAM,kBAAkB,QAAS,SAAS,OAAO,aAAa,KAAG,IA0FvE,CAAC"}
@@ -5,6 +5,30 @@ const zod_1 = require("zod");
5
5
  const helpers_1 = require("../../server/helpers");
6
6
  const predicate_1 = require("../../server/predicate");
7
7
  const protocol_1 = require("../../shared/protocol");
8
+ const evaluate = (dispatch, pred, isLeaf, leafPath, leafValue, leafOp, message) => {
9
+ if (!dispatch.ok) {
10
+ const failure = { error: dispatch.error, pass: false };
11
+ if (message)
12
+ failure.message = message;
13
+ return failure;
14
+ }
15
+ const pass = (0, predicate_1.evalPredicate)(dispatch.result, pred);
16
+ const payload = { pass };
17
+ if (isLeaf)
18
+ payload.actual = (0, predicate_1.resolvePath)(dispatch.result, leafPath);
19
+ if (!pass) {
20
+ if (isLeaf) {
21
+ payload.expected = leafValue;
22
+ payload.op = leafOp;
23
+ if (leafPath)
24
+ payload.path = leafPath;
25
+ }
26
+ if (message)
27
+ payload.message = message;
28
+ payload.result = dispatch.result;
29
+ }
30
+ return payload;
31
+ };
8
32
  const registerAssertTool = (mcp, ctx) => {
9
33
  mcp.registerTool('assert', {
10
34
  annotations: {
@@ -13,9 +37,14 @@ const registerAssertTool = (mcp, ctx) => {
13
37
  },
14
38
  description: `Single-shot assertion over a tool's result. Same predicate vocabulary (including { all / any / not }) as wait_until, but one attempt and a standardized diff on failure.
15
39
 
16
- Returns { pass: true, actual? } on success — actual is the path-resolved value for leaf predicates, omitted for compound.
17
- Returns { pass: false, actual, expected?, op?, path?, message?, result } on predicate failure.
18
- Returns { pass: false, error, message? } when the tool dispatch itself threw.
40
+ Single client (clientId omitted or a string):
41
+ Returns { pass: true, actual? } on success — actual is the path-resolved value for leaf predicates, omitted for compound.
42
+ Returns { pass: false, actual, expected?, op?, path?, message?, result } on predicate failure.
43
+ Returns { pass: false, error, message? } when the tool dispatch itself threw.
44
+
45
+ Broadcast (clientId is an array — asserts on each client in parallel):
46
+ Returns { pass, perClient: [{ clientId, pass, actual?, expected?, op?, path?, message?, result?, error? }, ...] }
47
+ with overall pass = every client's assertion passed.
19
48
 
20
49
  Useful after wait_until as a checkpoint — the pair reads "do action → wait → assert" which produces a clean audit trail in session logs.`,
21
50
  inputSchema: {
@@ -23,7 +52,10 @@ Useful after wait_until as a checkpoint — the pair reads "do action → wait
23
52
  .union([zod_1.z.string(), zod_1.z.record(zod_1.z.string(), zod_1.z.unknown())])
24
53
  .optional()
25
54
  .describe('Arguments for the asserted tool — object or JSON string.'),
26
- clientId: zod_1.z.string().optional().describe('Target client ID, same semantics as `call`.'),
55
+ clientId: zod_1.z
56
+ .union([zod_1.z.string(), zod_1.z.array(zod_1.z.string())])
57
+ .optional()
58
+ .describe('Target client ID(s). String asserts on one client; `/body/flags` literal ("/^ios/") expands to every matching connected client; array mixes literals and regex strings. Broadcast forms run the assertion on each matched client in parallel. Same auto-resolution semantics as `call`.'),
27
59
  message: zod_1.z
28
60
  .string()
29
61
  .optional()
@@ -39,39 +71,38 @@ Useful after wait_until as a checkpoint — the pair reads "do action → wait
39
71
  const parsedArgs = (0, helpers_1.parseCallArgs)(args);
40
72
  if (!parsedArgs.ok)
41
73
  return (0, helpers_1.jsonError)(parsedArgs.error);
42
- const dispatch = await ctx.dispatchTool(tool, parsedArgs.args, clientId);
43
- if (!dispatch.ok) {
44
- return {
45
- content: [
46
- {
47
- text: JSON.stringify({ error: dispatch.error, message, pass: false }, null, 2),
48
- type: 'text',
49
- },
50
- ],
51
- };
52
- }
74
+ const clients = (0, helpers_1.parseClientIds)(clientId, ctx.bridge);
75
+ if (!clients.ok)
76
+ return (0, helpers_1.jsonError)(clients.error);
53
77
  const pred = predicate;
54
78
  const isLeaf = (0, predicate_1.isLeafPredicate)(pred);
55
79
  const leafPath = isLeaf ? pred.path : undefined;
56
80
  const leafValue = isLeaf ? pred.value : undefined;
57
81
  const leafOp = isLeaf ? pred.op : undefined;
58
- const pass = (0, predicate_1.evalPredicate)(dispatch.result, pred);
59
- const payload = { pass };
60
- if (isLeaf)
61
- payload.actual = (0, predicate_1.resolvePath)(dispatch.result, leafPath);
62
- if (!pass) {
63
- if (isLeaf) {
64
- payload.expected = leafValue;
65
- payload.op = leafOp;
66
- if (leafPath)
67
- payload.path = leafPath;
68
- }
69
- if (message)
70
- payload.message = message;
71
- payload.result = dispatch.result;
82
+ if (clients.mode === 'single') {
83
+ const dispatch = await ctx.dispatchTool(tool, parsedArgs.args, clients.clientId);
84
+ const payload = evaluate(dispatch, pred, isLeaf, leafPath, leafValue, leafOp, message);
85
+ return {
86
+ content: [{ text: JSON.stringify(payload, null, 2), type: 'text' }],
87
+ };
72
88
  }
89
+ const perClient = await Promise.all(clients.ids.map(async (id) => {
90
+ const dispatch = await ctx.dispatchTool(tool, parsedArgs.args, id);
91
+ const entry = evaluate(dispatch, pred, isLeaf, leafPath, leafValue, leafOp, message);
92
+ return { clientId: id, ...entry };
93
+ }));
94
+ const passedCount = perClient.reduce((n, entry) => {
95
+ return n + (entry.pass === true ? 1 : 0);
96
+ }, 0);
97
+ const failedCount = perClient.length - passedCount;
98
+ const pass = failedCount === 0;
73
99
  return {
74
- content: [{ text: JSON.stringify(payload, null, 2), type: 'text' }],
100
+ content: [
101
+ {
102
+ text: JSON.stringify({ failedCount, pass, passedCount, perClient }, null, 2),
103
+ type: 'text',
104
+ },
105
+ ],
75
106
  };
76
107
  });
77
108
  };
@@ -1 +1 @@
1
- {"version":3,"file":"assert.js","sourceRoot":"","sources":["../../../src/server/tools/assert.ts"],"names":[],"mappings":"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"}
1
+ {"version":3,"file":"assert.js","sourceRoot":"","sources":["../../../src/server/tools/assert.ts"],"names":[],"mappings":"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"}
@@ -1 +1 @@
1
- {"version":3,"file":"call.d.ts","sourceRoot":"","sources":["../../../src/server/tools/call.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,KAAK,SAAS,EAAE,MAAM,yCAAyC,CAAC;AAGzE,OAAO,EAA4B,KAAK,aAAa,EAAE,MAAM,kBAAkB,CAAC;AAGhF,eAAO,MAAM,gBAAgB,QAAS,SAAS,OAAO,aAAa,KAAG,IAsCrE,CAAC"}
1
+ {"version":3,"file":"call.d.ts","sourceRoot":"","sources":["../../../src/server/tools/call.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,KAAK,SAAS,EAAE,MAAM,yCAAyC,CAAC;AAGzE,OAAO,EAML,KAAK,aAAa,EACnB,MAAM,kBAAkB,CAAC;AAG1B,eAAO,MAAM,gBAAgB,QAAS,SAAS,OAAO,aAAa,KAAG,IAoDrE,CAAC"}
@@ -10,16 +10,16 @@ const registerCallTool = (mcp, ctx) => {
10
10
  openWorldHint: true,
11
11
  title: 'Call Tool',
12
12
  },
13
- description: 'Call a tool registered by a React Native app client. Use list_tools first to see available tools. When multiple clients are connected, specify clientId; otherwise it is auto-picked. `args` accepts either a plain object or a JSON string — objects are preferred to avoid escaping quotes.',
13
+ description: 'Call a tool registered by a React Native app client. Use list_tools first to see available tools. When multiple clients are connected, specify clientId; otherwise it is auto-picked. Pass an array (`["ios-1", "android-1"]`) or a `/regex/flags` literal (`"/^ios/"`) to broadcast the same call to several clients in parallel — useful for iOS↔Android parity checks. `args` accepts either a plain object or a JSON string — objects are preferred to avoid escaping quotes.',
14
14
  inputSchema: {
15
15
  args: zod_1.z
16
16
  .union([zod_1.z.string(), zod_1.z.record(zod_1.z.string(), zod_1.z.unknown())])
17
17
  .optional()
18
18
  .describe('Tool arguments as a plain object (e.g. { screen: "AUTH_LOGIN_SCREEN" }) or a JSON string.'),
19
19
  clientId: zod_1.z
20
- .string()
20
+ .union([zod_1.z.string(), zod_1.z.array(zod_1.z.string())])
21
21
  .optional()
22
- .describe('Target client ID (e.g. "ios-1", "android-1"). Optional when exactly one client is connected.'),
22
+ .describe('Target client ID(s). Plain string ("ios-1") selects one client. `/body/flags` literal ("/^ios/") matches connected IDs by regex and broadcasts to every match. Array ([ "ios-1", "/^android/" ]) accepts literals and regex strings mixed — entries are unioned and dedup\'d. Optional when exactly one client is connected.'),
23
23
  tool: zod_1.z
24
24
  .string()
25
25
  .describe(`Tool name in format "module${protocol_1.MODULE_SEPARATOR}method" (e.g. "navigation${protocol_1.MODULE_SEPARATOR}navigate")`),
@@ -28,10 +28,20 @@ const registerCallTool = (mcp, ctx) => {
28
28
  const parsed = (0, helpers_1.parseCallArgs)(args);
29
29
  if (!parsed.ok)
30
30
  return (0, helpers_1.jsonError)(parsed.error);
31
- const dispatch = await ctx.dispatchTool(tool, parsed.args, clientId);
32
- if (!dispatch.ok)
33
- return (0, helpers_1.jsonError)(dispatch.error);
34
- return { content: ctx.formatResult(dispatch.result) };
31
+ const clients = (0, helpers_1.parseClientIds)(clientId, ctx.bridge);
32
+ if (!clients.ok)
33
+ return (0, helpers_1.jsonError)(clients.error);
34
+ if (clients.mode === 'single') {
35
+ const dispatch = await ctx.dispatchTool(tool, parsed.args, clients.clientId);
36
+ if (!dispatch.ok)
37
+ return (0, helpers_1.jsonError)(dispatch.error);
38
+ return { content: ctx.formatResult(dispatch.result) };
39
+ }
40
+ const results = await Promise.all(clients.ids.map(async (id) => {
41
+ const result = await ctx.dispatchTool(tool, parsed.args, id);
42
+ return { clientId: id, result };
43
+ }));
44
+ return { content: (0, helpers_1.buildBroadcastContent)(results, ctx.formatResult) };
35
45
  });
36
46
  };
37
47
  exports.registerCallTool = registerCallTool;
@@ -1 +1 @@
1
- {"version":3,"file":"call.js","sourceRoot":"","sources":["../../../src/server/tools/call.ts"],"names":[],"mappings":";;;AACA,6BAAwB;AAExB,8CAAgF;AAChF,gDAAqD;AAE9C,MAAM,gBAAgB,GAAG,CAAC,GAAc,EAAE,GAAkB,EAAQ,EAAE;IAC3E,GAAG,CAAC,YAAY,CACd,MAAM,EACN;QACE,WAAW,EAAE;YACX,aAAa,EAAE,IAAI;YACnB,KAAK,EAAE,WAAW;SACnB;QACD,WAAW,EACT,+RAA+R;QACjS,WAAW,EAAE;YACX,IAAI,EAAE,OAAC;iBACJ,KAAK,CAAC,CAAC,OAAC,CAAC,MAAM,EAAE,EAAE,OAAC,CAAC,MAAM,CAAC,OAAC,CAAC,MAAM,EAAE,EAAE,OAAC,CAAC,OAAO,EAAE,CAAC,CAAC,CAAC;iBACtD,QAAQ,EAAE;iBACV,QAAQ,CACP,2FAA2F,CAC5F;YACH,QAAQ,EAAE,OAAC;iBACR,MAAM,EAAE;iBACR,QAAQ,EAAE;iBACV,QAAQ,CACP,8FAA8F,CAC/F;YACH,IAAI,EAAE,OAAC;iBACJ,MAAM,EAAE;iBACR,QAAQ,CACP,8BAA8B,2BAAgB,4BAA4B,2BAAgB,YAAY,CACvG;SACJ;KACF,EACD,KAAK,EAAE,EAAE,IAAI,EAAE,QAAQ,EAAE,IAAI,EAAE,EAAE,EAAE;QACjC,MAAM,MAAM,GAAG,IAAA,uBAAa,EAAC,IAAI,CAAC,CAAC;QACnC,IAAI,CAAC,MAAM,CAAC,EAAE;YAAE,OAAO,IAAA,mBAAS,EAAC,MAAM,CAAC,KAAK,CAAC,CAAC;QAC/C,MAAM,QAAQ,GAAG,MAAM,GAAG,CAAC,YAAY,CAAC,IAAI,EAAE,MAAM,CAAC,IAAI,EAAE,QAAQ,CAAC,CAAC;QACrE,IAAI,CAAC,QAAQ,CAAC,EAAE;YAAE,OAAO,IAAA,mBAAS,EAAC,QAAQ,CAAC,KAAK,CAAC,CAAC;QACnD,OAAO,EAAE,OAAO,EAAE,GAAG,CAAC,YAAY,CAAC,QAAQ,CAAC,MAAM,CAAC,EAAE,CAAC;IACxD,CAAC,CACF,CAAC;AACJ,CAAC,CAAC;AAtCW,QAAA,gBAAgB,oBAsC3B"}
1
+ {"version":3,"file":"call.js","sourceRoot":"","sources":["../../../src/server/tools/call.ts"],"names":[],"mappings":";;;AACA,6BAAwB;AAExB,8CAO0B;AAC1B,gDAAqD;AAE9C,MAAM,gBAAgB,GAAG,CAAC,GAAc,EAAE,GAAkB,EAAQ,EAAE;IAC3E,GAAG,CAAC,YAAY,CACd,MAAM,EACN;QACE,WAAW,EAAE;YACX,aAAa,EAAE,IAAI;YACnB,KAAK,EAAE,WAAW;SACnB;QACD,WAAW,EACT,mdAAmd;QACrd,WAAW,EAAE;YACX,IAAI,EAAE,OAAC;iBACJ,KAAK,CAAC,CAAC,OAAC,CAAC,MAAM,EAAE,EAAE,OAAC,CAAC,MAAM,CAAC,OAAC,CAAC,MAAM,EAAE,EAAE,OAAC,CAAC,OAAO,EAAE,CAAC,CAAC,CAAC;iBACtD,QAAQ,EAAE;iBACV,QAAQ,CACP,2FAA2F,CAC5F;YACH,QAAQ,EAAE,OAAC;iBACR,KAAK,CAAC,CAAC,OAAC,CAAC,MAAM,EAAE,EAAE,OAAC,CAAC,KAAK,CAAC,OAAC,CAAC,MAAM,EAAE,CAAC,CAAC,CAAC;iBACxC,QAAQ,EAAE;iBACV,QAAQ,CACP,8TAA8T,CAC/T;YACH,IAAI,EAAE,OAAC;iBACJ,MAAM,EAAE;iBACR,QAAQ,CACP,8BAA8B,2BAAgB,4BAA4B,2BAAgB,YAAY,CACvG;SACJ;KACF,EACD,KAAK,EAAE,EAAE,IAAI,EAAE,QAAQ,EAAE,IAAI,EAAE,EAAE,EAAE;QACjC,MAAM,MAAM,GAAG,IAAA,uBAAa,EAAC,IAAI,CAAC,CAAC;QACnC,IAAI,CAAC,MAAM,CAAC,EAAE;YAAE,OAAO,IAAA,mBAAS,EAAC,MAAM,CAAC,KAAK,CAAC,CAAC;QAE/C,MAAM,OAAO,GAAG,IAAA,wBAAc,EAAC,QAAQ,EAAE,GAAG,CAAC,MAAM,CAAC,CAAC;QACrD,IAAI,CAAC,OAAO,CAAC,EAAE;YAAE,OAAO,IAAA,mBAAS,EAAC,OAAO,CAAC,KAAK,CAAC,CAAC;QAEjD,IAAI,OAAO,CAAC,IAAI,KAAK,QAAQ,EAAE,CAAC;YAC9B,MAAM,QAAQ,GAAG,MAAM,GAAG,CAAC,YAAY,CAAC,IAAI,EAAE,MAAM,CAAC,IAAI,EAAE,OAAO,CAAC,QAAQ,CAAC,CAAC;YAC7E,IAAI,CAAC,QAAQ,CAAC,EAAE;gBAAE,OAAO,IAAA,mBAAS,EAAC,QAAQ,CAAC,KAAK,CAAC,CAAC;YACnD,OAAO,EAAE,OAAO,EAAE,GAAG,CAAC,YAAY,CAAC,QAAQ,CAAC,MAAM,CAAC,EAAE,CAAC;QACxD,CAAC;QAED,MAAM,OAAO,GAAwB,MAAM,OAAO,CAAC,GAAG,CACpD,OAAO,CAAC,GAAG,CAAC,GAAG,CAAC,KAAK,EAAE,EAAE,EAAE,EAAE;YAC3B,MAAM,MAAM,GAAG,MAAM,GAAG,CAAC,YAAY,CAAC,IAAI,EAAE,MAAM,CAAC,IAAI,EAAE,EAAE,CAAC,CAAC;YAC7D,OAAO,EAAE,QAAQ,EAAE,EAAE,EAAE,MAAM,EAAE,CAAC;QAClC,CAAC,CAAC,CACH,CAAC;QACF,OAAO,EAAE,OAAO,EAAE,IAAA,+BAAqB,EAAC,OAAO,EAAE,GAAG,CAAC,YAAY,CAAC,EAAE,CAAC;IACvE,CAAC,CACF,CAAC;AACJ,CAAC,CAAC;AApDW,QAAA,gBAAgB,oBAoD3B"}
@@ -1 +1 @@
1
- {"version":3,"file":"describeTool.d.ts","sourceRoot":"","sources":["../../../src/server/tools/describeTool.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,KAAK,SAAS,EAAE,MAAM,yCAAyC,CAAC;AAGzE,OAAO,EAIL,KAAK,aAAa,EAEnB,MAAM,kBAAkB,CAAC;AAG1B,eAAO,MAAM,wBAAwB,QAAS,SAAS,OAAO,aAAa,KAAG,IAyJ7E,CAAC"}
1
+ {"version":3,"file":"describeTool.d.ts","sourceRoot":"","sources":["../../../src/server/tools/describeTool.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,KAAK,SAAS,EAAE,MAAM,yCAAyC,CAAC;AAGzE,OAAO,EAKL,KAAK,aAAa,EAEnB,MAAM,kBAAkB,CAAC;AAG1B,eAAO,MAAM,wBAAwB,QAAS,SAAS,OAAO,aAAa,KAAG,IAgL7E,CAAC"}
@@ -10,17 +10,20 @@ const registerDescribeToolTool = (mcp, ctx) => {
10
10
  readOnlyHint: true,
11
11
  title: 'Describe Tool',
12
12
  },
13
- description: 'Fetch the full description and input schema for a single tool. Use this after list_tools to learn how to construct arguments for a tool before calling it. For host tools, clientId is ignored. For in-app tools, omit clientId to auto-pick the shared descriptor; specify it only when multiple clients have the same tool with different schemas.',
13
+ description: 'Fetch the full description and input schema for a single tool. Use this after list_tools to learn how to construct arguments for a tool before calling it. For host tools, clientId is ignored. For in-app tools, omit clientId to auto-pick the shared descriptor; pass a literal string to pin to one client, a `/body/flags` regex to narrow the canonicalisation pool to matching IDs, or an array (literals and regex mixed) for an explicit subset.',
14
14
  inputSchema: {
15
15
  clientId: zod_1.z
16
- .string()
16
+ .union([zod_1.z.string(), zod_1.z.array(zod_1.z.string())])
17
17
  .optional()
18
- .describe('Target client ID for in-app tools. Required only when multiple clients have the same tool with different schemas. Ignored for host tools.'),
18
+ .describe('Target client(s) for in-app tools. String pins to one client; `/body/flags` regex or array narrow the canonicalisation pool to matching IDs. Ignored for host tools.'),
19
19
  tool: zod_1.z
20
20
  .string()
21
21
  .describe(`Full tool name in the format "module${protocol_1.MODULE_SEPARATOR}method" (e.g. "navigation${protocol_1.MODULE_SEPARATOR}navigate", "host${protocol_1.MODULE_SEPARATOR}screenshot").`),
22
22
  },
23
23
  }, async ({ clientId, tool }) => {
24
+ const parsedClient = (0, helpers_1.parseClientIds)(clientId, ctx.bridge);
25
+ if (!parsedClient.ok)
26
+ return (0, helpers_1.jsonError)(parsedClient.error);
24
27
  // 1. Host tool path — resolved via hostToolMap, clientId is ignored
25
28
  const hostEntry = ctx.hostToolMap.get(tool);
26
29
  if (hostEntry) {
@@ -45,9 +48,10 @@ const registerDescribeToolTool = (mcp, ctx) => {
45
48
  ],
46
49
  };
47
50
  }
48
- // 2. Explicit clientId — look up the specific client
49
- if (clientId) {
50
- const client = ctx.bridge.getClient(clientId);
51
+ // 2. Explicit single clientId — pin to one client
52
+ if (parsedClient.mode === 'single' && parsedClient.clientId) {
53
+ const pinned = parsedClient.clientId;
54
+ const client = ctx.bridge.getClient(pinned);
51
55
  if (!client) {
52
56
  const available = ctx.bridge
53
57
  .listClients()
@@ -55,17 +59,17 @@ const registerDescribeToolTool = (mcp, ctx) => {
55
59
  return c.id;
56
60
  })
57
61
  .join(', ') || '(none)';
58
- return (0, helpers_1.jsonError)(`Client '${clientId}' not connected. Available: ${available}`);
62
+ return (0, helpers_1.jsonError)(`Client '${pinned}' not connected. Available: ${available}`);
59
63
  }
60
64
  const found = (0, helpers_1.findToolInClient)(client, tool);
61
65
  if (!found) {
62
- return (0, helpers_1.jsonError)(`Tool '${tool}' not found on client '${clientId}'.`);
66
+ return (0, helpers_1.jsonError)(`Tool '${tool}' not found on client '${pinned}'.`);
63
67
  }
64
68
  return {
65
69
  content: [
66
70
  {
67
71
  text: JSON.stringify({
68
- clientIds: [clientId],
72
+ clientIds: [pinned],
69
73
  description: found.description,
70
74
  inputSchema: found.inputSchema,
71
75
  name: tool,
@@ -76,8 +80,23 @@ const registerDescribeToolTool = (mcp, ctx) => {
76
80
  ],
77
81
  };
78
82
  }
79
- // 3. Auto-pick across all connected clients
80
- const clients = ctx.bridge.listClients();
83
+ // 3. Auto-pick across all connected clients (optionally narrowed by an
84
+ // explicit array of clientIds — only those clients participate in the
85
+ // canonicalisation pool).
86
+ const allClients = ctx.bridge.listClients();
87
+ const clients = parsedClient.mode === 'broadcast'
88
+ ? allClients.filter((c) => {
89
+ return parsedClient.ids.includes(c.id);
90
+ })
91
+ : allClients;
92
+ if (parsedClient.mode === 'broadcast' && clients.length === 0) {
93
+ const available = allClients
94
+ .map((c) => {
95
+ return c.id;
96
+ })
97
+ .join(', ') || '(none)';
98
+ return (0, helpers_1.jsonError)(`None of the requested clients (${parsedClient.ids.join(', ')}) are connected. Available: ${available}`);
99
+ }
81
100
  const matches = [];
82
101
  for (const c of clients) {
83
102
  const found = (0, helpers_1.findToolInClient)(c, tool);
@@ -1 +1 @@
1
- {"version":3,"file":"describeTool.js","sourceRoot":"","sources":["../../../src/server/tools/describeTool.ts"],"names":[],"mappings":"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"}
1
+ {"version":3,"file":"describeTool.js","sourceRoot":"","sources":["../../../src/server/tools/describeTool.ts"],"names":[],"mappings":"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"}
@@ -1 +1 @@
1
- {"version":3,"file":"listTools.d.ts","sourceRoot":"","sources":["../../../src/server/tools/listTools.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,KAAK,SAAS,EAAE,MAAM,yCAAyC,CAAC;AAGzE,OAAO,EAAqB,KAAK,aAAa,EAAkB,MAAM,kBAAkB,CAAC;AAGzF,eAAO,MAAM,qBAAqB,QAAS,SAAS,OAAO,aAAa,KAAG,IAwH1E,CAAC"}
1
+ {"version":3,"file":"listTools.d.ts","sourceRoot":"","sources":["../../../src/server/tools/listTools.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,KAAK,SAAS,EAAE,MAAM,yCAAyC,CAAC;AAGzE,OAAO,EAIL,KAAK,aAAa,EAEnB,MAAM,kBAAkB,CAAC;AAG1B,eAAO,MAAM,qBAAqB,QAAS,SAAS,OAAO,aAAa,KAAG,IAiI1E,CAAC"}
@@ -10,12 +10,12 @@ const registerListToolsTool = (mcp, ctx) => {
10
10
  readOnlyHint: true,
11
11
  title: 'List Tools',
12
12
  },
13
- description: 'Browse available tools with compact (schema-free) descriptions. Modules with identical shape across multiple clients are deduplicated into a single entry with a clientIds array. Use describe_tool to fetch the full input schema for a specific tool before calling it. Pass `module` to narrow to one module, `clientId` to narrow to one client, `compact: true` to drop long module-level descriptions.',
13
+ description: 'Browse available tools with compact (schema-free) descriptions. Modules with identical shape across multiple clients are deduplicated into a single entry with a clientIds array. Use describe_tool to fetch the full input schema for a specific tool before calling it. Pass `module` to narrow to one module, `clientId` to narrow to one client (string), several clients (array), or by regex (`"/^ios/"`), `compact: true` to drop long module-level descriptions.',
14
14
  inputSchema: {
15
15
  clientId: zod_1.z
16
- .string()
16
+ .union([zod_1.z.string(), zod_1.z.array(zod_1.z.string())])
17
17
  .optional()
18
- .describe('Narrow listing to a single client. Omit for all connected clients.'),
18
+ .describe('Narrow listing to a single client (string), a `/body/flags` regex over IDs, or several clients (array of literals and/or regex strings). Omit for all connected clients.'),
19
19
  compact: zod_1.z
20
20
  .boolean()
21
21
  .optional()
@@ -26,12 +26,19 @@ const registerListToolsTool = (mcp, ctx) => {
26
26
  .describe('Narrow listing to a single module name (e.g. "fiber_tree", "host"). Omit for all.'),
27
27
  },
28
28
  }, async ({ clientId, compact, module }) => {
29
+ const filter = (0, helpers_1.parseClientIds)(clientId, ctx.bridge);
30
+ if (!filter.ok)
31
+ return (0, helpers_1.jsonError)(filter.error);
29
32
  const allClients = ctx.bridge.listClients();
30
- const clients = clientId
33
+ const clients = filter.mode === 'broadcast'
31
34
  ? allClients.filter((c) => {
32
- return c.id === clientId;
35
+ return filter.ids.includes(c.id);
33
36
  })
34
- : allClients;
37
+ : filter.clientId
38
+ ? allClients.filter((c) => {
39
+ return c.id === filter.clientId;
40
+ })
41
+ : allClients;
35
42
  // Dedup tool groups across clients by canonical shape
36
43
  const dedupMap = new Map();
37
44
  for (const client of clients) {
@@ -1 +1 @@
1
- {"version":3,"file":"listTools.js","sourceRoot":"","sources":["../../../src/server/tools/listTools.ts"],"names":[],"mappings":"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"}
1
+ {"version":3,"file":"listTools.js","sourceRoot":"","sources":["../../../src/server/tools/listTools.ts"],"names":[],"mappings":"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"}
@@ -1 +1 @@
1
- {"version":3,"file":"waitUntil.d.ts","sourceRoot":"","sources":["../../../src/server/tools/waitUntil.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,KAAK,SAAS,EAAE,MAAM,yCAAyC,CAAC;AAGzE,OAAO,EAA4B,KAAK,aAAa,EAAE,MAAM,kBAAkB,CAAC;AAUhF,eAAO,MAAM,qBAAqB,QAAS,SAAS,OAAO,aAAa,KAAG,IAsH1E,CAAC"}
1
+ {"version":3,"file":"waitUntil.d.ts","sourceRoot":"","sources":["../../../src/server/tools/waitUntil.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,KAAK,SAAS,EAAE,MAAM,yCAAyC,CAAC;AAGzE,OAAO,EAA4C,KAAK,aAAa,EAAE,MAAM,kBAAkB,CAAC;AAwEhG,eAAO,MAAM,qBAAqB,QAAS,SAAS,OAAO,aAAa,KAAG,IAuJ1E,CAAC"}
@@ -5,6 +5,48 @@ const zod_1 = require("zod");
5
5
  const helpers_1 = require("../../server/helpers");
6
6
  const predicate_1 = require("../../server/predicate");
7
7
  const protocol_1 = require("../../shared/protocol");
8
+ const pollForClient = async (ctx, tool, args, clientId, pred, isLeaf, leafPath, timeout, interval) => {
9
+ const started = Date.now();
10
+ let attempts = 0;
11
+ let lastResult;
12
+ let lastError;
13
+ while (Date.now() - started < timeout) {
14
+ attempts += 1;
15
+ const dispatch = await ctx.dispatchTool(tool, args, clientId);
16
+ if (dispatch.ok) {
17
+ lastResult = dispatch.result;
18
+ if ((0, predicate_1.evalPredicate)(lastResult, pred)) {
19
+ const outcome = {
20
+ attempts,
21
+ elapsedMs: Date.now() - started,
22
+ ok: true,
23
+ };
24
+ if (isLeaf)
25
+ outcome.matched = (0, predicate_1.resolvePath)(lastResult, leafPath);
26
+ return outcome;
27
+ }
28
+ }
29
+ else {
30
+ lastError = dispatch.error;
31
+ }
32
+ const remaining = timeout - (Date.now() - started);
33
+ if (remaining <= 0)
34
+ break;
35
+ await new Promise((r) => {
36
+ return setTimeout(r, Math.min(interval, remaining));
37
+ });
38
+ }
39
+ return {
40
+ attempts,
41
+ elapsedMs: Date.now() - started,
42
+ lastError,
43
+ lastResult,
44
+ ok: false,
45
+ reason: lastError
46
+ ? `Last dispatch failed: ${lastError}`
47
+ : `Predicate did not hold within ${timeout}ms`,
48
+ };
49
+ };
8
50
  const registerWaitUntilTool = (mcp, ctx) => {
9
51
  mcp.registerTool('wait_until', {
10
52
  annotations: {
@@ -30,15 +72,23 @@ PREDICATE
30
72
  Example: { all: [{op:"equals", path:"name", value:"CART"}, {op:"gt", path:"items.length", value:0}] }
31
73
 
32
74
  RETURNS
33
- { ok: true, attempts, elapsedMs, matched? } on success — matched is the path-
34
- resolved value for leaf predicates, omitted for compound.
35
- { ok: false, reason, attempts, elapsedMs, lastResult, lastError? } on timeout.`,
75
+ Single client (clientId omitted or a string):
76
+ { ok: true, attempts, elapsedMs, matched? } on success — matched is the path-
77
+ resolved value for leaf predicates, omitted for compound.
78
+ { ok: false, reason, attempts, elapsedMs, lastResult, lastError? } on timeout.
79
+
80
+ Broadcast (clientId is an array — polls each client in parallel under the
81
+ shared timeout): { ok, perClient: [{ clientId, ok, attempts, elapsedMs, matched? | lastResult, lastError? }, ...] }
82
+ with overall ok = every client matched within its budget.`,
36
83
  inputSchema: {
37
84
  args: zod_1.z
38
85
  .union([zod_1.z.string(), zod_1.z.record(zod_1.z.string(), zod_1.z.unknown())])
39
86
  .optional()
40
87
  .describe('Arguments for the polled tool — object or JSON string.'),
41
- clientId: zod_1.z.string().optional().describe('Target client ID, same semantics as `call`.'),
88
+ clientId: zod_1.z
89
+ .union([zod_1.z.string(), zod_1.z.array(zod_1.z.string())])
90
+ .optional()
91
+ .describe('Target client ID(s). String polls one client; `/body/flags` literal ("/^ios/") expands to every matching connected client; array mixes literals and regex strings. Broadcast forms poll each matched client in parallel under the shared timeoutMs and report per-client outcomes. Same auto-resolution semantics as `call`.'),
42
92
  intervalMs: zod_1.z
43
93
  .number()
44
94
  .optional()
@@ -58,57 +108,64 @@ RETURNS
58
108
  const parsedArgs = (0, helpers_1.parseCallArgs)(args);
59
109
  if (!parsedArgs.ok)
60
110
  return (0, helpers_1.jsonError)(parsedArgs.error);
111
+ const clients = (0, helpers_1.parseClientIds)(clientId, ctx.bridge);
112
+ if (!clients.ok)
113
+ return (0, helpers_1.jsonError)(clients.error);
61
114
  const pred = predicate;
62
115
  const isLeaf = (0, predicate_1.isLeafPredicate)(pred);
63
116
  const leafPath = isLeaf ? pred.path : undefined;
64
117
  const timeout = Math.max(500, Math.min(60_000, timeoutMs ?? 10_000));
65
118
  const interval = Math.max(50, Math.min(5_000, intervalMs ?? 300));
66
- const started = Date.now();
67
- let attempts = 0;
68
- let lastResult;
69
- let lastError;
70
- while (Date.now() - started < timeout) {
71
- attempts += 1;
72
- const dispatch = await ctx.dispatchTool(tool, parsedArgs.args, clientId);
73
- if (dispatch.ok) {
74
- lastResult = dispatch.result;
75
- if ((0, predicate_1.evalPredicate)(lastResult, pred)) {
76
- const payload = {
77
- attempts,
78
- elapsedMs: Date.now() - started,
79
- ok: true,
80
- };
81
- if (isLeaf) {
82
- payload.matched = (0, predicate_1.resolvePath)(lastResult, leafPath);
83
- }
84
- return {
85
- content: [{ text: JSON.stringify(payload, null, 2), type: 'text' }],
86
- };
87
- }
119
+ if (clients.mode === 'single') {
120
+ const outcome = await pollForClient(ctx, tool, parsedArgs.args, clients.clientId, pred, isLeaf, leafPath, timeout, interval);
121
+ const payload = {
122
+ attempts: outcome.attempts,
123
+ elapsedMs: outcome.elapsedMs,
124
+ ok: outcome.ok,
125
+ };
126
+ if (outcome.ok) {
127
+ if (isLeaf)
128
+ payload.matched = outcome.matched;
88
129
  }
89
130
  else {
90
- lastError = dispatch.error;
131
+ payload.lastResult = outcome.lastResult;
132
+ payload.reason = outcome.reason;
133
+ if (outcome.lastError)
134
+ payload.lastError = outcome.lastError;
91
135
  }
92
- const remaining = timeout - (Date.now() - started);
93
- if (remaining <= 0)
94
- break;
95
- await new Promise((r) => {
96
- return setTimeout(r, Math.min(interval, remaining));
97
- });
136
+ return {
137
+ content: [{ text: JSON.stringify(payload, null, 2), type: 'text' }],
138
+ };
98
139
  }
140
+ const perClient = await Promise.all(clients.ids.map(async (id) => {
141
+ const outcome = await pollForClient(ctx, tool, parsedArgs.args, id, pred, isLeaf, leafPath, timeout, interval);
142
+ const entry = {
143
+ attempts: outcome.attempts,
144
+ clientId: id,
145
+ elapsedMs: outcome.elapsedMs,
146
+ ok: outcome.ok,
147
+ };
148
+ if (outcome.ok) {
149
+ if (isLeaf)
150
+ entry.matched = outcome.matched;
151
+ }
152
+ else {
153
+ entry.lastResult = outcome.lastResult;
154
+ entry.reason = outcome.reason;
155
+ if (outcome.lastError)
156
+ entry.lastError = outcome.lastError;
157
+ }
158
+ return entry;
159
+ }));
160
+ const okCount = perClient.reduce((n, entry) => {
161
+ return n + (entry.ok === true ? 1 : 0);
162
+ }, 0);
163
+ const failedCount = perClient.length - okCount;
164
+ const ok = failedCount === 0;
99
165
  return {
100
166
  content: [
101
167
  {
102
- text: JSON.stringify({
103
- attempts,
104
- elapsedMs: Date.now() - started,
105
- lastError,
106
- lastResult,
107
- ok: false,
108
- reason: lastError
109
- ? `Last dispatch failed: ${lastError}`
110
- : `Predicate did not hold within ${timeout}ms`,
111
- }, null, 2),
168
+ text: JSON.stringify({ failedCount, ok, okCount, perClient }, null, 2),
112
169
  type: 'text',
113
170
  },
114
171
  ],
@@ -1 +1 @@
1
- {"version":3,"file":"waitUntil.js","sourceRoot":"","sources":["../../../src/server/tools/waitUntil.ts"],"names":[],"mappings":"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"}
1
+ {"version":3,"file":"waitUntil.js","sourceRoot":"","sources":["../../../src/server/tools/waitUntil.ts"],"names":[],"mappings":";;;AACA,6BAAwB;AAExB,8CAAgG;AAChG,kDAM4B;AAC5B,gDAAqD;AAYrD,MAAM,aAAa,GAAG,KAAK,EACzB,GAAkB,EAClB,IAAY,EACZ,IAA6B,EAC7B,QAA4B,EAC5B,IAAe,EACf,MAAe,EACf,QAA4B,EAC5B,OAAe,EACf,QAAgB,EACM,EAAE;IACxB,MAAM,OAAO,GAAG,IAAI,CAAC,GAAG,EAAE,CAAC;IAC3B,IAAI,QAAQ,GAAG,CAAC,CAAC;IACjB,IAAI,UAAmB,CAAC;IACxB,IAAI,SAA6B,CAAC;IAElC,OAAO,IAAI,CAAC,GAAG,EAAE,GAAG,OAAO,GAAG,OAAO,EAAE,CAAC;QACtC,QAAQ,IAAI,CAAC,CAAC;QACd,MAAM,QAAQ,GAAG,MAAM,GAAG,CAAC,YAAY,CAAC,IAAI,EAAE,IAAI,EAAE,QAAQ,CAAC,CAAC;QAC9D,IAAI,QAAQ,CAAC,EAAE,EAAE,CAAC;YAChB,UAAU,GAAG,QAAQ,CAAC,MAAM,CAAC;YAC7B,IAAI,IAAA,yBAAa,EAAC,UAAU,EAAE,IAAI,CAAC,EAAE,CAAC;gBACpC,MAAM,OAAO,GAAgB;oBAC3B,QAAQ;oBACR,SAAS,EAAE,IAAI,CAAC,GAAG,EAAE,GAAG,OAAO;oBAC/B,EAAE,EAAE,IAAI;iBACT,CAAC;gBACF,IAAI,MAAM;oBAAE,OAAO,CAAC,OAAO,GAAG,IAAA,uBAAW,EAAC,UAAU,EAAE,QAAQ,CAAC,CAAC;gBAChE,OAAO,OAAO,CAAC;YACjB,CAAC;QACH,CAAC;aAAM,CAAC;YACN,SAAS,GAAG,QAAQ,CAAC,KAAK,CAAC;QAC7B,CAAC;QACD,MAAM,SAAS,GAAG,OAAO,GAAG,CAAC,IAAI,CAAC,GAAG,EAAE,GAAG,OAAO,CAAC,CAAC;QACnD,IAAI,SAAS,IAAI,CAAC;YAAE,MAAM;QAC1B,MAAM,IAAI,OAAO,CAAC,CAAC,CAAC,EAAE,EAAE;YACtB,OAAO,UAAU,CAAC,CAAC,EAAE,IAAI,CAAC,GAAG,CAAC,QAAQ,EAAE,SAAS,CAAC,CAAC,CAAC;QACtD,CAAC,CAAC,CAAC;IACL,CAAC;IAED,OAAO;QACL,QAAQ;QACR,SAAS,EAAE,IAAI,CAAC,GAAG,EAAE,GAAG,OAAO;QAC/B,SAAS;QACT,UAAU;QACV,EAAE,EAAE,KAAK;QACT,MAAM,EAAE,SAAS;YACf,CAAC,CAAC,yBAAyB,SAAS,EAAE;YACtC,CAAC,CAAC,iCAAiC,OAAO,IAAI;KACjD,CAAC;AACJ,CAAC,CAAC;AAEK,MAAM,qBAAqB,GAAG,CAAC,GAAc,EAAE,GAAkB,EAAQ,EAAE;IAChF,GAAG,CAAC,YAAY,CACd,YAAY,EACZ;QACE,WAAW,EAAE;YACX,aAAa,EAAE,IAAI;YACnB,KAAK,EAAE,YAAY;SACpB;QACD,WAAW,EAAE;;;;;;;;;;;;;;;;;;;;;;;;;;4DA0ByC;QACtD,WAAW,EAAE;YACX,IAAI,EAAE,OAAC;iBACJ,KAAK,CAAC,CAAC,OAAC,CAAC,MAAM,EAAE,EAAE,OAAC,CAAC,MAAM,CAAC,OAAC,CAAC,MAAM,EAAE,EAAE,OAAC,CAAC,OAAO,EAAE,CAAC,CAAC,CAAC;iBACtD,QAAQ,EAAE;iBACV,QAAQ,CAAC,wDAAwD,CAAC;YACrE,QAAQ,EAAE,OAAC;iBACR,KAAK,CAAC,CAAC,OAAC,CAAC,MAAM,EAAE,EAAE,OAAC,CAAC,KAAK,CAAC,OAAC,CAAC,MAAM,EAAE,CAAC,CAAC,CAAC;iBACxC,QAAQ,EAAE;iBACV,QAAQ,CACP,8TAA8T,CAC/T;YACH,UAAU,EAAE,OAAC;iBACV,MAAM,EAAE;iBACR,QAAQ,EAAE;iBACV,QAAQ,CAAC,6DAA6D,CAAC;YAC1E,SAAS,EAAE,OAAC;iBACT,WAAW,CAAC,EAAE,CAAC;iBACf,QAAQ,CACP,+HAA+H,CAChI;YACH,SAAS,EAAE,OAAC;iBACT,MAAM,EAAE;iBACR,QAAQ,EAAE;iBACV,QAAQ,CAAC,uDAAuD,CAAC;YACpE,IAAI,EAAE,OAAC;iBACJ,MAAM,EAAE;iBACR,QAAQ,CAAC,sCAAsC,2BAAgB,sBAAsB,CAAC;SAC1F;KACF,EACD,KAAK,EAAE,EAAE,IAAI,EAAE,QAAQ,EAAE,UAAU,EAAE,SAAS,EAAE,SAAS,EAAE,IAAI,EAAE,EAAE,EAAE;QACnE,MAAM,UAAU,GAAG,IAAA,uBAAa,EAAC,IAAI,CAAC,CAAC;QACvC,IAAI,CAAC,UAAU,CAAC,EAAE;YAAE,OAAO,IAAA,mBAAS,EAAC,UAAU,CAAC,KAAK,CAAC,CAAC;QAEvD,MAAM,OAAO,GAAG,IAAA,wBAAc,EAAC,QAAQ,EAAE,GAAG,CAAC,MAAM,CAAC,CAAC;QACrD,IAAI,CAAC,OAAO,CAAC,EAAE;YAAE,OAAO,IAAA,mBAAS,EAAC,OAAO,CAAC,KAAK,CAAC,CAAC;QAEjD,MAAM,IAAI,GAAG,SAAsB,CAAC;QACpC,MAAM,MAAM,GAAG,IAAA,2BAAe,EAAC,IAAI,CAAC,CAAC;QACrC,MAAM,QAAQ,GAAG,MAAM,CAAC,CAAC,CAAE,IAAsB,CAAC,IAAI,CAAC,CAAC,CAAC,SAAS,CAAC;QACnE,MAAM,OAAO,GAAG,IAAI,CAAC,GAAG,CAAC,GAAG,EAAE,IAAI,CAAC,GAAG,CAAC,MAAM,EAAE,SAAS,IAAI,MAAM,CAAC,CAAC,CAAC;QACrE,MAAM,QAAQ,GAAG,IAAI,CAAC,GAAG,CAAC,EAAE,EAAE,IAAI,CAAC,GAAG,CAAC,KAAK,EAAE,UAAU,IAAI,GAAG,CAAC,CAAC,CAAC;QAElE,IAAI,OAAO,CAAC,IAAI,KAAK,QAAQ,EAAE,CAAC;YAC9B,MAAM,OAAO,GAAG,MAAM,aAAa,CACjC,GAAG,EACH,IAAI,EACJ,UAAU,CAAC,IAAI,EACf,OAAO,CAAC,QAAQ,EAChB,IAAI,EACJ,MAAM,EACN,QAAQ,EACR,OAAO,EACP,QAAQ,CACT,CAAC;YACF,MAAM,OAAO,GAA4B;gBACvC,QAAQ,EAAE,OAAO,CAAC,QAAQ;gBAC1B,SAAS,EAAE,OAAO,CAAC,SAAS;gBAC5B,EAAE,EAAE,OAAO,CAAC,EAAE;aACf,CAAC;YACF,IAAI,OAAO,CAAC,EAAE,EAAE,CAAC;gBACf,IAAI,MAAM;oBAAE,OAAO,CAAC,OAAO,GAAG,OAAO,CAAC,OAAO,CAAC;YAChD,CAAC;iBAAM,CAAC;gBACN,OAAO,CAAC,UAAU,GAAG,OAAO,CAAC,UAAU,CAAC;gBACxC,OAAO,CAAC,MAAM,GAAG,OAAO,CAAC,MAAM,CAAC;gBAChC,IAAI,OAAO,CAAC,SAAS;oBAAE,OAAO,CAAC,SAAS,GAAG,OAAO,CAAC,SAAS,CAAC;YAC/D,CAAC;YACD,OAAO;gBACL,OAAO,EAAE,CAAC,EAAE,IAAI,EAAE,IAAI,CAAC,SAAS,CAAC,OAAO,EAAE,IAAI,EAAE,CAAC,CAAC,EAAE,IAAI,EAAE,MAAe,EAAE,CAAC;aAC7E,CAAC;QACJ,CAAC;QAED,MAAM,SAAS,GAAG,MAAM,OAAO,CAAC,GAAG,CACjC,OAAO,CAAC,GAAG,CAAC,GAAG,CAAC,KAAK,EAAE,EAAE,EAAE,EAAE;YAC3B,MAAM,OAAO,GAAG,MAAM,aAAa,CACjC,GAAG,EACH,IAAI,EACJ,UAAU,CAAC,IAAI,EACf,EAAE,EACF,IAAI,EACJ,MAAM,EACN,QAAQ,EACR,OAAO,EACP,QAAQ,CACT,CAAC;YACF,MAAM,KAAK,GAA4B;gBACrC,QAAQ,EAAE,OAAO,CAAC,QAAQ;gBAC1B,QAAQ,EAAE,EAAE;gBACZ,SAAS,EAAE,OAAO,CAAC,SAAS;gBAC5B,EAAE,EAAE,OAAO,CAAC,EAAE;aACf,CAAC;YACF,IAAI,OAAO,CAAC,EAAE,EAAE,CAAC;gBACf,IAAI,MAAM;oBAAE,KAAK,CAAC,OAAO,GAAG,OAAO,CAAC,OAAO,CAAC;YAC9C,CAAC;iBAAM,CAAC;gBACN,KAAK,CAAC,UAAU,GAAG,OAAO,CAAC,UAAU,CAAC;gBACtC,KAAK,CAAC,MAAM,GAAG,OAAO,CAAC,MAAM,CAAC;gBAC9B,IAAI,OAAO,CAAC,SAAS;oBAAE,KAAK,CAAC,SAAS,GAAG,OAAO,CAAC,SAAS,CAAC;YAC7D,CAAC;YACD,OAAO,KAAK,CAAC;QACf,CAAC,CAAC,CACH,CAAC;QAEF,MAAM,OAAO,GAAG,SAAS,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,KAAK,EAAE,EAAE;YAC5C,OAAO,CAAC,GAAG,CAAC,KAAK,CAAC,EAAE,KAAK,IAAI,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;QACzC,CAAC,EAAE,CAAC,CAAC,CAAC;QACN,MAAM,WAAW,GAAG,SAAS,CAAC,MAAM,GAAG,OAAO,CAAC;QAC/C,MAAM,EAAE,GAAG,WAAW,KAAK,CAAC,CAAC;QAC7B,OAAO;YACL,OAAO,EAAE;gBACP;oBACE,IAAI,EAAE,IAAI,CAAC,SAAS,CAAC,EAAE,WAAW,EAAE,EAAE,EAAE,OAAO,EAAE,SAAS,EAAE,EAAE,IAAI,EAAE,CAAC,CAAC;oBACtE,IAAI,EAAE,MAAe;iBACtB;aACF;SACF,CAAC;IACJ,CAAC,CACF,CAAC;AACJ,CAAC,CAAC;AAvJW,QAAA,qBAAqB,yBAuJhC"}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "react-native-mcp-kit",
3
- "version": "4.2.1",
3
+ "version": "4.4.0",
4
4
  "description": "MCP server for React Native — drive, inspect, and debug running RN apps from AI agents via the real OS gesture pipeline",
5
5
  "license": "MIT",
6
6
  "author": "Aliaksei Pranko",