Update telemetry to use new errors information (#5547)

* Properly get information from AstroError in telemetry

* Record telemetry for dev errors

* Fix tests

* ci(telemetry): Make sure we don't hit telemetry in tests

* fix(build): Fix types

* test: Remove unnecessary flag

* test(errors): Add test to make sure we don't have stacktraces inside error messages

* refactor(types): Type cast `getErrorDataByCode`'s return type so we don't have to cast to any later
This commit is contained in:
Erika 2023-01-04 17:34:11 +01:00 committed by GitHub
parent 2a57864195
commit 63745f8d5a
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
13 changed files with 130 additions and 24 deletions

View file

@ -990,6 +990,7 @@ export interface AstroSettings {
tsConfig: TsConfigJson | undefined; tsConfig: TsConfigJson | undefined;
tsConfigPath: string | undefined; tsConfigPath: string | undefined;
watchFiles: string[]; watchFiles: string[];
forceDisableTelemetry: boolean;
} }
export type AsyncRendererComponentFn<U> = ( export type AsyncRendererComponentFn<U> = (

View file

@ -18,6 +18,7 @@ export function createBaseSettings(config: AstroConfig): AstroSettings {
renderers: [jsxRenderer], renderers: [jsxRenderer],
scripts: [], scripts: [],
watchFiles: [], watchFiles: [],
forceDisableTelemetry: false,
}; };
} }

View file

@ -47,6 +47,7 @@ export interface CreateContainerParams {
// The string passed to --config and the resolved path // The string passed to --config and the resolved path
configFlag?: string; configFlag?: string;
configFlagPath?: string; configFlagPath?: string;
disableTelemetry?: boolean;
} }
export async function createContainer(params: CreateContainerParams = {}): Promise<Container> { export async function createContainer(params: CreateContainerParams = {}): Promise<Container> {
@ -55,8 +56,13 @@ export async function createContainer(params: CreateContainerParams = {}): Promi
logging = defaultLogging, logging = defaultLogging,
settings = await createDefaultDevSettings(params.userConfig, params.root), settings = await createDefaultDevSettings(params.userConfig, params.root),
fs = nodeFs, fs = nodeFs,
disableTelemetry,
} = params; } = params;
if (disableTelemetry) {
settings.forceDisableTelemetry = true;
}
// Initialize // Initialize
applyPolyfill(); applyPolyfill();
settings = await runHookConfigSetup({ settings = await runHookConfigSetup({
@ -143,11 +149,14 @@ export function isStarted(container: Container): boolean {
return !!container.viteServer.httpServer?.listening; return !!container.viteServer.httpServer?.listening;
} }
/**
* Only used in tests
*/
export async function runInContainer( export async function runInContainer(
params: CreateContainerParams, params: CreateContainerParams,
callback: (container: Container) => Promise<void> | void callback: (container: Container) => Promise<void> | void
) { ) {
const container = await createContainer(params); const container = await createContainer({ ...params, disableTelemetry: true });
try { try {
await callback(container); await callback(container);
} finally { } finally {

View file

@ -4,7 +4,7 @@
import type { ZodError } from 'zod'; import type { ZodError } from 'zod';
interface ErrorData { export interface ErrorData {
code: number; code: number;
title: string; title: string;
message?: string | ((...params: any) => string); message?: string | ((...params: any) => string);

View file

@ -1,6 +1,6 @@
import { DiagnosticCode } from '@astrojs/compiler/shared/diagnostics.js'; import { DiagnosticCode } from '@astrojs/compiler/shared/diagnostics.js';
import type { SSRError } from '../../@types/astro.js'; import type { SSRError } from '../../@types/astro.js';
import { AstroErrorCodes, AstroErrorData } from './errors-data.js'; import { AstroErrorCodes, AstroErrorData, ErrorData } from './errors-data.js';
/** /**
* Get the line and character based on the offset * Get the line and character based on the offset
@ -96,7 +96,7 @@ export function getErrorDataByCode(code: AstroErrorCodes | DiagnosticCode) {
if (entry) { if (entry) {
return { return {
name: entry[0], name: entry[0],
data: entry[1], data: entry[1] as ErrorData,
}; };
} }
} }

View file

@ -36,6 +36,7 @@ export function createDevelopmentEnvironment(
site: settings.config.site, site: settings.config.site,
ssr: settings.config.output === 'server', ssr: settings.config.output === 'server',
streaming: true, streaming: true,
telemetry: Boolean(settings.forceDisableTelemetry),
}); });
return { return {

View file

@ -21,6 +21,7 @@ export interface Environment {
site?: string; site?: string;
ssr: boolean; ssr: boolean;
streaming: boolean; streaming: boolean;
telemetry?: boolean;
} }
export type CreateEnvironmentArgs = Environment; export type CreateEnvironmentArgs = Environment;
@ -33,6 +34,9 @@ export type CreateBasicEnvironmentArgs = Partial<Environment> & {
logging: CreateEnvironmentArgs['logging']; logging: CreateEnvironmentArgs['logging'];
}; };
/**
* Only used in tests
*/
export function createBasicEnvironment(options: CreateBasicEnvironmentArgs): Environment { export function createBasicEnvironment(options: CreateBasicEnvironmentArgs): Environment {
const mode = options.mode ?? 'development'; const mode = options.mode ?? 'development';
return createEnvironment({ return createEnvironment({
@ -48,5 +52,6 @@ export function createBasicEnvironment(options: CreateBasicEnvironmentArgs): Env
routeCache: new RouteCache(options.logging, mode), routeCache: new RouteCache(options.logging, mode),
ssr: options.ssr ?? true, ssr: options.ssr ?? true,
streaming: options.streaming ?? true, streaming: options.streaming ?? true,
telemetry: false,
}); });
} }

View file

@ -1,10 +1,12 @@
import { ZodError } from 'zod'; import { ZodError } from 'zod';
import { AstroErrorData, ErrorWithMetadata } from '../core/errors/index.js'; import { AstroError, AstroErrorData, ErrorWithMetadata } from '../core/errors/index.js';
import { getErrorDataByCode } from '../core/errors/utils.js';
const EVENT_ERROR = 'ASTRO_CLI_ERROR'; const EVENT_ERROR = 'ASTRO_CLI_ERROR';
interface ErrorEventPayload { interface ErrorEventPayload {
code: number | undefined; code: number | undefined;
name: string;
isFatal: boolean; isFatal: boolean;
plugin?: string | undefined; plugin?: string | undefined;
cliCommand: string; cliCommand: string;
@ -22,10 +24,8 @@ interface ConfigErrorEventPayload extends ErrorEventPayload {
* content like a filename, filepath, or any other code-specific value. * content like a filename, filepath, or any other code-specific value.
* We also trim this value even further to just a few words. * We also trim this value even further to just a few words.
* *
* Our goal is to remove this entirely before v1.0.0 is released, as we work * This is only used for errors that do not come from us so we can get a basic
* to add a proper error code system (see AstroErrorCodes for examples). * and anonymous idea of what the error is about.
*
* TODO(fks): Remove around v1.0.0 release.
*/ */
const ANONYMIZE_MESSAGE_REGEX = /^(\w| )+/; const ANONYMIZE_MESSAGE_REGEX = /^(\w| )+/;
function anonymizeErrorMessage(msg: string): string | undefined { function anonymizeErrorMessage(msg: string): string | undefined {
@ -47,6 +47,7 @@ export function eventConfigError({
}): { eventName: string; payload: ConfigErrorEventPayload }[] { }): { eventName: string; payload: ConfigErrorEventPayload }[] {
const payload: ConfigErrorEventPayload = { const payload: ConfigErrorEventPayload = {
code: AstroErrorData.UnknownConfigError.code, code: AstroErrorData.UnknownConfigError.code,
name: 'ZodError',
isFatal, isFatal,
isConfig: true, isConfig: true,
cliCommand: cmd, cliCommand: cmd,
@ -64,12 +65,51 @@ export function eventError({
cmd: string; cmd: string;
isFatal: boolean; isFatal: boolean;
}): { eventName: string; payload: ErrorEventPayload }[] { }): { eventName: string; payload: ErrorEventPayload }[] {
const errorData =
AstroError.is(err) && err.errorCode ? getErrorDataByCode(err.errorCode)?.data : undefined;
const payload: ErrorEventPayload = { const payload: ErrorEventPayload = {
code: err.code || AstroErrorData.UnknownError.code, code: err.code || err.errorCode || AstroErrorData.UnknownError.code,
name: err.name,
plugin: err.plugin, plugin: err.plugin,
cliCommand: cmd, cliCommand: cmd,
isFatal: isFatal, isFatal: isFatal,
anonymousMessageHint: anonymizeErrorMessage(err.message), anonymousMessageHint:
errorData && errorData.message
? getSafeErrorMessage(errorData.message)
: anonymizeErrorMessage(err.message),
}; };
return [{ eventName: EVENT_ERROR, payload }]; return [{ eventName: EVENT_ERROR, payload }];
} }
/**
* Safely get the error message from an error, even if it's a function.
*/
// eslint-disable-next-line @typescript-eslint/ban-types
function getSafeErrorMessage(message: string | Function): string {
if (typeof message === 'string') {
return message;
} else {
return String.raw({
raw: extractStringFromFunction(message.toString()),
});
}
function extractStringFromFunction(func: string) {
const arrowIndex = func.indexOf('=>') + '=>'.length;
return func
.slice(arrowIndex)
.trim()
.slice(1, -1)
.replace(
/\${([^}]+)}/gm,
(str, match1) =>
`${match1
.split(/\.?(?=[A-Z])/)
.join('_')
.toUpperCase()}`
)
.replace(/\\`/g, '`');
}
}

View file

@ -8,6 +8,7 @@ import { createSafeError } from '../core/errors/index.js';
import { error } from '../core/logger/core.js'; import { error } from '../core/logger/core.js';
import * as msg from '../core/messages.js'; import * as msg from '../core/messages.js';
import { removeTrailingForwardSlash } from '../core/path.js'; import { removeTrailingForwardSlash } from '../core/path.js';
import { eventError, telemetry } from '../events/index.js';
import { runWithErrorHandling } from './controller.js'; import { runWithErrorHandling } from './controller.js';
import { handle500Response } from './response.js'; import { handle500Response } from './response.js';
import { handleRoute, matchRoute } from './route.js'; import { handleRoute, matchRoute } from './route.js';
@ -83,6 +84,10 @@ export async function handleRequest(
// Our error should already be complete, but let's try to add a bit more through some guesswork // Our error should already be complete, but let's try to add a bit more through some guesswork
const errorWithMetadata = collectErrorMetadata(err, config.root); const errorWithMetadata = collectErrorMetadata(err, config.root);
if (env.telemetry !== false) {
telemetry.record(eventError({ cmd: 'dev', err: errorWithMetadata, isFatal: false }));
}
error(env.logging, null, msg.formatErrorMessage(errorWithMetadata)); error(env.logging, null, msg.formatErrorMessage(errorWithMetadata));
handle500Response(moduleLoader, res, errorWithMetadata); handle500Response(moduleLoader, res, errorWithMetadata);

View file

@ -1,5 +1,6 @@
import { expect } from 'chai'; import { expect } from 'chai';
import { AstroErrorData } from '../dist/core/errors/errors-data.js'; import { AstroErrorData } from '../dist/core/errors/errors-data.js';
import { AstroError } from '../dist/core/errors/errors.js';
import * as events from '../dist/events/index.js'; import * as events from '../dist/events/index.js';
describe('Events', () => { describe('Events', () => {
@ -427,6 +428,7 @@ describe('Events', () => {
eventName: 'ASTRO_CLI_ERROR', eventName: 'ASTRO_CLI_ERROR',
payload: { payload: {
code: AstroErrorData.UnknownConfigError.code, code: AstroErrorData.UnknownConfigError.code,
name: 'ZodError',
isFatal: true, isFatal: true,
isConfig: true, isConfig: true,
cliCommand: 'COMMAND_NAME', cliCommand: 'COMMAND_NAME',
@ -451,6 +453,7 @@ describe('Events', () => {
payload: { payload: {
code: 1234, code: 1234,
plugin: 'TEST PLUGIN', plugin: 'TEST PLUGIN',
name: 'Error',
isFatal: true, isFatal: true,
cliCommand: 'COMMAND_NAME', cliCommand: 'COMMAND_NAME',
anonymousMessageHint: 'TEST ERROR MESSAGE', anonymousMessageHint: 'TEST ERROR MESSAGE',
@ -458,6 +461,30 @@ describe('Events', () => {
}); });
}); });
it('returns the expected event payload for AstroError', () => {
const [event] = events.eventError({
err: new AstroError({
...AstroErrorData.ClientAddressNotAvailable,
message: AstroErrorData.ClientAddressNotAvailable.message('mysuperadapter'),
}),
cmd: 'COMMAND_NAME',
isFatal: false,
});
expect(event).to.deep.equal({
eventName: 'ASTRO_CLI_ERROR',
payload: {
anonymousMessageHint:
'`Astro.clientAddress` is not available in the `ADAPTER_NAME` adapter. File an issue with the adapter to add support.',
cliCommand: 'COMMAND_NAME',
code: 3002,
isFatal: false,
name: 'ClientAddressNotAvailable',
plugin: undefined,
},
});
});
it('returns the expected event payload with a generic error', () => { it('returns the expected event payload with a generic error', () => {
const [event] = events.eventError({ const [event] = events.eventError({
err: new Error('TEST ERROR MESSAGE'), err: new Error('TEST ERROR MESSAGE'),
@ -468,6 +495,7 @@ describe('Events', () => {
eventName: 'ASTRO_CLI_ERROR', eventName: 'ASTRO_CLI_ERROR',
payload: { payload: {
code: AstroErrorData.UnknownError.code, code: AstroErrorData.UnknownError.code,
name: 'Error',
plugin: undefined, plugin: undefined,
isFatal: false, isFatal: false,
cliCommand: 'COMMAND_NAME', cliCommand: 'COMMAND_NAME',
@ -480,9 +508,26 @@ describe('Events', () => {
const [event] = events.eventError({ const [event] = events.eventError({
err: new Error('TEST ERROR MESSAGE: Sensitive data is "/Users/MYNAME/foo.astro"'), err: new Error('TEST ERROR MESSAGE: Sensitive data is "/Users/MYNAME/foo.astro"'),
cmd: 'COMMAND_NAME', cmd: 'COMMAND_NAME',
name: 'Error',
isFatal: true, isFatal: true,
}); });
expect(event.payload.anonymousMessageHint).to.equal('TEST ERROR MESSAGE'); expect(event.payload.anonymousMessageHint).to.equal('TEST ERROR MESSAGE');
}); });
it('properly exclude stack traces from anonymousMessageHint', () => {
// Some libraries/frameworks returns stack traces in the error message, make sure we don't include that in the anonymousMessageHint
const [event] = events.eventError({
err: new Error(`[postcss] /home/projects/github-ssfd5p/src/components/Counter.css:3:15: Missed semicolon
at Input.error (file:///home/projects/github-ssfd5p/node_modules/postcss/lib/input.js:148:16)
at Parser.checkMissedSemicolon (file:///home/projects/github-ssfd5p/node_modules/postcss/lib/parser.js:596:22)
at Parser.decl (file:///home/projects/github-ssfd5p/node_modules/postcss/lib/parser.js:279:12)
at Parser.other (file:///home/projects/github-ssfd5p/node_modules/postcss/lib/parser.js:128:18)
at Parser.parse (file:///home/projects/github-ssfd5p/node_modules/postcss/lib/parser.js:72:16)`),
cmd: 'COMMAND_NAME',
name: 'Error',
isFatal: true,
});
expect(event.payload.anonymousMessageHint).to.be.undefined;
});
}); });
}); });

View file

@ -1,17 +1,17 @@
import { execa } from 'execa';
import { polyfill } from '@astrojs/webapi'; import { polyfill } from '@astrojs/webapi';
import { execa } from 'execa';
import fastGlob from 'fast-glob';
import fs from 'fs'; import fs from 'fs';
import os from 'os';
import stripAnsi from 'strip-ansi';
import { fileURLToPath } from 'url'; import { fileURLToPath } from 'url';
import { sync } from '../dist/cli/sync/index.js';
import build from '../dist/core/build/index.js';
import { loadConfig } from '../dist/core/config/config.js'; import { loadConfig } from '../dist/core/config/config.js';
import { createSettings } from '../dist/core/config/index.js'; import { createSettings } from '../dist/core/config/index.js';
import dev from '../dist/core/dev/index.js'; import dev from '../dist/core/dev/index.js';
import build from '../dist/core/build/index.js';
import preview from '../dist/core/preview/index.js';
import { sync } from '../dist/cli/sync/index.js';
import { nodeLogDestination } from '../dist/core/logger/node.js'; import { nodeLogDestination } from '../dist/core/logger/node.js';
import os from 'os'; import preview from '../dist/core/preview/index.js';
import stripAnsi from 'strip-ansi';
import fastGlob from 'fast-glob';
// polyfill WebAPIs to globalThis for Node v12, Node v14, and Node v16 // polyfill WebAPIs to globalThis for Node v12, Node v14, and Node v16
polyfill(globalThis, { polyfill(globalThis, {
@ -243,7 +243,7 @@ const cliPath = fileURLToPath(new URL('../astro.js', import.meta.url));
/** Returns a process running the Astro CLI. */ /** Returns a process running the Astro CLI. */
export function cli(/** @type {string[]} */ ...args) { export function cli(/** @type {string[]} */ ...args) {
const spawned = execa('node', [cliPath, ...args]); const spawned = execa('node', [cliPath, ...args], { env: {'ASTRO_TELEMETRY_DISABLED': true}});
spawned.stdout.setEncoding('utf8'); spawned.stdout.setEncoding('utf8');

View file

@ -1,13 +1,13 @@
import { expect } from 'chai'; import { expect } from 'chai';
import { createLoader } from '../../../dist/core/module-loader/index.js';
import { createController, handleRequest } from '../../../dist/vite-plugin-astro-server/index.js';
import { createDefaultDevSettings } from '../../../dist/core/config/index.js'; import { createDefaultDevSettings } from '../../../dist/core/config/index.js';
import { createLoader } from '../../../dist/core/module-loader/index.js';
import { createBasicEnvironment } from '../../../dist/core/render/index.js'; import { createBasicEnvironment } from '../../../dist/core/render/index.js';
import { createRouteManifest } from '../../../dist/core/routing/index.js'; import { createRouteManifest } from '../../../dist/core/routing/index.js';
import { defaultLogging as logging } from '../../test-utils.js';
import { createComponent, render } from '../../../dist/runtime/server/index.js'; import { createComponent, render } from '../../../dist/runtime/server/index.js';
import { createRequestAndResponse, createFs, createAstroModule } from '../test-utils.js'; import { createController, handleRequest } from '../../../dist/vite-plugin-astro-server/index.js';
import { defaultLogging as logging } from '../../test-utils.js';
import { createAstroModule, createFs, createRequestAndResponse } from '../test-utils.js';
async function createDevEnvironment(overrides = {}) { async function createDevEnvironment(overrides = {}) {
const env = createBasicEnvironment({ const env = createBasicEnvironment({

View file

@ -145,7 +145,6 @@ export class AstroTelemetry {
// to preview what data would be sent. // to preview what data would be sent.
return Promise.resolve(); return Promise.resolve();
} }
return post({ return post({
context, context,
meta, meta,