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
79import { fork } from '@ionic/utils-subprocess';
import Debug from 'debug';
import { open, mkdirp } from 'fs-extra';
import { request } from 'https';
import { resolve } from 'path';
import type { Metric } from './telemetry';
import { ENV_PATHS } from './util/cli';
const debug = Debug('capacitor:ipc');
export interface TelemetryIPCMessage {
type: 'telemetry';
data: Metric<string, unknown>;
}
export type IPCMessage = TelemetryIPCMessage;
/**
* Send an IPC message to a forked process.
*/
export async function send(msg: IPCMessage): Promise<void> {
const dir = ENV_PATHS.log;
await mkdirp(dir);
const logPath = resolve(dir, 'ipc.log');
debug('Sending %O IPC message to forked process (logs: %O)', msg.type, logPath);
const fd = await open(logPath, 'a');
const p = fork(process.argv[1], ['๐ก'], { stdio: ['ignore', fd, fd, 'ipc'] });
p.send(msg);
p.disconnect();
p.unref();
}
/**
* Receive and handle an IPC message.
*
* Assume minimal context and keep external dependencies to a minimum.
*/
export async function receive(msg: IPCMessage): Promise<void> {
debug('Received %O IPC message', msg.type);
if (msg.type === 'telemetry') {
const now = new Date().toISOString();
const { data } = msg;
// This request is only made if telemetry is on.
const req = request(
{
hostname: 'api.ionicjs.com',
port: 443,
path: '/events/metrics',
method: 'POST',
headers: {
'Content-Type': 'application/json',
},
},
(response) => {
debug('Sent %O metric to events service (status: %O)', data.name, response.statusCode);
if (response.statusCode !== 204) {
response.on('data', (chunk) => {
debug('Bad response from events service. Request body: %O', chunk.toString());
});
}
},
);
const body = {
metrics: [data],
sent_at: now,
};
req.end(JSON.stringify(body));
}
}