vk-chat-bot
Version:
Package for easy creation of chat bots for VK communities (uses Callback API).
3 lines (2 loc) • 39.4 kB
JavaScript
;var __awaiter=this&&this.__awaiter||function(e,t,n,r){return new(n||(n=Promise))((function(s,a){function i(e){try{o(r.next(e))}catch(e){a(e)}}function l(e){try{o(r.throw(e))}catch(e){a(e)}}function o(e){var t;e.done?s(e.value):(t=e.value,t instanceof n?t:new n((function(e){e(t)}))).then(i,l)}o((r=r.apply(e,t||[])).next())}))},__importDefault=this&&this.__importDefault||function(e){return e&&e.__esModule?e:{default:e}};Object.defineProperty(exports,"__esModule",{value:!0});const escape_string_regexp_1=__importDefault(require("escape-string-regexp")),context_1=__importDefault(require("./api/context")),log_1=require("./extra/log");class Core{constructor(e,t,n,r){this.locked=!1,this.eventHandlers={message_new:null,message_reply:null,message_edit:null,message_typing_state:null,message_allow:null,message_deny:null,start:null,service_action:null,no_match:null,handler_error:null},this.exactPayloadHandlers={},this.dynPayloadHandlers=[],this.commandHandlers=[],this.regexHandlers=[],this.eventWarnings=!0,this.helpMessage="",this.api=e,this.stats=t,this.cmdPrefix=n,this.escapedCmdPrefix=escape_string_regexp_1.default(this.cmdPrefix),this.groupId=escape_string_regexp_1.default(r.toString()),this.registerMessageNewHandler()}noEventWarnings(){this.eventWarnings=!1,log_1.log().w("Warnings about missing event handlers were disabled").from("core").now()}lock(){this.locked=!0,this.generateHelpMessage()}on(e,t){this.isLocked()||(Object.keys(this.eventHandlers).includes(e)||log_1.log().e(`Cannot register a handler: unknown event type '${e}'`).from("core").now(),this.eventHandlers[e]?"message_new"===e?log_1.log().e("Cannot register a handler: handler for the `message_new` event is defined internally").from("core").now():log_1.log().e(`Cannot register a handler: duplicate handler for event '${e}'`).from("core").now():this.eventHandlers[e]=t)}payload(e,t){this.isLocked()||("function"!=typeof e?this.exactPayloadHandlers[JSON.stringify(e)]?log_1.log().e(`Cannot register a handler: duplicate handler for payload '${e}'`).from("core").now():this.exactPayloadHandlers[JSON.stringify(e)]=t:this.dynPayloadHandlers.push({tester:e,handler:t}))}cmd(e,t,n=""){this.isLocked()||this.commandHandlers.push({command:e,description:n,handler:t})}regex(e,t){this.isLocked()||this.regexHandlers.push({regex:e,handler:t})}parseRequest(e){return __awaiter(this,void 0,void 0,(function*(){const t=e.object,n=e.type,r=new context_1.default(this.api,n,t,t.text);yield this.event(n,r)}))}help(){return this.helpMessage}isLocked(){return this.locked&&log_1.log().w("Registering a handler while the bot is running is not allowed").from("core").now(),this.locked}event(e,t){return __awaiter(this,void 0,void 0,(function*(){if(this.stats.event(e),this.eventHandlers[e])try{yield this.eventHandlers[e](t),t.needsAutoSend()&&"message_new"!==e&&(yield t.send())}catch(n){log_1.log().w(`Error in handler: ${n}`).from("core").now(),"handler_error"!==e&&(yield this.event("handler_error",t))}else this.eventWarnings&&log_1.log().w(`No handler for event '${e}'`).from("core").now()}))}registerMessageNewHandler(){this.on("message_new",e=>__awaiter(this,void 0,void 0,(function*(){if(!e.obj.action)return(yield this.tryHandlePayload(e))||(yield this.tryHandleCommand(e))||(yield this.tryHandleRegex(e))?void(e.needsAutoSend()&&(yield e.send())):(log_1.log().w(`Don't know how to respond to ${JSON.stringify(e.msg).replace(/\n/g,"\\n")}, calling 'no_match' event`).from("core").now(),void(yield this.event("no_match",e)));yield this.event("service_action",e)})))}tryHandlePayload(e){return __awaiter(this,void 0,void 0,(function*(){const{payload:t}=e.obj;if(t){try{if("start"===JSON.parse(t).command)return yield this.event("start",e),e.noAutoSend(),!0}catch(e){}if(this.exactPayloadHandlers[t])return yield this.exactPayloadHandlers[t](e),!0;const n=this.dynPayloadHandlers.map(e=>{let n=null;try{n=JSON.parse(t)}catch(e){}return e.tester(t,n)?e:null}).filter(e=>e);if(n)return yield n[0].handler(e),!0}return!1}))}tryHandleCommand(e){return __awaiter(this,void 0,void 0,(function*(){const t=this.commandHandlers.map(t=>{const n=escape_string_regexp_1.default(t.command),r=new RegExp(`^( *\\[club${this.groupId}\\|.*\\])?( *${this.escapedCmdPrefix}${n})+`,"i");return r.test(e.msg)?{handler:t,msg:e.msg.replace(r,"")}:null}).filter(e=>e);if(t.length>0){const{handler:n,msg:r}=t[0];return e.msg=r,yield n.handler(e),!0}return!1}))}tryHandleRegex(e){return __awaiter(this,void 0,void 0,(function*(){const t=this.regexHandlers.filter(t=>t.regex.test(e.msg));return t.length>0&&(yield t[0].handler(e),!0)}))}generateHelpMessage(){let e="\n";this.commandHandlers.forEach(t=>{let n="";n+=this.cmdPrefix,n+=t.command,t.description&&(n+=" - ",n+=t.description),e+=`${n}\n`}),this.helpMessage=e}getHandlerCounts(){return{evt:Object.values(this.eventHandlers).filter(e=>e).length-1,cmd:this.commandHandlers.length,reg:this.regexHandlers.length,pld:Object.keys(this.exactPayloadHandlers).length+this.dynPayloadHandlers.length}}}exports.default=Core;
//# sourceMappingURL=data:application/json;charset=utf8;base64,{"version":3,"sources":["core.js","core.ts"],"names":["__awaiter","this","thisArg","_arguments","P","generator","Promise","resolve","reject","fulfilled","value","step","next","e","rejected","result","done","then","apply","__importDefault","mod","__esModule","default","Object","defineProperty","exports","escape_string_regexp_1","require","context_1","log_1","Core","[object Object]","api","stats","cmdPrefix","groupId","locked","eventHandlers","message_new","message_reply","message_edit","message_typing_state","message_allow","message_deny","start","service_action","no_match","handler_error","exactPayloadHandlers","dynPayloadHandlers","commandHandlers","regexHandlers","eventWarnings","helpMessage","escapedCmdPrefix","toString","registerMessageNewHandler","log","w","from","now","generateHelpMessage","event","handler","isLocked","keys","includes","payload","JSON","stringify","push","tester","command","description","regex","body","obj","object","type","$","text","name","needsAutoSend","send","error","on","action","tryHandlePayload","tryHandleCommand","tryHandleRegex","msg","replace","parse","noAutoSend","handlers","map","potentialHandler","parsed","filter","handlerObjs","cmd","cmdRegex","RegExp","test","length","forEach","helpEntry","evt","values","reg","pld"],"mappings":"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","file":"core.js","sourcesContent":[null,"import escapeRegex from 'escape-string-regexp';\nimport Context from './api/context';\nimport { log } from './extra/log';\n\nimport API from './api/api';\nimport Stats from './extra/stats';\n\nexport type Handler = ($: Context) => void | Promise<void>;\n\nexport type Payload = any; // eslint-disable-line @typescript-eslint/no-explicit-any\n\n/**\n * @param payloadJson `payloadJson` the stringified JSON of the payload (as given by Callback API)\n * @param payload `payload` the JS object that was made by parsing the `payloadJson`, if the\n * parsing was successful\n */\nexport type Tester = (payloadJson: string, payload?: Payload) => boolean;\n\ninterface DynPayloadHandler {\n    tester: Tester;\n    handler: Handler;\n}\n\ninterface CommandHandler {\n    command: string;\n    description: string;\n    handler: Handler;\n}\n\n/**\n * Dispatches message handling to appropriate handlers,\n * and is used for setting these handlers.\n *\n * Handlers for the `message_new` event will be searched in this order:\n * 1. If service action message => [[on]] handler for the `service_action` event\n * 1. If user pressed the `Start` button => [[on]] handler for the `start` event\n * 1. [[payload]]\n * 1. [[cmd]]\n * 1. [[regex]]\n *\n * For other events, a matching [[on]] handler will be called.\n */\nexport default class Core {\n    public readonly api: API;\n    public readonly stats: Stats;\n\n    /**\n     * Command prefix.\n     */\n    private cmdPrefix: string;\n\n    /**\n     * Group ID.\n     */\n    private groupId: string;\n\n    /**\n     * Command prefix, escaped for usage in regular expressions\n     */\n    private escapedCmdPrefix: string;\n\n    /**\n     * Is this `Core` locked?\n     */\n    private locked = false;\n\n    /**\n     * Handlers for events.\n     */\n    /* eslint-disable @typescript-eslint/camelcase */\n    private eventHandlers: { [key: string]: Handler } = {\n        // Callback API\n        message_new: null,\n        message_reply: null,\n        message_edit: null,\n        message_typing_state: null,\n        message_allow: null,\n        message_deny: null,\n\n        // Detected when parsing 'message_new' event\n        start: null,\n        service_action: null,\n\n        // Internal events\n        no_match: null,\n        handler_error: null,\n    };\n    /* eslint-enable @typescript-eslint/camelcase */\n\n    /**\n     * Exact payload handlers.\n     */\n    private exactPayloadHandlers: { [key: string]: Handler } = {};\n\n    /**\n     * Dynamic payload handlers.\n     * (those which use functions to determine whether a handler is suitable)\n     */\n    private dynPayloadHandlers: DynPayloadHandler[] = [];\n\n    /**\n     * Command handlers.\n     */\n    private commandHandlers: CommandHandler[] = [];\n\n    /**\n     * Regular expression handlers.\n     */\n    private regexHandlers: { regex: RegExp; handler: Handler }[] = [];\n\n    /**\n     * Are event warnings enabled?\n     */\n    private eventWarnings = true;\n\n    /**\n     * The help message.\n     */\n    private helpMessage = '';\n\n    /**\n     * Creates a new [[Core]].\n     * @param api [[API]] object\n     * @param stats [[Stats]] object\n     * @param cmdPrefix command prefix\n     * @param groupId group ID\n     */\n    public constructor(\n        api: API,\n        stats: Stats,\n        cmdPrefix: string,\n        groupId: string | number,\n    ) {\n        this.api = api;\n        this.stats = stats;\n        this.cmdPrefix = cmdPrefix;\n        this.escapedCmdPrefix = escapeRegex(this.cmdPrefix);\n        this.groupId = escapeRegex(groupId.toString());\n\n        this.registerMessageNewHandler();\n    }\n\n    /**\n     * Disables warnings about missing event handlers.\n     */\n    public noEventWarnings(): void {\n        this.eventWarnings = false;\n        log()\n            .w('Warnings about missing event handlers were disabled')\n            .from('core')\n            .now();\n    }\n\n    /**\n     * Locks this `Core`, so new handlers can't be added,\n     * and generates the help message for later usage.\n     */\n    public lock(): void {\n        this.locked = true;\n        this.generateHelpMessage();\n    }\n\n    /**\n     * Registers an event handler.\n     *\n     * Does not work for `message_new`, as its handler is defined by `vk-chat-bot` itself.\n     *\n     * ### Events\n     *\n     * #### Callback API Events\n     * Event | Description\n     * ---|---\n     * `message_allow` | User **allowed** sending messages to him/her\n     * `message_deny` | User **disallowed** sending messages to him/her\n     * `message_reply` | New **message sent** by community administrator (or by the bot itself)\n     * `message_edit` | **Message edited** by user\n     * `message_typing_state` | **User is typing** a message\n     *\n     * #### Other Events\n     * Event type | When handler is called\n     * ---|---\n     * `start` | If the message's payload is `{\"command\": \"start\"}` (i.e. `Start` button pressed)\n     * `service_action` | Service action message received\n     * `no_match` | When no matching `cmd()` or `regex()` handler is found\n     * `handler_error` | If an error is thrown in a handler\n     *\n     * #### The `service_action` event\n     * > The `$.obj.action` object contains information about the service action.\n     * > It contains the following fields:\n     *\n     * ```\n     * type (string) — action type, one of:\n     *    `chat_photo_update` — chat photo updated\n     *    `chat_photo_remove` — chat photo removed\n     *    `chat_create` — chat created\n     *    `chat_title_update` — chat title updated\n     *    `chat_invite_user` — user was invited to chat\n     *    `chat_kick_user` — user was kicked from the chat\n     *    `chat_pin_message` — a message was pinned\n     *    `chat_unpin_message` — a message was unpinned\n     *    `chat_invite_user_by_link` — user joined the chat by link\n     *\n     * member_id (integer):\n     *   user id (if > 0), which was invited or kicked (if < 0, see `email` field)\n     *     (`chat_invite_user`, `chat_invite_user_by_link`,  `chat_kick_user`)\n     *   user id, which pinned or unpinned a message\n     *     (`chat_pin_message`, `chat_unpin_message`)\n     *\n     * text (string):\n     *   chat name\n     *     (`chat_create`, `chat_title_update`)\n     *\n     * email (string):\n     *   email, which was invited or kicked\n     *     (`chat_invite_user`, `chat_kick_user`, member_id < 0)\n     *\n     * photo (object) — chat picture, contains:\n     *     photo_50 (string): URL of image 50 x 50 px\n     *     photo_100 (string): URL of image 100 x 100 px\n     *     photo_200 (string): URL of image 200 x 200 px\n     * ```\n     *\n     * @param - event name\n     * @param - function which will handle the message\n     *\n     * @example\n     * ```\n     * core.on('no_match', $ => {\n     *   $.text('I don\\'t know how to respond to your message.');\n     * });\n     * ```\n     */\n    public on(event: string, handler: Handler): void {\n        if (this.isLocked()) {\n            return;\n        }\n\n        if (!Object.keys(this.eventHandlers).includes(event)) {\n            log()\n                .e(`Cannot register a handler: unknown event type '${event}'`)\n                .from('core')\n                .now();\n        }\n\n        if (!this.eventHandlers[event]) {\n            this.eventHandlers[event] = handler;\n        } else if (event === 'message_new') {\n            log()\n                .e(\n                    'Cannot register a handler: handler for the `message_new` event is defined internally',\n                )\n                .from('core')\n                .now();\n        } else {\n            log()\n                .e(`Cannot register a handler: duplicate handler for event '${event}'`)\n                .from('core')\n                .now();\n        }\n    }\n\n    /**\n     * Registers a payload handler.\n     *\n     * **Note**: exact handlers are searched first, and only if they don't match,\n     * the search for a dynamic handler begins.\n     *\n     * @param payload - exact payload to handle,\n     * or a function (type [[Tester]]) which\n     * will determine whether to handle the payload or not.\n     * @param handler - function which will handle the message\n     *\n     * @example\n     * ```\n     * // -------> KEYBOARD (for sending the payload)\n     *\n     * // Create a keyboard\n     * const { colors, Keyboard, Button } = vk.kbd;\n     *\n     * var kbd = new Keyboard([\n     *     [\n     *         // Clicking on this button will send the payload {a: 'b'}\n     *         button.text('Test 1', colors.default, {a: 'b'}),\n     *         button.text('Test 2', colors.default, {a: 'b', c: 'd'})\n     *     ]\n     * ], false);\n     *\n     * // When asked, send the keyboard\n     * core.regex(/keyboard/i, $ => {\n     *    $.keyboard(kbd);\n     *    $.text('Here it is!');\n     * });\n     *\n     * // -------> EXACT PAYLOAD\n     * core.payload({a: 'b'}, $ => {\n     *    $.text('Received secret payload!');\n     * });\n     *\n     * // -------> DYNAMIC PAYLOAD\n     * // In this case, the handler will run only if the\n     * // payload's property `c` contains the value `d`.\n     * core.payload((payload, parsed) => {\n     *    if (parsed) { // If the payload is a valid JSON\n     *      return parsed.c === 'd';\n     *    } else {\n     *      return false;\n     *    }\n     * }, $ => {\n     *    $.text(`In message '${$.msg}', payload.c is 'd'!`);\n     * });\n     * ```\n     */\n    public payload(payload: Payload, handler: Handler): void {\n        if (this.isLocked()) {\n            return;\n        }\n\n        if (typeof payload !== 'function') {\n            // Exact payload match:\n\n            if (!this.exactPayloadHandlers[JSON.stringify(payload)]) {\n                this.exactPayloadHandlers[JSON.stringify(payload)] = handler;\n            } else {\n                log()\n                    .e(\n                        `Cannot register a handler: duplicate handler for payload '${payload}'`,\n                    )\n                    .from('core')\n                    .now();\n            }\n        } else {\n            // Dynamic payload match:\n\n            this.dynPayloadHandlers.push({\n                tester: payload,\n                handler: handler,\n            });\n        }\n    }\n\n    /**\n     * Registers a command handler.\n     *\n     * Handler is called if the message begins with `cmd_prefix`\n     * (defined in the parameters) **+** `command`\n     *\n     * @param command - command\n     * @param handler - function which will handle the message\n     * @param description - the description of what this command does,\n     * to be used in help messages.\n     *\n     * @example\n     * ```\n     * core.cmd('help', $ => {\n     *   // core.help() returns the help message\n     *   $.text('Test Bot' + core.help());\n     * }, 'shows the help message');\n     * ```\n     */\n    public cmd(\n        command: string,\n        handler: Handler,\n        description = '',\n    ): void {\n        if (this.isLocked()) {\n            return;\n        }\n\n        this.commandHandlers.push({\n            command,\n            description,\n            handler: handler,\n        });\n    }\n\n    /**\n     * Registers a regex handler.\n     *\n     * @param handler - function which will handle the message\n     *\n     * @example\n     * ```\n     * core.regex(/h(i|ello|ey)/i, $ => {\n     *    $.text('Hello, I am a test bot. You said: ' + $.msg);\n     * });\n     * ```\n     */\n    public regex(regex: RegExp, handler: Handler): void {\n        if (this.isLocked()) {\n            return;\n        }\n\n        this.regexHandlers.push({\n            regex,\n            handler: handler,\n        });\n    }\n\n    /**\n     * Parses the request, creates a [[Context]], and proceeds\n     * to call [[Core.event]] to handle the event\n     *\n     * @param body - body of the request, in parsed JSON\n     *\n     */\n    public async parseRequest(body: any /* TODO: body type? */): Promise<void> { // eslint-disable-line @typescript-eslint/no-explicit-any\n        const obj = body.object;\n        const event = body.type;\n\n        const $ = new Context(this.api, event, obj, obj.text);\n        await this.event(event, $);\n    }\n\n    /**\n     * Returns the help message.\n     */\n    public help(): string {\n        return this.helpMessage;\n    }\n\n    /**\n     * Indicates whether this `Core` is locked, and prints a message\n     * to notify the user if it is locked.\n     */\n    private isLocked(): boolean {\n        if (this.locked) {\n            log()\n                .w('Registering a handler while the bot is running is not allowed')\n                .from('core')\n                .now();\n        }\n\n        return this.locked;\n    }\n\n    /**\n     * Handles an event.\n     */\n    private async event(name: string, $: Context): Promise<void> {\n        this.stats.event(name);\n\n        if (this.eventHandlers[name]) {\n            try {\n                await this.eventHandlers[name]($);\n\n                if ($.needsAutoSend() && name !== 'message_new') {\n                    await $.send();\n                }\n            } catch (error) {\n                log()\n                    .w(`Error in handler: ${error}`)\n                    .from('core')\n                    .now();\n\n                if (name !== 'handler_error') {\n                    await this.event('handler_error', $);\n                }\n            }\n        } else if (this.eventWarnings) {\n            log()\n                .w(`No handler for event '${name}'`)\n                .from('core')\n                .now();\n        }\n    }\n\n    /**\n     * Registers a handler for `message_new` event.\n     */\n    private registerMessageNewHandler(): void {\n        this.on('message_new', async ($: Context): Promise<void> => {\n            // Check for 'service_action' event\n            if ($.obj.action) {\n                await this.event('service_action', $);\n                return;\n            }\n\n            // Handle regular message\n            if (!(await this.tryHandlePayload($))) {\n                if (!(await this.tryHandleCommand($))) {\n                    if (!(await this.tryHandleRegex($))) {\n                        log()\n                            .w(\n                                `Don't know how to respond to ${JSON.stringify($.msg).replace(\n                                    /\\n/g,\n                                    '\\\\n',\n                                )}, calling 'no_match' event`,\n                            )\n                            .from('core')\n                            .now();\n                        await this.event('no_match', $);\n                        return;\n                    }\n                }\n            }\n\n            if ($.needsAutoSend()) {\n                await $.send();\n            }\n        });\n    }\n\n    /**\n     * Tries to handle the message in the given `Context`\n     * with a payload handler.\n     *\n     * @return was the message handled?\n     */\n    private async tryHandlePayload($: Context): Promise<boolean> {\n        const { payload } = $.obj;\n        if (payload) {\n            // Check for 'start' event\n            try {\n                if (JSON.parse(payload).command === 'start') {\n                    await this.event('start', $);\n                    $.noAutoSend(); // Message sending was already handled by event\n                    return true;\n                }\n            } catch (e) {\n                /* JSON Parse Error */\n            }\n\n            // Check for exact payload handler\n            if (this.exactPayloadHandlers[payload]) {\n                await this.exactPayloadHandlers[payload]($);\n                return true;\n            }\n\n            // Check for dynamic payload handler\n            const handlers = this.dynPayloadHandlers\n                .map((potentialHandler): DynPayloadHandler => {\n                    let parsed = null;\n                    try {\n                        parsed = JSON.parse(payload);\n                    } catch (e) {\n                        /* JSON Parse Error */\n                    }\n\n                    if (potentialHandler.tester(payload, parsed)) {\n                        return potentialHandler;\n                    }\n\n                    return null;\n                })\n                .filter(e => e); // eslint-disable-line @typescript-eslint/explicit-function-return-type\n\n            if (handlers) {\n                await handlers[0].handler($);\n                return true;\n            }\n        }\n\n        return false;\n    }\n\n    /**\n     * Tries to handle the message in the given `Context`\n     * with a command handler.\n     *\n     * @return was the message handled?\n     */\n    private async tryHandleCommand($: Context): Promise<boolean> {\n        const handlerObjs = this.commandHandlers\n            .map((potentialHandler): { handler: CommandHandler; msg: string } => {\n                const cmd = escapeRegex(potentialHandler.command);\n                const cmdRegex = new RegExp(\n                    `^( *\\\\[club${this.groupId}\\\\|.*\\\\])?( *${this.escapedCmdPrefix}${cmd})+`, 'i',\n                );\n\n                if (cmdRegex.test($.msg)) {\n                    return {\n                        handler: potentialHandler,\n                        msg: $.msg.replace(cmdRegex, ''),\n                    };\n                }\n\n                return null;\n            })\n            .filter(e => e); // eslint-disable-line @typescript-eslint/explicit-function-return-type\n\n        if (handlerObjs.length > 0) {\n            const { handler, msg } = handlerObjs[0];\n\n            $.msg = msg; // eslint-disable-line no-param-reassign\n            await handler.handler($);\n            return true;\n        }\n\n        return false;\n    }\n\n    /**\n     * Tries to handle the message in the given `Context`\n     * with a regex handler.\n     *\n     * @return was the message handled?\n     */\n    private async tryHandleRegex($: Context): Promise<boolean> {\n        const handlers = this.regexHandlers.filter((potentialHandler): boolean =>\n            potentialHandler.regex.test($.msg),\n        );\n\n        if (handlers.length > 0) {\n            await handlers[0].handler($);\n            return true;\n        }\n\n        return false;\n    }\n\n    /**\n     * Generates the help message.\n     */\n    private generateHelpMessage(): void {\n        let helpMessage = '\\n';\n\n        this.commandHandlers.forEach((handler): void => {\n            let helpEntry = '';\n\n            helpEntry += this.cmdPrefix;\n            helpEntry += handler.command;\n\n            if (handler.description) {\n                helpEntry += ' - ';\n                helpEntry += handler.description;\n            }\n\n            helpMessage += `${helpEntry}\\n`;\n        });\n\n        this.helpMessage = helpMessage;\n    }\n\n    /**\n     * Returns handler counts, except `message_new` event since it is built-in.\n     */\n    public getHandlerCounts(): {\n        evt: number;\n        cmd: number;\n        reg: number;\n        pld: number;\n    } {\n        return {\n            // Does not count `message_new` event\n            evt: Object.values(this.eventHandlers).filter(e => e).length - 1, // eslint-disable-line @typescript-eslint/explicit-function-return-type\n            cmd: this.commandHandlers.length,\n            reg: this.regexHandlers.length,\n            pld: Object.keys(this.exactPayloadHandlers).length + this.dynPayloadHandlers.length,\n        }\n    }\n}\n"]}