2022-05-05 01:32:12 +02:00
|
|
|
Add support for telemetry endpoint
|
|
|
|
|
2022-09-09 19:05:30 +02:00
|
|
|
To test:
|
2022-12-22 18:25:28 +01:00
|
|
|
1. Create a mock API using [RequestBin](https://requestbin.io/) or [Beeceptor](https://beeceptor.com/)
|
2022-09-27 23:09:53 +02:00
|
|
|
2. Run code-server with `CS_TELEMETRY_URL` set:
|
2022-10-26 20:50:08 +02:00
|
|
|
i.e. `CS_TELEMETRY_URL="https://requestbin.io/1ebub9z1" ./code-server-<version>-macos-amd64/bin/code-server`
|
2022-11-09 23:10:03 +01:00
|
|
|
NOTE: it has to be a production build.
|
2022-09-27 23:09:53 +02:00
|
|
|
3. Load code-server in browser an do things (i.e. open a file)
|
|
|
|
4. Refresh RequestBin and you should see logs
|
2022-09-09 19:05:30 +02:00
|
|
|
|
2022-05-05 01:32:12 +02:00
|
|
|
Index: code-server/lib/vscode/src/vs/server/node/serverServices.ts
|
|
|
|
===================================================================
|
|
|
|
--- code-server.orig/lib/vscode/src/vs/server/node/serverServices.ts
|
|
|
|
+++ code-server/lib/vscode/src/vs/server/node/serverServices.ts
|
2023-10-19 02:50:21 +02:00
|
|
|
@@ -65,6 +65,7 @@ import { IExtensionsScannerService } fro
|
2022-06-21 23:51:46 +02:00
|
|
|
import { ExtensionsScannerService } from 'vs/server/node/extensionsScannerService';
|
2023-02-07 23:22:06 +01:00
|
|
|
import { IExtensionsProfileScannerService } from 'vs/platform/extensionManagement/common/extensionsProfileScannerService';
|
|
|
|
import { IUserDataProfilesService } from 'vs/platform/userDataProfile/common/userDataProfile';
|
2023-10-19 02:50:21 +02:00
|
|
|
+import { TelemetryClient } from 'vs/server/node/telemetryClient';
|
2022-06-21 23:51:46 +02:00
|
|
|
import { NullPolicyService } from 'vs/platform/policy/common/policy';
|
2022-08-17 03:26:19 +02:00
|
|
|
import { OneDataSystemAppender } from 'vs/platform/telemetry/node/1dsAppender';
|
2022-11-09 23:10:03 +01:00
|
|
|
import { LoggerService } from 'vs/platform/log/node/loggerService';
|
2024-05-07 05:27:34 +02:00
|
|
|
@@ -146,11 +147,23 @@ export async function setupServerService
|
2024-05-07 01:53:53 +02:00
|
|
|
const requestService = new RequestService(configurationService, environmentService, logService, loggerService);
|
|
|
|
services.set(IRequestService, requestService);
|
|
|
|
|
|
|
|
+ let isContainer = undefined;
|
|
|
|
+ try {
|
|
|
|
+ await Promises.stat('/run/.containerenv');
|
|
|
|
+ isContainer = true;
|
|
|
|
+ } catch (error) { /* Does not exist, probably. */ }
|
|
|
|
+ if (!isContainer) {
|
|
|
|
+ try {
|
|
|
|
+ const content = await Promises.readFile('/proc/self/cgroup', 'utf8')
|
|
|
|
+ isContainer = content.includes('docker');
|
|
|
|
+ } catch (error) { /* Permission denied, probably. */ }
|
|
|
|
+ }
|
|
|
|
+
|
2023-03-14 21:03:53 +01:00
|
|
|
let oneDsAppender: ITelemetryAppender = NullAppender;
|
2022-08-17 03:26:19 +02:00
|
|
|
const isInternal = isInternalTelemetry(productService, configurationService);
|
2022-05-05 01:32:12 +02:00
|
|
|
if (supportsTelemetry(productService, environmentService)) {
|
2023-10-19 02:50:21 +02:00
|
|
|
- if (!isLoggingOnly(productService, environmentService) && productService.aiConfig?.ariaKey) {
|
2024-05-07 05:27:34 +02:00
|
|
|
- oneDsAppender = new OneDataSystemAppender(requestService, isInternal, eventPrefix, null, productService.aiConfig.ariaKey);
|
|
|
|
+ if (!isLoggingOnly(productService, environmentService) && productService.telemetryEndpoint) {
|
|
|
|
+ oneDsAppender = new OneDataSystemAppender(requestService, isInternal, eventPrefix, null, () => new TelemetryClient(productService.telemetryEndpoint!, isContainer));
|
2023-09-21 01:33:28 +02:00
|
|
|
disposables.add(toDisposable(() => oneDsAppender?.flush())); // Ensure the AI appender is disposed so that it flushes remaining data
|
2022-05-05 01:32:12 +02:00
|
|
|
}
|
2024-05-07 05:27:34 +02:00
|
|
|
|
2022-05-05 01:32:12 +02:00
|
|
|
Index: code-server/lib/vscode/src/vs/server/node/telemetryClient.ts
|
|
|
|
===================================================================
|
|
|
|
--- /dev/null
|
|
|
|
+++ code-server/lib/vscode/src/vs/server/node/telemetryClient.ts
|
2024-05-07 01:53:53 +02:00
|
|
|
@@ -0,0 +1,53 @@
|
2022-08-17 03:26:19 +02:00
|
|
|
+import { AppInsightsCore, IExtendedTelemetryItem, ITelemetryItem } from '@microsoft/1ds-core-js';
|
2022-05-05 01:32:12 +02:00
|
|
|
+import * as https from 'https';
|
|
|
|
+import * as http from 'http';
|
|
|
|
+import * as os from 'os';
|
|
|
|
+
|
2022-08-17 03:26:19 +02:00
|
|
|
+export class TelemetryClient extends AppInsightsCore {
|
2024-05-07 01:53:53 +02:00
|
|
|
+ public constructor(
|
|
|
|
+ private readonly endpoint: string,
|
|
|
|
+ private readonly isContainer: Boolean | undefined) {
|
2022-08-17 03:26:19 +02:00
|
|
|
+ super();
|
2022-05-05 01:32:12 +02:00
|
|
|
+ }
|
|
|
|
+
|
2022-08-17 03:26:19 +02:00
|
|
|
+ public override track(item: IExtendedTelemetryItem | ITelemetryItem): void {
|
|
|
|
+ const options = item.baseData || {}
|
2022-05-05 01:32:12 +02:00
|
|
|
+ if (!options.properties) {
|
|
|
|
+ options.properties = {};
|
|
|
|
+ }
|
|
|
|
+ if (!options.measurements) {
|
|
|
|
+ options.measurements = {};
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ try {
|
|
|
|
+ const cpus = os.cpus();
|
|
|
|
+ options.measurements.cores = cpus.length;
|
|
|
|
+ options.properties['common.cpuModel'] = cpus[0].model;
|
|
|
|
+ } catch (error) {}
|
|
|
|
+
|
|
|
|
+ try {
|
|
|
|
+ options.measurements.memoryFree = os.freemem();
|
|
|
|
+ options.measurements.memoryTotal = os.totalmem();
|
|
|
|
+ } catch (error) {}
|
|
|
|
+
|
|
|
|
+ try {
|
|
|
|
+ options.properties['common.shell'] = os.userInfo().shell;
|
|
|
|
+ options.properties['common.release'] = os.release();
|
|
|
|
+ options.properties['common.arch'] = os.arch();
|
|
|
|
+ } catch (error) {}
|
|
|
|
+
|
2024-05-07 01:53:53 +02:00
|
|
|
+ options.properties['common.isContainer'] = this.isContainer;
|
|
|
|
+
|
2022-05-05 01:32:12 +02:00
|
|
|
+ try {
|
|
|
|
+ const request = (/^http:/.test(this.endpoint) ? http : https).request(this.endpoint, {
|
|
|
|
+ method: 'POST',
|
|
|
|
+ headers: {
|
|
|
|
+ 'Content-Type': 'application/json',
|
|
|
|
+ },
|
|
|
|
+ });
|
|
|
|
+ request.on('error', () => { /* We don't care. */ });
|
|
|
|
+ request.write(JSON.stringify(options));
|
|
|
|
+ request.end();
|
|
|
|
+ } catch (error) {}
|
|
|
|
+ }
|
|
|
|
+}
|
2022-06-21 23:51:46 +02:00
|
|
|
Index: code-server/lib/vscode/src/vs/server/node/webClientServer.ts
|
|
|
|
===================================================================
|
|
|
|
--- code-server.orig/lib/vscode/src/vs/server/node/webClientServer.ts
|
|
|
|
+++ code-server/lib/vscode/src/vs/server/node/webClientServer.ts
|
2024-05-07 02:58:07 +02:00
|
|
|
@@ -318,6 +318,8 @@ export class WebClientServer {
|
2023-05-16 01:44:03 +02:00
|
|
|
scope: vscodeBase + '/',
|
|
|
|
path: base + '/_static/out/browser/serviceWorker.js',
|
2022-06-21 23:51:46 +02:00
|
|
|
},
|
2023-05-16 01:44:03 +02:00
|
|
|
+ enableTelemetry: this._productService.enableTelemetry,
|
2024-05-07 02:58:07 +02:00
|
|
|
+ telemetryEndpoint: this._productService.telemetryEndpoint,
|
2023-05-16 01:44:03 +02:00
|
|
|
embedderIdentifier: 'server-distro',
|
|
|
|
extensionsGallery: this._productService.extensionsGallery,
|
|
|
|
};
|
2024-05-07 02:58:07 +02:00
|
|
|
Index: code-server/lib/vscode/src/vs/base/common/product.ts
|
|
|
|
===================================================================
|
|
|
|
--- code-server.orig/lib/vscode/src/vs/base/common/product.ts
|
|
|
|
+++ code-server/lib/vscode/src/vs/base/common/product.ts
|
|
|
|
@@ -64,6 +64,7 @@ export interface IProductConfiguration {
|
|
|
|
readonly path: string;
|
|
|
|
readonly scope: string;
|
|
|
|
}
|
|
|
|
+ readonly telemetryEndpoint?: string
|
|
|
|
|
|
|
|
readonly version: string;
|
|
|
|
readonly date?: string;
|
|
|
|
Index: code-server/lib/vscode/src/vs/platform/product/common/product.ts
|
|
|
|
===================================================================
|
|
|
|
--- code-server.orig/lib/vscode/src/vs/platform/product/common/product.ts
|
|
|
|
+++ code-server/lib/vscode/src/vs/platform/product/common/product.ts
|
|
|
|
@@ -55,7 +55,8 @@ else if (globalThis._VSCODE_PRODUCT_JSON
|
|
|
|
resourceUrlTemplate: "https://open-vsx.org/vscode/asset/{publisher}/{name}/{version}/Microsoft.VisualStudio.Code.WebResources/{path}",
|
|
|
|
controlUrl: "",
|
|
|
|
recommendationsUrl: "",
|
|
|
|
- })
|
|
|
|
+ }),
|
|
|
|
+ telemetryEndpoint: env.CS_TELEMETRY_URL || product.telemetryEndpoint || "https://v1.telemetry.coder.com/track",
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|