UNPKG

cc-terminal

Version:
191 lines 24.4 kB
import { Injectable } from '@angular/core'; import { Observable, Subject } from 'rxjs'; import { filter, map, takeUntil } from 'rxjs/operators'; import cloneDeep from 'lodash.clonedeep'; import * as i0 from "@angular/core"; export class CcTerminalService { constructor() { this.ngUnsubscribe = new Subject(); // https://stackoverflow.com/questions/38008334/angular-rxjs-when-should-i-unsubscribe-from-subscription/41177163#41177163 this.event = new Subject(); this.readyStore(); } fetch(url) { return Observable.create(observer => { // angular http lib does not support arrayBuffer hence XMLHTTP const req = new XMLHttpRequest(); req.open('get', url, true); req.responseType = 'arraybuffer'; req.onreadystatechange = function () { if (req.readyState === 4 && req.status === 200) { observer.next(req.response); observer.complete(); } }; req.send(); }); } // calls the next event with listener id on listening component and data to send broadcast(key, data) { this.event.next({ key, data }); } /** * @description - This function will register to execute the store whenever store is ready. */ readyStore() { this.on('store-ready').subscribe(_store => { this.store = _store; this.store.state$.subscribe(state => { console.log('StoreReady: In Service:', state); }); }); } // filters through active observers and maps data to a matching observer on(key) { return this.event.asObservable().pipe(filter((event) => event.key === key), map(event => event.data)); } /** * @description - Get the Current prompt */ getPrompt() { return this.prompt; } /** * @description - Get the Current store */ getStore() { console.log('here'); return this.store; } initPrompt(config) { this.prompt = {}; let _user, _path, _userPathSeparator, _promptEnd; config = config ? config.promptConfiguration : null; const build = () => { this.prompt.text = _user + _userPathSeparator + _path + _promptEnd; }; this.prompt.reset = () => { _user = config && config.user != null ? (config.user || '') : 'anon'; _path = config && config.path != null ? (config.path || '') : '\\'; _userPathSeparator = config && config.separator != null ? (config.separator || '') : '@'; _promptEnd = config && config.end != null ? (config.end || '') : ':>'; build(); }; this.prompt.text = ''; this.prompt.reset(); return this.prompt; } /** * @description - This function will help you to interpret your commands. * @param cmd - command */ interpret(cmd) { const prompt = this.getPrompt(); // this.store.state$.subscribe(state => { console.log(state); }); const command = (cmd.command || '').split(' '); let _command = null; this.store.state$.pipe(takeUntil(this.ngUnsubscribe)).subscribe(state => { _command = cloneDeep(state.commands.filter(// Remove the reference of command by making copy, to avoid modifying the command state (item) => { return item.name === command[0]; })[0] || null); }); if (_command && _command.name) { if (_command.callback && typeof _command.callback === 'function') { _command.callback(); } switch (_command.name) { case 'help': _command.details.result = _command.details.result.map((_result) => { if (_result && _result.text && typeof _result.text === 'function') { let text = _result.text(this.store.state.commands.map((c) => { return c.name; })); return { ..._result, text }; } else { return _result; } }); break; default: _command.details.result = _command.details.result.map((_result) => { if (_result && _result.text && typeof _result.text === 'function') { let text = (_result.text()).toString(); return { ..._result, text }; } else { return _result; } }); } _command.details.result.splice(0, 0, { text: prompt.text + cmd.command }); console.log('Final:', _command); this.broadcast('terminal-output', _command); } else { let result = ''; try { result = eval(cmd.command); // eval.call(null, cmd.command); if (result !== undefined) { this.broadcast('terminal-output', { details: { output: true, result: [ { text: prompt.text + cmd.command, }, { text: '' + result }, ], breakLine: true, } }); } } catch (e) { this.broadcast('terminal-output', { details: { output: true, result: [ { text: prompt.text + cmd.command, }, { text: '' + e, css: { color: 'red' } }, ], breakLine: true, } }); } } /** * @description - Regex for exact match command * note: we can add this in constants * TODO: We can design the exact match regex based command also. */ const regex = { alert: /^alert$/, }; // Example of how to work with regex based command // else if (regex.alert.test(command[0])) { // this.broadcast('terminal-output', { // details: { // output: true, // result: [ // { text: prompt.text + cmd.command, }, // { text: '' + e, css: { color: 'red' } }], // breakLine: true, // } // }); // command.splice(0, 1); // Remove command from command string // alert(command.join(' ')); // console.log(command); // } } ngOnDestroy() { // Clear Storage allocation of memory. this.ngUnsubscribe.next(true); this.ngUnsubscribe.complete(); } } /** @nocollapse */ CcTerminalService.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.3.3", ngImport: i0, type: CcTerminalService, deps: [], target: i0.ɵɵFactoryTarget.Injectable }); /** @nocollapse */ CcTerminalService.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "13.3.3", ngImport: i0, type: CcTerminalService, providedIn: 'root' }); i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.3.3", ngImport: i0, type: CcTerminalService, decorators: [{ type: Injectable, args: [{ providedIn: 'root' }] }], ctorParameters: function () { return []; } }); //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"cc-terminal.service.js","sourceRoot":"","sources":["../../../../projects/cc-terminal/src/lib/cc-terminal.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable, OnDestroy } from '@angular/core';\nimport { Observable, Subject } from 'rxjs';\nimport { filter, map, takeUntil } from 'rxjs/operators';\nimport { CommandStore } from './cc-terminal-command-store';\nimport cloneDeep from 'lodash.clonedeep';\n\ninterface BroadcastEvent {\n  key: any;\n  data?: any;\n}\n\n@Injectable({\n  providedIn: 'root'\n})\nexport class CcTerminalService implements OnDestroy {\n  private ngUnsubscribe = new Subject<boolean>(); // https://stackoverflow.com/questions/38008334/angular-rxjs-when-should-i-unsubscribe-from-subscription/41177163#41177163\n  prompt: any;\n  public store: CommandStore;\n  private event: Subject<BroadcastEvent>;\n\n  constructor() {\n    this.event = new Subject<BroadcastEvent>();\n    this.readyStore();\n  }\n\n  fetch(url: string) {\n\n    return Observable.create(observer => {\n      // angular http lib does not support arrayBuffer hence XMLHTTP\n      const req = new XMLHttpRequest();\n      req.open('get', url, true);\n      req.responseType = 'arraybuffer';\n      req.onreadystatechange = function () {\n        if (req.readyState === 4 && req.status === 200) {\n          observer.next(req.response);\n          observer.complete();\n        }\n      };\n      req.send();\n    });\n  }\n\n  // calls the next event with listener id on listening component and data to send\n  broadcast(key: any, data?: any) {\n    this.event.next({ key, data });\n  }\n\n  /**\n   * @description - This function will register to execute the store whenever store is ready.\n   */\n  readyStore() {\n    this.on<any>('store-ready').subscribe(_store => {\n      this.store = _store;\n      this.store.state$.subscribe(state => {\n        console.log('StoreReady: In Service:', state);\n      });\n    });\n  }\n\n  // filters through active observers and maps data to a matching observer\n  on<T>(key: any): Observable<T> {\n    return this.event.asObservable().pipe(filter((event) => event.key === key), map(event => <T>event.data));\n  }\n\n  /**\n   * @description - Get the Current prompt\n   */\n  public getPrompt() {\n    return this.prompt;\n  }\n\n  /**\n   * @description - Get the Current store\n   */\n  public getStore() {\n    console.log('here');\n    return this.store;\n  }\n\n  public initPrompt(config: any) {\n    this.prompt = {};\n    let _user: any, _path: any, _userPathSeparator: any, _promptEnd: any;\n    config = config ? config.promptConfiguration : null;\n    const build = () => {\n      this.prompt.text = _user + _userPathSeparator + _path + _promptEnd;\n    };\n    this.prompt.reset = () => {\n      _user = config && config.user != null ? (config.user || '') : 'anon';\n      _path = config && config.path != null ? (config.path || '') : '\\\\';\n      _userPathSeparator = config && config.separator != null ? (config.separator || '') : '@';\n      _promptEnd = config && config.end != null ? (config.end || '') : ':>';\n      build();\n    };\n    this.prompt.text = '';\n    this.prompt.reset();\n    return this.prompt;\n  }\n\n\n  /**\n   * @description - This function will help you to interpret your commands.\n   * @param cmd - command\n   */\n  public interpret(cmd: any) {\n    const prompt = this.getPrompt();\n    // this.store.state$.subscribe(state => { console.log(state); });\n    const command = (cmd.command || '').split(' ');\n    let _command = null;\n    this.store.state$.pipe(takeUntil(this.ngUnsubscribe)).subscribe(state => {\n      _command = cloneDeep(state.commands.filter( // Remove the reference of command by making copy, to avoid modifying the command state\n        (item) => {\n          return item.name === command[0];\n        })[0] || null);\n    });\n    if (_command && _command.name) {\n      if (_command.callback && typeof _command.callback === 'function') {\n        _command.callback();\n      }\n      switch (_command.name) {\n        case 'help':\n          _command.details.result = _command.details.result.map((_result) => {\n            if (_result && _result.text && typeof _result.text === 'function') {\n              let text = _result.text(this.store.state.commands.map((c) => { return c.name; }));\n              return { ..._result, text };\n            } else {\n              return _result;\n            }\n          });\n          break;\n        default:\n          _command.details.result = _command.details.result.map((_result) => {\n            if (_result && _result.text && typeof _result.text === 'function') {\n              let text = (_result.text()).toString();\n              return { ..._result, text };\n            } else {\n              return _result;\n            }\n          });\n      }\n      _command.details.result.splice(0, 0, { text: prompt.text + cmd.command });\n      console.log('Final:', _command);\n      this.broadcast('terminal-output', _command);\n    } else {\n      let result = '';\n      try {\n        result = eval(cmd.command); // eval.call(null, cmd.command);\n        if (result !== undefined) {\n          this.broadcast('terminal-output', {\n            details: {\n              output: true,\n              result: [\n                { text: prompt.text + cmd.command, },\n                { text: '' + result },\n              ],\n              breakLine: true,\n            }\n          });\n        }\n      } catch (e) {\n        this.broadcast('terminal-output', {\n          details: {\n            output: true,\n            result: [\n              { text: prompt.text + cmd.command, },\n              { text: '' + e, css: { color: 'red' } },\n            ],\n            breakLine: true,\n          }\n        });\n      }\n    }\n\n    /**\n     * @description - Regex for exact match command\n     *   note: we can add this in constants\n     *  TODO: We can design the exact match regex based command also.\n     */\n    const regex = { // We can design the exact match regex based command also.\n      alert: /^alert$/,\n    };\n\n    // Example of how to work with regex based command\n    // else if (regex.alert.test(command[0])) {\n    // this.broadcast('terminal-output', {\n    //   details: {\n    //   output: true,\n    //   result: [\n    //      { text: prompt.text + cmd.command, },\n    //      { text: '' + e, css: { color: 'red' } }],\n    //     breakLine: true,\n    //   }\n    // });\n    //   command.splice(0, 1); // Remove command from command string\n    //   alert(command.join(' '));\n    //   console.log(command);\n    // }\n  }\n\n  ngOnDestroy() {\n    // Clear Storage allocation of memory.\n    this.ngUnsubscribe.next(true);\n    this.ngUnsubscribe.complete();\n  }\n\n}\n"]}