UNPKG

@chatie/angular

Version:

Wechaty Component NgModule

356 lines 44.6 kB
import { __awaiter } from "tslib"; import { BehaviorSubject, Observable, Subject, } from 'rxjs'; import { filter, share, } from 'rxjs/operators'; import { Brolog } from 'brolog'; import { StateSwitch } from 'state-switch'; export var ReadyState; (function (ReadyState) { ReadyState[ReadyState["CLOSED"] = WebSocket.CLOSED] = "CLOSED"; ReadyState[ReadyState["CLOSING"] = WebSocket.CLOSING] = "CLOSING"; ReadyState[ReadyState["CONNECTING"] = WebSocket.CONNECTING] = "CONNECTING"; ReadyState[ReadyState["OPEN"] = WebSocket.OPEN] = "OPEN"; })(ReadyState || (ReadyState = {})); export class IoService { constructor() { this.autoReconnect = true; this.log = Brolog.instance(); this.CONNECT_TIMEOUT = 10 * 1000; // 10 seconds this.ENDPOINT = 'wss://api.chatie.io/v0/websocket/token/'; this.PROTOCOL = 'web|0.0.1'; this.sendBuffer = []; this.log.verbose('IoService', 'constructor()'); } get readyState() { return this._readyState.asObservable(); } init() { return __awaiter(this, void 0, void 0, function* () { this.log.verbose('IoService', 'init()'); if (this.state) { throw new Error('re-init'); } this.snapshot = { readyState: ReadyState.CLOSED, event: null, }; this._readyState = new BehaviorSubject(ReadyState.CLOSED); this.state = new StateSwitch('IoService', this.log); this.state.setLog(this.log); try { yield this.initStateDealer(); yield this.initRxSocket(); } catch (e) { this.log.silly('IoService', 'init() exception: %s', e.message); throw e; } this.readyState.subscribe(s => { this.log.silly('IoService', 'init() readyState.subscribe(%s)', ReadyState[s]); this.snapshot.readyState = s; }); // IMPORTANT: subscribe to event and make it HOT! this.event.subscribe(s => { this.log.silly('IoService', 'init() event.subscribe({name:%s})', s.name); this.snapshot.event = s; }); return; }); } token(newToken) { this.log.silly('IoService', 'token(%s)', newToken); if (newToken) { this._token = newToken; return; } return this._token; } start() { return __awaiter(this, void 0, void 0, function* () { this.log.verbose('IoService', 'start() with token:%s', this._token); if (!this._token) { throw new Error('start() without token'); } if (this.state.on()) { throw new Error('state is already ON'); } if (this.state.pending()) { throw new Error('state is pending'); } this.state.on('pending'); this.autoReconnect = true; try { yield this.connectRxSocket(); this.state.on(true); } catch (e) { this.log.warn('IoService', 'start() failed:%s', e.message); this.state.off(true); } }); } stop() { return __awaiter(this, void 0, void 0, function* () { this.log.verbose('IoService', 'stop()'); if (this.state.off()) { this.log.warn('IoService', 'stop() state is already off'); if (this.state.pending()) { throw new Error('state pending() is true'); } return; } this.state.off('pending'); this.autoReconnect = false; if (!this._websocket) { throw new Error('no websocket'); } yield this.socketClose(1000, 'IoService.stop()'); this.state.off(true); return; }); } restart() { return __awaiter(this, void 0, void 0, function* () { this.log.verbose('IoService', 'restart()'); try { yield this.stop(); yield this.start(); } catch (e) { this.log.error('IoService', 'restart() error:%s', e.message); throw e; } return; }); } initStateDealer() { this.log.verbose('IoService', 'initStateDealer()'); const isReadyStateOpen = (s) => s === ReadyState.OPEN; this.readyState.pipe(filter(isReadyStateOpen)) .subscribe(open => this.stateOnOpen()); } /** * Creates a subject from the specified observer and observable. * - https://github.com/Reactive-Extensions/RxJS/blob/master/doc/api/subjects/subject.md * Create an Rx.Subject using Subject.create that allows onNext without subscription * A socket implementation (example, don't use) * - http://stackoverflow.com/a/34862286/1123955 */ initRxSocket() { return __awaiter(this, void 0, void 0, function* () { this.log.verbose('IoService', 'initRxSocket()'); if (this.event) { throw new Error('re-init is not permitted'); } // 1. Mobile Originated. moObserver.next() means mobile is sending this.moObserver = { next: this.socketSend.bind(this), error: this.socketClose.bind(this), complete: this.socketClose.bind(this), }; // 2. Mobile Terminated. mtObserver.next() means mobile is receiving const observable = new Observable((observer) => { this.log.verbose('IoService', 'initRxSocket() Observable.create()'); this.mtObserver = observer; return this.socketClose.bind(this); }); // 3. Subject for MO & MT Observers this.event = Subject.create(this.moObserver, observable.pipe(share())); }); } connectRxSocket() { return __awaiter(this, void 0, void 0, function* () { this.log.verbose('IoService', 'connectRxSocket()'); // FIXME: check & close the old one if (this._websocket) { throw new Error('already has a websocket'); } // if (this.state.target() !== 'open' // || this.state.current() !== 'open' // || this.state.stable() if (this.state.off()) { throw new Error('switch state is off'); } else if (!this.state.pending()) { throw new Error('switch state is already ON'); } this._websocket = new WebSocket(this.endPoint(), this.PROTOCOL); this.socketUpdateState(); const onOpenPromise = new Promise((resolve, reject) => { this.log.verbose('IoService', 'connectRxSocket() Promise()'); const id = setTimeout(() => { this._websocket = null; const e = new Error('rxSocket connect timeout after ' + Math.round(this.CONNECT_TIMEOUT / 1000)); reject(e); }, this.CONNECT_TIMEOUT); // timeout for connect websocket this._websocket.onopen = (e) => { this.log.verbose('IoService', 'connectRxSocket() Promise() WebSocket.onOpen() resolve()'); this.socketUpdateState(); clearTimeout(id); resolve(); }; }); // Handle the payload this._websocket.onmessage = this.socketOnMessage.bind(this); // Deal the event this._websocket.onerror = this.socketOnError.bind(this); this._websocket.onclose = this.socketOnClose.bind(this); return onOpenPromise; }); } endPoint() { const url = this.ENDPOINT + this._token; this.log.verbose('IoService', 'endPoint() => %s', url); return url; } /****************************************************************** * * State Event Listeners * */ stateOnOpen() { this.log.verbose('IoService', 'stateOnOpen()'); this.socketSendBuffer(); this.rpcUpdate('from stateOnOpen()'); } /****************************************************************** * * Io RPC Methods * */ rpcDing(payload) { return __awaiter(this, void 0, void 0, function* () { this.log.verbose('IoService', 'ding(%s)', payload); const e = { name: 'ding', payload, }; this.event.next(e); // TODO: get the return value }); } rpcUpdate(payload) { return __awaiter(this, void 0, void 0, function* () { this.event.next({ name: 'update', payload, }); }); } /****************************************************************** * * Socket Actions * */ socketClose(code, reason) { return __awaiter(this, void 0, void 0, function* () { this.log.verbose('IoService', 'socketClose()'); if (!this._websocket) { throw new Error('no websocket'); } this._websocket.close(code, reason); this.socketUpdateState(); const future = new Promise(resolve => { this.readyState.pipe(filter(s => s === ReadyState.CLOSED)) .subscribe(resolve); }); yield future; return; }); } socketSend(ioEvent) { this.log.silly('IoService', 'socketSend({name:%s, payload:%s})', ioEvent.name, ioEvent.payload); if (!this._websocket) { this.log.silly('IoService', 'socketSend() no _websocket'); } const strEvt = JSON.stringify(ioEvent); this.sendBuffer.push(strEvt); // XXX can move this to onOpen? this.socketSendBuffer(); } socketSendBuffer() { this.log.silly('IoService', 'socketSendBuffer() length:%s', this.sendBuffer.length); if (!this._websocket) { throw new Error('socketSendBuffer(): no _websocket'); } if (this._websocket.readyState !== WebSocket.OPEN) { this.log.warn('IoService', 'socketSendBuffer() readyState is not OPEN, send job delayed.'); return; } while (this.sendBuffer.length) { const buf = this.sendBuffer.shift(); this.log.silly('IoService', 'socketSendBuffer() sending(%s)', buf); this._websocket.send(buf); } } socketUpdateState() { var _a; this.log.verbose('IoService', 'socketUpdateState() is %s', ReadyState[(_a = this._websocket) === null || _a === void 0 ? void 0 : _a.readyState]); if (!this._websocket) { this.log.error('IoService', 'socketUpdateState() no _websocket'); return; } this._readyState.next(this._websocket.readyState); } /****************************************************************** * * Socket Events Listener * */ socketOnMessage(message) { this.log.verbose('IoService', 'onMessage({data: %s})', message.data); const data = message.data; // WebSocket data const ioEvent = { name: 'raw', payload: data, }; // this is default io event for unknown format message try { const obj = JSON.parse(data); ioEvent.name = obj.name; ioEvent.payload = obj.payload; } catch (e) { this.log.warn('IoService', 'onMessage parse message fail. save as RAW'); } this.mtObserver.next(ioEvent); } socketOnError(event) { this.log.silly('IoService', 'socketOnError(%s)', event); // this._websocket = null } /** * https://developer.mozilla.org/en-US/docs/Web/API/CloseEvent * code: 1006 CLOSE_ABNORMAL * - Reserved. Used to indicate that a connection was closed abnormally * (that is, with no close frame being sent) when a status code is expected. */ socketOnClose(closeEvent) { this.log.verbose('IoService', 'socketOnClose({code:%s, reason:%s, returnValue:%s})', closeEvent.code, closeEvent.reason, closeEvent.returnValue); this.socketUpdateState(); /** * reconnect inside onClose */ if (this.autoReconnect) { this.state.on('pending'); setTimeout(() => __awaiter(this, void 0, void 0, function* () { try { yield this.connectRxSocket(); this.state.on(true); } catch (e) { this.log.warn('IoService', 'socketOnClose() autoReconnect() exception: %s', e); this.state.off(true); } }), 1000); } else { this.state.off(true); } this._websocket = null; if (!closeEvent.wasClean) { this.log.warn('IoService', 'socketOnClose() event.wasClean FALSE'); // TODO emit error } } } //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"io.js","sourceRoot":"","sources":["../../../../src/wechaty/io.ts"],"names":[],"mappings":"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","sourcesContent":["import { VERSION }      from '../config'\n\nimport {\n  BehaviorSubject,\n  Observable,\n  Observer,\n  Subject,\n}                   from 'rxjs'\nimport {\n  filter,\n  share,\n}                   from 'rxjs/operators'\n\nimport { Brolog }       from 'brolog'\nimport { StateSwitch }  from 'state-switch'\n\nexport type WechatyEventName =\n    'scan'\n  | 'login' | 'logout'\n  | 'reset' | 'shutdown'\n  | 'ding'  | 'dong'\n  | 'message'\n  | 'heartbeat'\n  | 'update'\n  | 'error'\n\nexport type ServerEventName = 'sys'\n                            | 'botie'\n\nexport type IoEventName = 'raw' | WechatyEventName | ServerEventName\n\nexport interface IoEvent {\n  name: IoEventName,\n  payload: any,\n}\n\nexport enum ReadyState {\n  CLOSED      = WebSocket.CLOSED,\n  CLOSING     = WebSocket.CLOSING,\n  CONNECTING  = WebSocket.CONNECTING,\n  OPEN        = WebSocket.OPEN,\n}\n\nexport interface IoServiceSnapshot {\n  readyState: ReadyState\n  event:     IoEvent\n}\n\nexport class IoService {\n  // https://github.com/ReactiveX/rxjs/blob/master/src/observable/dom/WebSocketSubject.ts\n  public event: Subject<IoEvent>\n\n  private _readyState: BehaviorSubject<ReadyState>\n  public get readyState() {\n    return this._readyState.asObservable()\n  }\n\n  public snapshot: IoServiceSnapshot\n\n  private autoReconnect = true\n  private log = Brolog.instance()\n\n  private readonly CONNECT_TIMEOUT = 10 * 1000 // 10 seconds\n  private readonly ENDPOINT = 'wss://api.chatie.io/v0/websocket/token/'\n  private readonly PROTOCOL = 'web|0.0.1'\n\n  private _token: string // FIXME possible be `undefined`\n  private _websocket: WebSocket | null\n  private moObserver: Observer<IoEvent> // Mobile Originated. moObserver.next() means mobile is sending\n  private mtObserver: Observer<IoEvent> // Mobile Terminated. mtObserver.next() means mobile is receiving\n  private sendBuffer: string[] = []\n\n  private state: StateSwitch\n\n  constructor() {\n    this.log.verbose('IoService', 'constructor()')\n  }\n\n  public async init(): Promise<void> {\n    this.log.verbose('IoService', 'init()')\n\n    if (this.state) {\n      throw new Error('re-init')\n    }\n\n    this.snapshot = {\n      readyState: ReadyState.CLOSED,\n      event:     null,\n    }\n\n    this._readyState = new BehaviorSubject<ReadyState>(ReadyState.CLOSED)\n    this.state = new StateSwitch('IoService', this.log)\n    this.state.setLog(this.log)\n\n    try {\n      await this.initStateDealer()\n      await this.initRxSocket()\n    } catch (e) {\n      this.log.silly('IoService', 'init() exception: %s', e.message)\n      throw e\n    }\n\n    this.readyState.subscribe(s => {\n      this.log.silly('IoService', 'init() readyState.subscribe(%s)', ReadyState[s])\n      this.snapshot.readyState = s\n    })\n    // IMPORTANT: subscribe to event and make it HOT!\n    this.event.subscribe(s => {\n      this.log.silly('IoService', 'init() event.subscribe({name:%s})', s.name)\n      this.snapshot.event = s\n    })\n\n    return\n  }\n\n  public token(): string\n  public token(newToken: string): void\n\n  public token(newToken?: string): string | void {\n    this.log.silly('IoService', 'token(%s)', newToken)\n    if (newToken) {\n      this._token = newToken\n      return\n    }\n    return this._token\n  }\n\n  async start(): Promise<void> {\n    this.log.verbose('IoService', 'start() with token:%s', this._token)\n\n    if (!this._token) {\n      throw new Error('start() without token')\n    }\n\n    if (this.state.on()) {\n      throw new Error('state is already ON')\n    }\n    if (this.state.pending()) {\n      throw new Error('state is pending')\n    }\n\n    this.state.on('pending')\n\n    this.autoReconnect = true\n\n    try {\n      await this.connectRxSocket()\n      this.state.on(true)\n    } catch (e) {\n      this.log.warn('IoService', 'start() failed:%s', e.message)\n\n      this.state.off(true)\n    }\n  }\n\n  async stop(): Promise<void> {\n    this.log.verbose('IoService', 'stop()')\n\n    if (this.state.off()) {\n      this.log.warn('IoService', 'stop() state is already off')\n      if (this.state.pending()) {\n        throw new Error('state pending() is true')\n      }\n      return\n    }\n\n    this.state.off('pending')\n\n    this.autoReconnect = false\n\n    if (!this._websocket) {\n      throw new Error('no websocket')\n    }\n\n    await this.socketClose(1000, 'IoService.stop()')\n    this.state.off(true)\n\n    return\n  }\n\n  public async restart(): Promise<void> {\n    this.log.verbose('IoService', 'restart()')\n    try {\n      await this.stop()\n      await this.start()\n    } catch (e) {\n      this.log.error('IoService', 'restart() error:%s', e.message)\n      throw e\n    }\n    return\n  }\n\n  private initStateDealer() {\n    this.log.verbose('IoService', 'initStateDealer()')\n\n    const isReadyStateOpen = (s: ReadyState) => s === ReadyState.OPEN\n\n    this.readyState.pipe(\n      filter(isReadyStateOpen),\n    )\n      .subscribe(open => this.stateOnOpen())\n  }\n\n  /**\n   * Creates a subject from the specified observer and observable.\n   *  - https://github.com/Reactive-Extensions/RxJS/blob/master/doc/api/subjects/subject.md\n   * Create an Rx.Subject using Subject.create that allows onNext without subscription\n   *   A socket implementation (example, don't use)\n   *  - http://stackoverflow.com/a/34862286/1123955\n   */\n  private async initRxSocket(): Promise<void> {\n    this.log.verbose('IoService', 'initRxSocket()')\n\n    if (this.event) {\n      throw new Error('re-init is not permitted')\n    }\n\n    // 1. Mobile Originated. moObserver.next() means mobile is sending\n    this.moObserver = {\n      next:     this.socketSend.bind(this),\n      error:    this.socketClose.bind(this),\n      complete: this.socketClose.bind(this),\n    }\n\n    // 2. Mobile Terminated. mtObserver.next() means mobile is receiving\n    const observable = new Observable((observer: Observer<IoEvent>) => {\n      this.log.verbose('IoService', 'initRxSocket() Observable.create()')\n      this.mtObserver = observer\n\n      return this.socketClose.bind(this)\n    })\n\n    // 3. Subject for MO & MT Observers\n    this.event = Subject.create(this.moObserver, observable.pipe(share()))\n\n  }\n\n  private async connectRxSocket(): Promise<void> {\n    this.log.verbose('IoService', 'connectRxSocket()')\n\n    // FIXME: check & close the old one\n    if (this._websocket) {\n      throw new Error('already has a websocket')\n    }\n\n    // if (this.state.target() !== 'open'\n    //   || this.state.current() !== 'open'\n    //   || this.state.stable()\n    if (this.state.off()) {\n      throw new Error('switch state is off')\n    } else if (!this.state.pending()) {\n      throw new Error('switch state is already ON')\n    }\n\n    this._websocket = new WebSocket(this.endPoint(), this.PROTOCOL)\n    this.socketUpdateState()\n\n    const onOpenPromise = new Promise<void>((resolve, reject) => {\n      this.log.verbose('IoService', 'connectRxSocket() Promise()')\n\n      const id = setTimeout(() => {\n        this._websocket = null\n        const e = new Error('rxSocket connect timeout after '\n                            + Math.round(this.CONNECT_TIMEOUT / 1000),\n                          )\n        reject(e)\n      }, this.CONNECT_TIMEOUT) // timeout for connect websocket\n\n      this._websocket.onopen = (e) => {\n        this.log.verbose('IoService', 'connectRxSocket() Promise() WebSocket.onOpen() resolve()')\n        this.socketUpdateState()\n        clearTimeout(id)\n        resolve()\n      }\n    })\n\n    // Handle the payload\n    this._websocket.onmessage = this.socketOnMessage.bind(this)\n    // Deal the event\n    this._websocket.onerror   = this.socketOnError.bind(this)\n    this._websocket.onclose   = this.socketOnClose.bind(this)\n\n    return onOpenPromise\n  }\n\n  private endPoint(): string {\n    const url = this.ENDPOINT + this._token\n    this.log.verbose('IoService', 'endPoint() => %s', url)\n    return url\n  }\n\n  /******************************************************************\n   *\n   * State Event Listeners\n   *\n   */\n  private stateOnOpen() {\n    this.log.verbose('IoService', 'stateOnOpen()')\n\n    this.socketSendBuffer()\n    this.rpcUpdate('from stateOnOpen()')\n  }\n\n  /******************************************************************\n   *\n   * Io RPC Methods\n   *\n   */\n  async rpcDing(payload: any): Promise<any> {\n    this.log.verbose('IoService', 'ding(%s)', payload)\n\n    const e: IoEvent = {\n      name: 'ding',\n      payload,\n    }\n    this.event.next(e)\n    // TODO: get the return value\n  }\n\n  async rpcUpdate(payload: any): Promise<void> {\n    this.event.next({\n      name:     'update',\n      payload,\n    })\n  }\n\n  /******************************************************************\n   *\n   * Socket Actions\n   *\n   */\n  private async socketClose(code?: number, reason?: string): Promise<void> {\n    this.log.verbose('IoService', 'socketClose()')\n\n    if (!this._websocket) {\n      throw new Error('no websocket')\n    }\n\n    this._websocket.close(code, reason)\n    this.socketUpdateState()\n\n    const future = new Promise(resolve => {\n      this.readyState.pipe(\n        filter(s => s === ReadyState.CLOSED),\n      )\n      .subscribe(resolve)\n    })\n    await future\n\n    return\n  }\n\n  private socketSend(ioEvent: IoEvent) {\n    this.log.silly('IoService', 'socketSend({name:%s, payload:%s})', ioEvent.name, ioEvent.payload)\n\n    if (!this._websocket) {\n      this.log.silly('IoService', 'socketSend() no _websocket')\n    }\n\n    const strEvt = JSON.stringify(ioEvent)\n    this.sendBuffer.push(strEvt)\n\n    // XXX can move this to onOpen?\n    this.socketSendBuffer()\n  }\n\n  private socketSendBuffer(): void {\n    this.log.silly('IoService', 'socketSendBuffer() length:%s', this.sendBuffer.length)\n\n    if (!this._websocket) {\n      throw new Error('socketSendBuffer(): no _websocket')\n    }\n\n    if (this._websocket.readyState !== WebSocket.OPEN) {\n      this.log.warn('IoService', 'socketSendBuffer() readyState is not OPEN, send job delayed.')\n      return\n    }\n\n    while (this.sendBuffer.length) {\n      const buf = this.sendBuffer.shift()\n      this.log.silly('IoService', 'socketSendBuffer() sending(%s)', buf)\n      this._websocket.send(buf)\n    }\n  }\n\n  private socketUpdateState() {\n    this.log.verbose('IoService', 'socketUpdateState() is %s',\n      ReadyState[this._websocket?.readyState],\n    )\n\n    if (!this._websocket) {\n      this.log.error('IoService', 'socketUpdateState() no _websocket')\n      return\n    }\n\n    this._readyState.next(this._websocket.readyState)\n  }\n\n  /******************************************************************\n   *\n   * Socket Events Listener\n   *\n   */\n  private socketOnMessage(message: MessageEvent) {\n    this.log.verbose('IoService', 'onMessage({data: %s})', message.data)\n\n    const data = message.data // WebSocket data\n\n    const ioEvent: IoEvent = {\n      name:     'raw',\n      payload:  data,\n    } // this is default io event for unknown format message\n\n    try {\n      const obj = JSON.parse(data)\n      ioEvent.name = obj.name\n      ioEvent.payload = obj.payload\n    } catch (e) {\n      this.log.warn('IoService', 'onMessage parse message fail. save as RAW')\n    }\n\n    this.mtObserver.next(ioEvent)\n  }\n\n  private socketOnError(event: Event) {\n    this.log.silly('IoService', 'socketOnError(%s)', event)\n    // this._websocket = null\n  }\n\n  /**\n   * https://developer.mozilla.org/en-US/docs/Web/API/CloseEvent\n   * code: 1006\tCLOSE_ABNORMAL\n   *  - Reserved. Used to indicate that a connection was closed abnormally\n   *    (that is, with no close frame being sent) when a status code is expected.\n   */\n  private socketOnClose(closeEvent: CloseEvent) {\n    this.log.verbose('IoService', 'socketOnClose({code:%s, reason:%s, returnValue:%s})',\n                                  closeEvent.code,\n                                  closeEvent.reason,\n                                  closeEvent.returnValue,\n                    )\n    this.socketUpdateState()\n    /**\n     * reconnect inside onClose\n     */\n    if (this.autoReconnect) {\n      this.state.on('pending')\n      setTimeout(async () => {\n        try {\n          await this.connectRxSocket()\n          this.state.on(true)\n        } catch (e) {\n          this.log.warn('IoService', 'socketOnClose() autoReconnect() exception: %s', e)\n          this.state.off(true)\n        }\n      }, 1000)\n    } else {\n      this.state.off(true)\n    }\n    this._websocket = null\n\n    if (!closeEvent.wasClean) {\n      this.log.warn('IoService', 'socketOnClose() event.wasClean FALSE')\n      // TODO emit error\n    }\n  }\n}\n"]}