UNPKG

vk-chat-bot

Version:

Package for easy creation of chat bots for VK communities (uses Callback API).

3 lines (2 loc) 21.3 kB
"use strict";var __awaiter=this&&this.__awaiter||function(e,t,s,r){return new(s||(s=Promise))((function(o,i){function n(e){try{a(r.next(e))}catch(e){i(e)}}function u(e){try{a(r.throw(e))}catch(e){i(e)}}function a(e){var t;e.done?o(e.value):(t=e.value,t instanceof s?t:new s((function(e){e(t)}))).then(n,u)}a((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 crypto_1=__importDefault(require("crypto")),request_promise_1=__importDefault(require("request-promise")),log_1=require("../extra/log"),API_VERSION="5.95",API_QUOTA=20;class API{constructor(e,t){this.queue=[],this.isQueueProcessing=!1,this.vkToken=e,this.stats=t,this.checkPermissions().then(e=>{log_1.log().i(e).from("api").now()}).catch(e=>{log_1.log().w(e).from("api").now()}),setInterval(()=>{this.isQueueProcessing||(this.isQueueProcessing=!0,this.processQueue().then(()=>{this.isQueueProcessing=!1}).catch(e=>{log_1.log().w(e).from("api").now(),this.isQueueProcessing=!1}))},1e3)}scheduleCall(e,t){return __awaiter(this,void 0,void 0,(function*(){return new Promise((s,r)=>{this.queue.push({method:e,params:t,resolve:s,reject:r})})}))}call(e,t){return __awaiter(this,void 0,void 0,(function*(){const s={uri:`https://api.vk.com/method/${encodeURIComponent(e)}`,json:!0,qs:Object.assign({access_token:this.vkToken,v:API_VERSION},t)},r=request_promise_1.default(s);return r.catch(t=>{log_1.log().w(`Error occured while calling API method '${e}': ${t}`).from("api").now()}),r}))}send(e,t,s,r){return __awaiter(this,void 0,void 0,(function*(){const o={peer_id:e.toString(),random_id:BigInt.asIntN(32,BigInt(`0x${crypto_1.default.randomBytes(6).toString("hex")}`)).toString()};return t&&(o.message=t),s&&(o.attachment=s),r&&(o.keyboard=r),new Promise(e=>{this.scheduleCall("messages.send",o).then(()=>{this.stats.sent(),e()}).catch(t=>{log_1.log().w(t).from("api").now(),e()})})}))}checkPermissions(){return __awaiter(this,void 0,void 0,(function*(){const e=yield this.scheduleCall("groups.getTokenPermissions",{}),{permissions:t}=e;let s=!1;return t.forEach(e=>{"messages"===e.name&&(s=!0)}),s?Promise.resolve("Token permission `messages` is present"):Promise.reject(new Error("Token permission `messages` is missing. Bot will be unable to send any messages"))}))}processQueue(){return __awaiter(this,void 0,void 0,(function*(){if(this.queue){for(let e=1;e<=API_QUOTA&&0!==this.queue.length;e+=1){const e=this.queue.shift(),t=yield this.call(e.method,e.params);if(void 0!==t.response&&null!==t.response)e.resolve(t.response);else if(t.error){const s=t.error.error_code,r=t.error.error_msg;e.reject(`An API call to method '${e.method}' failed due to an API error #${s}: ${r}`)}else e.reject(`An API call to method '${e.method}' failed due to an unknown API error. The API responded with: ${JSON.stringify(t)}`)}return Promise.resolve()}return Promise.reject(new Error("No queue for API calls found"))}))}}exports.default=API; //# sourceMappingURL=data:application/json;charset=utf8;base64,{"version":3,"sources":["api/api.js","api/api.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","crypto_1","require","request_promise_1","log_1","API_VERSION","API_QUOTA","API","[object Object]","vkToken","stats","queue","isQueueProcessing","checkPermissions","log","i","from","now","catch","w","setInterval","processQueue","method","params","push","options","uri","encodeURIComponent","json","qs","assign","access_token","v","promise","err","pid","message","attachment","keyboard","peer_id","toString","random_id","BigInt","asIntN","randomBytes","scheduleCall","sent","response","permissions","ok","forEach","permission","name","Error","length","shift","call","undefined","error","errorCode","error_code","errorMsg","error_msg","JSON","stringify"],"mappings":"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","file":"api/api.js","sourcesContent":[null,"import crypto from 'crypto';\nimport request from 'request-promise';\nimport { log } from '../extra/log';\nimport Stats from '../extra/stats';\n\n/**\n * VK API version used by API.\n */\nconst API_VERSION = '5.95';\n/**\n * API quota, in requests per second\n */\nconst API_QUOTA = 20;\n\n/**\n * Used to call API methods.\n *\n * You can get the [[API]] object from a [[Context]] object:\n * ```js\n * // Assuming your Context object is $\n * var api = $.api\n * ```\n *\n * Or from [[Core]] (after initialization with [[bot]]:\n * ```js\n * var api = core.api\n * ```\n */\nexport default class API {\n    /**\n     * VK API token.\n     */\n    private vkToken: string;\n\n    /**\n     * Stats object.\n     */\n    private stats: Stats;\n\n    /**\n     * Queue of scheduled API calls.\n     */\n    private queue: {\n        method: string;\n        params: { [key: string]: string };\n        resolve: Function;\n        reject: Function;\n    }[] = [];\n\n    /**\n     * Is the queue being processed now?\n     */\n    private isQueueProcessing = false;\n\n    /**\n     * Creates a new [[API]].\n     * @param vkToken VK API token\n     * @param stats [[Stats]] object\n     */\n    public constructor(vkToken: string, stats: Stats) {\n        this.vkToken = vkToken;\n        this.stats = stats;\n\n        // Check permissions\n        this.checkPermissions()\n            .then((e): void => {\n                log()\n                    .i(e)\n                    .from('api')\n                    .now();\n            })\n            .catch((e): void => {\n                log()\n                    .w(e)\n                    .from('api')\n                    .now();\n            });\n\n        // Start the queue processing\n        setInterval((): void => {\n            if (!this.isQueueProcessing) {\n                this.isQueueProcessing = true;\n                this.processQueue()\n                    .then((): void => {\n                        this.isQueueProcessing = false;\n                    })\n                    .catch((e): void => {\n                        log()\n                            .w(e)\n                            .from('api')\n                            .now();\n                        this.isQueueProcessing = false;\n                    });\n            }\n        }, 1000);\n    }\n\n    /**\n     * Schedules a call to a VK API Method.\n     *\n     * After the call completes, a check will be performed to see if the call was successful or not,\n     * and in the latter case a warning will be logged.\n     *\n     * @param method VK API method name\n     * @param params parameters for the method, `access_token` and `v` will be added automatically\n     *\n     * @return promise, which resolves with `json.response` when the request is completed\n     * and a response is given, and rejects if an error happened\n     *\n     * @example\n     * ```\n     * core.cmd('info', async $ => {\n     *    var uid = $.obj.from_id;\n     *\n     *    // Call VK API to get information about the user\n     *    var response = await $.api.scheduleCall('users.get', { user_ids: uid });\n     *    var userInfo = response[0];\n     *\n     *    var name = userInfo.first_name;\n     *    var surname = userInfo.last_name;\n     *\n     *  $.text(`User ID: ${uid}\\nName: ${name} ${surname}`);\n     * });\n     * ```\n     */\n    public async scheduleCall(method: string, params: { [key: string]: string }): Promise<any> { // eslint-disable-line @typescript-eslint/no-explicit-any\n        return new Promise((resolve, reject): void => {\n            this.queue.push({\n                method,\n                params,\n                resolve,\n                reject,\n            });\n        });\n    }\n\n    /**\n     * Call a VK API Method.\n     *\n     * **It is highly recommended to use [[scheduleCall]]\n     * instead to not exceed the API quota and to check whether the call was successful or not!**\n     *\n     * @param method VK API method name\n     * @param params parameters for the method, `access_token` and `v` will be added automatically\n     *\n     * @example\n     * ```\n     * core.cmd('info', async $ => {\n     *    var uid = $.obj.from_id;\n     *\n     *    // Call VK API to get information about the user\n     *    var json = await $.api.call('users.get', { user_ids: uid });\n     *    var userInfo = json.response[0];\n     *\n     *    var name = userInfo.first_name;\n     *    var surname = userInfo.last_name;\n     *\n     *  $.text(`User ID: ${uid}\\nName: ${name} ${surname}`);\n     * });\n     * ```\n     */\n    public async call(\n        method: string,\n        params: { [key: string]: string },\n    ): Promise<any> { // eslint-disable-line @typescript-eslint/no-explicit-any\n        const url = `https://api.vk.com/method/${encodeURIComponent(method)}`;\n\n        const options = {\n            uri: url,\n            json: true,\n            qs: {\n                access_token: this.vkToken, // eslint-disable-line @typescript-eslint/camelcase\n                v: API_VERSION,\n                ...params,\n            },\n        };\n\n        const promise = request(options);\n\n        promise.catch((err: Error): void => {\n            log()\n                .w(`Error occured while calling API method '${method}': ${err}`)\n                .from('api')\n                .now();\n        });\n\n        return promise;\n    }\n\n    /**\n     * Sends a message to a user via Peer ID.\n     *\n     * **Note that it is much easier to use the [[Context]] object passed to handlers\n     * to compose and send messages, keyboards and attachments!**\n     *\n     * @param pid peer ID\n     * @param message message text **(required, if attachment is empty)**\n     * @param attachment list of attachments, comma-separated\n     * (see [VK API Docs](https://vk.com/dev/messages.send) for further information)\n     * **(required if message is empty)**\n     * @param keyboard json of keyboard\n     *\n     * @example\n     * ```\n     * await api.send(1, 'Hello!', 'photo6492_456240778')\n     * ```\n     */\n    public async send(\n        pid: string | number,\n        message: string,\n        attachment: string,\n        keyboard: string,\n    ): Promise<void> {\n        /* global BigInt */\n\n        const params: {\n            peer_id: string;\n            message?: string;\n            attachment?: string;\n            keyboard?: string;\n            random_id: string;\n        } = {\n            peer_id: pid.toString(), // eslint-disable-line @typescript-eslint/camelcase\n            random_id: BigInt.asIntN( // eslint-disable-line @typescript-eslint/camelcase\n                32,\n                BigInt(`0x${crypto.randomBytes(6).toString('hex')}`),\n            ).toString(),\n        };\n\n        if (message) {\n            params.message = message;\n        }\n        if (attachment) {\n            params.attachment = attachment;\n        }\n        if (keyboard) {\n            params.keyboard = keyboard;\n        }\n\n        return new Promise((resolve): void => {\n            this.scheduleCall('messages.send', params)\n                .then((): void => {\n                    this.stats.sent();\n                    resolve();\n                })\n                .catch((e): void => {\n                    log()\n                        .w(e)\n                        .from('api')\n                        .now();\n                    resolve();\n                });\n        });\n    }\n\n    /**\n     * Checks if the required permissions for bot to work properly are present,\n     * and emits a warning if that is not the case.\n     */\n    private async checkPermissions(): Promise<string> {\n        // Check if the token has the required permissions\n        const response = await this.scheduleCall('groups.getTokenPermissions', {});\n\n        const { permissions } = response;\n\n        let ok = false;\n        permissions.forEach((permission: any): void => { // eslint-disable-line @typescript-eslint/no-explicit-any\n            if (permission.name === 'messages') {\n                ok = true;\n            }\n        });\n\n        if (!ok) {\n            return Promise.reject(\n                new Error(\n                    'Token permission `messages` is missing. Bot will be unable to send any messages',\n                ),\n            );\n        }\n        return Promise.resolve('Token permission `messages` is present');\n    }\n\n    /**\n     * Move forward through the queue, processing at most [[API_QUOTA]] items\n     */\n    private async processQueue(): Promise<void> {\n        if (this.queue) {\n            for (let i = 1; i <= API_QUOTA; i += 1) {\n                if (this.queue.length === 0) {\n                    break;\n                }\n\n                const e = this.queue.shift();\n\n                /* eslint-disable-next-line no-await-in-loop */\n                const json = await this.call(e.method, e.params);\n\n                if (json.response !== undefined && json.response !== null) {\n                    e.resolve(json.response);\n                } else if (json.error) {\n                    const errorCode = json.error.error_code;\n                    const errorMsg = json.error.error_msg;\n\n                    e.reject(\n                        `An API call to method '${e.method}' failed due to an API error #${errorCode}: ${errorMsg}`,\n                    );\n                } else {\n                    e.reject(\n                        `An API call to method '${e.method}' failed due to an unknown API error. The API responded with: ${JSON.stringify(json)}`,\n                    );\n                }\n            }\n\n            return Promise.resolve();\n        }\n\n        return Promise.reject(new Error('No queue for API calls found'));\n    }\n}\n"]}