UNPKG

@qodalis/cli-curl

Version:

Provide utility functions for the library curl

215 lines 30.1 kB
import { Injectable } from '@angular/core'; import { DefaultLibraryAuthor, } from '@qodalis/cli-core'; import { LIBRARY_VERSION } from '../version'; import * as i0 from "@angular/core"; export class CliCurlCommandProcessor { constructor() { this.command = 'curl'; this.description = 'A command-line tool to execute HTTP requests on your server. Supports GET, POST, PUT, DELETE, headers, and body data.'; this.author = DefaultLibraryAuthor; this.version = LIBRARY_VERSION; this.processors = []; this.metadata = { icon: '🌐', }; this.registerSubProcessors(); } async processCommand(command, context) { context.executor.showHelp(command, context); } async initialize(context) { } registerSubProcessors() { this.processors = [ { command: 'get', description: 'Perform an HTTP GET request', valueRequired: true, parameters: [ { name: 'header', aliases: ['H'], type: 'array', description: 'Add custom headers. Accept multiple headers.', required: false, }, { name: 'proxy', type: 'boolean', description: 'Use procy', required: false, }, ], processCommand: async (command, context) => { await this.executeRequest('GET', command, context); }, }, { command: 'post', description: 'Perform an HTTP POST request', valueRequired: true, parameters: [ { name: 'header', aliases: ['H'], type: 'array', description: 'Add custom headers', required: false, }, { name: 'data', aliases: ['d'], type: 'string', description: 'Request body', required: false, }, { name: 'proxy', type: 'boolean', description: 'Use procy', required: false, }, ], processCommand: async (command, context) => { await this.executeRequest('POST', command, context); }, }, { command: 'put', description: 'Perform an HTTP PUT request', valueRequired: true, parameters: [ { name: 'header', aliases: ['H'], type: 'array', description: 'Add custom headers', required: false, }, { name: 'data', aliases: ['d'], type: 'string', description: 'Request body', required: false, }, { name: 'proxy', type: 'boolean', description: 'Use procy', required: false, }, ], processCommand: async (command, context) => { await this.executeRequest('PUT', command, context); }, }, { command: 'delete', description: 'Perform an HTTP DELETE request', valueRequired: true, parameters: [ { name: 'header', aliases: ['H'], type: 'array', description: 'Add custom headers', required: false, }, { name: 'proxy', type: 'boolean', description: 'Use procy', required: false, }, ], processCommand: async (command, context) => { await this.executeRequest('DELETE', command, context); }, }, ]; } async executeRequest(method, command, context) { const url = command.value; const headers = command.args['header'] || command.args['H'] || []; const data = command.args['data'] || command.args['d']; const verbose = !!command.args['verbose']; const useProxy = !!command.args['proxy']; if (!url) { context.writer.writeError('URL is required.'); return; } // Prepare headers const headersObject = headers.reduce((acc, header) => { const [key, value] = header.split(':').map((str) => str.trim()); if (key && value) acc[key] = value; return acc; }, {}); // Prepare request options const options = { method, headers: headersObject, body: data ? JSON.stringify(JSON.parse(data)) : undefined, }; try { const requestUrl = useProxy ? this.rewriteUrlToProxy(url) : url; const response = await fetch(requestUrl, options); const text = await response.text(); context.writer.writeSuccess('Request successful:'); if (verbose) { context.writer.writeln(`Status: ${response.status}`); context.writer.writeln(`Headers: ${JSON.stringify(response.headers, null, 2)}`); } context.writer.writeln(text); context.process.output(text); } catch (error) { context.writer.writeError(`Request failed: ${error}`); context.process.exit(-1); } finally { context.writer.writeln(); context.writer.writeInfo('Equivalent curl command:'); context.writer.writeln(this.generateCurlCommand(url, method, headers, data)); } } generateCurlCommand(url, method, headers, data) { const headerString = headers.map((h) => `-H "${h}"`).join(' '); const dataString = data ? `-d '${data}'` : ''; return `curl -X ${method} ${headerString} ${dataString} "${url}"`; } rewriteUrlToProxy(originalUrl) { const regex = /^(https?):\/\/([^\/]+)(\/.*)?$/i; const match = originalUrl.match(regex); if (!match) { throw new Error('Invalid URL provided'); } const scheme = match[1]; // 'http' or 'https' const domain = match[2]; // domain.com const path = match[3] || '/'; // /path or '/' return `https://proxy.qodalis.com/proxy/${scheme}/${domain}${path}`; } writeDescription(context) { context.writer.writeln(this.description); context.writer.writeln(); context.writer.writeln('Usage:'); context.writer.writeln(' curl <method> <url> [options]'); context.writer.writeln(); context.writer.writeln('Options:'); context.writer.writeln(' -H, --header Add custom headers (e.g., -H="Authorization: Bearer <token>")'); context.writer.writeln(' -d, --data Add request body'); context.writer.writeln(' --verbose Print detailed response'); context.writer.writeln(); context.writer.writeln('Examples:'); context.writer.writeln(' curl get https://api.example.com/users'); context.writer.writeln(' curl post https://api.example.com/users -d=\'{"name":"John"}\' -H="Content-Type: application/json"'); context.writer.writeln(); context.writer.writeWarning('Note: The server must allow CORS for this tool to work.'); } static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: CliCurlCommandProcessor, deps: [], target: i0.ɵɵFactoryTarget.Injectable }); } static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: CliCurlCommandProcessor }); } } i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: CliCurlCommandProcessor, decorators: [{ type: Injectable }], ctorParameters: function () { return []; } }); //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"cli-curl-command-processor.js","sourceRoot":"","sources":["../../../../../projects/curl/src/lib/processors/cli-curl-command-processor.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable } from '@angular/core';\nimport {\n    CliProcessCommand,\n    CliProcessorMetadata,\n    DefaultLibraryAuthor,\n    ICliCommandProcessor,\n    ICliExecutionContext,\n} from '@qodalis/cli-core';\nimport { LIBRARY_VERSION } from '../version';\n\n@Injectable()\nexport class CliCurlCommandProcessor implements ICliCommandProcessor {\n    command = 'curl';\n\n    description =\n        'A command-line tool to execute HTTP requests on your server. Supports GET, POST, PUT, DELETE, headers, and body data.';\n\n    author = DefaultLibraryAuthor;\n\n    version = LIBRARY_VERSION;\n\n    processors?: ICliCommandProcessor[] | undefined = [];\n\n    metadata?: CliProcessorMetadata | undefined = {\n        icon: '🌐',\n    };\n\n    constructor() {\n        this.registerSubProcessors();\n    }\n\n    async processCommand(\n        command: CliProcessCommand,\n        context: ICliExecutionContext,\n    ): Promise<void> {\n        context.executor.showHelp(command, context);\n    }\n\n    async initialize(context: ICliExecutionContext): Promise<void> {}\n\n    private registerSubProcessors(): void {\n        this.processors = [\n            {\n                command: 'get',\n                description: 'Perform an HTTP GET request',\n                valueRequired: true,\n                parameters: [\n                    {\n                        name: 'header',\n                        aliases: ['H'],\n                        type: 'array',\n                        description:\n                            'Add custom headers. Accept multiple headers.',\n                        required: false,\n                    },\n                    {\n                        name: 'proxy',\n                        type: 'boolean',\n                        description: 'Use procy',\n                        required: false,\n                    },\n                ],\n                processCommand: async (command, context) => {\n                    await this.executeRequest('GET', command, context);\n                },\n            },\n            {\n                command: 'post',\n                description: 'Perform an HTTP POST request',\n                valueRequired: true,\n                parameters: [\n                    {\n                        name: 'header',\n                        aliases: ['H'],\n                        type: 'array',\n                        description: 'Add custom headers',\n                        required: false,\n                    },\n                    {\n                        name: 'data',\n                        aliases: ['d'],\n                        type: 'string',\n                        description: 'Request body',\n                        required: false,\n                    },\n                    {\n                        name: 'proxy',\n                        type: 'boolean',\n                        description: 'Use procy',\n                        required: false,\n                    },\n                ],\n                processCommand: async (command, context) => {\n                    await this.executeRequest('POST', command, context);\n                },\n            },\n            {\n                command: 'put',\n                description: 'Perform an HTTP PUT request',\n                valueRequired: true,\n                parameters: [\n                    {\n                        name: 'header',\n                        aliases: ['H'],\n                        type: 'array',\n                        description: 'Add custom headers',\n                        required: false,\n                    },\n                    {\n                        name: 'data',\n                        aliases: ['d'],\n                        type: 'string',\n                        description: 'Request body',\n                        required: false,\n                    },\n                    {\n                        name: 'proxy',\n                        type: 'boolean',\n                        description: 'Use procy',\n                        required: false,\n                    },\n                ],\n                processCommand: async (command, context) => {\n                    await this.executeRequest('PUT', command, context);\n                },\n            },\n            {\n                command: 'delete',\n                description: 'Perform an HTTP DELETE request',\n                valueRequired: true,\n                parameters: [\n                    {\n                        name: 'header',\n                        aliases: ['H'],\n                        type: 'array',\n                        description: 'Add custom headers',\n                        required: false,\n                    },\n                    {\n                        name: 'proxy',\n                        type: 'boolean',\n                        description: 'Use procy',\n                        required: false,\n                    },\n                ],\n                processCommand: async (command, context) => {\n                    await this.executeRequest('DELETE', command, context);\n                },\n            },\n        ];\n    }\n\n    private async executeRequest(\n        method: string,\n        command: CliProcessCommand,\n        context: ICliExecutionContext,\n    ): Promise<void> {\n        const url = command.value;\n        const headers = command.args['header'] || command.args['H'] || [];\n        const data = command.args['data'] || command.args['d'];\n        const verbose = !!command.args['verbose'];\n        const useProxy = !!command.args['proxy'];\n\n        if (!url) {\n            context.writer.writeError('URL is required.');\n            return;\n        }\n\n        // Prepare headers\n        const headersObject = headers.reduce(\n            (acc: Record<string, string>, header: string) => {\n                const [key, value] = header.split(':').map((str) => str.trim());\n                if (key && value) acc[key] = value;\n                return acc;\n            },\n            {},\n        );\n\n        // Prepare request options\n        const options: RequestInit = {\n            method,\n            headers: headersObject,\n            body: data ? JSON.stringify(JSON.parse(data)) : undefined,\n        };\n\n        try {\n            const requestUrl = useProxy ? this.rewriteUrlToProxy(url) : url;\n\n            const response = await fetch(requestUrl, options);\n            const text = await response.text();\n\n            context.writer.writeSuccess('Request successful:');\n            if (verbose) {\n                context.writer.writeln(`Status: ${response.status}`);\n                context.writer.writeln(\n                    `Headers: ${JSON.stringify(response.headers, null, 2)}`,\n                );\n            }\n            context.writer.writeln(text);\n\n            context.process.output(text);\n        } catch (error) {\n            context.writer.writeError(`Request failed: ${error}`);\n            context.process.exit(-1);\n        } finally {\n            context.writer.writeln();\n            context.writer.writeInfo('Equivalent curl command:');\n            context.writer.writeln(\n                this.generateCurlCommand(url, method, headers, data),\n            );\n        }\n    }\n\n    private generateCurlCommand(\n        url: string,\n        method: string,\n        headers: string[],\n        data?: string,\n    ): string {\n        const headerString = headers.map((h) => `-H \"${h}\"`).join(' ');\n        const dataString = data ? `-d '${data}'` : '';\n        return `curl -X ${method} ${headerString} ${dataString} \"${url}\"`;\n    }\n\n    private rewriteUrlToProxy(originalUrl: string): string {\n        const regex = /^(https?):\\/\\/([^\\/]+)(\\/.*)?$/i;\n\n        const match = originalUrl.match(regex);\n\n        if (!match) {\n            throw new Error('Invalid URL provided');\n        }\n\n        const scheme = match[1]; // 'http' or 'https'\n        const domain = match[2]; // domain.com\n        const path = match[3] || '/'; // /path or '/'\n\n        return `https://proxy.qodalis.com/proxy/${scheme}/${domain}${path}`;\n    }\n\n    writeDescription(context: ICliExecutionContext): void {\n        context.writer.writeln(this.description!);\n        context.writer.writeln();\n        context.writer.writeln('Usage:');\n        context.writer.writeln('  curl <method> <url> [options]');\n        context.writer.writeln();\n        context.writer.writeln('Options:');\n        context.writer.writeln(\n            '  -H, --header     Add custom headers (e.g., -H=\"Authorization: Bearer <token>\")',\n        );\n        context.writer.writeln('  -d, --data       Add request body');\n        context.writer.writeln('  --verbose    Print detailed response');\n        context.writer.writeln();\n        context.writer.writeln('Examples:');\n        context.writer.writeln('  curl get https://api.example.com/users');\n        context.writer.writeln(\n            '  curl post https://api.example.com/users -d=\\'{\"name\":\"John\"}\\' -H=\"Content-Type: application/json\"',\n        );\n        context.writer.writeln();\n\n        context.writer.writeWarning(\n            'Note: The server must allow CORS for this tool to work.',\n        );\n    }\n}\n"]}