UNPKG

@qodalis/cli-server-logs

Version:

An Angular CLI extension for server logs.

152 lines 21.3 kB
import { Injectable } from '@angular/core'; import * as signalR from '@microsoft/signalr'; import { DefaultLibraryAuthor, highlightTextWithBg, toQueryString, } from '@qodalis/cli-core'; import * as i0 from "@angular/core"; const levels = ['verbose', 'debug', 'information', 'warning', 'error', 'fatal']; export class CliLogsCommandProcessor { constructor() { this.command = 'server-logs'; this.description = 'Show live logs'; this.processors = []; this.metadata = { requireServer: true, }; this.parameters = [ { name: 'pattern', type: 'string', description: 'The regex pattern to search for in the logs', required: false, validator: (value) => { const isValid = this.isValidRegex(value); return { valid: isValid, message: isValid ? undefined : 'Invalid regex pattern provided', }; }, }, { name: 'level', type: 'string', description: 'The log level to filter by, e.g. ' + levels.join(', '), required: false, validator: (value) => { const isValid = levels.includes(value); return { valid: isValid, message: isValid ? undefined : 'Invalid log level provided, must be one of: ' + levels.join(', '), }; }, }, { name: 'server', type: 'string', description: 'The server to connect to, e.g. "http://localhost:5000"', required: false, }, { name: 'hub', type: 'string', description: 'The hub to connect to, e.g. "loghub" (default) or "loghub2"', required: false, }, { name: 'file', type: 'boolean', description: 'Export logs to a file', required: false, }, ]; this.author = DefaultLibraryAuthor; this.processors?.push({ command: 'live', description: this.description, parameters: this.parameters, processCommand: this.processCommand.bind(this), writeDescription: this.writeDescription.bind(this), }); } async processCommand(command, context) { let qs = '?'; const args = this.excludeKeys(command.args, ['server', 'hub', 'file']); if (Object.keys(args).length > 0) { qs += toQueryString(args); } const hub = command.args['hub'] || 'loghub'; let server = command.args['server'] || ''; server = server.replace(/\/+$/, ''); const url = `${server}/${hub}${qs}`; console.log('Connecting to:', url); this.hubConnection = new signalR.HubConnectionBuilder() .withUrl(url) .build(); const buffer = []; await this.hubConnection .start() .then(() => { console.log('SignalR connection started'); context.writer.writeWarning('Connected to live logs'); if (qs.length) { Object.keys(args).forEach((key) => { context.writer.writeWarning(`Filtering logs by: ${key}=${command.args[key]}`); }); } let firstLog = true; let index = 0; this.hubConnection.on('log', (log) => { if (firstLog) { context.writer.writeln(''); } buffer.push(log); context.writer.writeln(`\x1b[33m${++index}\x1b[0m. ` + (command.args['pattern'] ? highlightTextWithBg(log, new RegExp(command.args['pattern'], 'g')) : log)); firstLog = false; }); const subscription = context.onAbort.subscribe(() => { this.hubConnection.stop(); context.writer.writeWarning('Disconnected from live logs'); if (command.args['file']) { const filename = `logs-${new Date().toISOString()}.txt`; context.writer.writeToFile(filename, buffer.join('\n')); } subscription.unsubscribe(); }); }) .catch((err) => { console.error('Error starting SignalR connection:', err); context.writer.writeError('Failed to connect to live logs'); context.writer.writeError(err?.toString()); }); } writeDescription(context) { context.writer.writeln('Show live logs'); } excludeKeys(record, keysToExclude) { return Object.fromEntries(Object.entries(record).filter(([key]) => !keysToExclude.includes(key))); } isValidRegex(pattern) { try { new RegExp(pattern); return true; } catch (e) { return false; } } static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: CliLogsCommandProcessor, deps: [], target: i0.ɵɵFactoryTarget.Injectable }); } static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: CliLogsCommandProcessor, providedIn: 'root' }); } } i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: CliLogsCommandProcessor, decorators: [{ type: Injectable, args: [{ providedIn: 'root', }] }], ctorParameters: function () { return []; } }); //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"cli-logs-command-processor.js","sourceRoot":"","sources":["../../../../../projects/server-logs/src/lib/processors/cli-logs-command-processor.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable } from '@angular/core';\nimport * as signalR from '@microsoft/signalr';\nimport {\n    CliProcessCommand,\n    CliProcessorMetadata,\n    DefaultLibraryAuthor,\n    highlightTextWithBg,\n    ICliCommandAuthor,\n    ICliCommandParameterDescriptor,\n    ICliCommandProcessor,\n    ICliExecutionContext,\n    toQueryString,\n} from '@qodalis/cli-core';\n\nconst levels = ['verbose', 'debug', 'information', 'warning', 'error', 'fatal'];\n\n@Injectable({\n    providedIn: 'root',\n})\nexport class CliLogsCommandProcessor implements ICliCommandProcessor {\n    command = 'server-logs';\n\n    description?: string | undefined = 'Show live logs';\n\n    processors?: ICliCommandProcessor[] | undefined = [];\n\n    metadata?: CliProcessorMetadata | undefined = {\n        requireServer: true,\n    };\n\n    parameters?: ICliCommandParameterDescriptor[] | undefined = [\n        {\n            name: 'pattern',\n            type: 'string',\n            description: 'The regex pattern to search for in the logs',\n            required: false,\n            validator: (value: string) => {\n                const isValid = this.isValidRegex(value);\n                return {\n                    valid: isValid,\n                    message: isValid\n                        ? undefined\n                        : 'Invalid regex pattern provided',\n                };\n            },\n        },\n        {\n            name: 'level',\n            type: 'string',\n            description:\n                'The log level to filter by, e.g. ' + levels.join(', '),\n            required: false,\n            validator: (value: string) => {\n                const isValid = levels.includes(value);\n\n                return {\n                    valid: isValid,\n                    message: isValid\n                        ? undefined\n                        : 'Invalid log level provided, must be one of: ' +\n                          levels.join(', '),\n                };\n            },\n        },\n        {\n            name: 'server',\n            type: 'string',\n            description:\n                'The server to connect to, e.g. \"http://localhost:5000\"',\n            required: false,\n        },\n        {\n            name: 'hub',\n            type: 'string',\n            description:\n                'The hub to connect to, e.g. \"loghub\" (default) or \"loghub2\"',\n            required: false,\n        },\n        {\n            name: 'file',\n            type: 'boolean',\n            description: 'Export logs to a file',\n            required: false,\n        },\n    ];\n\n    author?: ICliCommandAuthor | undefined = DefaultLibraryAuthor;\n\n    private hubConnection!: signalR.HubConnection;\n\n    constructor() {\n        this.processors?.push({\n            command: 'live',\n            description: this.description,\n            parameters: this.parameters,\n            processCommand: this.processCommand.bind(this),\n            writeDescription: this.writeDescription.bind(this),\n        });\n    }\n\n    async processCommand(\n        command: CliProcessCommand,\n        context: ICliExecutionContext,\n    ): Promise<void> {\n        let qs = '?';\n\n        const args = this.excludeKeys(command.args, ['server', 'hub', 'file']);\n\n        if (Object.keys(args).length > 0) {\n            qs += toQueryString(args);\n        }\n\n        const hub = command.args['hub'] || 'loghub';\n\n        let server = command.args['server'] || '';\n        server = server.replace(/\\/+$/, '');\n\n        const url = `${server}/${hub}${qs}`;\n\n        console.log('Connecting to:', url);\n\n        this.hubConnection = new signalR.HubConnectionBuilder()\n            .withUrl(url)\n            .build();\n\n        const buffer: string[] = [];\n\n        await this.hubConnection\n            .start()\n            .then(() => {\n                console.log('SignalR connection started');\n\n                context.writer.writeWarning('Connected to live logs');\n                if (qs.length) {\n                    Object.keys(args).forEach((key) => {\n                        context.writer.writeWarning(\n                            `Filtering logs by: ${key}=${command.args[key]}`,\n                        );\n                    });\n                }\n\n                let firstLog = true;\n                let index = 0;\n\n                this.hubConnection.on('log', (log: string) => {\n                    if (firstLog) {\n                        context.writer.writeln('');\n                    }\n\n                    buffer.push(log);\n\n                    context.writer.writeln(\n                        `\\x1b[33m${++index}\\x1b[0m. ` +\n                            (command.args['pattern']\n                                ? highlightTextWithBg(\n                                      log,\n                                      new RegExp(command.args['pattern'], 'g'),\n                                  )\n                                : log),\n                    );\n                    firstLog = false;\n                });\n\n                const subscription = context.onAbort.subscribe(() => {\n                    this.hubConnection.stop();\n                    context.writer.writeWarning('Disconnected from live logs');\n\n                    if (command.args['file']) {\n                        const filename = `logs-${new Date().toISOString()}.txt`;\n                        context.writer.writeToFile(filename, buffer.join('\\n'));\n                    }\n\n                    subscription.unsubscribe();\n                });\n            })\n            .catch((err) => {\n                console.error('Error starting SignalR connection:', err);\n                context.writer.writeError('Failed to connect to live logs');\n                context.writer.writeError(err?.toString());\n            });\n    }\n\n    writeDescription(context: ICliExecutionContext): void {\n        context.writer.writeln('Show live logs');\n    }\n\n    private excludeKeys<T extends Record<string, any>>(\n        record: T,\n        keysToExclude: string[],\n    ): Partial<T> {\n        return Object.fromEntries(\n            Object.entries(record).filter(\n                ([key]) => !keysToExclude.includes(key),\n            ),\n        ) as Partial<T>;\n    }\n\n    private isValidRegex(pattern: string): boolean {\n        try {\n            new RegExp(pattern);\n            return true;\n        } catch (e) {\n            return false;\n        }\n    }\n}\n"]}