-
Notifications
You must be signed in to change notification settings - Fork 180
/
Copy pathremote.ts
125 lines (113 loc) · 3.83 KB
/
remote.ts
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
// access main process remote modules via attachments to `global`
import type { AxiosRequestConfig, AxiosResponse } from 'axios'
import type {
Remote,
NotifyTopic,
NotifyResponseData,
IPCSafeFormData,
} from './types'
const emptyRemote: Remote = {} as any
export const remote: Remote = new Proxy(emptyRemote, {
get(_target, propName: string): unknown {
console.assert(
(global as any).APP_SHELL_REMOTE,
'Expected APP_SHELL_REMOTE to be attached to global scope; is app-shell/src/preload.ts properly configured?'
)
console.assert(
propName in (global as any).APP_SHELL_REMOTE,
`Expected APP_SHELL_REMOTE.${propName} to exist, is app-shell/src/preload.ts properly configured?`
)
return (global as any).APP_SHELL_REMOTE[propName] as Remote
},
})
// FormData and File objects can't be sent through invoke().
// This converts them into simpler objects that can be.
// app-shell will convert them back.
async function proxyFormData(formData: FormData): Promise<IPCSafeFormData> {
const result: IPCSafeFormData = []
for (const [name, value] of formData.entries()) {
if (value instanceof File) {
result.push({
type: 'file',
name,
// todo(mm, 2024-04-24): Send just the (full) filename instead of the file
// contents, to avoid the IPC message ballooning into several MB.
value: await value.arrayBuffer(),
filename: value.name,
})
} else {
result.push({ type: 'string', name, value })
}
}
return result
}
export async function appShellRequestor<Data>(
config: AxiosRequestConfig
): Promise<AxiosResponse<Data>> {
const { data } = config
const formDataProxy =
data instanceof FormData
? { proxiedFormData: await proxyFormData(data) }
: data
const configProxy = { ...config, data: formDataProxy }
const result = await remote.ipcRenderer.invoke('usb:request', configProxy)
if (result?.error != null) {
throw result.error
}
return result
}
interface CallbackStore {
[hostname: string]: {
[topic in NotifyTopic]: Array<(data: NotifyResponseData) => void>
}
}
const callbackStore: CallbackStore = {}
interface AppShellListener {
hostname: string
notifyTopic: NotifyTopic
/* The callback MUST be memoized so it's identity is stable between calls from the same location. */
callback: (data: NotifyResponseData) => void
isDismounting?: boolean
}
export function appShellListener({
hostname,
notifyTopic,
callback,
isDismounting = false,
}: AppShellListener): CallbackStore {
// The shell emits general messages to ALL_TOPICS, typically errors, and all listeners must handle those messages.
const topics: NotifyTopic[] = [notifyTopic, 'ALL_TOPICS'] as const
topics.forEach(topic => {
if (isDismounting) {
const callbacks = callbackStore[hostname]?.[topic]
if (callbacks != null) {
callbackStore[hostname][topic] = callbacks.filter(cb => cb !== callback)
if (!callbackStore[hostname][topic].length) {
// eslint-disable-next-line @typescript-eslint/no-dynamic-delete
delete callbackStore[hostname][topic]
if (!Object.keys(callbackStore[hostname]).length) {
// eslint-disable-next-line @typescript-eslint/no-dynamic-delete
delete callbackStore[hostname]
}
}
}
} else {
callbackStore[hostname] = callbackStore[hostname] ?? {}
callbackStore[hostname][topic] ??= []
// Assumes callback is memoized.
if (!callbackStore[hostname][topic].includes(callback)) {
callbackStore[hostname][topic].push(callback)
}
}
})
return callbackStore
}
// Instantiate the notify listener at runtime.
remote.ipcRenderer.on(
'notify',
(_, shellHostname, shellTopic, shellMessage) => {
callbackStore[shellHostname]?.[shellTopic]?.forEach(cb => {
cb(shellMessage)
})
}
)