From 050a2dc2087b703332b8e54f29489344a9d780a4 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?berislav=20grgi=C4=8Dak?= Date: Thu, 29 Feb 2024 08:39:33 +0100 Subject: [PATCH] Add logging support to Playground (#1035) ## What is this PR doing? This PR adds support for collecting PHP logs and WordPress debug logs in Playground. After each request logs are printed with `console.debug` in the browser. Support for Node, WP-now, and VSCode will be added in separate PRs. We also plan to add more Playground logs in the future to help us improve the stability of WordPress Playground. ## What problem is it solving? When Playground crashes it's hard to know why it happened. By adding access to PHP logs and users will be able to debug their sites. Playground developers will benefit from having Playground and PHP logs in a single place and a shared format. ## How is the problem addressed? On `request.end` we collect logs from the current PHP request and send them back with the event. These logs are caught by the Logger and printed. Additionally, the logger catches some of the Playground errors using regular `window` events. ## Testing Instructions - Checkout this branch - Start Playground `npm run dev` - [Open Playground in your browser](http://localhost:5400/website-server/?url=/wp-admin/post-new.php&networking=yes) - Confirm that the debug log output is recorded in the browser console after each request - Add `throw new Exception('This is a test exception');` after this file of https://github.com/WordPress/wordpress-playground/blob/c23dd927a92501fa94fd38d6eebc0d0ff39699eb/packages/playground/remote/src/lib/playground-mu-plugin/playground-includes/playground_logger.php#L9 - Refresh Playground and confirm that the PHP log is printed. This confirms that we can collect logs even if Playground fails to run - Replace the Exception from /playground_logger.php with this code ``` add_action('init', function () { if (isset($_GET['post'])) { throw new Exception('This is a test exception'); } }); ``` - Refresh Playground - Confirm that it loads - Go to a post edit page using wp-admin - Confirm that the post edit page crashes and the debug log is printed --- package-lock.json | 12 ++ packages/php-wasm/logger/.eslintrc.json | 18 ++ packages/php-wasm/logger/README.md | 0 packages/php-wasm/logger/package.json | 28 +++ packages/php-wasm/logger/project.json | 50 +++++ packages/php-wasm/logger/src/index.ts | 1 + packages/php-wasm/logger/src/logger.ts | 188 ++++++++++++++++++ packages/php-wasm/logger/tsconfig.json | 23 +++ packages/php-wasm/logger/tsconfig.lib.json | 10 + packages/php-wasm/logger/tsconfig.spec.json | 19 ++ packages/php-wasm/logger/vite.config.ts | 48 +++++ .../playground/blueprints/src/lib/compile.ts | 6 +- packages/playground/client/src/index.ts | 2 + packages/playground/node/src/index.ts | 5 +- .../lib/playground-mu-plugin/0-playground.php | 36 ++-- .../remote/src/lib/worker-thread.ts | 3 + packages/playground/website/src/main.tsx | 7 +- tsconfig.base.json | 1 + 18 files changed, 437 insertions(+), 20 deletions(-) create mode 100644 packages/php-wasm/logger/.eslintrc.json create mode 100644 packages/php-wasm/logger/README.md create mode 100644 packages/php-wasm/logger/package.json create mode 100644 packages/php-wasm/logger/project.json create mode 100644 packages/php-wasm/logger/src/index.ts create mode 100644 packages/php-wasm/logger/src/logger.ts create mode 100644 packages/php-wasm/logger/tsconfig.json create mode 100644 packages/php-wasm/logger/tsconfig.lib.json create mode 100644 packages/php-wasm/logger/tsconfig.spec.json create mode 100644 packages/php-wasm/logger/vite.config.ts diff --git a/package-lock.json b/package-lock.json index 98df7737ba..8f71df2eaf 100644 --- a/package-lock.json +++ b/package-lock.json @@ -13990,6 +13990,10 @@ "resolved": "packages/php-wasm/fs-journal", "link": true }, + "node_modules/@php-wasm/logger": { + "resolved": "packages/php-wasm/logger", + "link": true + }, "node_modules/@php-wasm/node": { "resolved": "packages/php-wasm/node", "link": true @@ -44944,6 +44948,14 @@ "npm": ">=8.11.0" } }, + "packages/php-wasm/logger": { + "version": "0.0.1", + "license": "GPL-2.0-or-later", + "engines": { + "node": ">=18.18.2", + "npm": ">=8.11.0" + } + }, "packages/php-wasm/node": { "name": "@php-wasm/node", "version": "0.6.3", diff --git a/packages/php-wasm/logger/.eslintrc.json b/packages/php-wasm/logger/.eslintrc.json new file mode 100644 index 0000000000..79fd7c1d98 --- /dev/null +++ b/packages/php-wasm/logger/.eslintrc.json @@ -0,0 +1,18 @@ +{ + "extends": ["../../../.eslintrc.json"], + "ignorePatterns": ["!**/*"], + "overrides": [ + { + "files": ["*.ts", "*.tsx", "*.js", "*.jsx"], + "rules": {} + }, + { + "files": ["*.ts", "*.tsx"], + "rules": {} + }, + { + "files": ["*.js", "*.jsx"], + "rules": {} + } + ] +} diff --git a/packages/php-wasm/logger/README.md b/packages/php-wasm/logger/README.md new file mode 100644 index 0000000000..e69de29bb2 diff --git a/packages/php-wasm/logger/package.json b/packages/php-wasm/logger/package.json new file mode 100644 index 0000000000..8c346b0991 --- /dev/null +++ b/packages/php-wasm/logger/package.json @@ -0,0 +1,28 @@ +{ + "name": "@php-wasm/logger", + "version": "0.0.1", + "description": "A logger for PHP-wasm clients like Playground and WP-now.", + "repository": { + "type": "git", + "url": "https://github.com/WordPress/wordpress-playground" + }, + "homepage": "https://developer.wordpress.org/playground", + "author": "The WordPress contributors", + "typedoc": { + "entryPoint": "./src/index.ts", + "readmeFile": "./README.md", + "displayName": "@php-wasm/logger", + "tsconfig": "./tsconfig.lib.json" + }, + "publishConfig": { + "access": "public", + "directory": "../../../dist/packages/php-wasm/logger" + }, + "license": "GPL-2.0-or-later", + "main": "index.cjs", + "types": "index.d.ts", + "engines": { + "node": ">=18.18.2", + "npm": ">=8.11.0" + } +} diff --git a/packages/php-wasm/logger/project.json b/packages/php-wasm/logger/project.json new file mode 100644 index 0000000000..d9a1f51f3f --- /dev/null +++ b/packages/php-wasm/logger/project.json @@ -0,0 +1,50 @@ +{ + "name": "php-wasm-logger", + "$schema": "../../../node_modules/nx/schemas/project-schema.json", + "sourceRoot": "packages/php-wasm/logger/src", + "projectType": "library", + "targets": { + "build": { + "executor": "@nx/vite:build", + "outputs": ["{options.outputPath}"], + "options": { + "outputPath": "dist/packages/php-wasm/logger" + } + }, + "test": { + "executor": "nx:noop", + "dependsOn": ["test:vite"] + }, + "test:esmcjs": { + "executor": "@wp-playground/nx-extensions:assert-built-esm-and-cjs", + "options": { + "outputPath": "dist/packages/php-wasm/logger" + }, + "dependsOn": ["build"] + }, + "test:vite": { + "executor": "@nx/vite:test", + "outputs": ["{workspaceRoot}/coverage/packages/php-wasm/logger"], + "options": { + "passWithNoTests": true, + "reportsDirectory": "../../../coverage/packages/php-wasm/logger" + } + }, + "lint": { + "executor": "@nx/linter:eslint", + "outputs": ["{options.outputFile}"], + "options": { + "lintFilePatterns": ["packages/php-wasm/logger/**/*.ts"] + } + } + }, + "typecheck": { + "executor": "nx:run-commands", + "options": { + "commands": [ + "tsc -p packages/php-wasm/logger/tsconfig.lib.json --noEmit", + "tsc -p packages/php-wasm/logger/tsconfig.spec.json --noEmit" + ] + } + } +} diff --git a/packages/php-wasm/logger/src/index.ts b/packages/php-wasm/logger/src/index.ts new file mode 100644 index 0000000000..1ff09efd40 --- /dev/null +++ b/packages/php-wasm/logger/src/index.ts @@ -0,0 +1 @@ +export * from './logger'; diff --git a/packages/php-wasm/logger/src/logger.ts b/packages/php-wasm/logger/src/logger.ts new file mode 100644 index 0000000000..777edd1038 --- /dev/null +++ b/packages/php-wasm/logger/src/logger.ts @@ -0,0 +1,188 @@ +import { UniversalPHP } from '@php-wasm/universal/src/lib/universal-php'; +/** + * Log severity levels. + */ +export type LogSeverity = 'debug' | 'info' | 'warn' | 'error' | 'fatal'; + +/** + * A logger for Playground. + */ +export class Logger { + private readonly LOG_PREFIX = 'Playground'; + + /** + * Whether the window events are connected. + */ + private windowConnected = false; + + /** + * The length of the last PHP log. + */ + private lastPHPLogLength = 0; + + /** + * The path to the error log file. + */ + private errorLogPath = '/wordpress/wp-content/debug.log'; + + constructor(errorLogPath?: string) { + if (errorLogPath) { + this.errorLogPath = errorLogPath; + } + } + + /** + * Read the WordPress debug.log file and return its content. + * + * @param UniversalPHP playground instance + * @returns string The content of the debug.log file + */ + private async getRequestPhpErrorLog(playground: UniversalPHP) { + if (!(await playground.fileExists(this.errorLogPath))) { + return ''; + } + return await playground.readFileAsText(this.errorLogPath); + } + + /** + * Log Windows errors. + * + * @param ErrorEvent event + */ + private logWindowError(event: ErrorEvent) { + this.log( + `${event.message} in ${event.filename} on line ${event.lineno}:${event.colno}`, + 'fatal' + ); + } + + /** + * Log unhandled promise rejections. + * + * @param PromiseRejectionEvent event + */ + private logUnhandledRejection(event: PromiseRejectionEvent) { + this.log(`${event.reason.stack}`, 'fatal'); + } + + /** + * Register a listener for the window error events and log the data. + */ + public addWindowErrorListener() { + // Ensure that the window events are connected only once. + if (this.windowConnected) { + return; + } + if (typeof window === 'undefined') { + return; + } + + window.addEventListener('error', this.logWindowError.bind(this)); + window.addEventListener( + 'unhandledrejection', + this.logUnhandledRejection.bind(this) + ); + window.addEventListener( + 'rejectionhandled', + this.logUnhandledRejection.bind(this) + ); + this.windowConnected = true; + } + + /** + * Register a listener for the request.end event and log the data. + * @param UniversalPHP playground instance + */ + public addPlaygroundRequestEndListener(playground: UniversalPHP) { + playground.addEventListener('request.end', async () => { + const log = await this.getRequestPhpErrorLog(playground); + if (log.length > this.lastPHPLogLength) { + this.logRaw(log.substring(this.lastPHPLogLength)); + this.lastPHPLogLength = log.length; + } + }); + } + + /** + * Get UTC date in the PHP log format https://github.com/php/php-src/blob/master/main/main.c#L849 + * + * @param date + * @returns string + */ + private formatLogDate(date: Date): string { + const formattedDate = new Intl.DateTimeFormat('en-GB', { + year: 'numeric', + month: 'short', + day: '2-digit', + timeZone: 'UTC', + }) + .format(date) + .replace(/ /g, '-'); + + const formattedTime = new Intl.DateTimeFormat('en-GB', { + hour: '2-digit', + minute: '2-digit', + second: '2-digit', + hour12: false, + timeZone: 'UTC', + timeZoneName: 'short', + }).format(date); + return formattedDate + ' ' + formattedTime; + } + + /** + * Format log message and severity and log it. + * @param string message + * @param LogSeverity severity + */ + public formatMessage(message: string, severity: LogSeverity): string { + const now = this.formatLogDate(new Date()); + return `[${now}] ${this.LOG_PREFIX} ${severity}: ${message}`; + } + + /** + * Log message with severity and timestamp. + * @param string message + * @param LogSeverity severity + */ + public log(message: string, severity?: LogSeverity): void { + if (severity === undefined) { + severity = 'info'; + } + const log = this.formatMessage(message, severity); + this.logRaw(log); + } + + /** + * Log message without severity and timestamp. + * @param string log + */ + public logRaw(log: string): void { + console.debug(log); + } +} + +/** + * The logger instance. + */ +export const logger: Logger = new Logger(); + +/** + * Collect errors from JavaScript window events like error and log them. + * @param loggerInstance The logger instance + */ +export function collectWindowErrors(loggerInstance: Logger) { + loggerInstance.addWindowErrorListener(); +} + +/** + * Collect PHP logs from the error_log file and log them. + * @param UniversalPHP playground instance + * @param loggerInstance The logger instance + */ +export function collectPhpLogs( + loggerInstance: Logger, + playground: UniversalPHP +) { + loggerInstance.addPlaygroundRequestEndListener(playground); +} diff --git a/packages/php-wasm/logger/tsconfig.json b/packages/php-wasm/logger/tsconfig.json new file mode 100644 index 0000000000..a6f5201524 --- /dev/null +++ b/packages/php-wasm/logger/tsconfig.json @@ -0,0 +1,23 @@ +{ + "extends": "../../../tsconfig.base.json", + "compilerOptions": { + "forceConsistentCasingInFileNames": true, + "strict": true, + "noImplicitOverride": true, + "allowSyntheticDefaultImports": true, + "noPropertyAccessFromIndexSignature": true, + "noImplicitReturns": true, + "noFallthroughCasesInSwitch": true, + "types": ["vitest"] + }, + "files": [], + "include": [], + "references": [ + { + "path": "./tsconfig.lib.json" + }, + { + "path": "./tsconfig.spec.json" + } + ] +} diff --git a/packages/php-wasm/logger/tsconfig.lib.json b/packages/php-wasm/logger/tsconfig.lib.json new file mode 100644 index 0000000000..829b0bc14c --- /dev/null +++ b/packages/php-wasm/logger/tsconfig.lib.json @@ -0,0 +1,10 @@ +{ + "extends": "./tsconfig.json", + "compilerOptions": { + "outDir": "../../../dist/out-tsc", + "declaration": true, + "types": ["node"] + }, + "include": ["src/**/*.ts"], + "exclude": ["jest.config.ts", "src/**/*.spec.ts", "src/**/*.test.ts"] +} diff --git a/packages/php-wasm/logger/tsconfig.spec.json b/packages/php-wasm/logger/tsconfig.spec.json new file mode 100644 index 0000000000..eb23daacbc --- /dev/null +++ b/packages/php-wasm/logger/tsconfig.spec.json @@ -0,0 +1,19 @@ +{ + "extends": "./tsconfig.json", + "compilerOptions": { + "outDir": "../../dist/out-tsc", + "types": ["vitest/globals", "vitest/importMeta", "vite/client", "node"] + }, + "include": [ + "vite.config.ts", + "src/**/*.test.ts", + "src/**/*.spec.ts", + "src/**/*.test.tsx", + "src/**/*.spec.tsx", + "src/**/*.test.js", + "src/**/*.spec.js", + "src/**/*.test.jsx", + "src/**/*.spec.jsx", + "src/**/*.d.ts" + ] +} diff --git a/packages/php-wasm/logger/vite.config.ts b/packages/php-wasm/logger/vite.config.ts new file mode 100644 index 0000000000..09ac745a38 --- /dev/null +++ b/packages/php-wasm/logger/vite.config.ts @@ -0,0 +1,48 @@ +/// +import { defineConfig } from 'vite'; + +import dts from 'vite-plugin-dts'; +import { join } from 'path'; + +// eslint-disable-next-line @nx/enforce-module-boundaries +import { viteTsConfigPaths } from '../../vite-ts-config-paths'; + +export default defineConfig({ + cacheDir: '../../../node_modules/.vite/php-wasm-logger', + + plugins: [ + dts({ + entryRoot: 'src', + tsconfigPath: join(__dirname, 'tsconfig.lib.json'), + }), + + viteTsConfigPaths({ + root: '../../../', + }), + ], + + // Configuration for building your library. + // See: https://vitejs.dev/guide/build.html#library-mode + build: { + lib: { + // Could also be a dictionary or array of multiple entry points. + entry: 'src/index.ts', + name: 'php-wasm-logger', + fileName: 'index', + formats: ['es', 'cjs'], + }, + rollupOptions: { + // External packages that should not be bundled into your library. + external: [], + }, + }, + + test: { + globals: true, + cache: { + dir: '../../../node_modules/.vitest', + }, + environment: 'jsdom', + include: ['src/**/*.{test,spec}.{js,mjs,cjs,ts,mts,cts,jsx,tsx}'], + }, +}); diff --git a/packages/playground/blueprints/src/lib/compile.ts b/packages/playground/blueprints/src/lib/compile.ts index 0e3e614463..f7a7d4a954 100644 --- a/packages/playground/blueprints/src/lib/compile.ts +++ b/packages/playground/blueprints/src/lib/compile.ts @@ -240,10 +240,8 @@ export function compileBlueprint( throw new Error( `Error when executing the blueprint step #${i} (${JSON.stringify( step - )}). Inspect the "cause" property of this error for more details`, - { - cause: e, - } + )}) ${e instanceof Error ? `: ${e.message}` : e}`, + { cause: e } ); } } diff --git a/packages/playground/client/src/index.ts b/packages/playground/client/src/index.ts index deb097b8e3..e5fc712f1f 100644 --- a/packages/playground/client/src/index.ts +++ b/packages/playground/client/src/index.ts @@ -38,6 +38,7 @@ import { import { consumeAPI } from '@php-wasm/web'; import { ProgressTracker } from '@php-wasm/progress'; import { PlaygroundClient } from '@wp-playground/remote'; +import { collectPhpLogs, logger } from '@php-wasm/logger'; export interface StartPlaygroundOptions { iframe: HTMLIFrameElement; remoteUrl: string; @@ -94,6 +95,7 @@ export async function startPlaygroundWeb({ }), progressTracker ); + collectPhpLogs(logger, playground); await runBlueprintSteps(compiled, playground); progressTracker.finish(); diff --git a/packages/playground/node/src/index.ts b/packages/playground/node/src/index.ts index 82b413e90f..9fd9c8d4bd 100644 --- a/packages/playground/node/src/index.ts +++ b/packages/playground/node/src/index.ts @@ -13,6 +13,7 @@ import { } from '@wp-playground/blueprints'; import { NodePHP } from '@php-wasm/node'; import { UniversalPHP } from '@php-wasm/universal'; +import { collectPhpLogs, logger } from '@php-wasm/logger'; export interface NodePlaygroundOptions { blueprint?: Blueprint; @@ -43,6 +44,8 @@ export async function startPlaygroundNode( }, }); + collectPhpLogs(logger, playground); + await defineSiteUrl(playground, { siteUrl: options.serverUrl, }); @@ -64,7 +67,7 @@ export async function allowWpOrgHosts( // Needed because gethostbyname( 'wordpress.org' ) returns // a private network IP address for some reason. add_filter( 'allowed_redirect_hosts', function( $deprecated = '' ) { - return array( + return array( 'wordpress.org', 'api.wordpress.org', 'downloads.wordpress.org', diff --git a/packages/playground/remote/src/lib/playground-mu-plugin/0-playground.php b/packages/playground/remote/src/lib/playground-mu-plugin/0-playground.php index 3a20b5b4e7..1d1e73a93a 100644 --- a/packages/playground/remote/src/lib/playground-mu-plugin/0-playground.php +++ b/packages/playground/remote/src/lib/playground-mu-plugin/0-playground.php @@ -1,7 +1,6 @@ @@ -29,19 +28,19 @@ function ( $message ) { ); /** - * Because the in-browser Playground doesn't have access to the internet, - * network-dependent features like directories don't work. Normally, you'll + * Because the in-browser Playground doesn't have access to the internet, + * network-dependent features like directories don't work. Normally, you'll * see a confusing message like "An unexpected error occurred." This mu-plugin * makes it more clear that the feature is not yet supported. - * + * * https://github.com/WordPress/wordpress-playground/issues/498 - * + * * Added styling to hide the Popular tags section of the Plugins page * and the nonfunctional Try Again button (both Plugins and Themes) that's * appended when the message is displayed. - * + * * https://github.com/WordPress/wordpress-playground/issues/927 - * + * */ add_action('admin_head', function () { echo '