@qodalis/cli-server-logs
Version:
An Angular CLI extension for server logs.
152 lines • 21.3 kB
JavaScript
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"]}