@limrun/ui 0.9.0-rc.5 → 0.9.0-rc.7

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.
Files changed (73) hide show
  1. package/README.md +9 -0
  2. package/dist/components/device-install/device-install-dialog.d.ts +5 -0
  3. package/dist/components/device-install/index.d.ts +2 -0
  4. package/dist/components/inspect-overlay.d.ts +1 -0
  5. package/dist/components/remote-control.d.ts +13 -2
  6. package/dist/core/ax-tree.d.ts +2 -0
  7. package/dist/core/device-install/apple/client.d.ts +17 -0
  8. package/dist/core/device-install/apple/crypto.d.ts +20 -0
  9. package/dist/core/device-install/apple/gsa-srp.d.ts +26 -0
  10. package/dist/core/device-install/apple/index.d.ts +5 -0
  11. package/dist/core/device-install/apple/provisioning.d.ts +161 -0
  12. package/dist/core/device-install/apple/relay.d.ts +29 -0
  13. package/dist/core/device-install/index.d.ts +4 -0
  14. package/dist/core/device-install/operations/index.d.ts +6 -0
  15. package/dist/core/device-install/operations/limbuild-client.d.ts +28 -0
  16. package/dist/core/device-install/operations/operations.d.ts +32 -0
  17. package/dist/core/device-install/operations/relay-client.d.ts +25 -0
  18. package/dist/core/device-install/operations/relay-protocol.d.ts +27 -0
  19. package/dist/core/device-install/operations/usbmux.d.ts +32 -0
  20. package/dist/core/device-install/operations/webusb.d.ts +21 -0
  21. package/dist/core/device-install/storage/browser-storage.d.ts +44 -0
  22. package/dist/core/device-install/storage/index.d.ts +1 -0
  23. package/dist/core/device-install/types.d.ts +48 -0
  24. package/dist/device-install/index.cjs +1 -0
  25. package/dist/device-install/index.d.ts +3 -0
  26. package/dist/device-install/index.js +78 -0
  27. package/dist/device-install/react.cjs +1 -0
  28. package/dist/device-install/react.d.ts +1 -0
  29. package/dist/device-install/react.js +4 -0
  30. package/dist/device-install-dialog-86RDdoK9.js +2 -0
  31. package/dist/device-install-dialog-CnyDWf0q.mjs +462 -0
  32. package/dist/device-install-dialog.css +1 -0
  33. package/dist/hooks/index.d.ts +1 -0
  34. package/dist/hooks/use-device-install.d.ts +73 -0
  35. package/dist/index.cjs +1 -1
  36. package/dist/index.css +1 -1
  37. package/dist/index.d.ts +3 -1
  38. package/dist/index.js +737 -703
  39. package/dist/use-device-install-CbGVvwPp.js +31 -0
  40. package/dist/use-device-install-j1Gekpl4.mjs +13623 -0
  41. package/package.json +15 -2
  42. package/src/components/device-install/device-install-dialog.css +325 -0
  43. package/src/components/device-install/device-install-dialog.tsx +513 -0
  44. package/src/components/device-install/index.ts +2 -0
  45. package/src/components/inspect-overlay.css +6 -0
  46. package/src/components/inspect-overlay.tsx +46 -15
  47. package/src/components/remote-control.tsx +16 -2
  48. package/src/core/ax-tree.test.ts +124 -0
  49. package/src/core/ax-tree.ts +107 -0
  50. package/src/core/device-install/apple/client.ts +152 -0
  51. package/src/core/device-install/apple/crypto.ts +202 -0
  52. package/src/core/device-install/apple/gsa-srp.ts +127 -0
  53. package/src/core/device-install/apple/index.ts +5 -0
  54. package/src/core/device-install/apple/provisioning.ts +298 -0
  55. package/src/core/device-install/apple/relay.ts +221 -0
  56. package/src/core/device-install/index.ts +4 -0
  57. package/src/core/device-install/operations/index.ts +6 -0
  58. package/src/core/device-install/operations/limbuild-client.ts +104 -0
  59. package/src/core/device-install/operations/operations.ts +217 -0
  60. package/src/core/device-install/operations/relay-client.ts +255 -0
  61. package/src/core/device-install/operations/relay-protocol.ts +71 -0
  62. package/src/core/device-install/operations/usbmux.ts +270 -0
  63. package/src/core/device-install/operations/webusb-dom.d.ts +54 -0
  64. package/src/core/device-install/operations/webusb.ts +105 -0
  65. package/src/core/device-install/storage/browser-storage.ts +263 -0
  66. package/src/core/device-install/storage/index.ts +1 -0
  67. package/src/core/device-install/types.ts +65 -0
  68. package/src/device-install/index.ts +3 -0
  69. package/src/device-install/react.ts +1 -0
  70. package/src/hooks/index.ts +1 -0
  71. package/src/hooks/use-device-install.ts +1210 -0
  72. package/src/index.ts +4 -0
  73. package/vite.config.ts +6 -2
@@ -0,0 +1,104 @@
1
+ import type { BuildLogLine, DeviceInstallBuildStatus } from '../types';
2
+
3
+ export type LimbuildInfo = {
4
+ homeDir?: string;
5
+ lastBuildConfig?: {
6
+ bundleId?: string;
7
+ sdk?: string;
8
+ };
9
+ };
10
+
11
+ export type StartSignedDeviceBuildOptions = {
12
+ limbuildApiUrl: string;
13
+ token?: string;
14
+ certificateP12Base64: string;
15
+ certificatePassword: string;
16
+ provisioningProfileBase64: string;
17
+ };
18
+
19
+ export type BuildLogEventsOptions = {
20
+ limbuildApiUrl: string;
21
+ execId: string;
22
+ token?: string;
23
+ onLine: (line: BuildLogLine) => void;
24
+ onStatus: (status: DeviceInstallBuildStatus) => void;
25
+ onError?: (error: Error) => void;
26
+ };
27
+
28
+ export async function fetchLimbuildInfo(limbuildApiUrl: string, token?: string) {
29
+ const url = new URL(`${limbuildApiUrl}/info`);
30
+ if (token) {
31
+ url.searchParams.set('token', token);
32
+ }
33
+ const response = await fetch(url.toString(), {
34
+ headers: token ? { Authorization: `Bearer ${token}` } : undefined,
35
+ });
36
+ if (!response.ok) {
37
+ const body = await response.text();
38
+ throw new Error(`Info request failed: HTTP ${response.status} ${body}`);
39
+ }
40
+ return (await response.json()) as LimbuildInfo;
41
+ }
42
+
43
+ export async function startSignedDeviceBuild({
44
+ limbuildApiUrl,
45
+ token,
46
+ certificateP12Base64,
47
+ certificatePassword,
48
+ provisioningProfileBase64,
49
+ }: StartSignedDeviceBuildOptions) {
50
+ const url = new URL(`${limbuildApiUrl}/exec`);
51
+ if (token) {
52
+ url.searchParams.set('token', token);
53
+ }
54
+ const response = await fetch(url.toString(), {
55
+ method: 'POST',
56
+ headers: {
57
+ 'Content-Type': 'application/json',
58
+ ...(token ? { Authorization: `Bearer ${token}` } : {}),
59
+ },
60
+ body: JSON.stringify({
61
+ command: 'xcodebuild',
62
+ xcodebuild: { sdk: 'iphoneos' },
63
+ signing: {
64
+ certificateP12Base64,
65
+ certificatePassword,
66
+ provisioningProfileBase64,
67
+ },
68
+ }),
69
+ });
70
+ if (!response.ok) {
71
+ const body = await response.text();
72
+ throw new Error(`Build request failed: HTTP ${response.status} ${body}`);
73
+ }
74
+ return (await response.json()) as { execId?: string };
75
+ }
76
+
77
+ export function watchBuildLogEvents({
78
+ limbuildApiUrl,
79
+ execId,
80
+ token,
81
+ onLine,
82
+ onStatus,
83
+ onError,
84
+ }: BuildLogEventsOptions) {
85
+ const url = new URL(`${limbuildApiUrl}/exec/${execId}/events`);
86
+ if (token) {
87
+ url.searchParams.set('token', token);
88
+ }
89
+ const events = new EventSource(url.toString());
90
+ onStatus('running');
91
+ events.addEventListener('command', (event) => onLine({ type: 'command', data: event.data }));
92
+ events.addEventListener('stdout', (event) => onLine({ type: 'stdout', data: event.data }));
93
+ events.addEventListener('stderr', (event) => onLine({ type: 'stderr', data: event.data }));
94
+ events.addEventListener('exitCode', (event) => {
95
+ const code = parseInt(event.data, 10);
96
+ onStatus(code === 0 ? 'succeeded' : code < 0 ? 'cancelled' : 'failed');
97
+ events.close();
98
+ });
99
+ events.onerror = () => {
100
+ events.close();
101
+ onError?.(new Error('Build log stream closed before completion.'));
102
+ };
103
+ return () => events.close();
104
+ }
@@ -0,0 +1,217 @@
1
+ /// <reference path="./webusb-dom.d.ts" />
2
+
3
+ import type { DeviceHello, DeviceInstallLog, PairRecordPayload } from '../types';
4
+ import { RelayClient } from './relay-client';
5
+ import { closeUsbmuxSession, createUsbmuxSession, type UsbmuxSession } from './usbmux';
6
+ import { claimUsbmux, findUsbmuxCandidates, requestAppleDevice, type UsbmuxCandidate } from './webusb';
7
+
8
+ export type DeviceRelayTarget = {
9
+ device: USBDevice;
10
+ candidate: UsbmuxCandidate;
11
+ session?: UsbmuxSession;
12
+ claimedInterfaceNumber?: number;
13
+ hello: DeviceHello;
14
+ };
15
+
16
+ export type RequestUSBAccessOptions = {
17
+ log: DeviceInstallLog;
18
+ };
19
+
20
+ export type StartPairingRelayOptions = {
21
+ limbuildApiUrl: string;
22
+ token?: string;
23
+ log: DeviceInstallLog;
24
+ target: DeviceRelayTarget;
25
+ };
26
+
27
+ export type StartInstallRelayOptions = StartPairingRelayOptions & {
28
+ pairRecord: PairRecordPayload;
29
+ };
30
+
31
+ export async function requestUSBAccess({ log }: RequestUSBAccessOptions) {
32
+ log('Selecting USB device');
33
+ const device = await requestAppleDevice();
34
+ const target = makeDeviceRelayTarget(device);
35
+ log(
36
+ 'Selected USB device',
37
+ `${device.manufacturerName ?? ''} ${device.productName ?? ''} ${device.serialNumber ?? ''}`.trim(),
38
+ );
39
+ return target;
40
+ }
41
+
42
+ export async function startPairingRelay({ limbuildApiUrl, token, log, target }: StartPairingRelayOptions) {
43
+ const deviceRelayUrl = deviceRelayWebSocketUrl(limbuildApiUrl, token);
44
+ let relay: RelayClient | undefined;
45
+ try {
46
+ relay = await connectRelay(deviceRelayUrl, target, log);
47
+ const pairRecord = await relay.startPairing();
48
+ return { relay, pairRecord, target };
49
+ } catch (error) {
50
+ relay?.close();
51
+ throw error;
52
+ }
53
+ }
54
+
55
+ export async function startInstallRelay({
56
+ limbuildApiUrl,
57
+ token,
58
+ log,
59
+ target,
60
+ pairRecord,
61
+ }: StartInstallRelayOptions) {
62
+ const deviceRelayUrl = deviceRelayWebSocketUrl(limbuildApiUrl, token);
63
+ let relay: RelayClient | undefined;
64
+ try {
65
+ relay = await connectRelay(deviceRelayUrl, target, log);
66
+ await relay.startInstall(pairRecord);
67
+ return relay;
68
+ } catch (error) {
69
+ relay?.close();
70
+ throw error;
71
+ }
72
+ }
73
+
74
+ export async function closeDeviceRelayTarget(target: DeviceRelayTarget | undefined, log?: DeviceInstallLog) {
75
+ if (!target) return;
76
+ if (target.session) {
77
+ closeUsbmuxSession(target.session);
78
+ target.session = undefined;
79
+ }
80
+ if (target.claimedInterfaceNumber !== undefined) {
81
+ try {
82
+ await target.device.releaseInterface(target.claimedInterfaceNumber);
83
+ log?.('Released usbmux interface');
84
+ } catch (error) {
85
+ log?.('USB interface release failed', errorMessage(error));
86
+ } finally {
87
+ target.claimedInterfaceNumber = undefined;
88
+ }
89
+ }
90
+ if (target.device.opened) {
91
+ try {
92
+ await target.device.close();
93
+ log?.('Closed USB device');
94
+ } catch (error) {
95
+ log?.('USB device close failed', errorMessage(error));
96
+ }
97
+ }
98
+ }
99
+
100
+ async function connectRelay(deviceRelayUrl: string, target: DeviceRelayTarget, log: DeviceInstallLog) {
101
+ await ensureUsbmuxSession(target, log);
102
+ const relay = new RelayClient(deviceRelayUrl, target.session!, target.hello, log);
103
+ await relay.connect();
104
+ return relay;
105
+ }
106
+
107
+ async function ensureUsbmuxSession(target: DeviceRelayTarget, log: DeviceInstallLog) {
108
+ if (target.session) return;
109
+ try {
110
+ target.candidate = await claimBestUsbmuxCandidate(target, log);
111
+ target.session = await createUsbmuxSession(target.device, target.candidate);
112
+ log('Created usbmux session');
113
+ } catch (error) {
114
+ await closeDeviceRelayTarget(target, log);
115
+ throw error;
116
+ }
117
+ }
118
+
119
+ function makeDeviceRelayTarget(device: USBDevice): DeviceRelayTarget {
120
+ return {
121
+ device,
122
+ candidate: pickUsbmuxCandidate(device),
123
+ hello: {
124
+ serialNumber: device.serialNumber,
125
+ productName: device.productName,
126
+ manufacturerName: device.manufacturerName,
127
+ productId: device.productId,
128
+ vendorId: device.vendorId,
129
+ },
130
+ };
131
+ }
132
+
133
+ async function claimBestUsbmuxCandidate(target: DeviceRelayTarget, log: DeviceInstallLog) {
134
+ const candidates = orderedUsbmuxCandidates(target.device);
135
+ if (candidates.length === 0) throw new Error('No Apple usbmux interface found.');
136
+ let lastError: unknown;
137
+ for (const candidate of candidates) {
138
+ for (const attempt of [1, 2]) {
139
+ try {
140
+ if (!target.device.opened) {
141
+ await target.device.open();
142
+ }
143
+ log(
144
+ 'Claiming usbmux interface',
145
+ `configuration ${candidate.configurationValue}, interface ${candidate.interfaceNumber}, alternate ${candidate.alternateSetting}, attempt ${attempt}`,
146
+ );
147
+ await claimUsbmux(target.device, candidate);
148
+ target.claimedInterfaceNumber = candidate.interfaceNumber;
149
+ log(
150
+ 'Claimed usbmux interface',
151
+ `configuration ${candidate.configurationValue}, interface ${candidate.interfaceNumber}`,
152
+ );
153
+ return candidate;
154
+ } catch (error) {
155
+ lastError = error;
156
+ log(
157
+ 'USB interface claim failed',
158
+ `configuration ${candidate.configurationValue}, interface ${candidate.interfaceNumber}: ${errorMessage(error)}`,
159
+ );
160
+ await resetUSBDevice(target);
161
+ await sleep(250);
162
+ }
163
+ }
164
+ }
165
+ throw lastError instanceof Error ? lastError : new Error('Unable to claim any Apple usbmux interface.');
166
+ }
167
+
168
+ function pickUsbmuxCandidate(device: USBDevice) {
169
+ const candidate = orderedUsbmuxCandidates(device)[0];
170
+ if (!candidate) throw new Error('No Apple usbmux interface found.');
171
+ return candidate;
172
+ }
173
+
174
+ function orderedUsbmuxCandidates(device: USBDevice) {
175
+ const candidates = findUsbmuxCandidates(device);
176
+ const activeConfigurationValue = device.configuration?.configurationValue;
177
+ return [
178
+ ...candidates.filter((item) => item.configurationValue === activeConfigurationValue),
179
+ ...candidates.filter((item) => item.configurationValue !== activeConfigurationValue),
180
+ ];
181
+ }
182
+
183
+ export function deviceRelayWebSocketUrl(limbuildApiUrl: string, token?: string) {
184
+ const url = new URL(limbuildApiUrl);
185
+ url.protocol = url.protocol === 'https:' ? 'wss:' : 'ws:';
186
+ url.pathname = `${url.pathname.replace(/\/$/, '')}/device/ws`;
187
+ if (token) {
188
+ url.searchParams.set('token', token);
189
+ }
190
+ return url.toString();
191
+ }
192
+
193
+ function errorMessage(error: unknown) {
194
+ return error instanceof Error ? error.message : String(error);
195
+ }
196
+
197
+ async function resetUSBDevice(target: DeviceRelayTarget) {
198
+ if (target.claimedInterfaceNumber !== undefined) {
199
+ try {
200
+ await target.device.releaseInterface(target.claimedInterfaceNumber);
201
+ } catch {
202
+ // Best effort: claim failures often mean there is nothing we own to release.
203
+ }
204
+ target.claimedInterfaceNumber = undefined;
205
+ }
206
+ if (target.device.opened) {
207
+ try {
208
+ await target.device.close();
209
+ } catch {
210
+ // Best effort before reopening for the next candidate/attempt.
211
+ }
212
+ }
213
+ }
214
+
215
+ function sleep(ms: number) {
216
+ return new Promise((resolve) => setTimeout(resolve, ms));
217
+ }
@@ -0,0 +1,255 @@
1
+ import type { DeviceHello, DeviceInstallLog, PairRecordPayload } from '../types';
2
+ import { decodeFrame, decodeJson, encodeFrame, encodeJson, RelayMessageType } from './relay-protocol';
3
+ import {
4
+ openStream,
5
+ receiveStreamData,
6
+ sendStreamData,
7
+ type UsbmuxSession,
8
+ type UsbmuxStream,
9
+ } from './usbmux';
10
+
11
+ type OpenStreamPayload = {
12
+ port: number;
13
+ };
14
+
15
+ type ProgressPayload = {
16
+ message: string;
17
+ };
18
+
19
+ export class RelayClient {
20
+ private socket?: WebSocket;
21
+ private streams = new Map<number, UsbmuxStream>();
22
+ private frameQueue = Promise.resolve();
23
+ private closed = false;
24
+ private pairRecordWaiter?: {
25
+ resolve: (record: PairRecordPayload) => void;
26
+ reject: (error: Error) => void;
27
+ };
28
+
29
+ constructor(
30
+ private readonly webSocketUrl: string,
31
+ private readonly session: UsbmuxSession,
32
+ private readonly deviceHello: DeviceHello,
33
+ private readonly log: DeviceInstallLog,
34
+ ) {}
35
+
36
+ async connect() {
37
+ const socket = new WebSocket(this.webSocketUrl);
38
+ socket.binaryType = 'arraybuffer';
39
+ this.socket = socket;
40
+ socket.onclose = () => {
41
+ this.closed = true;
42
+ this.log('Relay socket closed');
43
+ if (this.pairRecordWaiter) {
44
+ this.pairRecordWaiter.reject(new Error('Relay socket closed'));
45
+ this.pairRecordWaiter = undefined;
46
+ }
47
+ };
48
+ await new Promise<void>((resolve, reject) => {
49
+ socket.onopen = () => resolve();
50
+ socket.onerror = () => reject(new Error('WebSocket connection failed'));
51
+ });
52
+ socket.onmessage = (event) => {
53
+ const data = event.data instanceof ArrayBuffer ? new Uint8Array(event.data) : new Uint8Array();
54
+ this.enqueueFrame(data);
55
+ };
56
+ await this.send({
57
+ type: RelayMessageType.DeviceHello,
58
+ requestId: 0,
59
+ streamId: 0,
60
+ payload: encodeJson(this.deviceHello),
61
+ });
62
+ this.log('Connected WebSocket relay');
63
+ }
64
+
65
+ async startPairing() {
66
+ const recordPromise = new Promise<PairRecordPayload>((resolve, reject) => {
67
+ this.pairRecordWaiter = { resolve, reject };
68
+ });
69
+ await this.send({
70
+ type: RelayMessageType.StartPairing,
71
+ requestId: 0,
72
+ streamId: 0,
73
+ payload: encodeJson({}),
74
+ });
75
+ this.log('Pairing requested');
76
+ return recordPromise;
77
+ }
78
+
79
+ async startInstall(pairRecord: PairRecordPayload) {
80
+ await this.send({
81
+ type: RelayMessageType.StartInstall,
82
+ requestId: 0,
83
+ streamId: 0,
84
+ payload: encodeJson(pairRecord),
85
+ });
86
+ this.log('Installation requested');
87
+ }
88
+
89
+ close() {
90
+ this.closed = true;
91
+ this.socket?.close();
92
+ }
93
+
94
+ private enqueueFrame(data: Uint8Array) {
95
+ this.frameQueue = this.frameQueue.then(() => this.handleFrame(decodeFrame(data))).catch((error) => {
96
+ this.log('Relay frame handling failed', error instanceof Error ? error.message : String(error));
97
+ });
98
+ }
99
+
100
+ private async handleFrame(frame: ReturnType<typeof decodeFrame>) {
101
+ switch (frame.type) {
102
+ case RelayMessageType.OpenStream:
103
+ await this.handleOpenStream(
104
+ frame.requestId,
105
+ frame.streamId,
106
+ decodeJson<OpenStreamPayload>(frame.payload).port,
107
+ );
108
+ break;
109
+ case RelayMessageType.StreamData: {
110
+ const stream = this.streams.get(frame.streamId);
111
+ if (!stream) throw new Error(`Unknown stream ${frame.streamId}`);
112
+ await sendStreamData(stream, frame.payload);
113
+ break;
114
+ }
115
+ case RelayMessageType.StreamClose:
116
+ this.streams.delete(frame.streamId);
117
+ break;
118
+ case RelayMessageType.InstallProgress:
119
+ this.log(formatInstallProgress(decodeJson<ProgressPayload>(frame.payload).message));
120
+ break;
121
+ case RelayMessageType.Error:
122
+ this.handleError(frame.payload);
123
+ break;
124
+ case RelayMessageType.PairRecordReady:
125
+ this.handlePairRecordReady(decodeJson<PairRecordPayload>(frame.payload));
126
+ break;
127
+ case RelayMessageType.Ping:
128
+ await this.send({
129
+ type: RelayMessageType.Pong,
130
+ requestId: frame.requestId,
131
+ streamId: 0,
132
+ payload: new Uint8Array(),
133
+ });
134
+ break;
135
+ }
136
+ }
137
+
138
+ private handleError(payload: Uint8Array) {
139
+ const message = decodeServerError(payload);
140
+ this.log('Server error', message);
141
+ if (this.pairRecordWaiter) {
142
+ this.pairRecordWaiter.reject(new Error(message));
143
+ this.pairRecordWaiter = undefined;
144
+ }
145
+ }
146
+
147
+ private handlePairRecordReady(record: PairRecordPayload) {
148
+ this.log('Pair record received', record.udid);
149
+ this.pairRecordWaiter?.resolve(record);
150
+ this.pairRecordWaiter = undefined;
151
+ }
152
+
153
+ private async handleOpenStream(requestId: number, streamId: number, port: number) {
154
+ try {
155
+ const stream = await openStream(this.session, port);
156
+ this.streams.set(streamId, stream);
157
+ await this.send({
158
+ type: RelayMessageType.OpenResult,
159
+ requestId,
160
+ streamId,
161
+ payload: encodeJson({ ok: true }),
162
+ });
163
+ this.log(`Opened device stream ${streamId} to port ${port}`);
164
+ void this.pumpDeviceToServer(streamId, stream);
165
+ } catch (error) {
166
+ this.log(`Open device stream ${streamId} failed`, error instanceof Error ? error.message : String(error));
167
+ await this.send({
168
+ type: RelayMessageType.OpenResult,
169
+ requestId,
170
+ streamId,
171
+ payload: encodeJson({
172
+ ok: false,
173
+ error: error instanceof Error ? error.message : String(error),
174
+ }),
175
+ });
176
+ }
177
+ }
178
+
179
+ private async pumpDeviceToServer(streamId: number, stream: UsbmuxStream) {
180
+ try {
181
+ for (;;) {
182
+ const data = await receiveStreamData(stream);
183
+ if (this.closed) return;
184
+ await this.send({
185
+ type: RelayMessageType.StreamData,
186
+ requestId: 0,
187
+ streamId,
188
+ payload: data,
189
+ });
190
+ }
191
+ } catch (error) {
192
+ this.log(`Device stream ${streamId} closed`, error instanceof Error ? error.message : String(error));
193
+ await this.send({
194
+ type: RelayMessageType.StreamClose,
195
+ requestId: 0,
196
+ streamId,
197
+ payload: encodeJson({
198
+ reason: error instanceof Error ? error.message : String(error),
199
+ }),
200
+ });
201
+ }
202
+ }
203
+
204
+ private async send(frame: Parameters<typeof encodeFrame>[0]) {
205
+ if (this.closed) return;
206
+ if (!this.socket || this.socket.readyState !== WebSocket.OPEN) {
207
+ this.closed = true;
208
+ return;
209
+ }
210
+ this.socket.send(encodeFrame(frame));
211
+ }
212
+ }
213
+
214
+ function decodeServerError(payload: Uint8Array) {
215
+ const text = new TextDecoder().decode(payload);
216
+ try {
217
+ const parsed = JSON.parse(text) as { error?: string };
218
+ return userFacingServerError(parsed.error ?? text);
219
+ } catch {
220
+ return userFacingServerError(text);
221
+ }
222
+ }
223
+
224
+ function userFacingServerError(message: string) {
225
+ return message.replace(/libimobiledevice/g, '').trim();
226
+ }
227
+
228
+ function formatInstallProgress(message: string) {
229
+ const prefix = 'install status: ';
230
+ if (!message.startsWith(prefix)) {
231
+ return message;
232
+ }
233
+ const xml = message.slice(prefix.length);
234
+ const doc = new DOMParser().parseFromString(xml, 'application/xml');
235
+ const dict = doc.querySelector('plist > dict');
236
+ if (!dict) {
237
+ return message;
238
+ }
239
+ const values = readPlistDict(dict);
240
+ const status = values.Status ?? 'Unknown';
241
+ const percent = values.PercentComplete ? `${values.PercentComplete}% ` : '';
242
+ return `Install progress: ${percent}${status}`;
243
+ }
244
+
245
+ function readPlistDict(dict: Element) {
246
+ const result: Record<string, string> = {};
247
+ const children = Array.from(dict.children);
248
+ for (let index = 0; index < children.length; index += 2) {
249
+ const key = children[index];
250
+ const value = children[index + 1];
251
+ if (!key || key.tagName !== 'key' || !value) continue;
252
+ result[key.textContent ?? ''] = value.textContent ?? '';
253
+ }
254
+ return result;
255
+ }
@@ -0,0 +1,71 @@
1
+ export const RELAY_PROTOCOL_VERSION = 1;
2
+ export const RELAY_HEADER_BYTES = 16;
3
+
4
+ export const RelayMessageType = {
5
+ DeviceHello: 1,
6
+ OpenStream: 2,
7
+ OpenResult: 3,
8
+ StreamData: 4,
9
+ StreamClose: 5,
10
+ InstallProgress: 6,
11
+ Error: 7,
12
+ Ping: 8,
13
+ Pong: 9,
14
+ StartPairing: 10,
15
+ StartInstall: 11,
16
+ PairRecordReady: 12,
17
+ } as const;
18
+
19
+ export type RelayMessageType = (typeof RelayMessageType)[keyof typeof RelayMessageType];
20
+
21
+ export type RelayFrame = {
22
+ type: RelayMessageType;
23
+ requestId: number;
24
+ streamId: number;
25
+ payload: Uint8Array;
26
+ };
27
+
28
+ export function encodeFrame(frame: RelayFrame) {
29
+ const result = new Uint8Array(RELAY_HEADER_BYTES + frame.payload.byteLength);
30
+ const view = new DataView(result.buffer);
31
+ view.setUint8(0, RELAY_PROTOCOL_VERSION);
32
+ view.setUint8(1, frame.type);
33
+ view.setUint8(2, 0);
34
+ view.setUint8(3, 0);
35
+ view.setUint32(4, frame.requestId);
36
+ view.setUint32(8, frame.streamId);
37
+ view.setUint32(12, frame.payload.byteLength);
38
+ result.set(frame.payload, RELAY_HEADER_BYTES);
39
+ return result;
40
+ }
41
+
42
+ export function decodeFrame(data: Uint8Array): RelayFrame {
43
+ if (data.byteLength < RELAY_HEADER_BYTES) {
44
+ throw new Error(`Relay frame too short: ${data.byteLength}`);
45
+ }
46
+ const view = new DataView(data.buffer, data.byteOffset, data.byteLength);
47
+ const version = view.getUint8(0);
48
+ if (version !== RELAY_PROTOCOL_VERSION) {
49
+ throw new Error(`Unsupported relay protocol version ${version}`);
50
+ }
51
+ const payloadLength = view.getUint32(12);
52
+ if (data.byteLength !== RELAY_HEADER_BYTES + payloadLength) {
53
+ throw new Error(
54
+ `Relay frame length mismatch: got ${data.byteLength}, expected ${RELAY_HEADER_BYTES + payloadLength}`,
55
+ );
56
+ }
57
+ return {
58
+ type: view.getUint8(1) as RelayMessageType,
59
+ requestId: view.getUint32(4),
60
+ streamId: view.getUint32(8),
61
+ payload: data.slice(RELAY_HEADER_BYTES),
62
+ };
63
+ }
64
+
65
+ export function encodeJson(value: unknown) {
66
+ return new TextEncoder().encode(JSON.stringify(value));
67
+ }
68
+
69
+ export function decodeJson<T>(payload: Uint8Array): T {
70
+ return JSON.parse(new TextDecoder().decode(payload)) as T;
71
+ }