import Axios, {AxiosResponse} from 'axios'; import * as qs from 'qs'; import {Connection as Interfaces, WebSocketConnection} from './interfaces'; const fatalErrors = [2, 3, 4, 9, 30, 31, 33, 39, 40, 62, -4]; const dieErrors = [9, 30, 31, 39]; async function queryApi(this: void, endpoint: string, data: object): Promise { return Axios.post(`https://www.f-list.net/json/api/${endpoint}`, qs.stringify(data)); } export default class Connection implements Interfaces.Connection { character: string; vars: Interfaces.Vars & {[key: string]: string} = {}; //tslint:disable-line:no-any protected socket: WebSocketConnection | undefined = undefined; private messageHandlers: {[key in keyof Interfaces.ServerCommands]?: Interfaces.CommandHandler[]} = {}; private connectionHandlers: {[key in Interfaces.EventType]?: Interfaces.EventHandler[]} = {}; private errorHandlers: ((error: Error) => void)[] = []; private ticket: string; private cleanClose = false; private reconnectTimer: NodeJS.Timer; private ticketProvider: Interfaces.TicketProvider; private reconnectDelay = 0; private isReconnect = false; constructor(private readonly clientName: string, private readonly version: string, private readonly socketProvider: new() => WebSocketConnection, private readonly account: string, ticketProvider: Interfaces.TicketProvider | string) { this.ticketProvider = typeof ticketProvider === 'string' ? async() => this.getTicket(ticketProvider) : ticketProvider; } async connect(character: string): Promise { this.cleanClose = false; this.isReconnect = this.character === character; this.character = character; try { this.ticket = await this.ticketProvider(); } catch(e) { for(const handler of this.errorHandlers) handler(e); await this.invokeHandlers('closed', true); this.reconnect(); return; } await this.invokeHandlers('connecting', this.isReconnect); if(this.cleanClose) { this.cleanClose = false; await this.invokeHandlers('closed', false); return; } const socket = this.socket = new this.socketProvider(); socket.onOpen(() => { this.send('IDN', { account: this.account, character: this.character, cname: this.clientName, cversion: this.version, method: 'ticket', ticket: this.ticket }); }); socket.onMessage(async(msg: string) => { const type = msg.substr(0, 3); const data = msg.length > 6 ? JSON.parse(msg.substr(4)) : undefined; return this.handleMessage(type, data); }); socket.onClose(async() => { if(!this.cleanClose) this.reconnect(); this.socket = undefined; await this.invokeHandlers('closed', !this.cleanClose); }); socket.onError((error: Error) => { for(const handler of this.errorHandlers) handler(error); }); return new Promise((resolve) => { const handler = () => { resolve(); this.offEvent('connected', handler); }; this.onEvent('connected', handler); }); } private reconnect(): void { this.reconnectTimer = setTimeout(async() => this.connect(this.character), this.reconnectDelay); this.reconnectDelay = this.reconnectDelay >= 30000 ? 60000 : this.reconnectDelay >= 10000 ? 30000 : 10000; } close(): void { clearTimeout(this.reconnectTimer); this.cleanClose = true; if(this.socket !== undefined) this.socket.close(); } async queryApi(endpoint: string, data?: {account?: string, ticket?: string}): Promise { if(data === undefined) data = {}; data.account = this.account; data.ticket = this.ticket; let res = (await queryApi(endpoint, data)).data; if(res.error === 'Invalid ticket.' || res.error === 'Your login ticket has expired (five minutes) or no ticket requested.') { data.ticket = this.ticket = await this.ticketProvider(); res = (await queryApi(endpoint, data)).data; } if(res.error !== '') { const error = new Error(res.error); (error).request = true; throw error; } return res; } onError(handler: (error: Error) => void): void { this.errorHandlers.push(handler); } onEvent(type: Interfaces.EventType, handler: Interfaces.EventHandler): void { let handlers = this.connectionHandlers[type]; if(handlers === undefined) handlers = this.connectionHandlers[type] = []; handlers.push(handler); } offEvent(type: Interfaces.EventType, handler: Interfaces.EventHandler): void { const handlers = this.connectionHandlers[type]; if(handlers === undefined) return; handlers.splice(handlers.indexOf(handler), 1); } onMessage(type: K, handler: Interfaces.CommandHandler): void { let handlers = [] | undefined>this.messageHandlers[type]; if(handlers === undefined) handlers = this.messageHandlers[type] = []; handlers.push(handler); } offMessage(type: K, handler: Interfaces.CommandHandler): void { const handlers = [] | undefined>this.messageHandlers[type]; if(handlers === undefined) return; handlers.splice(handlers.indexOf(handler), 1); } send(command: K, data?: Interfaces.ClientCommands[K]): void { if(this.socket !== undefined) this.socket.send(command + (data !== undefined ? ` ${JSON.stringify(data)}` : '')); } //tslint:disable:no-unsafe-any no-any protected async handleMessage(type: T, data: any): Promise { const time = new Date(); const handlers = [] | undefined>this.messageHandlers[type]; if(handlers !== undefined) for(const handler of handlers) await handler(data, time); switch(type) { case 'VAR': this.vars[data.variable] = data.value; break; case 'PIN': this.send('PIN'); break; case 'ERR': if(fatalErrors.indexOf(data.number) !== -1) { const error = new Error(data.message); for(const handler of this.errorHandlers) handler(error); if(dieErrors.indexOf(data.number) !== -1) this.close(); else this.socket!.close(); } break; case 'NLN': if(data.identity === this.character) { await this.invokeHandlers('connected', this.isReconnect); this.reconnectDelay = 0; } } } //tslint:enable private async getTicket(password: string): Promise { const data = <{ticket?: string, error: string}>(await Axios.post('https://www.f-list.net/json/getApiTicket.php', qs.stringify( {account: this.account, password, no_friends: true, no_bookmarks: true, no_characters: true}))).data; if(data.ticket !== undefined) return data.ticket; throw new Error(data.error); } private async invokeHandlers(type: Interfaces.EventType, isReconnect: boolean): Promise { const handlers = this.connectionHandlers[type]; if(handlers === undefined) return; for(const handler of handlers) await handler(isReconnect); } }