From 839428b949ed7da2a4e3fdcbdef0b75abc44c9ea Mon Sep 17 00:00:00 2001 From: MAmamiyaAhmad Date: Sat, 1 Feb 2025 18:51:57 +0700 Subject: [PATCH 01/11] New Update --- .env | 2 + .gitignore | 4 - LICENSE | 21 - Procfile | 1 - README.md | 39 -- bot.js | 261 ----------- package-lock.json | 834 +++++----------------------------- package.json | 36 +- src/bot.ts | 16 + src/config.ts | 4 + src/handlers/language.ts | 6 + src/handlers/start.ts | 26 ++ src/middlewares/middleware.ts | 20 + src/utils.ts | 32 ++ tsconfig.json | 0 15 files changed, 221 insertions(+), 1081 deletions(-) create mode 100644 .env delete mode 100644 .gitignore delete mode 100644 LICENSE delete mode 100644 Procfile delete mode 100644 bot.js create mode 100644 src/bot.ts create mode 100644 src/config.ts create mode 100644 src/handlers/language.ts create mode 100644 src/handlers/start.ts create mode 100644 src/middlewares/middleware.ts create mode 100644 src/utils.ts create mode 100644 tsconfig.json diff --git a/.env b/.env new file mode 100644 index 0000000..fefc2a5 --- /dev/null +++ b/.env @@ -0,0 +1,2 @@ +BOT_TOKEN=952920973:AAGFpuzir-eRj1Sx0R1m9uVT6SD4ml3gFqs +CHANNEL_ID=@bm_update diff --git a/.gitignore b/.gitignore deleted file mode 100644 index 1fc7361..0000000 --- a/.gitignore +++ /dev/null @@ -1,4 +0,0 @@ -.env -config.json -node_modules -*.png \ No newline at end of file diff --git a/LICENSE b/LICENSE deleted file mode 100644 index 76df676..0000000 --- a/LICENSE +++ /dev/null @@ -1,21 +0,0 @@ -MIT License - -Copyright (c) 2018 Abhinav Kasamsetty - -Permission is hereby granted, free of charge, to any person obtaining a copy -of this software and associated documentation files (the "Software"), to deal -in the Software without restriction, including without limitation the rights -to use, copy, modify, merge, publish, distribute, sublicense, and/or sell -copies of the Software, and to permit persons to whom the Software is -furnished to do so, subject to the following conditions: - -The above copyright notice and this permission notice shall be included in all -copies or substantial portions of the Software. - -THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR -IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, -FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE -AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER -LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, -OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE -SOFTWARE. diff --git a/Procfile b/Procfile deleted file mode 100644 index 8753747..0000000 --- a/Procfile +++ /dev/null @@ -1 +0,0 @@ -worker: node bot.js \ No newline at end of file diff --git a/README.md b/README.md index a51bcf4..e69de29 100644 --- a/README.md +++ b/README.md @@ -1,39 +0,0 @@ -# crypto-bot - -Telegram bot that uses the CoinMarketCap API and the Binance API to get information on cryptocurrencies and the global cryptocurrency market. - -Talk to @coinmarketcap_info_bot on Telegram! - -## Installation -1. Clone the repo to your computer and install the required dependencies. - ```console - git clone https://github.com/abhinavk99/crypto-bot.git - cd crypto-bot - npm install - ``` -2. Get Binance API key and secret from your Binance user settings. -3. Get a Telegram bot token from [@BotFather](https://t.me/BotFather). -4. Make a file called `.env` in the repo directory. -5. Copy/paste the below into the file. - ``` - TELEGRAM_TOKEN=Token here - BINANCE_KEY=Key here - BINANCE_SECRET=Secret here - ``` -6. Put your Telegram token and Binance API key and secret where it says to in the config. Do not put quotes around the tokens. -7. Run the bot. - ```console - node bot.js - ``` - -## Commands -| Command | Description | Usage | Example | -| --- | --- | --- | --- | -| /info | Get info on coin with that name | /info `` | `/info Bitcoin` | -| /info | Get info on coin with that CoinMarketCap rank | /info `` | `/info 11` | -| /global | Get total market information | /global | `/global` | -| / | Get latest Binance ticker price for a coin | /`` | `/ETH` | -| /chart | Get chart on historical price for a coin | /chart `` | `/chart Bitcoin` | - -## Releases -Releases start at v1.1.0 because I didn't know how to use git tags before then and manually changed the versions in package.json \ No newline at end of file diff --git a/bot.js b/bot.js deleted file mode 100644 index 01b1966..0000000 --- a/bot.js +++ /dev/null @@ -1,261 +0,0 @@ -require('dotenv').config(); -const TeleBot = require('telebot'); -const binance = require('node-binance-api'); -const fetch = require('node-fetch'); -const webshot = require('webshot'); -const fs = require('fs'); - -const bot = new TeleBot(process.env.TELEGRAM_TOKEN); -binance.options({ - 'APIKEY': process.env.BINANCE_KEY, - 'APISECRET': process.env.BINANCE_SECRET -}); - -// 10 API calls a minute are allowed -var calls = 0; - -// Base CoinMarketCap API url -const baseUrl = 'https://api.coinmarketcap.com/v1/'; - -// Caching CMC data to account for repeated calls -var cache = {}; - -// Caching Binance ticker data and time of most recent call -var bin = []; - -// Constants for bot error message responses -const tooMuch = 'You\'re using the bot too much!'; // Prevent overuse of bot calls -const noCurrency = 'No currency found with that name.'; // Currency not found in /info -const notNumber = 'A ticker can\'t be a number.'; // Ticker input was a number in / -const noTicker = 'Ticker not found.' // Ticker not found in / - -bot.on('/start', (msg) => { - msg.reply.text('/info for information on the coin with that name\n' - + '/info for information on the coin with that rank\n' - + '/global for total market information\n' - + '/ for latest Binance ticker price\n' - + '/chart for chart on historical prices on the coin with that name'); -}); - -// Ticker information from CoinMarketCap -bot.on(/^\/info (.+)$/i, (msg, props) => { - if (calls > 10) { - return msg.reply.text(tooMuch, {asReply: true}); - } - calls++; - var text = props.match[1].substring(5); - // Checks if the same argument has been passed into the command in the last 5 minutes - if (text in cache && Math.floor((new Date() - - new Date(parseInt(cache[text]['last_updated']) * 1000)) / - 60000 % 60) < 5) { - return msg.reply.text(formatInfo(cache[text]), {asReply: true}); - } else { - if (isNaN(text)) { - // Bot replies with information on the currency if found - fetch(baseUrl + 'ticker/' + text.toLowerCase() + '/').then((res) => { - return res.json(); - }).then((info) => { - // If currency found, info is a JS object wrapped in an array - // If not found, info is just a JS object - console.log(info); - // Info[0] is a JS object if currency found, otherwise it is undefined - if (info[0]) { - cache[text] = info[0]; - return msg.reply.text(formatInfo(info[0]), {asReply: true}); - } else { - return msg.reply.text(noCurrency, {asReply: true}); - } - }); - } else { - fetch(baseUrl + 'ticker/?limit=' + text).then((res) => { - return res.json(); - }).then((info) => { - // Info is an array of JS objects - console.log(info); - cache['global'] = info[parseInt(text) - 1]; - return msg.reply.text(formatInfo(info[parseInt(text) - 1]), - {asReply: true}); - }); - } - } -}); - -// Total market information from CoinMarketCap -bot.on('/global', (msg) => { - // Current time - var d = new Date(); - if (calls > 10) { - return msg.reply.text(tooMuch, {asReply: true}); - } - calls++; - // Checks if global command has been called in last 5 minutes - if ('global' in cache && Math.floor((new Date() - - new Date(parseInt(cache['global']['last_updated']) * 1000)) / - 60000 % 60) < 5) { - return msg.reply.text(formatGlobalInfo(cache['global']), {asReply: true}); - } - fetch(baseUrl + '/global/').then((res) => { - return res.json(); - }).then((info) => { - // Info is a JS object - console.log(info); - cache['global'] = info; - return msg.reply.text(formatGlobalInfo(info), {asReply: true}); - }); -}); - -// Latest exchange price from Binance -bot.on(/^\/(.+)$/i, (msg, props) => { - var text = props.match[1].toLowerCase(); - console.log(props); - // Accounts for not responding to one of the other commands - if (!text.startsWith('global') - && !text.startsWith('info') - && !props.match[0].startsWith('/chart') - && /^[a-zA-Z]+$/.test(text) - && text.length < 5) { - if (calls > 10) { - return msg.reply.text(tooMuch, {asReply: true}); - } - calls++; - // Checks if command has been called in the past 5 minutes - if (bin[0] !== undefined && Math.floor((new Date() - bin[1]) / 60000 % 60) < 5) { - return msg.reply.text(formatBinanceInfo(bin[0], text.toUpperCase()), - {asReply: true}); - } else { - if (isNaN(text)) { - binance.prices((ticker) => { - bin[1] = new Date(); - bin[0] = ticker; - console.log('Called Binance API'); - return msg.reply.text(formatBinanceInfo(ticker, text.toUpperCase()), - {asReply: true}); - }); - } else { - return msg.reply.text(notNumber, {asReply: true}); - } - } - } -}); - -bot.on(/^\/chart (.+)$/i, (msg, props) => { - return msg.reply.text('Deprecated', {asReply: true}); - - if (calls > 10) { - return msg.reply.text(tooMuch, { asReply: true }); - } - calls++; - var text = props.match[1].toLowerCase(); - - webshot(`https://coinmarketcap.com/currencies/${text}/#charts`, `${text}.png`, { - shotSize: { width: 'window', height: 630 }, - shotOffset: { left: 35, right: 70, top: 690, bottom: 0 } - }, err => { - return msg.reply.photo(`${text}.png`, {asReply: true}); - }); -}); - -bot.start(); - -// Formats the output of the json for better readability -function formatInfo(info) { - var output = info['name'] + ' (' + info['symbol'] + ')\n'; - output += ('CoinMarketCap ID: ' + info['id'] + '\n') - output += ('CoinMarketCap Rank: ' + info['rank'] + '\n'); - output += ('https://coinmarketcap.com/currencies/' + info['id'] + '/\n\n'); - - output += ('Price USD: $' + formatNum(info['price_usd']) + '\n'); - output += ('Price BTC: ' + info['price_btc'] + ' BTC\n\n'); - - output += ('Market Cap: $' + formatNum(info['market_cap_usd']) + '\n'); - output += ('24h Volume: $' + formatNum(info['24h_volume_usd']) + '\n'); - output += ('Available Supply: ' + formatNum(info['available_supply']) + '\n'); - output += ('Total Supply: ' + formatNum(info['total_supply'])+ '\n'); - if (info['max_supply']) { - output += ('Maximum Supply: ' + formatNum(info['max_supply']) + '\n'); - } - - output += ('\nChange 1h: ' + formatNum(info['percent_change_1h']) + '%\n'); - output += ('Change 24h: ' + formatNum(info['percent_change_24h']) + '%\n'); - output += ('Change 7d: ' + formatNum(info['percent_change_7d']) + '%\n\n'); - - return output + 'Last Updated: ' - + new Date(parseInt(info['last_updated']) * 1000).toString(); -} - -// Formats the output of the json for global CMC data -function formatGlobalInfo(info) { - var output = 'Total Market Cap: $' - + parseInt(info['total_market_cap_usd']).toLocaleString() + '\n'; - output += ('Total 24h Volume: $' - + parseInt(info['total_24h_volume_usd']).toLocaleString() + '\n'); - output += ('Bitcoin Percentage of Market Cap: ' - + info['bitcoin_percentage_of_market_cap'] + '%\n\n'); - - output += ('Number of Active Currencies: ' + info['active_currencies'] + '\n'); - output += ('Number of Active Assets: ' + info['active_assets'] + '\n'); - output += ('Number of Active Markets: ' + info['active_markets'] + '\n\n'); - - output += ('https://coinmarketcap.com/charts/' + '\n\n'); - - return output + 'Last Updated: ' - + new Date(parseInt(info['last_updated']) * 1000).toString(); -} - -// Formats the output for Binance exchange price -function formatBinanceInfo(ticker, text) { - // The keys in the ticker object are exchanges in format equivalent to VENETH - var tradingPairs = ['ETH', 'BTC', 'BNB']; - var output = ''; - tradingPairs.forEach((item) => { - var exc = text + item; - // Checks if there exists an exchange in the ticker with each of the pairs - if (ticker[exc]) { - output += (ticker[exc] + ' ' + exc.replace(text, text + '/') + ' '); - output += ('($' + formatBin(ticker[exc], ticker[item + 'USDT']) + ')\n'); - } - }); - if (ticker[text + 'USDT']) { - output += (formatNum(ticker[text + 'USDT']) + ' ' + text + '/USDT\n'); - } - return ((output == '') ? noTicker : output); -} - -// Formats number string -function formatNum(str) { - return parseFloat(str).toLocaleString(); -} - -// Formats the displayed estimated dollar price for Binance exchange values -function formatBin(price1, price2) { - return (parseFloat(price1) * parseFloat(price2)).toLocaleString(); -} - -// Resets number of calls to 0 every minute -resetNumCalls(); -setInterval(resetNumCalls, 60000); -function resetNumCalls() { - console.log('Resetting number of calls at ' + new Date().toString()); - calls = 0; -} - -// Resets the caches every 2 hours -resetCaches(); -setInterval(resetCaches, 7200000); -function resetCaches() { - console.log('Resetting caches at ' + new Date().toString()); - cache = {}; - bin = []; - // Deletes all files ending with .png - fs.readdir('./', (err, files) => { - if (err) throw err; - - for (const file of files) { - if (file.endsWith('.png')) { - fs.unlink(file, err => { - if (err) throw err; - }); - } - } - }); -} \ No newline at end of file diff --git a/package-lock.json b/package-lock.json index 9d3a5e7..bf4d760 100644 --- a/package-lock.json +++ b/package-lock.json @@ -1,761 +1,131 @@ { - "name": "crypto-bot", - "version": "1.2.0", - "lockfileVersion": 1, + "name": "TGBot", + "lockfileVersion": 3, "requires": true, - "dependencies": { - "ajv": { - "version": "5.5.2", - "resolved": "https://registry.npmjs.org/ajv/-/ajv-5.5.2.tgz", - "integrity": "sha1-c7Xuyj+rZT49P5Qis0GtQiBdyWU=", - "requires": { - "co": "4.6.0", - "fast-deep-equal": "1.0.0", - "fast-json-stable-stringify": "2.0.0", - "json-schema-traverse": "0.3.1" - } - }, - "asn1": { - "version": "0.2.3", - "resolved": "https://registry.npmjs.org/asn1/-/asn1-0.2.3.tgz", - "integrity": "sha1-2sh4dxPJlmhJ/IGAd36+nB3fO4Y=" - }, - "assert-plus": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/assert-plus/-/assert-plus-1.0.0.tgz", - "integrity": "sha1-8S4PPF13sLHN2RRpQuTpbB5N1SU=" - }, - "async-limiter": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/async-limiter/-/async-limiter-1.0.0.tgz", - "integrity": "sha512-jp/uFnooOiO+L211eZOoSyzpOITMXx1rBITauYykG3BRYPu8h0UcxsPNB04RR5vo4Tyz3+ay17tR6JVf9qzYWg==" - }, - "asynckit": { - "version": "0.4.0", - "resolved": "https://registry.npmjs.org/asynckit/-/asynckit-0.4.0.tgz", - "integrity": "sha1-x57Zf380y48robyXkLzDZkdLS3k=" - }, - "aws-sign2": { - "version": "0.7.0", - "resolved": "https://registry.npmjs.org/aws-sign2/-/aws-sign2-0.7.0.tgz", - "integrity": "sha1-tG6JCTSpWR8tL2+G1+ap8bP+dqg=" - }, - "aws4": { - "version": "1.6.0", - "resolved": "https://registry.npmjs.org/aws4/-/aws4-1.6.0.tgz", - "integrity": "sha1-g+9cqGCysy5KDe7e6MdxudtXRx4=" - }, - "bcrypt-pbkdf": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/bcrypt-pbkdf/-/bcrypt-pbkdf-1.0.1.tgz", - "integrity": "sha1-Y7xdy2EzG5K8Bf1SiVPDNGKgb40=", - "optional": true, - "requires": { - "tweetnacl": "0.14.5" - } - }, - "boom": { - "version": "4.3.1", - "resolved": "https://registry.npmjs.org/boom/-/boom-4.3.1.tgz", - "integrity": "sha1-T4owBctKfjiJ90kDD9JbluAdLjE=", - "requires": { - "hoek": "4.2.0" - } - }, - "caseless": { - "version": "0.12.0", - "resolved": "https://registry.npmjs.org/caseless/-/caseless-0.12.0.tgz", - "integrity": "sha1-G2gcIf+EAzyCZUMJBolCDRhxUdw=" - }, - "co": { - "version": "4.6.0", - "resolved": "https://registry.npmjs.org/co/-/co-4.6.0.tgz", - "integrity": "sha1-bqa989hTrlTMuOR7+gvz+QMfsYQ=" - }, - "combined-stream": { - "version": "1.0.5", - "resolved": "https://registry.npmjs.org/combined-stream/-/combined-stream-1.0.5.tgz", - "integrity": "sha1-k4NwpXtKUd6ix3wV1cX9+JUWQAk=", - "requires": { - "delayed-stream": "1.0.0" - } - }, - "concat-stream": { - "version": "1.6.0", - "resolved": "https://registry.npmjs.org/concat-stream/-/concat-stream-1.6.0.tgz", - "integrity": "sha1-CqxmL9Ur54lk1VMvaUeE5wEQrPc=", - "optional": true, - "requires": { - "inherits": "2.0.3", - "readable-stream": "2.3.6", - "typedarray": "0.0.6" - } - }, - "core-util-is": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/core-util-is/-/core-util-is-1.0.2.tgz", - "integrity": "sha1-tf1UIgqivFq1eqtxQMlAdUUDwac=" - }, - "cross-spawn": { - "version": "0.2.9", - "resolved": "https://registry.npmjs.org/cross-spawn/-/cross-spawn-0.2.9.tgz", - "integrity": "sha1-vWf5bAfvtjA7f+lMHpefiEeOCjk=", - "requires": { - "lru-cache": "2.7.3" - } - }, - "cryptiles": { - "version": "3.1.2", - "resolved": "https://registry.npmjs.org/cryptiles/-/cryptiles-3.1.2.tgz", - "integrity": "sha1-qJ+7Ig9c4l7FboxKqKT9e1sNKf4=", - "requires": { - "boom": "5.2.0" - }, + "packages": { + "": { "dependencies": { - "boom": { - "version": "5.2.0", - "resolved": "https://registry.npmjs.org/boom/-/boom-5.2.0.tgz", - "integrity": "sha512-Z5BTk6ZRe4tXXQlkqftmsAUANpXmuwlsF5Oov8ThoMbQRzdGTA1ngYRW160GexgOgjsFOKJz0LYhoNi+2AMBUw==", - "requires": { - "hoek": "4.2.0" - } - } - } - }, - "dashdash": { - "version": "1.14.1", - "resolved": "https://registry.npmjs.org/dashdash/-/dashdash-1.14.1.tgz", - "integrity": "sha1-hTz6D3y+L+1d4gMmuN1YEDX24vA=", - "requires": { - "assert-plus": "1.0.0" - } - }, - "debug": { - "version": "2.6.9", - "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz", - "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==", - "optional": true, - "requires": { - "ms": "2.0.0" - } - }, - "delayed-stream": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/delayed-stream/-/delayed-stream-1.0.0.tgz", - "integrity": "sha1-3zrhmayt+31ECqrgsp4icrJOxhk=" - }, - "dotenv": { - "version": "5.0.1", - "resolved": "https://registry.npmjs.org/dotenv/-/dotenv-5.0.1.tgz", - "integrity": "sha512-4As8uPrjfwb7VXC+WnLCbXK7y+Ueb2B3zgNCePYfhxS1PYeaO1YTeplffTEcbfLhvFNGLAz90VvJs9yomG7bow==" - }, - "ecc-jsbn": { - "version": "0.1.1", - "resolved": "https://registry.npmjs.org/ecc-jsbn/-/ecc-jsbn-0.1.1.tgz", - "integrity": "sha1-D8c6ntXw1Tw4GTOYUj735UN3dQU=", - "optional": true, - "requires": { - "jsbn": "0.1.1" - } - }, - "es6-promise": { - "version": "4.2.4", - "resolved": "https://registry.npmjs.org/es6-promise/-/es6-promise-4.2.4.tgz", - "integrity": "sha512-/NdNZVJg+uZgtm9eS3O6lrOLYmQag2DjdEXuPaHlZ6RuVqgqaVZfgYCepEIKsLqwdQArOPtC3XzRLqGGfT8KQQ==", - "optional": true - }, - "extend": { - "version": "3.0.1", - "resolved": "https://registry.npmjs.org/extend/-/extend-3.0.1.tgz", - "integrity": "sha1-p1Xqe8Gt/MWjHOfnYtuq3F5jZEQ=" - }, - "extract-zip": { - "version": "1.6.6", - "resolved": "https://registry.npmjs.org/extract-zip/-/extract-zip-1.6.6.tgz", - "integrity": "sha1-EpDt6NINCHK0Kf0/NRyhKOxe+Fw=", - "optional": true, - "requires": { - "concat-stream": "1.6.0", - "debug": "2.6.9", - "mkdirp": "0.5.0", - "yauzl": "2.4.1" + "dotenv": "^16.4.7", + "grammy": "^1.34.1" } }, - "extsprintf": { - "version": "1.3.0", - "resolved": "https://registry.npmjs.org/extsprintf/-/extsprintf-1.3.0.tgz", - "integrity": "sha1-lpGEQOMEGnpBT4xS48V06zw+HgU=" - }, - "fast-deep-equal": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/fast-deep-equal/-/fast-deep-equal-1.0.0.tgz", - "integrity": "sha1-liVqO8l1WV6zbYLpkp0GDYk0Of8=" - }, - "fast-json-stable-stringify": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/fast-json-stable-stringify/-/fast-json-stable-stringify-2.0.0.tgz", - "integrity": "sha1-1RQsDK7msRifh9OnYREGT4bIu/I=" - }, - "fd-slicer": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/fd-slicer/-/fd-slicer-1.0.1.tgz", - "integrity": "sha1-i1vL2ewyfFBBv5qwI/1nUPEXfmU=", - "optional": true, - "requires": { - "pend": "1.2.0" - } + "node_modules/@grammyjs/types": { + "version": "3.18.0", + "resolved": "https://registry.npmjs.org/@grammyjs/types/-/types-3.18.0.tgz", + "integrity": "sha512-/VpQcT6Z0+Lw6H6jv2JvnDmqB1/9rjbFdDBlPBPUW3kQ1bWfjzfaiVk2+J2eiid3o3sLaPp3ZK0XjEI1gWNc3g==", + "license": "MIT" }, - "forever-agent": { - "version": "0.6.1", - "resolved": "https://registry.npmjs.org/forever-agent/-/forever-agent-0.6.1.tgz", - "integrity": "sha1-+8cfDEGt6zf5bFd60e1C2P2sypE=" - }, - "form-data": { - "version": "2.3.1", - "resolved": "https://registry.npmjs.org/form-data/-/form-data-2.3.1.tgz", - "integrity": "sha1-b7lPvXGIUwbXPRXMSX/kzE7NRL8=", - "requires": { - "asynckit": "0.4.0", - "combined-stream": "1.0.5", - "mime-types": "2.1.17" + "node_modules/abort-controller": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/abort-controller/-/abort-controller-3.0.0.tgz", + "integrity": "sha512-h8lQ8tacZYnR3vNQTgibj+tODHI5/+l06Au2Pcriv/Gmet0eaj4TwWH41sO9wnHDiQsEj19q0drzdWdeAHtweg==", + "license": "MIT", + "dependencies": { + "event-target-shim": "^5.0.0" + }, + "engines": { + "node": ">=6.5" } }, - "fs": { - "version": "0.0.1-security", - "resolved": "https://registry.npmjs.org/fs/-/fs-0.0.1-security.tgz", - "integrity": "sha1-invTcYa23d84E/I4WLV+yq9eQdQ=" - }, - "fs-extra": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/fs-extra/-/fs-extra-1.0.0.tgz", - "integrity": "sha1-zTzl9+fLYUWIP8rjGR6Yd/hYeVA=", - "optional": true, - "requires": { - "graceful-fs": "4.1.11", - "jsonfile": "2.4.0", - "klaw": "1.3.1" - }, + "node_modules/debug": { + "version": "4.4.0", + "resolved": "https://registry.npmjs.org/debug/-/debug-4.4.0.tgz", + "integrity": "sha512-6WTZ/IxCY/T6BALoZHaE4ctp9xm+Z5kY/pzYaCHRFeyVhojxlrm+46y68HA6hr0TcwEssoxNiDEUJQjfPZ/RYA==", + "license": "MIT", "dependencies": { - "graceful-fs": { - "version": "4.1.11", - "resolved": "https://registry.npmjs.org/graceful-fs/-/graceful-fs-4.1.11.tgz", - "integrity": "sha1-Dovf5NHduIVNZOBOp8AOKgJuVlg=", + "ms": "^2.1.3" + }, + "engines": { + "node": ">=6.0" + }, + "peerDependenciesMeta": { + "supports-color": { "optional": true } } }, - "getpass": { - "version": "0.1.7", - "resolved": "https://registry.npmjs.org/getpass/-/getpass-0.1.7.tgz", - "integrity": "sha1-Xv+OPmhNVprkyysSgmBOi6YhSfo=", - "requires": { - "assert-plus": "1.0.0" - } - }, - "graceful-fs": { - "version": "3.0.11", - "resolved": "https://registry.npmjs.org/graceful-fs/-/graceful-fs-3.0.11.tgz", - "integrity": "sha1-dhPHeKGv6mLyXGMKCG1/Osu92Bg=", - "requires": { - "natives": "1.1.3" - } - }, - "har-schema": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/har-schema/-/har-schema-2.0.0.tgz", - "integrity": "sha1-qUwiJOvKwEeCoNkDVSHyRzW37JI=" - }, - "har-validator": { - "version": "5.0.3", - "resolved": "https://registry.npmjs.org/har-validator/-/har-validator-5.0.3.tgz", - "integrity": "sha1-ukAsJmGU8VlW7xXg/PJCmT9qff0=", - "requires": { - "ajv": "5.5.2", - "har-schema": "2.0.0" - } - }, - "hasha": { - "version": "2.2.0", - "resolved": "https://registry.npmjs.org/hasha/-/hasha-2.2.0.tgz", - "integrity": "sha1-eNfL/B5tZjA/55g3NlmEUXsvbuE=", - "optional": true, - "requires": { - "is-stream": "1.1.0", - "pinkie-promise": "2.0.1" - } - }, - "hawk": { - "version": "6.0.2", - "resolved": "https://registry.npmjs.org/hawk/-/hawk-6.0.2.tgz", - "integrity": "sha512-miowhl2+U7Qle4vdLqDdPt9m09K6yZhkLDTWGoUiUzrQCn+mHHSmfJgAyGaLRZbPmTqfFFjRV1QWCW0VWUJBbQ==", - "requires": { - "boom": "4.3.1", - "cryptiles": "3.1.2", - "hoek": "4.2.0", - "sntp": "2.1.0" - } - }, - "hoek": { - "version": "4.2.0", - "resolved": "https://registry.npmjs.org/hoek/-/hoek-4.2.0.tgz", - "integrity": "sha512-v0XCLxICi9nPfYrS9RL8HbYnXi9obYAeLbSP00BmnZwCK9+Ih9WOjoZ8YoHCoav2csqn4FOz4Orldsy2dmDwmQ==" - }, - "http-signature": { - "version": "1.2.0", - "resolved": "https://registry.npmjs.org/http-signature/-/http-signature-1.2.0.tgz", - "integrity": "sha1-muzZJRFHcvPZW2WmCruPfBj7rOE=", - "requires": { - "assert-plus": "1.0.0", - "jsprim": "1.4.1", - "sshpk": "1.13.1" + "node_modules/dotenv": { + "version": "16.4.7", + "resolved": "https://registry.npmjs.org/dotenv/-/dotenv-16.4.7.tgz", + "integrity": "sha512-47qPchRCykZC03FhkYAhrvwU4xDBFIj1QPqaarj6mdM/hgUzfPHcpkHJOn3mJAufFeeAxAzeGsr5X0M4k6fLZQ==", + "license": "BSD-2-Clause", + "engines": { + "node": ">=12" + }, + "funding": { + "url": "https://dotenvx.com" } }, - "inherits": { - "version": "2.0.3", - "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.3.tgz", - "integrity": "sha1-Yzwsg+PaQqUC9SRmAiSA9CCCYd4=" - }, - "is-stream": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/is-stream/-/is-stream-1.1.0.tgz", - "integrity": "sha1-EtSj3U5o4Lec6428hBc66A2RykQ=", - "optional": true - }, - "is-typedarray": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/is-typedarray/-/is-typedarray-1.0.0.tgz", - "integrity": "sha1-5HnICFjfDBsR3dppQPlgEfzaSpo=" - }, - "isarray": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/isarray/-/isarray-1.0.0.tgz", - "integrity": "sha1-u5NdSFgsuhaMBoNJV6VKPgcSTxE=", - "optional": true - }, - "isexe": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/isexe/-/isexe-2.0.0.tgz", - "integrity": "sha1-6PvzdNxVb/iUehDcsFctYz8s+hA=", - "optional": true - }, - "isstream": { - "version": "0.1.2", - "resolved": "https://registry.npmjs.org/isstream/-/isstream-0.1.2.tgz", - "integrity": "sha1-R+Y/evVa+m+S4VAOaQ64uFKcCZo=" - }, - "jsbn": { - "version": "0.1.1", - "resolved": "https://registry.npmjs.org/jsbn/-/jsbn-0.1.1.tgz", - "integrity": "sha1-peZUwuWi3rXyAdls77yoDA7y9RM=", - "optional": true - }, - "json-schema": { - "version": "0.2.3", - "resolved": "https://registry.npmjs.org/json-schema/-/json-schema-0.2.3.tgz", - "integrity": "sha1-tIDIkuWaLwWVTOcnvT8qTogvnhM=" - }, - "json-schema-traverse": { - "version": "0.3.1", - "resolved": "https://registry.npmjs.org/json-schema-traverse/-/json-schema-traverse-0.3.1.tgz", - "integrity": "sha1-NJptRMU6Ud6JtAgFxdXlm0F9M0A=" - }, - "json-stringify-safe": { + "node_modules/event-target-shim": { "version": "5.0.1", - "resolved": "https://registry.npmjs.org/json-stringify-safe/-/json-stringify-safe-5.0.1.tgz", - "integrity": "sha1-Epai1Y/UXxmg9s4B1lcB4sc1tus=" - }, - "jsonfile": { - "version": "2.4.0", - "resolved": "https://registry.npmjs.org/jsonfile/-/jsonfile-2.4.0.tgz", - "integrity": "sha1-NzaitCi4e72gzIO1P6PWM6NcKug=", - "optional": true, - "requires": { - "graceful-fs": "4.1.11" - }, + "resolved": "https://registry.npmjs.org/event-target-shim/-/event-target-shim-5.0.1.tgz", + "integrity": "sha512-i/2XbnSz/uxRCU6+NdVJgKWDTM427+MqYbkQzD321DuCQJUqOuJKIA0IM2+W2xtYHdKOmZ4dR6fExsd4SXL+WQ==", + "license": "MIT", + "engines": { + "node": ">=6" + } + }, + "node_modules/grammy": { + "version": "1.34.1", + "resolved": "https://registry.npmjs.org/grammy/-/grammy-1.34.1.tgz", + "integrity": "sha512-0ypt5d1bna8CqSANlDCg2ce1sx87oDm9gjyPRHWYpAgvjy3WYW9oEmMllAYCWCZ3vHDULW5x14jBMXai51ar0Q==", + "license": "MIT", "dependencies": { - "graceful-fs": { - "version": "4.1.11", - "resolved": "https://registry.npmjs.org/graceful-fs/-/graceful-fs-4.1.11.tgz", - "integrity": "sha1-Dovf5NHduIVNZOBOp8AOKgJuVlg=", - "optional": true - } - } - }, - "jsprim": { - "version": "1.4.1", - "resolved": "https://registry.npmjs.org/jsprim/-/jsprim-1.4.1.tgz", - "integrity": "sha1-MT5mvB5cwG5Di8G3SZwuXFastqI=", - "requires": { - "assert-plus": "1.0.0", - "extsprintf": "1.3.0", - "json-schema": "0.2.3", - "verror": "1.10.0" + "@grammyjs/types": "3.18.0", + "abort-controller": "^3.0.0", + "debug": "^4.3.4", + "node-fetch": "^2.7.0" + }, + "engines": { + "node": "^12.20.0 || >=14.13.1" } }, - "kew": { - "version": "0.7.0", - "resolved": "https://registry.npmjs.org/kew/-/kew-0.7.0.tgz", - "integrity": "sha1-edk9LTM2PW/dKXCzNdkUGtWR15s=", - "optional": true + "node_modules/ms": { + "version": "2.1.3", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.3.tgz", + "integrity": "sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA==", + "license": "MIT" }, - "klaw": { - "version": "1.3.1", - "resolved": "https://registry.npmjs.org/klaw/-/klaw-1.3.1.tgz", - "integrity": "sha1-QIhDO0azsbolnXh4XY6W9zugJDk=", - "optional": true, - "requires": { - "graceful-fs": "4.1.11" - }, + "node_modules/node-fetch": { + "version": "2.7.0", + "resolved": "https://registry.npmjs.org/node-fetch/-/node-fetch-2.7.0.tgz", + "integrity": "sha512-c4FRfUm/dbcWZ7U+1Wq0AwCyFL+3nt2bEw05wfxSz+DWpWsitgmSgYmy2dQdWyKC1694ELPqMs/YzUSNozLt8A==", + "license": "MIT", "dependencies": { - "graceful-fs": { - "version": "4.1.11", - "resolved": "https://registry.npmjs.org/graceful-fs/-/graceful-fs-4.1.11.tgz", - "integrity": "sha1-Dovf5NHduIVNZOBOp8AOKgJuVlg=", + "whatwg-url": "^5.0.0" + }, + "engines": { + "node": "4.x || >=6.0.0" + }, + "peerDependencies": { + "encoding": "^0.1.0" + }, + "peerDependenciesMeta": { + "encoding": { "optional": true } } }, - "lru-cache": { - "version": "2.7.3", - "resolved": "https://registry.npmjs.org/lru-cache/-/lru-cache-2.7.3.tgz", - "integrity": "sha1-bUUk6LlV+V1PW1iFHOId1y+06VI=" - }, - "mime-db": { - "version": "1.30.0", - "resolved": "https://registry.npmjs.org/mime-db/-/mime-db-1.30.0.tgz", - "integrity": "sha1-dMZD2i3Z1qRTmZY0ZbJtXKfXHwE=" - }, - "mime-types": { - "version": "2.1.17", - "resolved": "https://registry.npmjs.org/mime-types/-/mime-types-2.1.17.tgz", - "integrity": "sha1-Cdejk/A+mVp5+K+Fe3Cp4KsWVXo=", - "requires": { - "mime-db": "1.30.0" - } - }, - "minimist": { - "version": "0.0.8", - "resolved": "https://registry.npmjs.org/minimist/-/minimist-0.0.8.tgz", - "integrity": "sha1-hX/Kv8M5fSYluCKCYuhqp6ARsF0=", - "optional": true - }, - "mkdirp": { - "version": "0.5.0", - "resolved": "https://registry.npmjs.org/mkdirp/-/mkdirp-0.5.0.tgz", - "integrity": "sha1-HXMHam35hs2TROFecfzAWkyavxI=", - "optional": true, - "requires": { - "minimist": "0.0.8" - } - }, - "ms": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", - "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g=", - "optional": true - }, - "natives": { - "version": "1.1.3", - "resolved": "https://registry.npmjs.org/natives/-/natives-1.1.3.tgz", - "integrity": "sha512-BZGSYV4YOLxzoTK73l0/s/0sH9l8SHs2ocReMH1f8JYSh5FUWu4ZrKCpJdRkWXV6HFR/pZDz7bwWOVAY07q77g==" - }, - "node-binance-api": { - "version": "0.3.9", - "resolved": "https://registry.npmjs.org/node-binance-api/-/node-binance-api-0.3.9.tgz", - "integrity": "sha512-bOfFCKh2LGMeIxS2QjC10IhVivQ/lixto+5lMyYSJyVJ8ep5FHhJtx53MFHgQUNBKrKLyvTt/1/T4jpYmXiMNQ==", - "requires": { - "request": "2.83.0", - "ws": "3.3.3" - } - }, - "node-fetch": { - "version": "2.1.2", - "resolved": "https://registry.npmjs.org/node-fetch/-/node-fetch-2.1.2.tgz", - "integrity": "sha1-q4hOjn5X44qUR1POxwb3iNF2i7U=" - }, - "oauth-sign": { - "version": "0.8.2", - "resolved": "https://registry.npmjs.org/oauth-sign/-/oauth-sign-0.8.2.tgz", - "integrity": "sha1-Rqarfwrq2N6unsBWV4C31O/rnUM=" - }, - "os-tmpdir": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/os-tmpdir/-/os-tmpdir-1.0.2.tgz", - "integrity": "sha1-u+Z0BseaqFxc/sdm/lc0VV36EnQ=" - }, - "pend": { - "version": "1.2.0", - "resolved": "https://registry.npmjs.org/pend/-/pend-1.2.0.tgz", - "integrity": "sha1-elfrVQpng/kRUzH89GY9XI4AelA=", - "optional": true - }, - "performance-now": { - "version": "2.1.0", - "resolved": "https://registry.npmjs.org/performance-now/-/performance-now-2.1.0.tgz", - "integrity": "sha1-Ywn04OX6kT7BxpMHrjZLSzd8nns=" - }, - "phantomjs-prebuilt": { - "version": "2.1.16", - "resolved": "https://registry.npmjs.org/phantomjs-prebuilt/-/phantomjs-prebuilt-2.1.16.tgz", - "integrity": "sha1-79ISpKOWbTZHaE6ouniFSb4q7+8=", - "optional": true, - "requires": { - "es6-promise": "4.2.4", - "extract-zip": "1.6.6", - "fs-extra": "1.0.0", - "hasha": "2.2.0", - "kew": "0.7.0", - "progress": "1.1.8", - "request": "2.83.0", - "request-progress": "2.0.1", - "which": "1.3.0" - } - }, - "pinkie": { - "version": "2.0.4", - "resolved": "https://registry.npmjs.org/pinkie/-/pinkie-2.0.4.tgz", - "integrity": "sha1-clVrgM+g1IqXToDnckjoDtT3+HA=", - "optional": true + "node_modules/tr46": { + "version": "0.0.3", + "resolved": "https://registry.npmjs.org/tr46/-/tr46-0.0.3.tgz", + "integrity": "sha512-N3WMsuqV66lT30CrXNbEjx4GEwlow3v6rr4mCcv6prnfwhS01rkgyFdjPNBYd9br7LpXV1+Emh01fHnq2Gdgrw==", + "license": "MIT" }, - "pinkie-promise": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/pinkie-promise/-/pinkie-promise-2.0.1.tgz", - "integrity": "sha1-ITXW36ejWMBprJsXh3YogihFD/o=", - "optional": true, - "requires": { - "pinkie": "2.0.4" - } - }, - "process-nextick-args": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/process-nextick-args/-/process-nextick-args-2.0.0.tgz", - "integrity": "sha512-MtEC1TqN0EU5nephaJ4rAtThHtC86dNN9qCuEhtshvpVBkAW5ZO7BASN9REnF9eoXGcRub+pFuKEpOHE+HbEMw==", - "optional": true - }, - "progress": { - "version": "1.1.8", - "resolved": "https://registry.npmjs.org/progress/-/progress-1.1.8.tgz", - "integrity": "sha1-4mDHj2Fhzdmw5WzD4Khd4Xx6V74=", - "optional": true - }, - "punycode": { - "version": "1.4.1", - "resolved": "https://registry.npmjs.org/punycode/-/punycode-1.4.1.tgz", - "integrity": "sha1-wNWmOycYgArY4esPpSachN1BhF4=" - }, - "qs": { - "version": "6.5.1", - "resolved": "https://registry.npmjs.org/qs/-/qs-6.5.1.tgz", - "integrity": "sha512-eRzhrN1WSINYCDCbrz796z37LOe3m5tmW7RQf6oBntukAG1nmovJvhnwHHRMAfeoItc1m2Hk02WER2aQ/iqs+A==" - }, - "readable-stream": { - "version": "2.3.6", - "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-2.3.6.tgz", - "integrity": "sha512-tQtKA9WIAhBF3+VLAseyMqZeBjW0AHJoxOtYqSUZNJxauErmLbVm2FW1y+J/YA9dUrAC39ITejlZWhVIwawkKw==", - "optional": true, - "requires": { - "core-util-is": "1.0.2", - "inherits": "2.0.3", - "isarray": "1.0.0", - "process-nextick-args": "2.0.0", - "safe-buffer": "5.1.1", - "string_decoder": "1.1.1", - "util-deprecate": "1.0.2" - } - }, - "request": { - "version": "2.83.0", - "resolved": "https://registry.npmjs.org/request/-/request-2.83.0.tgz", - "integrity": "sha512-lR3gD69osqm6EYLk9wB/G1W/laGWjzH90t1vEa2xuxHD5KUrSzp9pUSfTm+YC5Nxt2T8nMPEvKlhbQayU7bgFw==", - "requires": { - "aws-sign2": "0.7.0", - "aws4": "1.6.0", - "caseless": "0.12.0", - "combined-stream": "1.0.5", - "extend": "3.0.1", - "forever-agent": "0.6.1", - "form-data": "2.3.1", - "har-validator": "5.0.3", - "hawk": "6.0.2", - "http-signature": "1.2.0", - "is-typedarray": "1.0.0", - "isstream": "0.1.2", - "json-stringify-safe": "5.0.1", - "mime-types": "2.1.17", - "oauth-sign": "0.8.2", - "performance-now": "2.1.0", - "qs": "6.5.1", - "safe-buffer": "5.1.1", - "stringstream": "0.0.5", - "tough-cookie": "2.3.3", - "tunnel-agent": "0.6.0", - "uuid": "3.1.0" - } - }, - "request-progress": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/request-progress/-/request-progress-2.0.1.tgz", - "integrity": "sha1-XTa7V5YcZzqlt4jbyBQf3yO0Tgg=", - "optional": true, - "requires": { - "throttleit": "1.0.0" - } - }, - "safe-buffer": { - "version": "5.1.1", - "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.1.1.tgz", - "integrity": "sha512-kKvNJn6Mm93gAczWVJg7wH+wGYWNrDHdWvpUmHyEsgCtIwwo3bqPtV4tR5tuPaUhTOo/kvhVwd8XwwOllGYkbg==" - }, - "sntp": { - "version": "2.1.0", - "resolved": "https://registry.npmjs.org/sntp/-/sntp-2.1.0.tgz", - "integrity": "sha512-FL1b58BDrqS3A11lJ0zEdnJ3UOKqVxawAkF3k7F0CVN7VQ34aZrV+G8BZ1WC9ZL7NyrwsW0oviwsWDgRuVYtJg==", - "requires": { - "hoek": "4.2.0" - } - }, - "sshpk": { - "version": "1.13.1", - "resolved": "https://registry.npmjs.org/sshpk/-/sshpk-1.13.1.tgz", - "integrity": "sha1-US322mKHFEMW3EwY/hzx2UBzm+M=", - "requires": { - "asn1": "0.2.3", - "assert-plus": "1.0.0", - "bcrypt-pbkdf": "1.0.1", - "dashdash": "1.14.1", - "ecc-jsbn": "0.1.1", - "getpass": "0.1.7", - "jsbn": "0.1.1", - "tweetnacl": "0.14.5" - } - }, - "string_decoder": { - "version": "1.1.1", - "resolved": "https://registry.npmjs.org/string_decoder/-/string_decoder-1.1.1.tgz", - "integrity": "sha512-n/ShnvDi6FHbbVfviro+WojiFzv+s8MPMHBczVePfUpDJLwoLT0ht1l4YwBCbi8pJAveEEdnkHyPyTP/mzRfwg==", - "optional": true, - "requires": { - "safe-buffer": "5.1.1" - } - }, - "stringstream": { - "version": "0.0.5", - "resolved": "https://registry.npmjs.org/stringstream/-/stringstream-0.0.5.tgz", - "integrity": "sha1-TkhM1N5aC7vuGORjB3EKioFiGHg=" - }, - "telebot": { - "version": "1.2.3", - "resolved": "https://registry.npmjs.org/telebot/-/telebot-1.2.3.tgz", - "integrity": "sha512-7boCJohw6Y/V5WuNH2HlNXxHTg0jPP27zMNhCsxJh0NaMWbayPDNs1it2W4OyGp1C7NZtJVitQG12FoIlh10/g==", - "requires": { - "request": "2.83.0" - } - }, - "throttleit": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/throttleit/-/throttleit-1.0.0.tgz", - "integrity": "sha1-nnhYNtr0Z0MUWlmEtiaNgoUorGw=", - "optional": true - }, - "tmp": { - "version": "0.0.33", - "resolved": "https://registry.npmjs.org/tmp/-/tmp-0.0.33.tgz", - "integrity": "sha512-jRCJlojKnZ3addtTOjdIqoRuPEKBvNXcGYqzO6zWZX8KfKEpnGY5jfggJQ3EjKuu8D4bJRr0y+cYJFmYbImXGw==", - "requires": { - "os-tmpdir": "1.0.2" - } - }, - "tough-cookie": { - "version": "2.3.3", - "resolved": "https://registry.npmjs.org/tough-cookie/-/tough-cookie-2.3.3.tgz", - "integrity": "sha1-C2GKVWW23qkL80JdBNVe3EdadWE=", - "requires": { - "punycode": "1.4.1" - } - }, - "tunnel-agent": { - "version": "0.6.0", - "resolved": "https://registry.npmjs.org/tunnel-agent/-/tunnel-agent-0.6.0.tgz", - "integrity": "sha1-J6XeoGs2sEoKmWZ3SykIaPD8QP0=", - "requires": { - "safe-buffer": "5.1.1" - } - }, - "tweetnacl": { - "version": "0.14.5", - "resolved": "https://registry.npmjs.org/tweetnacl/-/tweetnacl-0.14.5.tgz", - "integrity": "sha1-WuaBd/GS1EViadEIr6k/+HQ/T2Q=", - "optional": true - }, - "typedarray": { - "version": "0.0.6", - "resolved": "https://registry.npmjs.org/typedarray/-/typedarray-0.0.6.tgz", - "integrity": "sha1-hnrHTjhkGHsdPUfZlqeOxciDB3c=", - "optional": true - }, - "ultron": { - "version": "1.1.1", - "resolved": "https://registry.npmjs.org/ultron/-/ultron-1.1.1.tgz", - "integrity": "sha512-UIEXBNeYmKptWH6z8ZnqTeS8fV74zG0/eRU9VGkpzz+LIJNs8W/zM/L+7ctCkRrgbNnnR0xxw4bKOr0cW0N0Og==" - }, - "util-deprecate": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/util-deprecate/-/util-deprecate-1.0.2.tgz", - "integrity": "sha1-RQ1Nyfpw3nMnYvvS1KKJgUGaDM8=", - "optional": true - }, - "uuid": { - "version": "3.1.0", - "resolved": "https://registry.npmjs.org/uuid/-/uuid-3.1.0.tgz", - "integrity": "sha512-DIWtzUkw04M4k3bf1IcpS2tngXEL26YUD2M0tMDUpnUrz2hgzUBlD55a4FjdLGPvfHxS6uluGWvaVEqgBcVa+g==" - }, - "verror": { - "version": "1.10.0", - "resolved": "https://registry.npmjs.org/verror/-/verror-1.10.0.tgz", - "integrity": "sha1-OhBcoXBTr1XW4nDB+CiGguGNpAA=", - "requires": { - "assert-plus": "1.0.0", - "core-util-is": "1.0.2", - "extsprintf": "1.3.0" - } - }, - "webshot": { - "version": "0.18.0", - "resolved": "https://registry.npmjs.org/webshot/-/webshot-0.18.0.tgz", - "integrity": "sha1-BX5pJbw5cK6X7tVvwjEYV4y8/cM=", - "requires": { - "cross-spawn": "0.2.9", - "graceful-fs": "3.0.11", - "phantomjs-prebuilt": "2.1.16", - "tmp": "0.0.33" - } - }, - "which": { - "version": "1.3.0", - "resolved": "https://registry.npmjs.org/which/-/which-1.3.0.tgz", - "integrity": "sha512-xcJpopdamTuY5duC/KnTTNBraPK54YwpenP4lzxU8H91GudWpFv38u0CKjclE1Wi2EH2EDz5LRcHcKbCIzqGyg==", - "optional": true, - "requires": { - "isexe": "2.0.0" - } - }, - "ws": { - "version": "3.3.3", - "resolved": "https://registry.npmjs.org/ws/-/ws-3.3.3.tgz", - "integrity": "sha512-nnWLa/NwZSt4KQJu51MYlCcSQ5g7INpOrOMt4XV8j4dqTXdmlUmSHQ8/oLC069ckre0fRsgfvsKwbTdtKLCDkA==", - "requires": { - "async-limiter": "1.0.0", - "safe-buffer": "5.1.1", - "ultron": "1.1.1" - } - }, - "yauzl": { - "version": "2.4.1", - "resolved": "https://registry.npmjs.org/yauzl/-/yauzl-2.4.1.tgz", - "integrity": "sha1-lSj0QtqxsihOWLQ3m7GU4i4MQAU=", - "optional": true, - "requires": { - "fd-slicer": "1.0.1" + "node_modules/webidl-conversions": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/webidl-conversions/-/webidl-conversions-3.0.1.tgz", + "integrity": "sha512-2JAn3z8AR6rjK8Sm8orRC0h/bcl/DqL7tRPdGZ4I1CjdF+EaMLmYxBHyXuKL849eucPFhvBoxMsflfOb8kxaeQ==", + "license": "BSD-2-Clause" + }, + "node_modules/whatwg-url": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/whatwg-url/-/whatwg-url-5.0.0.tgz", + "integrity": "sha512-saE57nupxk6v3HY35+jzBwYa0rKSy0XR8JSxZPwgLr7ys0IBzhGviA1/TUGJLmSVqs8pb9AnvICXEuOHLprYTw==", + "license": "MIT", + "dependencies": { + "tr46": "~0.0.3", + "webidl-conversions": "^3.0.0" } } } diff --git a/package.json b/package.json index d0b59fc..6b4e138 100644 --- a/package.json +++ b/package.json @@ -1,29 +1,19 @@ { - "name": "crypto-bot", - "description": "Telegram bot that gets information on cryptocurrencies", - "version": "1.2.0", - "main": "bot.js", + + "name": "tgbot", + "version": "1.0.0", + "description": "", + "main": "index.js", + "devDependencies": {}, "scripts": { - "start": "node bot.js", "test": "echo \"Error: no test specified\" && exit 1" }, + "keywords": [], + "author": "", + "license": "ISC", + "type": "commonjs", "dependencies": { - "dotenv": "^5.0.1", - "fs": "0.0.1-security", - "node-binance-api": "^0.3.0", - "node-fetch": "^2.1.2", - "telebot": "^1.2.3", - "webshot": "^0.18.0" - }, - "devDependencies": {}, - "repository": { - "type": "git", - "url": "git+https://github.com/abhinavk99/crypto-bot.git" - }, - "author": "Abhinav Kasamsetty (http://github.com/abhinavk99)", - "license": "MIT", - "bugs": { - "url": "https://github.com/abhinavk99/crypto-bot/issues" - }, - "homepage": "https://github.com/abhinavk99/crypto-bot#readme" + "dotenv": "^16.4.7", + "grammy": "^1.34.1" + } } diff --git a/src/bot.ts b/src/bot.ts new file mode 100644 index 0000000..474c745 --- /dev/null +++ b/src/bot.ts @@ -0,0 +1,16 @@ +import { Bot } from "grammy"; +import { BOT_TOKEN } from "./config"; +import { startHandler } from "./handlers/start"; +import { languageHandler } from "./handlers/language"; + +const bot = new Bot(BOT_TOKEN); + +// Command /start dengan auto-detect bot username +bot.command("start", async (ctx) => startHandler(ctx, bot)); + +// Handle pilihan bahasa +bot.callbackQuery(/.*/, languageHandler); + +// Jalankan bot +bot.start(); +console.log("🤖 Bot is running..."); diff --git a/src/config.ts b/src/config.ts new file mode 100644 index 0000000..ab33cb5 --- /dev/null +++ b/src/config.ts @@ -0,0 +1,4 @@ +import * as dotenv from "dotenv"; +dotenv.config(); + +export const BOT_TOKEN = process.env.BOT_TOKEN as string; diff --git a/src/handlers/language.ts b/src/handlers/language.ts new file mode 100644 index 0000000..bec38b7 --- /dev/null +++ b/src/handlers/language.ts @@ -0,0 +1,6 @@ +import { Context } from "grammy"; + +export async function languageHandler(ctx: Context) { + const selectedLanguage = ctx.match; + await ctx.reply(`✅ You have selected: ${selectedLanguage}`); +} diff --git a/src/handlers/start.ts b/src/handlers/start.ts new file mode 100644 index 0000000..4eec29c --- /dev/null +++ b/src/handlers/start.ts @@ -0,0 +1,26 @@ +import { Context, Bot } from "grammy"; +import { getLanguageKeyboard, getBotUsername } from "../utils"; + +export async function startHandler(ctx: Context, bot: Bot) { + const userId = ctx.from?.id; + const firstName = ctx.from?.first_name; + const username = ctx.from?.username || "Tidak Ada"; + + // Ambil nama bot secara otomatis + const botUsername = await getBotUsername(bot); + + const message = `🌏 Choose Language\n` + + `•❂•─•─•❂•─•❂••❂•─•❂•─•─•❂•\n\n` + + `🆔 Name: ${firstName}\n` + + ` \u251c Username: @${username}\n` + + ` \u251c ID Telegram: ${userId}\n` + + ` \u2514 Link: Click Here\n\n` + + `👁‍🗨 Hello ${firstName}, Welcome to Bot Monitoring\n🌏 Please Choose your Language\n\n` + + `•❂•─•─•❂•─•❂••❂•─•❂•─•─•❂•\n\n`; + + await ctx.replyWithPhoto("https://imgur.com/a/mXrMZsI", { + caption: message, + parse_mode: "HTML", + reply_markup: getLanguageKeyboard(), + }); +} diff --git a/src/middlewares/middleware.ts b/src/middlewares/middleware.ts new file mode 100644 index 0000000..c47d127 --- /dev/null +++ b/src/middlewares/middleware.ts @@ -0,0 +1,20 @@ +import { Context, NextFunction } from "grammy"; +import { CHANNEL_ID } from "./config"; + +export async function checkJoin(ctx: Context, next: NextFunction) { + if (!ctx.from) return; + + try { + const chatMember = await ctx.api.getChatMember(CHANNEL_ID, ctx.from.id); + if (["member", "administrator", "creator"].includes(chatMember.status)) { + return next(); + } else { + await ctx.reply("❌ Anda harus bergabung dengan channel terlebih dahulu!\n\n🔗 Join di sini: https://t.me/YOUR_CHANNEL", { + parse_mode: "HTML", + }); + } + } catch (error) { + console.error("Error checking join status:", error); + await ctx.reply("Terjadi kesalahan saat memeriksa keanggotaan. Coba lagi nanti."); + } +} diff --git a/src/utils.ts b/src/utils.ts new file mode 100644 index 0000000..5b33aa6 --- /dev/null +++ b/src/utils.ts @@ -0,0 +1,32 @@ +import { InlineKeyboard } from "grammy"; +import { Bot } from "grammy"; + +export function getLanguageKeyboard() { + return new InlineKeyboard() + .text("🇬🇧 English", "English") + .text("🇩🇪 Deutsch", "Germany") + .text("🇮🇩 Indonesian", "Indonesian") + .row() + .text("🇷🇺 Русский", "Russian") + .text("🇮🇹 Italiano", "Italy") + .text("🇰🇷 한국어", "Korean") + .row() + .text("🇲🇾 Malay", "Malaysian") + .text("🇪🇦 Español", "Spanish") + .text("🇫🇷 France", "France") + .row() + .text("🇧🇷 Português", "Brazilian") + .text("🇵🇭 Tagalog", "Philippines") + .text("🇮🇳 हिन्दी", "Hindi") + .row() + .text("🇸🇦 العربية", "Arabian") + .text("🇯🇵 日本", "Japan") + .text("🇵🇱 Polska", "Poland") + .row() + .text("🇨🇳 中文", "Chinese"); +} + +export async function getBotUsername(bot: Bot) { + const botInfo = await bot.api.getMe(); + return botInfo.username; +} diff --git a/tsconfig.json b/tsconfig.json new file mode 100644 index 0000000..e69de29 From 37bda9673f701b26ddb774b2333d63f29677be17 Mon Sep 17 00:00:00 2001 From: M Amamiya Ahmad <134295603+MAmamiyaAhmad@users.noreply.github.com> Date: Sat, 1 Feb 2025 18:58:28 +0700 Subject: [PATCH 02/11] TGStoreBot LICENSE --- LICENSE | 21 +++++++++++++++++++++ 1 file changed, 21 insertions(+) create mode 100644 LICENSE diff --git a/LICENSE b/LICENSE new file mode 100644 index 0000000..b2f091c --- /dev/null +++ b/LICENSE @@ -0,0 +1,21 @@ +MIT License + +Copyright (c) 2025 M Amamiya Ahmad + +Permission is hereby granted, free of charge, to any person obtaining a copy +of this software and associated documentation files (the "Software"), to deal +in the Software without restriction, including without limitation the rights +to use, copy, modify, merge, publish, distribute, sublicense, and/or sell +copies of the Software, and to permit persons to whom the Software is +furnished to do so, subject to the following conditions: + +The above copyright notice and this permission notice shall be included in all +copies or substantial portions of the Software. + +THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR +IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, +FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE +AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER +LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, +OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE +SOFTWARE. From 0707bbd05ae43035ced1ba5229844121e28c06c1 Mon Sep 17 00:00:00 2001 From: M Amamiya Ahmad <134295603+MAmamiyaAhmad@users.noreply.github.com> Date: Sat, 1 Feb 2025 19:02:11 +0700 Subject: [PATCH 03/11] README.md --- README.md | 23 +++++++++++++++++++++++ 1 file changed, 23 insertions(+) diff --git a/README.md b/README.md index e69de29..efdb43e 100644 --- a/README.md +++ b/README.md @@ -0,0 +1,23 @@ +# Telegram Store Bot + +## Overview +Telegram Store Bot is a powerful bot built using **Node.js (TypeScript)** and the **grammY framework**, designed for **selling digital products** directly within Telegram. This bot enables store owners to manage products, process transactions, and engage with customers efficiently. + +The bot supports **multiple roles**: +- **Super Admin & Admin**: Manage store settings, products, and users. +- **User**: Browse, purchase products, and track transactions. + +## Features +- **🏠 Main Menu**: Provides easy navigation with the following sections: + - **Dashboard** + - **Store/Buy Product** + - **Referral & Bonus System** + - **📊 Statistics** + - **About Us & Extra Menu** +- **Multi-language Support**: Users can choose from multiple languages, including English, Indonesian, Chinese, Japanese, Russian, Portuguese, Vietnamese, and more. +- **Interactive Buttons**: Utilizes both **keyboard buttons** and **inline buttons** for seamless user interaction. +- **Message Formatting**: Supports **HTML** and **Markdown** for rich text display. + +## Conclusion +Telegram Store Bot is a feature-rich e-commerce solution for selling digital products on Telegram. It offers seamless navigation, multiple languages, interactive buttons, and a structured **Main Menu** to ensure an engaging user experience. This bot is highly customizable and optimized for **performance and scalability**. + From 63fd45d16cc44ccc55588621e2fb2992429da500 Mon Sep 17 00:00:00 2001 From: MAmamiyaAhmad Date: Sun, 2 Feb 2025 01:10:02 +0700 Subject: [PATCH 04/11] Update --- .gitignore | 5 + package-lock.json | 1046 +++++++++++++++++++++++++++++---- package.json | 34 +- src/middlewares/middleware.ts | 19 - tsconfig.json | 12 + 5 files changed, 967 insertions(+), 149 deletions(-) create mode 100644 .gitignore diff --git a/.gitignore b/.gitignore new file mode 100644 index 0000000..9c7f08b --- /dev/null +++ b/.gitignore @@ -0,0 +1,5 @@ +# Ignore node_modules folder +node_modules/ + +# Ignore dist folder (hasil kompilasi TypeScript) +dist/ diff --git a/package-lock.json b/package-lock.json index bf4d760..68c8237 100644 --- a/package-lock.json +++ b/package-lock.json @@ -1,132 +1,926 @@ { - "name": "TGBot", + + "name": "tgbot", + + "version": "1.0.0", + "lockfileVersion": 3, + "requires": true, + "packages": { - "": { - "dependencies": { - "dotenv": "^16.4.7", - "grammy": "^1.34.1" - } - }, - "node_modules/@grammyjs/types": { - "version": "3.18.0", - "resolved": "https://registry.npmjs.org/@grammyjs/types/-/types-3.18.0.tgz", - "integrity": "sha512-/VpQcT6Z0+Lw6H6jv2JvnDmqB1/9rjbFdDBlPBPUW3kQ1bWfjzfaiVk2+J2eiid3o3sLaPp3ZK0XjEI1gWNc3g==", - "license": "MIT" - }, - "node_modules/abort-controller": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/abort-controller/-/abort-controller-3.0.0.tgz", - "integrity": "sha512-h8lQ8tacZYnR3vNQTgibj+tODHI5/+l06Au2Pcriv/Gmet0eaj4TwWH41sO9wnHDiQsEj19q0drzdWdeAHtweg==", - "license": "MIT", - "dependencies": { - "event-target-shim": "^5.0.0" - }, - "engines": { - "node": ">=6.5" - } - }, - "node_modules/debug": { - "version": "4.4.0", - "resolved": "https://registry.npmjs.org/debug/-/debug-4.4.0.tgz", - "integrity": "sha512-6WTZ/IxCY/T6BALoZHaE4ctp9xm+Z5kY/pzYaCHRFeyVhojxlrm+46y68HA6hr0TcwEssoxNiDEUJQjfPZ/RYA==", - "license": "MIT", - "dependencies": { - "ms": "^2.1.3" - }, - "engines": { - "node": ">=6.0" - }, - "peerDependenciesMeta": { - "supports-color": { - "optional": true - } - } - }, - "node_modules/dotenv": { - "version": "16.4.7", - "resolved": "https://registry.npmjs.org/dotenv/-/dotenv-16.4.7.tgz", - "integrity": "sha512-47qPchRCykZC03FhkYAhrvwU4xDBFIj1QPqaarj6mdM/hgUzfPHcpkHJOn3mJAufFeeAxAzeGsr5X0M4k6fLZQ==", - "license": "BSD-2-Clause", - "engines": { - "node": ">=12" - }, - "funding": { - "url": "https://dotenvx.com" - } - }, - "node_modules/event-target-shim": { - "version": "5.0.1", - "resolved": "https://registry.npmjs.org/event-target-shim/-/event-target-shim-5.0.1.tgz", - "integrity": "sha512-i/2XbnSz/uxRCU6+NdVJgKWDTM427+MqYbkQzD321DuCQJUqOuJKIA0IM2+W2xtYHdKOmZ4dR6fExsd4SXL+WQ==", - "license": "MIT", - "engines": { - "node": ">=6" - } - }, - "node_modules/grammy": { - "version": "1.34.1", - "resolved": "https://registry.npmjs.org/grammy/-/grammy-1.34.1.tgz", - "integrity": "sha512-0ypt5d1bna8CqSANlDCg2ce1sx87oDm9gjyPRHWYpAgvjy3WYW9oEmMllAYCWCZ3vHDULW5x14jBMXai51ar0Q==", - "license": "MIT", - "dependencies": { - "@grammyjs/types": "3.18.0", - "abort-controller": "^3.0.0", - "debug": "^4.3.4", - "node-fetch": "^2.7.0" - }, - "engines": { - "node": "^12.20.0 || >=14.13.1" - } - }, - "node_modules/ms": { - "version": "2.1.3", - "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.3.tgz", - "integrity": "sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA==", - "license": "MIT" - }, - "node_modules/node-fetch": { - "version": "2.7.0", - "resolved": "https://registry.npmjs.org/node-fetch/-/node-fetch-2.7.0.tgz", - "integrity": "sha512-c4FRfUm/dbcWZ7U+1Wq0AwCyFL+3nt2bEw05wfxSz+DWpWsitgmSgYmy2dQdWyKC1694ELPqMs/YzUSNozLt8A==", - "license": "MIT", - "dependencies": { - "whatwg-url": "^5.0.0" - }, - "engines": { - "node": "4.x || >=6.0.0" - }, - "peerDependencies": { - "encoding": "^0.1.0" - }, - "peerDependenciesMeta": { - "encoding": { - "optional": true - } - } - }, - "node_modules/tr46": { - "version": "0.0.3", - "resolved": "https://registry.npmjs.org/tr46/-/tr46-0.0.3.tgz", - "integrity": "sha512-N3WMsuqV66lT30CrXNbEjx4GEwlow3v6rr4mCcv6prnfwhS01rkgyFdjPNBYd9br7LpXV1+Emh01fHnq2Gdgrw==", - "license": "MIT" - }, - "node_modules/webidl-conversions": { - "version": "3.0.1", - "resolved": "https://registry.npmjs.org/webidl-conversions/-/webidl-conversions-3.0.1.tgz", - "integrity": "sha512-2JAn3z8AR6rjK8Sm8orRC0h/bcl/DqL7tRPdGZ4I1CjdF+EaMLmYxBHyXuKL849eucPFhvBoxMsflfOb8kxaeQ==", - "license": "BSD-2-Clause" - }, - "node_modules/whatwg-url": { - "version": "5.0.0", - "resolved": "https://registry.npmjs.org/whatwg-url/-/whatwg-url-5.0.0.tgz", - "integrity": "sha512-saE57nupxk6v3HY35+jzBwYa0rKSy0XR8JSxZPwgLr7ys0IBzhGviA1/TUGJLmSVqs8pb9AnvICXEuOHLprYTw==", - "license": "MIT", - "dependencies": { - "tr46": "~0.0.3", - "webidl-conversions": "^3.0.0" - } - } + + + "": { + + + + "name": "tgbot", + + + + "version": "1.0.0", + + + + "license": "ISC", + + + + "dependencies": { + + + + + "dotenv": "^16.4.7", + + + + + "grammy": "^1.34.1" + + + + }, + + + + "devDependencies": { + + + + + "@types/node": "^22.13.0", + + + + + "@types/node-fetch": "^2.6.12" + + + + } + + + }, + + + "node_modules/@grammyjs/types": { + + + + "version": "3.18.0", + + + + "resolved": "https://registry.npmjs.org/@grammyjs/types/-/types-3.18.0.tgz", + + + + "integrity": "sha512-/VpQcT6Z0+Lw6H6jv2JvnDmqB1/9rjbFdDBlPBPUW3kQ1bWfjzfaiVk2+J2eiid3o3sLaPp3ZK0XjEI1gWNc3g==", + + + + "license": "MIT" + + + }, + + + "node_modules/@types/node": { + + + + "version": "22.13.0", + + + + "resolved": "https://registry.npmjs.org/@types/node/-/node-22.13.0.tgz", + + + + "integrity": "sha512-ClIbNe36lawluuvq3+YYhnIN2CELi+6q8NpnM7PYp4hBn/TatfboPgVSm2rwKRfnV2M+Ty9GWDFI64KEe+kysA==", + + + + "dev": true, + + + + "license": "MIT", + + + + "dependencies": { + + + + + "undici-types": "~6.20.0" + + + + } + + + }, + + + "node_modules/@types/node-fetch": { + + + + "version": "2.6.12", + + + + "resolved": "https://registry.npmjs.org/@types/node-fetch/-/node-fetch-2.6.12.tgz", + + + + "integrity": "sha512-8nneRWKCg3rMtF69nLQJnOYUcbafYeFSjqkw3jCRLsqkWFlHaoQrr5mXmofFGOx3DKn7UfmBMyov8ySvLRVldA==", + + + + "dev": true, + + + + "license": "MIT", + + + + "dependencies": { + + + + + "@types/node": "*", + + + + + "form-data": "^4.0.0" + + + + } + + + }, + + + "node_modules/abort-controller": { + + + + "version": "3.0.0", + + + + "resolved": "https://registry.npmjs.org/abort-controller/-/abort-controller-3.0.0.tgz", + + + + "integrity": "sha512-h8lQ8tacZYnR3vNQTgibj+tODHI5/+l06Au2Pcriv/Gmet0eaj4TwWH41sO9wnHDiQsEj19q0drzdWdeAHtweg==", + + + + "license": "MIT", + + + + "dependencies": { + + + + + "event-target-shim": "^5.0.0" + + + + }, + + + + "engines": { + + + + + "node": ">=6.5" + + + + } + + + }, + + + "node_modules/asynckit": { + + + + "version": "0.4.0", + + + + "resolved": "https://registry.npmjs.org/asynckit/-/asynckit-0.4.0.tgz", + + + + "integrity": "sha512-Oei9OH4tRh0YqU3GxhX79dM/mwVgvbZJaSNaRk+bshkj0S5cfHcgYakreBjrHwatXKbz+IoIdYLxrKim2MjW0Q==", + + + + "dev": true, + + + + "license": "MIT" + + + }, + + + "node_modules/combined-stream": { + + + + "version": "1.0.8", + + + + "resolved": "https://registry.npmjs.org/combined-stream/-/combined-stream-1.0.8.tgz", + + + + "integrity": "sha512-FQN4MRfuJeHf7cBbBMJFXhKSDq+2kAArBlmRBvcvFE5BB1HZKXtSFASDhdlz9zOYwxh8lDdnvmMOe/+5cdoEdg==", + + + + "dev": true, + + + + "license": "MIT", + + + + "dependencies": { + + + + + "delayed-stream": "~1.0.0" + + + + }, + + + + "engines": { + + + + + "node": ">= 0.8" + + + + } + + + }, + + + "node_modules/debug": { + + + + "version": "4.4.0", + + + + "resolved": "https://registry.npmjs.org/debug/-/debug-4.4.0.tgz", + + + + "integrity": "sha512-6WTZ/IxCY/T6BALoZHaE4ctp9xm+Z5kY/pzYaCHRFeyVhojxlrm+46y68HA6hr0TcwEssoxNiDEUJQjfPZ/RYA==", + + + + "license": "MIT", + + + + "dependencies": { + + + + + "ms": "^2.1.3" + + + + }, + + + + "engines": { + + + + + "node": ">=6.0" + + + + }, + + + + "peerDependenciesMeta": { + + + + + "supports-color": { + + + + + + "optional": true + + + + + } + + + + } + + + }, + + + "node_modules/delayed-stream": { + + + + "version": "1.0.0", + + + + "resolved": "https://registry.npmjs.org/delayed-stream/-/delayed-stream-1.0.0.tgz", + + + + "integrity": "sha512-ZySD7Nf91aLB0RxL4KGrKHBXl7Eds1DAmEdcoVawXnLD7SDhpNgtuII2aAkg7a7QS41jxPSZ17p4VdGnMHk3MQ==", + + + + "dev": true, + + + + "license": "MIT", + + + + "engines": { + + + + + "node": ">=0.4.0" + + + + } + + + }, + + + "node_modules/dotenv": { + + + + "version": "16.4.7", + + + + "resolved": "https://registry.npmjs.org/dotenv/-/dotenv-16.4.7.tgz", + + + + "integrity": "sha512-47qPchRCykZC03FhkYAhrvwU4xDBFIj1QPqaarj6mdM/hgUzfPHcpkHJOn3mJAufFeeAxAzeGsr5X0M4k6fLZQ==", + + + + "license": "BSD-2-Clause", + + + + "engines": { + + + + + "node": ">=12" + + + + }, + + + + "funding": { + + + + + "url": "https://dotenvx.com" + + + + } + + + }, + + + "node_modules/event-target-shim": { + + + + "version": "5.0.1", + + + + "resolved": "https://registry.npmjs.org/event-target-shim/-/event-target-shim-5.0.1.tgz", + + + + "integrity": "sha512-i/2XbnSz/uxRCU6+NdVJgKWDTM427+MqYbkQzD321DuCQJUqOuJKIA0IM2+W2xtYHdKOmZ4dR6fExsd4SXL+WQ==", + + + + "license": "MIT", + + + + "engines": { + + + + + "node": ">=6" + + + + } + + + }, + + + "node_modules/form-data": { + + + + "version": "4.0.1", + + + + "resolved": "https://registry.npmjs.org/form-data/-/form-data-4.0.1.tgz", + + + + "integrity": "sha512-tzN8e4TX8+kkxGPK8D5u0FNmjPUjw3lwC9lSLxxoB/+GtsJG91CO8bSWy73APlgAZzZbXEYZJuxjkHH2w+Ezhw==", + + + + "dev": true, + + + + "license": "MIT", + + + + "dependencies": { + + + + + "asynckit": "^0.4.0", + + + + + "combined-stream": "^1.0.8", + + + + + "mime-types": "^2.1.12" + + + + }, + + + + "engines": { + + + + + "node": ">= 6" + + + + } + + + }, + + + "node_modules/grammy": { + + + + "version": "1.34.1", + + + + "resolved": "https://registry.npmjs.org/grammy/-/grammy-1.34.1.tgz", + + + + "integrity": "sha512-0ypt5d1bna8CqSANlDCg2ce1sx87oDm9gjyPRHWYpAgvjy3WYW9oEmMllAYCWCZ3vHDULW5x14jBMXai51ar0Q==", + + + + "license": "MIT", + + + + "dependencies": { + + + + + "@grammyjs/types": "3.18.0", + + + + + "abort-controller": "^3.0.0", + + + + + "debug": "^4.3.4", + + + + + "node-fetch": "^2.7.0" + + + + }, + + + + "engines": { + + + + + "node": "^12.20.0 || >=14.13.1" + + + + } + + + }, + + + "node_modules/mime-db": { + + + + "version": "1.52.0", + + + + "resolved": "https://registry.npmjs.org/mime-db/-/mime-db-1.52.0.tgz", + + + + "integrity": "sha512-sPU4uV7dYlvtWJxwwxHD0PuihVNiE7TyAbQ5SWxDCB9mUYvOgroQOwYQQOKPJ8CIbE+1ETVlOoK1UC2nU3gYvg==", + + + + "dev": true, + + + + "license": "MIT", + + + + "engines": { + + + + + "node": ">= 0.6" + + + + } + + + }, + + + "node_modules/mime-types": { + + + + "version": "2.1.35", + + + + "resolved": "https://registry.npmjs.org/mime-types/-/mime-types-2.1.35.tgz", + + + + "integrity": "sha512-ZDY+bPm5zTTF+YpCrAU9nK0UgICYPT0QtT1NZWFv4s++TNkcgVaT0g6+4R2uI4MjQjzysHB1zxuWL50hzaeXiw==", + + + + "dev": true, + + + + "license": "MIT", + + + + "dependencies": { + + + + + "mime-db": "1.52.0" + + + + }, + + + + "engines": { + + + + + "node": ">= 0.6" + + + + } + + + }, + + + "node_modules/ms": { + + + + "version": "2.1.3", + + + + "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.3.tgz", + + + + "integrity": "sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA==", + + + + "license": "MIT" + + + }, + + + "node_modules/node-fetch": { + + + + "version": "2.7.0", + + + + "resolved": "https://registry.npmjs.org/node-fetch/-/node-fetch-2.7.0.tgz", + + + + "integrity": "sha512-c4FRfUm/dbcWZ7U+1Wq0AwCyFL+3nt2bEw05wfxSz+DWpWsitgmSgYmy2dQdWyKC1694ELPqMs/YzUSNozLt8A==", + + + + "license": "MIT", + + + + "dependencies": { + + + + + "whatwg-url": "^5.0.0" + + + + }, + + + + "engines": { + + + + + "node": "4.x || >=6.0.0" + + + + }, + + + + "peerDependencies": { + + + + + "encoding": "^0.1.0" + + + + }, + + + + "peerDependenciesMeta": { + + + + + "encoding": { + + + + + + "optional": true + + + + + } + + + + } + + + }, + + + "node_modules/tr46": { + + + + "version": "0.0.3", + + + + "resolved": "https://registry.npmjs.org/tr46/-/tr46-0.0.3.tgz", + + + + "integrity": "sha512-N3WMsuqV66lT30CrXNbEjx4GEwlow3v6rr4mCcv6prnfwhS01rkgyFdjPNBYd9br7LpXV1+Emh01fHnq2Gdgrw==", + + + + "license": "MIT" + + + }, + + + "node_modules/undici-types": { + + + + "version": "6.20.0", + + + + "resolved": "https://registry.npmjs.org/undici-types/-/undici-types-6.20.0.tgz", + + + + "integrity": "sha512-Ny6QZ2Nju20vw1SRHe3d9jVu6gJ+4e3+MMpqu7pqE5HT6WsTSlce++GQmK5UXS8mzV8DSYHrQH+Xrf2jVcuKNg==", + + + + "dev": true, + + + + "license": "MIT" + + + }, + + + "node_modules/webidl-conversions": { + + + + "version": "3.0.1", + + + + "resolved": "https://registry.npmjs.org/webidl-conversions/-/webidl-conversions-3.0.1.tgz", + + + + "integrity": "sha512-2JAn3z8AR6rjK8Sm8orRC0h/bcl/DqL7tRPdGZ4I1CjdF+EaMLmYxBHyXuKL849eucPFhvBoxMsflfOb8kxaeQ==", + + + + "license": "BSD-2-Clause" + + + }, + + + "node_modules/whatwg-url": { + + + + "version": "5.0.0", + + + + "resolved": "https://registry.npmjs.org/whatwg-url/-/whatwg-url-5.0.0.tgz", + + + + "integrity": "sha512-saE57nupxk6v3HY35+jzBwYa0rKSy0XR8JSxZPwgLr7ys0IBzhGviA1/TUGJLmSVqs8pb9AnvICXEuOHLprYTw==", + + + + "license": "MIT", + + + + "dependencies": { + + + + + "tr46": "~0.0.3", + + + + + "webidl-conversions": "^3.0.0" + + + + } + + + } + } } diff --git a/package.json b/package.json index 6b4e138..5c9ed1f 100644 --- a/package.json +++ b/package.json @@ -1,19 +1,45 @@ { "name": "tgbot", + "version": "1.0.0", + "description": "", + "main": "index.js", - "devDependencies": {}, + + "devDependencies": { + + + "@types/node": "^22.13.0", + + + "@types/node-fetch": "^2.6.12" + + }, + "scripts": { - "test": "echo \"Error: no test specified\" && exit 1" + + + "test": "echo \"Error: no test specified\" && exit 1" + }, + "keywords": [], + "author": "", + "license": "ISC", + "type": "commonjs", + "dependencies": { - "dotenv": "^16.4.7", - "grammy": "^1.34.1" + + + "dotenv": "^16.4.7", + + + "grammy": "^1.34.1" + } } diff --git a/src/middlewares/middleware.ts b/src/middlewares/middleware.ts index c47d127..e492b4a 100644 --- a/src/middlewares/middleware.ts +++ b/src/middlewares/middleware.ts @@ -1,20 +1 @@ import { Context, NextFunction } from "grammy"; -import { CHANNEL_ID } from "./config"; - -export async function checkJoin(ctx: Context, next: NextFunction) { - if (!ctx.from) return; - - try { - const chatMember = await ctx.api.getChatMember(CHANNEL_ID, ctx.from.id); - if (["member", "administrator", "creator"].includes(chatMember.status)) { - return next(); - } else { - await ctx.reply("❌ Anda harus bergabung dengan channel terlebih dahulu!\n\n🔗 Join di sini: https://t.me/YOUR_CHANNEL", { - parse_mode: "HTML", - }); - } - } catch (error) { - console.error("Error checking join status:", error); - await ctx.reply("Terjadi kesalahan saat memeriksa keanggotaan. Coba lagi nanti."); - } -} diff --git a/tsconfig.json b/tsconfig.json index e69de29..95c89cd 100644 --- a/tsconfig.json +++ b/tsconfig.json @@ -0,0 +1,12 @@ +{ + "compilerOptions": { + "target": "ESNext", + "module": "CommonJS", + "outDir": "./dist", + "rootDir": "./src", + "strict": true, + "types": ["node"] + }, + "include": ["src/**/*.ts", "node_modules/grammy/types/**/*.d.ts"] + } + \ No newline at end of file From 9334932b47a894c5ba8cfb60f2ca518bf1172588 Mon Sep 17 00:00:00 2001 From: MAmamiyaAhmad Date: Tue, 4 Feb 2025 20:21:26 +0700 Subject: [PATCH 05/11] Update --- .env | 3 +- package-lock.json | 3586 ++++++++++++++++++++------- package.json | 73 +- src/bot.ts | 26 +- src/config.ts | 4 - src/config/config.ts | 6 + src/handlers/language.ts | 19 +- src/handlers/start.ts | 49 +- src/languages/arabic.json | 12 + src/languages/chinese.json | 12 + src/languages/english.json | 12 + src/languages/french.json | 12 + src/languages/german.json | 12 + src/languages/hindi.json | 12 + src/languages/indonesian.json | 12 + src/languages/italian.json | 12 + src/languages/japanese.json | 12 + src/languages/korean.json | 12 + src/languages/malaysian.json | 12 + src/languages/philippine.json | 12 + src/languages/polish.json | 12 + src/languages/portuguese.json | 12 + src/languages/russian.json | 12 + src/languages/spanish.json | 12 + src/middlewares/middleware.ts | 1 - src/{utils.ts => utils/language.ts} | 0 src/utils/logger.ts | 31 + tsconfig.json | 26 +- 28 files changed, 3018 insertions(+), 998 deletions(-) delete mode 100644 src/config.ts create mode 100644 src/config/config.ts create mode 100644 src/languages/arabic.json create mode 100644 src/languages/chinese.json create mode 100644 src/languages/english.json create mode 100644 src/languages/french.json create mode 100644 src/languages/german.json create mode 100644 src/languages/hindi.json create mode 100644 src/languages/indonesian.json create mode 100644 src/languages/italian.json create mode 100644 src/languages/japanese.json create mode 100644 src/languages/korean.json create mode 100644 src/languages/malaysian.json create mode 100644 src/languages/philippine.json create mode 100644 src/languages/polish.json create mode 100644 src/languages/portuguese.json create mode 100644 src/languages/russian.json create mode 100644 src/languages/spanish.json delete mode 100644 src/middlewares/middleware.ts rename src/{utils.ts => utils/language.ts} (100%) create mode 100644 src/utils/logger.ts diff --git a/.env b/.env index fefc2a5..1007d54 100644 --- a/.env +++ b/.env @@ -1,2 +1,3 @@ BOT_TOKEN=952920973:AAGFpuzir-eRj1Sx0R1m9uVT6SD4ml3gFqs -CHANNEL_ID=@bm_update +SUPER_ADMIN_ID=601954832 +ADMIN_IDS=123456789,987654321 \ No newline at end of file diff --git a/package-lock.json b/package-lock.json index 68c8237..9c66d28 100644 --- a/package-lock.json +++ b/package-lock.json @@ -1,926 +1,2674 @@ { - - "name": "tgbot", - + "name": "tgstorebot", "version": "1.0.0", - "lockfileVersion": 3, - "requires": true, - "packages": { - - - "": { - - - - "name": "tgbot", - - - - "version": "1.0.0", - - - - "license": "ISC", - - - - "dependencies": { - - - - - "dotenv": "^16.4.7", - - - - - "grammy": "^1.34.1" - - - - }, - - - - "devDependencies": { - - - - - "@types/node": "^22.13.0", - - - - - "@types/node-fetch": "^2.6.12" - - - - } - - - }, - - - "node_modules/@grammyjs/types": { - - - - "version": "3.18.0", - - - - "resolved": "https://registry.npmjs.org/@grammyjs/types/-/types-3.18.0.tgz", - - - - "integrity": "sha512-/VpQcT6Z0+Lw6H6jv2JvnDmqB1/9rjbFdDBlPBPUW3kQ1bWfjzfaiVk2+J2eiid3o3sLaPp3ZK0XjEI1gWNc3g==", - - - - "license": "MIT" - - - }, - - - "node_modules/@types/node": { - - - - "version": "22.13.0", - - - - "resolved": "https://registry.npmjs.org/@types/node/-/node-22.13.0.tgz", - - - - "integrity": "sha512-ClIbNe36lawluuvq3+YYhnIN2CELi+6q8NpnM7PYp4hBn/TatfboPgVSm2rwKRfnV2M+Ty9GWDFI64KEe+kysA==", - - - - "dev": true, - - - - "license": "MIT", - - - - "dependencies": { - - - - - "undici-types": "~6.20.0" - - - - } - - - }, - - - "node_modules/@types/node-fetch": { - - - - "version": "2.6.12", - - - - "resolved": "https://registry.npmjs.org/@types/node-fetch/-/node-fetch-2.6.12.tgz", - - - - "integrity": "sha512-8nneRWKCg3rMtF69nLQJnOYUcbafYeFSjqkw3jCRLsqkWFlHaoQrr5mXmofFGOx3DKn7UfmBMyov8ySvLRVldA==", - - - - "dev": true, - - - - "license": "MIT", - - - - "dependencies": { - - - - - "@types/node": "*", - - - - - "form-data": "^4.0.0" - - - - } - - - }, - - - "node_modules/abort-controller": { - - - - "version": "3.0.0", - - - - "resolved": "https://registry.npmjs.org/abort-controller/-/abort-controller-3.0.0.tgz", - - - - "integrity": "sha512-h8lQ8tacZYnR3vNQTgibj+tODHI5/+l06Au2Pcriv/Gmet0eaj4TwWH41sO9wnHDiQsEj19q0drzdWdeAHtweg==", - - - - "license": "MIT", - - - - "dependencies": { - - - - - "event-target-shim": "^5.0.0" - - - - }, - - - - "engines": { - - - - - "node": ">=6.5" - - - - } - - - }, - - - "node_modules/asynckit": { - - - - "version": "0.4.0", - - - - "resolved": "https://registry.npmjs.org/asynckit/-/asynckit-0.4.0.tgz", - - - - "integrity": "sha512-Oei9OH4tRh0YqU3GxhX79dM/mwVgvbZJaSNaRk+bshkj0S5cfHcgYakreBjrHwatXKbz+IoIdYLxrKim2MjW0Q==", - - - - "dev": true, - - - - "license": "MIT" - - - }, - - - "node_modules/combined-stream": { - - - - "version": "1.0.8", - - - - "resolved": "https://registry.npmjs.org/combined-stream/-/combined-stream-1.0.8.tgz", - - - - "integrity": "sha512-FQN4MRfuJeHf7cBbBMJFXhKSDq+2kAArBlmRBvcvFE5BB1HZKXtSFASDhdlz9zOYwxh8lDdnvmMOe/+5cdoEdg==", - - - - "dev": true, - - - - "license": "MIT", - - - - "dependencies": { - - - - - "delayed-stream": "~1.0.0" - - - - }, - - - - "engines": { - - - - - "node": ">= 0.8" - - - - } - - - }, - - - "node_modules/debug": { - - - - "version": "4.4.0", - - - - "resolved": "https://registry.npmjs.org/debug/-/debug-4.4.0.tgz", - - - - "integrity": "sha512-6WTZ/IxCY/T6BALoZHaE4ctp9xm+Z5kY/pzYaCHRFeyVhojxlrm+46y68HA6hr0TcwEssoxNiDEUJQjfPZ/RYA==", - - - - "license": "MIT", - - - - "dependencies": { - - - - - "ms": "^2.1.3" - - - - }, - - - - "engines": { - - - - - "node": ">=6.0" - - - - }, - - - - "peerDependenciesMeta": { - - - - - "supports-color": { - - - - - - "optional": true - - - - - } - - - - } - - - }, - - - "node_modules/delayed-stream": { - - - - "version": "1.0.0", - - - - "resolved": "https://registry.npmjs.org/delayed-stream/-/delayed-stream-1.0.0.tgz", - - - - "integrity": "sha512-ZySD7Nf91aLB0RxL4KGrKHBXl7Eds1DAmEdcoVawXnLD7SDhpNgtuII2aAkg7a7QS41jxPSZ17p4VdGnMHk3MQ==", - - - - "dev": true, - - - - "license": "MIT", - - - - "engines": { - - - - - "node": ">=0.4.0" - - - - } - - - }, - - - "node_modules/dotenv": { - - - - "version": "16.4.7", - - - - "resolved": "https://registry.npmjs.org/dotenv/-/dotenv-16.4.7.tgz", - - - - "integrity": "sha512-47qPchRCykZC03FhkYAhrvwU4xDBFIj1QPqaarj6mdM/hgUzfPHcpkHJOn3mJAufFeeAxAzeGsr5X0M4k6fLZQ==", - - - - "license": "BSD-2-Clause", - - - - "engines": { - - - - - "node": ">=12" - - - - }, - - - - "funding": { - - - - - "url": "https://dotenvx.com" - - - - } - - - }, - - - "node_modules/event-target-shim": { - - - - "version": "5.0.1", - - - - "resolved": "https://registry.npmjs.org/event-target-shim/-/event-target-shim-5.0.1.tgz", - - - - "integrity": "sha512-i/2XbnSz/uxRCU6+NdVJgKWDTM427+MqYbkQzD321DuCQJUqOuJKIA0IM2+W2xtYHdKOmZ4dR6fExsd4SXL+WQ==", - - - - "license": "MIT", - - - - "engines": { - - - - - "node": ">=6" - - - - } - - - }, - - - "node_modules/form-data": { - - - - "version": "4.0.1", - - - - "resolved": "https://registry.npmjs.org/form-data/-/form-data-4.0.1.tgz", - - - - "integrity": "sha512-tzN8e4TX8+kkxGPK8D5u0FNmjPUjw3lwC9lSLxxoB/+GtsJG91CO8bSWy73APlgAZzZbXEYZJuxjkHH2w+Ezhw==", - - - - "dev": true, - - - - "license": "MIT", - - - - "dependencies": { - - - - - "asynckit": "^0.4.0", - - - - - "combined-stream": "^1.0.8", - - - - - "mime-types": "^2.1.12" - - - - }, - - - - "engines": { - - - - - "node": ">= 6" - - - - } - - - }, - - - "node_modules/grammy": { - - - - "version": "1.34.1", - - - - "resolved": "https://registry.npmjs.org/grammy/-/grammy-1.34.1.tgz", - - - - "integrity": "sha512-0ypt5d1bna8CqSANlDCg2ce1sx87oDm9gjyPRHWYpAgvjy3WYW9oEmMllAYCWCZ3vHDULW5x14jBMXai51ar0Q==", - - - - "license": "MIT", - - - - "dependencies": { - - - - - "@grammyjs/types": "3.18.0", - - - - - "abort-controller": "^3.0.0", - - - - - "debug": "^4.3.4", - - - - - "node-fetch": "^2.7.0" - - - - }, - - - - "engines": { - - - - - "node": "^12.20.0 || >=14.13.1" - - - - } - - - }, - - - "node_modules/mime-db": { - - - - "version": "1.52.0", - - - - "resolved": "https://registry.npmjs.org/mime-db/-/mime-db-1.52.0.tgz", - - - - "integrity": "sha512-sPU4uV7dYlvtWJxwwxHD0PuihVNiE7TyAbQ5SWxDCB9mUYvOgroQOwYQQOKPJ8CIbE+1ETVlOoK1UC2nU3gYvg==", - - - - "dev": true, - - - - "license": "MIT", - - - - "engines": { - - - - - "node": ">= 0.6" - - - - } - - - }, - - - "node_modules/mime-types": { - - - - "version": "2.1.35", - - - - "resolved": "https://registry.npmjs.org/mime-types/-/mime-types-2.1.35.tgz", - - - - "integrity": "sha512-ZDY+bPm5zTTF+YpCrAU9nK0UgICYPT0QtT1NZWFv4s++TNkcgVaT0g6+4R2uI4MjQjzysHB1zxuWL50hzaeXiw==", - - - - "dev": true, - - - - "license": "MIT", - - - - "dependencies": { - - - - - "mime-db": "1.52.0" - - - - }, - - - - "engines": { - - - - - "node": ">= 0.6" - - - - } - - - }, - - - "node_modules/ms": { - - - - "version": "2.1.3", - - - - "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.3.tgz", - - - - "integrity": "sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA==", - - - - "license": "MIT" - - - }, - - - "node_modules/node-fetch": { - - - - "version": "2.7.0", - - - - "resolved": "https://registry.npmjs.org/node-fetch/-/node-fetch-2.7.0.tgz", - - - - "integrity": "sha512-c4FRfUm/dbcWZ7U+1Wq0AwCyFL+3nt2bEw05wfxSz+DWpWsitgmSgYmy2dQdWyKC1694ELPqMs/YzUSNozLt8A==", - - - - "license": "MIT", - - - - "dependencies": { - - - - - "whatwg-url": "^5.0.0" - - - - }, - - - - "engines": { - - - - - "node": "4.x || >=6.0.0" - - - - }, - - - - "peerDependencies": { - - - - - "encoding": "^0.1.0" - - - - }, - - - - "peerDependenciesMeta": { - - - - - "encoding": { - - - - - - "optional": true - - - - - } - - - - } - - - }, - - - "node_modules/tr46": { - - - - "version": "0.0.3", - - - - "resolved": "https://registry.npmjs.org/tr46/-/tr46-0.0.3.tgz", - - - - "integrity": "sha512-N3WMsuqV66lT30CrXNbEjx4GEwlow3v6rr4mCcv6prnfwhS01rkgyFdjPNBYd9br7LpXV1+Emh01fHnq2Gdgrw==", - - - - "license": "MIT" - - - }, - - - "node_modules/undici-types": { - - - - "version": "6.20.0", - - - - "resolved": "https://registry.npmjs.org/undici-types/-/undici-types-6.20.0.tgz", - - - - "integrity": "sha512-Ny6QZ2Nju20vw1SRHe3d9jVu6gJ+4e3+MMpqu7pqE5HT6WsTSlce++GQmK5UXS8mzV8DSYHrQH+Xrf2jVcuKNg==", - - - - "dev": true, - - - - "license": "MIT" - - - }, - - - "node_modules/webidl-conversions": { - - - - "version": "3.0.1", - - - - "resolved": "https://registry.npmjs.org/webidl-conversions/-/webidl-conversions-3.0.1.tgz", - - - - "integrity": "sha512-2JAn3z8AR6rjK8Sm8orRC0h/bcl/DqL7tRPdGZ4I1CjdF+EaMLmYxBHyXuKL849eucPFhvBoxMsflfOb8kxaeQ==", - - - - "license": "BSD-2-Clause" - - - }, - - - "node_modules/whatwg-url": { - - - - "version": "5.0.0", - - - - "resolved": "https://registry.npmjs.org/whatwg-url/-/whatwg-url-5.0.0.tgz", - - - - "integrity": "sha512-saE57nupxk6v3HY35+jzBwYa0rKSy0XR8JSxZPwgLr7ys0IBzhGviA1/TUGJLmSVqs8pb9AnvICXEuOHLprYTw==", - - - - "license": "MIT", - - - - "dependencies": { - - - - - "tr46": "~0.0.3", - - - - - "webidl-conversions": "^3.0.0" - - - - } - - - } - + "": { + "name": "tgstorebot", + "version": "1.0.0", + "license": "MIT", + "dependencies": { + "better-sqlite3": "^11.8.1", + "dotenv": "^10.0.0", + "grammy": "^1.34.1", + "node-fetch": "^3.0.0", + "winston": "^3.17.0" + }, + "devDependencies": { + "@grammyjs/types": "^3.18.0", + "@types/better-sqlite3": "^7.6.12", + "@types/node": "^16.18.126", + "@typescript-eslint/eslint-plugin": "^8.23.0", + "@typescript-eslint/parser": "^8.23.0", + "eslint": "^9.19.0", + "ncp": "^2.0.0", + "ts-node": "^10.9.2", + "typescript": "^4.9.5" + } + }, + "node_modules/@colors/colors": { + "version": "1.6.0", + "resolved": "https://registry.npmjs.org/@colors/colors/-/colors-1.6.0.tgz", + "integrity": "sha512-Ir+AOibqzrIsL6ajt3Rz3LskB7OiMVHqltZmspbW/TJuTVuyOMirVqAkjfY6JISiLHgyNqicAC8AyHHGzNd/dA==", + "license": "MIT", + "engines": { + "node": ">=0.1.90" + } + }, + "node_modules/@cspotcode/source-map-support": { + "version": "0.8.1", + "resolved": "https://registry.npmjs.org/@cspotcode/source-map-support/-/source-map-support-0.8.1.tgz", + "integrity": "sha512-IchNf6dN4tHoMFIn/7OE8LWZ19Y6q/67Bmf6vnGREv8RSbBVb9LPJxEcnwrcwX6ixSvaiGoomAUvu4YSxXrVgw==", + "dev": true, + "license": "MIT", + "dependencies": { + "@jridgewell/trace-mapping": "0.3.9" + }, + "engines": { + "node": ">=12" + } + }, + "node_modules/@dabh/diagnostics": { + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/@dabh/diagnostics/-/diagnostics-2.0.3.tgz", + "integrity": "sha512-hrlQOIi7hAfzsMqlGSFyVucrx38O+j6wiGOf//H2ecvIEqYN4ADBSS2iLMh5UFyDunCNniUIPk/q3riFv45xRA==", + "license": "MIT", + "dependencies": { + "colorspace": "1.1.x", + "enabled": "2.0.x", + "kuler": "^2.0.0" + } + }, + "node_modules/@eslint-community/eslint-utils": { + "version": "4.4.1", + "resolved": "https://registry.npmjs.org/@eslint-community/eslint-utils/-/eslint-utils-4.4.1.tgz", + "integrity": "sha512-s3O3waFUrMV8P/XaF/+ZTp1X9XBZW1a4B97ZnjQF2KYWaFD2A8KyFBsrsfSjEmjn3RGWAIuvlneuZm3CUK3jbA==", + "dev": true, + "license": "MIT", + "dependencies": { + "eslint-visitor-keys": "^3.4.3" + }, + "engines": { + "node": "^12.22.0 || ^14.17.0 || >=16.0.0" + }, + "funding": { + "url": "https://opencollective.com/eslint" + }, + "peerDependencies": { + "eslint": "^6.0.0 || ^7.0.0 || >=8.0.0" + } + }, + "node_modules/@eslint-community/regexpp": { + "version": "4.12.1", + "resolved": "https://registry.npmjs.org/@eslint-community/regexpp/-/regexpp-4.12.1.tgz", + "integrity": "sha512-CCZCDJuduB9OUkFkY2IgppNZMi2lBQgD2qzwXkEia16cge2pijY/aXi96CJMquDMn3nJdlPV1A5KrJEXwfLNzQ==", + "dev": true, + "license": "MIT", + "engines": { + "node": "^12.0.0 || ^14.0.0 || >=16.0.0" + } + }, + "node_modules/@eslint/config-array": { + "version": "0.19.2", + "resolved": "https://registry.npmjs.org/@eslint/config-array/-/config-array-0.19.2.tgz", + "integrity": "sha512-GNKqxfHG2ySmJOBSHg7LxeUx4xpuCoFjacmlCoYWEbaPXLwvfIjixRI12xCQZeULksQb23uiA8F40w5TojpV7w==", + "dev": true, + "license": "Apache-2.0", + "dependencies": { + "@eslint/object-schema": "^2.1.6", + "debug": "^4.3.1", + "minimatch": "^3.1.2" + }, + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + } + }, + "node_modules/@eslint/config-array/node_modules/brace-expansion": { + "version": "1.1.11", + "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.11.tgz", + "integrity": "sha512-iCuPHDFgrHX7H2vEI/5xpz07zSHB00TpugqhmYtVmMO6518mCuRMoOYFldEBl0g187ufozdaHgWKcYFb61qGiA==", + "dev": true, + "license": "MIT", + "dependencies": { + "balanced-match": "^1.0.0", + "concat-map": "0.0.1" + } + }, + "node_modules/@eslint/config-array/node_modules/minimatch": { + "version": "3.1.2", + "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.1.2.tgz", + "integrity": "sha512-J7p63hRiAjw1NDEww1W7i37+ByIrOWO5XQQAzZ3VOcL0PNybwpfmV/N05zFAzwQ9USyEcX6t3UO+K5aqBQOIHw==", + "dev": true, + "license": "ISC", + "dependencies": { + "brace-expansion": "^1.1.7" + }, + "engines": { + "node": "*" + } + }, + "node_modules/@eslint/core": { + "version": "0.10.0", + "resolved": "https://registry.npmjs.org/@eslint/core/-/core-0.10.0.tgz", + "integrity": "sha512-gFHJ+xBOo4G3WRlR1e/3G8A6/KZAH6zcE/hkLRCZTi/B9avAG365QhFA8uOGzTMqgTghpn7/fSnscW++dpMSAw==", + "dev": true, + "license": "Apache-2.0", + "dependencies": { + "@types/json-schema": "^7.0.15" + }, + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + } + }, + "node_modules/@eslint/eslintrc": { + "version": "3.2.0", + "resolved": "https://registry.npmjs.org/@eslint/eslintrc/-/eslintrc-3.2.0.tgz", + "integrity": "sha512-grOjVNN8P3hjJn/eIETF1wwd12DdnwFDoyceUJLYYdkpbwq3nLi+4fqrTAONx7XDALqlL220wC/RHSC/QTI/0w==", + "dev": true, + "license": "MIT", + "dependencies": { + "ajv": "^6.12.4", + "debug": "^4.3.2", + "espree": "^10.0.1", + "globals": "^14.0.0", + "ignore": "^5.2.0", + "import-fresh": "^3.2.1", + "js-yaml": "^4.1.0", + "minimatch": "^3.1.2", + "strip-json-comments": "^3.1.1" + }, + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + }, + "funding": { + "url": "https://opencollective.com/eslint" + } + }, + "node_modules/@eslint/eslintrc/node_modules/brace-expansion": { + "version": "1.1.11", + "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.11.tgz", + "integrity": "sha512-iCuPHDFgrHX7H2vEI/5xpz07zSHB00TpugqhmYtVmMO6518mCuRMoOYFldEBl0g187ufozdaHgWKcYFb61qGiA==", + "dev": true, + "license": "MIT", + "dependencies": { + "balanced-match": "^1.0.0", + "concat-map": "0.0.1" + } + }, + "node_modules/@eslint/eslintrc/node_modules/minimatch": { + "version": "3.1.2", + "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.1.2.tgz", + "integrity": "sha512-J7p63hRiAjw1NDEww1W7i37+ByIrOWO5XQQAzZ3VOcL0PNybwpfmV/N05zFAzwQ9USyEcX6t3UO+K5aqBQOIHw==", + "dev": true, + "license": "ISC", + "dependencies": { + "brace-expansion": "^1.1.7" + }, + "engines": { + "node": "*" + } + }, + "node_modules/@eslint/js": { + "version": "9.19.0", + "resolved": "https://registry.npmjs.org/@eslint/js/-/js-9.19.0.tgz", + "integrity": "sha512-rbq9/g38qjfqFLOVPvwjIvFFdNziEC5S65jmjPw5r6A//QH+W91akh9irMwjDN8zKUTak6W9EsAv4m/7Wnw0UQ==", + "dev": true, + "license": "MIT", + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + } + }, + "node_modules/@eslint/object-schema": { + "version": "2.1.6", + "resolved": "https://registry.npmjs.org/@eslint/object-schema/-/object-schema-2.1.6.tgz", + "integrity": "sha512-RBMg5FRL0I0gs51M/guSAj5/e14VQ4tpZnQNWwuDT66P14I43ItmPfIZRhO9fUVIPOAQXU47atlywZ/czoqFPA==", + "dev": true, + "license": "Apache-2.0", + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + } + }, + "node_modules/@eslint/plugin-kit": { + "version": "0.2.5", + "resolved": "https://registry.npmjs.org/@eslint/plugin-kit/-/plugin-kit-0.2.5.tgz", + "integrity": "sha512-lB05FkqEdUg2AA0xEbUz0SnkXT1LcCTa438W4IWTUh4hdOnVbQyOJ81OrDXsJk/LSiJHubgGEFoR5EHq1NsH1A==", + "dev": true, + "license": "Apache-2.0", + "dependencies": { + "@eslint/core": "^0.10.0", + "levn": "^0.4.1" + }, + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + } + }, + "node_modules/@grammyjs/types": { + "version": "3.18.0", + "resolved": "https://registry.npmjs.org/@grammyjs/types/-/types-3.18.0.tgz", + "integrity": "sha512-/VpQcT6Z0+Lw6H6jv2JvnDmqB1/9rjbFdDBlPBPUW3kQ1bWfjzfaiVk2+J2eiid3o3sLaPp3ZK0XjEI1gWNc3g==", + "license": "MIT" + }, + "node_modules/@humanfs/core": { + "version": "0.19.1", + "resolved": "https://registry.npmjs.org/@humanfs/core/-/core-0.19.1.tgz", + "integrity": "sha512-5DyQ4+1JEUzejeK1JGICcideyfUbGixgS9jNgex5nqkW+cY7WZhxBigmieN5Qnw9ZosSNVC9KQKyb+GUaGyKUA==", + "dev": true, + "license": "Apache-2.0", + "engines": { + "node": ">=18.18.0" + } + }, + "node_modules/@humanfs/node": { + "version": "0.16.6", + "resolved": "https://registry.npmjs.org/@humanfs/node/-/node-0.16.6.tgz", + "integrity": "sha512-YuI2ZHQL78Q5HbhDiBA1X4LmYdXCKCMQIfw0pw7piHJwyREFebJUvrQN4cMssyES6x+vfUbx1CIpaQUKYdQZOw==", + "dev": true, + "license": "Apache-2.0", + "dependencies": { + "@humanfs/core": "^0.19.1", + "@humanwhocodes/retry": "^0.3.0" + }, + "engines": { + "node": ">=18.18.0" + } + }, + "node_modules/@humanfs/node/node_modules/@humanwhocodes/retry": { + "version": "0.3.1", + "resolved": "https://registry.npmjs.org/@humanwhocodes/retry/-/retry-0.3.1.tgz", + "integrity": "sha512-JBxkERygn7Bv/GbN5Rv8Ul6LVknS+5Bp6RgDC/O8gEBU/yeH5Ui5C/OlWrTb6qct7LjjfT6Re2NxB0ln0yYybA==", + "dev": true, + "license": "Apache-2.0", + "engines": { + "node": ">=18.18" + }, + "funding": { + "type": "github", + "url": "https://github.com/sponsors/nzakas" + } + }, + "node_modules/@humanwhocodes/module-importer": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/@humanwhocodes/module-importer/-/module-importer-1.0.1.tgz", + "integrity": "sha512-bxveV4V8v5Yb4ncFTT3rPSgZBOpCkjfK0y4oVVVJwIuDVBRMDXrPyXRL988i5ap9m9bnyEEjWfm5WkBmtffLfA==", + "dev": true, + "license": "Apache-2.0", + "engines": { + "node": ">=12.22" + }, + "funding": { + "type": "github", + "url": "https://github.com/sponsors/nzakas" + } + }, + "node_modules/@humanwhocodes/retry": { + "version": "0.4.1", + "resolved": "https://registry.npmjs.org/@humanwhocodes/retry/-/retry-0.4.1.tgz", + "integrity": "sha512-c7hNEllBlenFTHBky65mhq8WD2kbN9Q6gk0bTk8lSBvc554jpXSkST1iePudpt7+A/AQvuHs9EMqjHDXMY1lrA==", + "dev": true, + "license": "Apache-2.0", + "engines": { + "node": ">=18.18" + }, + "funding": { + "type": "github", + "url": "https://github.com/sponsors/nzakas" + } + }, + "node_modules/@jridgewell/resolve-uri": { + "version": "3.1.2", + "resolved": "https://registry.npmjs.org/@jridgewell/resolve-uri/-/resolve-uri-3.1.2.tgz", + "integrity": "sha512-bRISgCIjP20/tbWSPWMEi54QVPRZExkuD9lJL+UIxUKtwVJA8wW1Trb1jMs1RFXo1CBTNZ/5hpC9QvmKWdopKw==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6.0.0" + } + }, + "node_modules/@jridgewell/sourcemap-codec": { + "version": "1.5.0", + "resolved": "https://registry.npmjs.org/@jridgewell/sourcemap-codec/-/sourcemap-codec-1.5.0.tgz", + "integrity": "sha512-gv3ZRaISU3fjPAgNsriBRqGWQL6quFx04YMPW/zD8XMLsU32mhCCbfbO6KZFLjvYpCZ8zyDEgqsgf+PwPaM7GQ==", + "dev": true, + "license": "MIT" + }, + "node_modules/@jridgewell/trace-mapping": { + "version": "0.3.9", + "resolved": "https://registry.npmjs.org/@jridgewell/trace-mapping/-/trace-mapping-0.3.9.tgz", + "integrity": "sha512-3Belt6tdc8bPgAtbcmdtNJlirVoTmEb5e2gC94PnkwEW9jI6CAHUeoG85tjWP5WquqfavoMtMwiG4P926ZKKuQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "@jridgewell/resolve-uri": "^3.0.3", + "@jridgewell/sourcemap-codec": "^1.4.10" + } + }, + "node_modules/@nodelib/fs.scandir": { + "version": "2.1.5", + "resolved": "https://registry.npmjs.org/@nodelib/fs.scandir/-/fs.scandir-2.1.5.tgz", + "integrity": "sha512-vq24Bq3ym5HEQm2NKCr3yXDwjc7vTsEThRDnkp2DK9p1uqLR+DHurm/NOTo0KG7HYHU7eppKZj3MyqYuMBf62g==", + "dev": true, + "license": "MIT", + "dependencies": { + "@nodelib/fs.stat": "2.0.5", + "run-parallel": "^1.1.9" + }, + "engines": { + "node": ">= 8" + } + }, + "node_modules/@nodelib/fs.stat": { + "version": "2.0.5", + "resolved": "https://registry.npmjs.org/@nodelib/fs.stat/-/fs.stat-2.0.5.tgz", + "integrity": "sha512-RkhPPp2zrqDAQA/2jNhnztcPAlv64XdhIp7a7454A5ovI7Bukxgt7MX7udwAu3zg1DcpPU0rz3VV1SeaqvY4+A==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">= 8" + } + }, + "node_modules/@nodelib/fs.walk": { + "version": "1.2.8", + "resolved": "https://registry.npmjs.org/@nodelib/fs.walk/-/fs.walk-1.2.8.tgz", + "integrity": "sha512-oGB+UxlgWcgQkgwo8GcEGwemoTFt3FIO9ababBmaGwXIoBKZ+GTy0pP185beGg7Llih/NSHSV2XAs1lnznocSg==", + "dev": true, + "license": "MIT", + "dependencies": { + "@nodelib/fs.scandir": "2.1.5", + "fastq": "^1.6.0" + }, + "engines": { + "node": ">= 8" + } + }, + "node_modules/@tsconfig/node10": { + "version": "1.0.11", + "resolved": "https://registry.npmjs.org/@tsconfig/node10/-/node10-1.0.11.tgz", + "integrity": "sha512-DcRjDCujK/kCk/cUe8Xz8ZSpm8mS3mNNpta+jGCA6USEDfktlNvm1+IuZ9eTcDbNk41BHwpHHeW+N1lKCz4zOw==", + "dev": true, + "license": "MIT" + }, + "node_modules/@tsconfig/node12": { + "version": "1.0.11", + "resolved": "https://registry.npmjs.org/@tsconfig/node12/-/node12-1.0.11.tgz", + "integrity": "sha512-cqefuRsh12pWyGsIoBKJA9luFu3mRxCA+ORZvA4ktLSzIuCUtWVxGIuXigEwO5/ywWFMZ2QEGKWvkZG1zDMTag==", + "dev": true, + "license": "MIT" + }, + "node_modules/@tsconfig/node14": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/@tsconfig/node14/-/node14-1.0.3.tgz", + "integrity": "sha512-ysT8mhdixWK6Hw3i1V2AeRqZ5WfXg1G43mqoYlM2nc6388Fq5jcXyr5mRsqViLx/GJYdoL0bfXD8nmF+Zn/Iow==", + "dev": true, + "license": "MIT" + }, + "node_modules/@tsconfig/node16": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/@tsconfig/node16/-/node16-1.0.4.tgz", + "integrity": "sha512-vxhUy4J8lyeyinH7Azl1pdd43GJhZH/tP2weN8TntQblOY+A0XbT8DJk1/oCPuOOyg/Ja757rG0CgHcWC8OfMA==", + "dev": true, + "license": "MIT" + }, + "node_modules/@types/better-sqlite3": { + "version": "7.6.12", + "resolved": "https://registry.npmjs.org/@types/better-sqlite3/-/better-sqlite3-7.6.12.tgz", + "integrity": "sha512-fnQmj8lELIj7BSrZQAdBMHEHX8OZLYIHXqAKT1O7tDfLxaINzf00PMjw22r3N/xXh0w/sGHlO6SVaCQ2mj78lg==", + "dev": true, + "license": "MIT", + "dependencies": { + "@types/node": "*" + } + }, + "node_modules/@types/estree": { + "version": "1.0.6", + "resolved": "https://registry.npmjs.org/@types/estree/-/estree-1.0.6.tgz", + "integrity": "sha512-AYnb1nQyY49te+VRAVgmzfcgjYS91mY5P0TKUDCLEM+gNnA+3T6rWITXRLYCpahpqSQbN5cE+gHpnPyXjHWxcw==", + "dev": true, + "license": "MIT" + }, + "node_modules/@types/json-schema": { + "version": "7.0.15", + "resolved": "https://registry.npmjs.org/@types/json-schema/-/json-schema-7.0.15.tgz", + "integrity": "sha512-5+fP8P8MFNC+AyZCDxrB2pkZFPGzqQWUzpSeuuVLvm8VMcorNYavBqoFcxK8bQz4Qsbn4oUEEem4wDLfcysGHA==", + "dev": true, + "license": "MIT" + }, + "node_modules/@types/node": { + "version": "16.18.126", + "resolved": "https://registry.npmjs.org/@types/node/-/node-16.18.126.tgz", + "integrity": "sha512-OTcgaiwfGFBKacvfwuHzzn1KLxH/er8mluiy8/uM3sGXHaRe73RrSIj01jow9t4kJEW633Ov+cOexXeiApTyAw==", + "dev": true, + "license": "MIT" + }, + "node_modules/@types/triple-beam": { + "version": "1.3.5", + "resolved": "https://registry.npmjs.org/@types/triple-beam/-/triple-beam-1.3.5.tgz", + "integrity": "sha512-6WaYesThRMCl19iryMYP7/x2OVgCtbIVflDGFpWnb9irXI3UjYE4AzmYuiUKY1AJstGijoY+MgUszMgRxIYTYw==", + "license": "MIT" + }, + "node_modules/@typescript-eslint/eslint-plugin": { + "version": "8.23.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/eslint-plugin/-/eslint-plugin-8.23.0.tgz", + "integrity": "sha512-vBz65tJgRrA1Q5gWlRfvoH+w943dq9K1p1yDBY2pc+a1nbBLZp7fB9+Hk8DaALUbzjqlMfgaqlVPT1REJdkt/w==", + "dev": true, + "license": "MIT", + "dependencies": { + "@eslint-community/regexpp": "^4.10.0", + "@typescript-eslint/scope-manager": "8.23.0", + "@typescript-eslint/type-utils": "8.23.0", + "@typescript-eslint/utils": "8.23.0", + "@typescript-eslint/visitor-keys": "8.23.0", + "graphemer": "^1.4.0", + "ignore": "^5.3.1", + "natural-compare": "^1.4.0", + "ts-api-utils": "^2.0.1" + }, + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/typescript-eslint" + }, + "peerDependencies": { + "@typescript-eslint/parser": "^8.0.0 || ^8.0.0-alpha.0", + "eslint": "^8.57.0 || ^9.0.0", + "typescript": ">=4.8.4 <5.8.0" + } + }, + "node_modules/@typescript-eslint/parser": { + "version": "8.23.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/parser/-/parser-8.23.0.tgz", + "integrity": "sha512-h2lUByouOXFAlMec2mILeELUbME5SZRN/7R9Cw2RD2lRQQY08MWMM+PmVVKKJNK1aIwqTo9t/0CvOxwPbRIE2Q==", + "dev": true, + "license": "MIT", + "dependencies": { + "@typescript-eslint/scope-manager": "8.23.0", + "@typescript-eslint/types": "8.23.0", + "@typescript-eslint/typescript-estree": "8.23.0", + "@typescript-eslint/visitor-keys": "8.23.0", + "debug": "^4.3.4" + }, + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/typescript-eslint" + }, + "peerDependencies": { + "eslint": "^8.57.0 || ^9.0.0", + "typescript": ">=4.8.4 <5.8.0" + } + }, + "node_modules/@typescript-eslint/scope-manager": { + "version": "8.23.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/scope-manager/-/scope-manager-8.23.0.tgz", + "integrity": "sha512-OGqo7+dXHqI7Hfm+WqkZjKjsiRtFUQHPdGMXzk5mYXhJUedO7e/Y7i8AK3MyLMgZR93TX4bIzYrfyVjLC+0VSw==", + "dev": true, + "license": "MIT", + "dependencies": { + "@typescript-eslint/types": "8.23.0", + "@typescript-eslint/visitor-keys": "8.23.0" + }, + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/typescript-eslint" + } + }, + "node_modules/@typescript-eslint/type-utils": { + "version": "8.23.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/type-utils/-/type-utils-8.23.0.tgz", + "integrity": "sha512-iIuLdYpQWZKbiH+RkCGc6iu+VwscP5rCtQ1lyQ7TYuKLrcZoeJVpcLiG8DliXVkUxirW/PWlmS+d6yD51L9jvA==", + "dev": true, + "license": "MIT", + "dependencies": { + "@typescript-eslint/typescript-estree": "8.23.0", + "@typescript-eslint/utils": "8.23.0", + "debug": "^4.3.4", + "ts-api-utils": "^2.0.1" + }, + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/typescript-eslint" + }, + "peerDependencies": { + "eslint": "^8.57.0 || ^9.0.0", + "typescript": ">=4.8.4 <5.8.0" + } + }, + "node_modules/@typescript-eslint/types": { + "version": "8.23.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/types/-/types-8.23.0.tgz", + "integrity": "sha512-1sK4ILJbCmZOTt9k4vkoulT6/y5CHJ1qUYxqpF1K/DBAd8+ZUL4LlSCxOssuH5m4rUaaN0uS0HlVPvd45zjduQ==", + "dev": true, + "license": "MIT", + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/typescript-eslint" + } + }, + "node_modules/@typescript-eslint/typescript-estree": { + "version": "8.23.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/typescript-estree/-/typescript-estree-8.23.0.tgz", + "integrity": "sha512-LcqzfipsB8RTvH8FX24W4UUFk1bl+0yTOf9ZA08XngFwMg4Kj8A+9hwz8Cr/ZS4KwHrmo9PJiLZkOt49vPnuvQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "@typescript-eslint/types": "8.23.0", + "@typescript-eslint/visitor-keys": "8.23.0", + "debug": "^4.3.4", + "fast-glob": "^3.3.2", + "is-glob": "^4.0.3", + "minimatch": "^9.0.4", + "semver": "^7.6.0", + "ts-api-utils": "^2.0.1" + }, + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/typescript-eslint" + }, + "peerDependencies": { + "typescript": ">=4.8.4 <5.8.0" + } + }, + "node_modules/@typescript-eslint/utils": { + "version": "8.23.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/utils/-/utils-8.23.0.tgz", + "integrity": "sha512-uB/+PSo6Exu02b5ZEiVtmY6RVYO7YU5xqgzTIVZwTHvvK3HsL8tZZHFaTLFtRG3CsV4A5mhOv+NZx5BlhXPyIA==", + "dev": true, + "license": "MIT", + "dependencies": { + "@eslint-community/eslint-utils": "^4.4.0", + "@typescript-eslint/scope-manager": "8.23.0", + "@typescript-eslint/types": "8.23.0", + "@typescript-eslint/typescript-estree": "8.23.0" + }, + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/typescript-eslint" + }, + "peerDependencies": { + "eslint": "^8.57.0 || ^9.0.0", + "typescript": ">=4.8.4 <5.8.0" + } + }, + "node_modules/@typescript-eslint/visitor-keys": { + "version": "8.23.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/visitor-keys/-/visitor-keys-8.23.0.tgz", + "integrity": "sha512-oWWhcWDLwDfu++BGTZcmXWqpwtkwb5o7fxUIGksMQQDSdPW9prsSnfIOZMlsj4vBOSrcnjIUZMiIjODgGosFhQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "@typescript-eslint/types": "8.23.0", + "eslint-visitor-keys": "^4.2.0" + }, + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/typescript-eslint" + } + }, + "node_modules/@typescript-eslint/visitor-keys/node_modules/eslint-visitor-keys": { + "version": "4.2.0", + "resolved": "https://registry.npmjs.org/eslint-visitor-keys/-/eslint-visitor-keys-4.2.0.tgz", + "integrity": "sha512-UyLnSehNt62FFhSwjZlHmeokpRK59rcz29j+F1/aDgbkbRTk7wIc9XzdoasMUbRNKDM0qQt/+BJ4BrpFeABemw==", + "dev": true, + "license": "Apache-2.0", + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + }, + "funding": { + "url": "https://opencollective.com/eslint" + } + }, + "node_modules/abort-controller": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/abort-controller/-/abort-controller-3.0.0.tgz", + "integrity": "sha512-h8lQ8tacZYnR3vNQTgibj+tODHI5/+l06Au2Pcriv/Gmet0eaj4TwWH41sO9wnHDiQsEj19q0drzdWdeAHtweg==", + "license": "MIT", + "dependencies": { + "event-target-shim": "^5.0.0" + }, + "engines": { + "node": ">=6.5" + } + }, + "node_modules/acorn": { + "version": "8.14.0", + "resolved": "https://registry.npmjs.org/acorn/-/acorn-8.14.0.tgz", + "integrity": "sha512-cl669nCJTZBsL97OF4kUQm5g5hC2uihk0NxY3WENAC0TYdILVkAyHymAntgxGkl7K+t0cXIrH5siy5S4XkFycA==", + "dev": true, + "license": "MIT", + "bin": { + "acorn": "bin/acorn" + }, + "engines": { + "node": ">=0.4.0" + } + }, + "node_modules/acorn-jsx": { + "version": "5.3.2", + "resolved": "https://registry.npmjs.org/acorn-jsx/-/acorn-jsx-5.3.2.tgz", + "integrity": "sha512-rq9s+JNhf0IChjtDXxllJ7g41oZk5SlXtp0LHwyA5cejwn7vKmKp4pPri6YEePv2PU65sAsegbXtIinmDFDXgQ==", + "dev": true, + "license": "MIT", + "peerDependencies": { + "acorn": "^6.0.0 || ^7.0.0 || ^8.0.0" + } + }, + "node_modules/acorn-walk": { + "version": "8.3.4", + "resolved": "https://registry.npmjs.org/acorn-walk/-/acorn-walk-8.3.4.tgz", + "integrity": "sha512-ueEepnujpqee2o5aIYnvHU6C0A42MNdsIDeqy5BydrkuC5R1ZuUFnm27EeFJGoEHJQgn3uleRvmTXaJgfXbt4g==", + "dev": true, + "license": "MIT", + "dependencies": { + "acorn": "^8.11.0" + }, + "engines": { + "node": ">=0.4.0" + } + }, + "node_modules/ajv": { + "version": "6.12.6", + "resolved": "https://registry.npmjs.org/ajv/-/ajv-6.12.6.tgz", + "integrity": "sha512-j3fVLgvTo527anyYyJOGTYJbG+vnnQYvE0m5mmkc1TK+nxAppkCLMIL0aZ4dblVCNoGShhm+kzE4ZUykBoMg4g==", + "dev": true, + "license": "MIT", + "dependencies": { + "fast-deep-equal": "^3.1.1", + "fast-json-stable-stringify": "^2.0.0", + "json-schema-traverse": "^0.4.1", + "uri-js": "^4.2.2" + }, + "funding": { + "type": "github", + "url": "https://github.com/sponsors/epoberezkin" + } + }, + "node_modules/ansi-styles": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", + "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", + "dev": true, + "license": "MIT", + "dependencies": { + "color-convert": "^2.0.1" + }, + "engines": { + "node": ">=8" + }, + "funding": { + "url": "https://github.com/chalk/ansi-styles?sponsor=1" + } + }, + "node_modules/arg": { + "version": "4.1.3", + "resolved": "https://registry.npmjs.org/arg/-/arg-4.1.3.tgz", + "integrity": "sha512-58S9QDqG0Xx27YwPSt9fJxivjYl432YCwfDMfZ+71RAqUrZef7LrKQZ3LHLOwCS4FLNBplP533Zx895SeOCHvA==", + "dev": true, + "license": "MIT" + }, + "node_modules/argparse": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/argparse/-/argparse-2.0.1.tgz", + "integrity": "sha512-8+9WqebbFzpX9OR+Wa6O29asIogeRMzcGtAINdpMHHyAg10f05aSFVBbcEqGf/PXw1EjAZ+q2/bEBg3DvurK3Q==", + "dev": true, + "license": "Python-2.0" + }, + "node_modules/async": { + "version": "3.2.6", + "resolved": "https://registry.npmjs.org/async/-/async-3.2.6.tgz", + "integrity": "sha512-htCUDlxyyCLMgaM3xXg0C0LW2xqfuQ6p05pCEIsXuyQ+a1koYKTuBMzRNwmybfLgvJDMd0r1LTn4+E0Ti6C2AA==", + "license": "MIT" + }, + "node_modules/balanced-match": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-1.0.2.tgz", + "integrity": "sha512-3oSeUO0TMV67hN1AmbXsK4yaqU7tjiHlbxRDZOpH0KW9+CeX4bRAaX0Anxt0tx2MrpRpWwQaPwIlISEJhYU5Pw==", + "dev": true, + "license": "MIT" + }, + "node_modules/base64-js": { + "version": "1.5.1", + "resolved": "https://registry.npmjs.org/base64-js/-/base64-js-1.5.1.tgz", + "integrity": "sha512-AKpaYlHn8t4SVbOHCy+b5+KKgvR4vrsD8vbvrbiQJps7fKDTkjkDry6ji0rUJjC0kzbNePLwzxq8iypo41qeWA==", + "funding": [ + { + "type": "github", + "url": "https://github.com/sponsors/feross" + }, + { + "type": "patreon", + "url": "https://www.patreon.com/feross" + }, + { + "type": "consulting", + "url": "https://feross.org/support" + } + ], + "license": "MIT" + }, + "node_modules/better-sqlite3": { + "version": "11.8.1", + "resolved": "https://registry.npmjs.org/better-sqlite3/-/better-sqlite3-11.8.1.tgz", + "integrity": "sha512-9BxNaBkblMjhJW8sMRZxnxVTRgbRmssZW0Oxc1MPBTfiR+WW21e2Mk4qu8CzrcZb1LwPCnFsfDEzq+SNcBU8eg==", + "hasInstallScript": true, + "license": "MIT", + "dependencies": { + "bindings": "^1.5.0", + "prebuild-install": "^7.1.1" + } + }, + "node_modules/bindings": { + "version": "1.5.0", + "resolved": "https://registry.npmjs.org/bindings/-/bindings-1.5.0.tgz", + "integrity": "sha512-p2q/t/mhvuOj/UeLlV6566GD/guowlr0hHxClI0W9m7MWYkL1F0hLo+0Aexs9HSPCtR1SXQ0TD3MMKrXZajbiQ==", + "license": "MIT", + "dependencies": { + "file-uri-to-path": "1.0.0" + } + }, + "node_modules/bl": { + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/bl/-/bl-4.1.0.tgz", + "integrity": "sha512-1W07cM9gS6DcLperZfFSj+bWLtaPGSOHWhPiGzXmvVJbRLdG82sH/Kn8EtW1VqWVA54AKf2h5k5BbnIbwF3h6w==", + "license": "MIT", + "dependencies": { + "buffer": "^5.5.0", + "inherits": "^2.0.4", + "readable-stream": "^3.4.0" + } + }, + "node_modules/brace-expansion": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-2.0.1.tgz", + "integrity": "sha512-XnAIvQ8eM+kC6aULx6wuQiwVsnzsi9d3WxzV3FpWTGA19F621kwdbsAcFKXgKUHZWsy+mY6iL1sHTxWEFCytDA==", + "dev": true, + "license": "MIT", + "dependencies": { + "balanced-match": "^1.0.0" + } + }, + "node_modules/braces": { + "version": "3.0.3", + "resolved": "https://registry.npmjs.org/braces/-/braces-3.0.3.tgz", + "integrity": "sha512-yQbXgO/OSZVD2IsiLlro+7Hf6Q18EJrKSEsdoMzKePKXct3gvD8oLcOQdIzGupr5Fj+EDe8gO/lxc1BzfMpxvA==", + "dev": true, + "license": "MIT", + "dependencies": { + "fill-range": "^7.1.1" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/buffer": { + "version": "5.7.1", + "resolved": "https://registry.npmjs.org/buffer/-/buffer-5.7.1.tgz", + "integrity": "sha512-EHcyIPBQ4BSGlvjB16k5KgAJ27CIsHY/2JBmCRReo48y9rQ3MaUzWX3KVlBa4U7MyX02HdVj0K7C3WaB3ju7FQ==", + "funding": [ + { + "type": "github", + "url": "https://github.com/sponsors/feross" + }, + { + "type": "patreon", + "url": "https://www.patreon.com/feross" + }, + { + "type": "consulting", + "url": "https://feross.org/support" + } + ], + "license": "MIT", + "dependencies": { + "base64-js": "^1.3.1", + "ieee754": "^1.1.13" + } + }, + "node_modules/callsites": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/callsites/-/callsites-3.1.0.tgz", + "integrity": "sha512-P8BjAsXvZS+VIDUI11hHCQEv74YT67YUi5JJFNWIqL235sBmjX4+qx9Muvls5ivyNENctx46xQLQ3aTuE7ssaQ==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6" + } + }, + "node_modules/chalk": { + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", + "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", + "dev": true, + "license": "MIT", + "dependencies": { + "ansi-styles": "^4.1.0", + "supports-color": "^7.1.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/chalk/chalk?sponsor=1" + } + }, + "node_modules/chownr": { + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/chownr/-/chownr-1.1.4.tgz", + "integrity": "sha512-jJ0bqzaylmJtVnNgzTeSOs8DPavpbYgEr/b0YL8/2GO3xJEhInFmhKMUnEJQjZumK7KXGFhUy89PrsJWlakBVg==", + "license": "ISC" + }, + "node_modules/color": { + "version": "3.2.1", + "resolved": "https://registry.npmjs.org/color/-/color-3.2.1.tgz", + "integrity": "sha512-aBl7dZI9ENN6fUGC7mWpMTPNHmWUSNan9tuWN6ahh5ZLNk9baLJOnSMlrQkHcrfFgz2/RigjUVAjdx36VcemKA==", + "license": "MIT", + "dependencies": { + "color-convert": "^1.9.3", + "color-string": "^1.6.0" + } + }, + "node_modules/color-convert": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", + "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "color-name": "~1.1.4" + }, + "engines": { + "node": ">=7.0.0" + } + }, + "node_modules/color-name": { + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", + "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", + "license": "MIT" + }, + "node_modules/color-string": { + "version": "1.9.1", + "resolved": "https://registry.npmjs.org/color-string/-/color-string-1.9.1.tgz", + "integrity": "sha512-shrVawQFojnZv6xM40anx4CkoDP+fZsw/ZerEMsW/pyzsRbElpsL/DBVW7q3ExxwusdNXI3lXpuhEZkzs8p5Eg==", + "license": "MIT", + "dependencies": { + "color-name": "^1.0.0", + "simple-swizzle": "^0.2.2" + } + }, + "node_modules/color/node_modules/color-convert": { + "version": "1.9.3", + "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-1.9.3.tgz", + "integrity": "sha512-QfAUtd+vFdAtFQcC8CCyYt1fYWxSqAiK2cSD6zDB8N3cpsEBAvRxp9zOGg6G/SHHJYAT88/az/IuDGALsNVbGg==", + "license": "MIT", + "dependencies": { + "color-name": "1.1.3" + } + }, + "node_modules/color/node_modules/color-name": { + "version": "1.1.3", + "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.3.tgz", + "integrity": "sha512-72fSenhMw2HZMTVHeCA9KCmpEIbzWiQsjN+BHcBbS9vr1mtt+vJjPdksIBNUmKAW8TFUDPJK5SUU3QhE9NEXDw==", + "license": "MIT" + }, + "node_modules/colorspace": { + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/colorspace/-/colorspace-1.1.4.tgz", + "integrity": "sha512-BgvKJiuVu1igBUF2kEjRCZXol6wiiGbY5ipL/oVPwm0BL9sIpMIzM8IK7vwuxIIzOXMV3Ey5w+vxhm0rR/TN8w==", + "license": "MIT", + "dependencies": { + "color": "^3.1.3", + "text-hex": "1.0.x" + } + }, + "node_modules/concat-map": { + "version": "0.0.1", + "resolved": "https://registry.npmjs.org/concat-map/-/concat-map-0.0.1.tgz", + "integrity": "sha512-/Srv4dswyQNBfohGpz9o6Yb3Gz3SrUDqBH5rTuhGR7ahtlbYKnVxw2bCFMRljaA7EXHaXZ8wsHdodFvbkhKmqg==", + "dev": true, + "license": "MIT" + }, + "node_modules/create-require": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/create-require/-/create-require-1.1.1.tgz", + "integrity": "sha512-dcKFX3jn0MpIaXjisoRvexIJVEKzaq7z2rZKxf+MSr9TkdmHmsU4m2lcLojrj/FHl8mk5VxMmYA+ftRkP/3oKQ==", + "dev": true, + "license": "MIT" + }, + "node_modules/cross-spawn": { + "version": "7.0.6", + "resolved": "https://registry.npmjs.org/cross-spawn/-/cross-spawn-7.0.6.tgz", + "integrity": "sha512-uV2QOWP2nWzsy2aMp8aRibhi9dlzF5Hgh5SHaB9OiTGEyDTiJJyx0uy51QXdyWbtAHNua4XJzUKca3OzKUd3vA==", + "dev": true, + "license": "MIT", + "dependencies": { + "path-key": "^3.1.0", + "shebang-command": "^2.0.0", + "which": "^2.0.1" + }, + "engines": { + "node": ">= 8" + } + }, + "node_modules/data-uri-to-buffer": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/data-uri-to-buffer/-/data-uri-to-buffer-4.0.1.tgz", + "integrity": "sha512-0R9ikRb668HB7QDxT1vkpuUBtqc53YyAwMwGeUFKRojY/NWKvdZ+9UYtRfGmhqNbRkTSVpMbmyhXipFFv2cb/A==", + "license": "MIT", + "engines": { + "node": ">= 12" + } + }, + "node_modules/debug": { + "version": "4.4.0", + "resolved": "https://registry.npmjs.org/debug/-/debug-4.4.0.tgz", + "integrity": "sha512-6WTZ/IxCY/T6BALoZHaE4ctp9xm+Z5kY/pzYaCHRFeyVhojxlrm+46y68HA6hr0TcwEssoxNiDEUJQjfPZ/RYA==", + "license": "MIT", + "dependencies": { + "ms": "^2.1.3" + }, + "engines": { + "node": ">=6.0" + }, + "peerDependenciesMeta": { + "supports-color": { + "optional": true + } + } + }, + "node_modules/decompress-response": { + "version": "6.0.0", + "resolved": "https://registry.npmjs.org/decompress-response/-/decompress-response-6.0.0.tgz", + "integrity": "sha512-aW35yZM6Bb/4oJlZncMH2LCoZtJXTRxES17vE3hoRiowU2kWHaJKFkSBDnDR+cm9J+9QhXmREyIfv0pji9ejCQ==", + "license": "MIT", + "dependencies": { + "mimic-response": "^3.1.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/deep-extend": { + "version": "0.6.0", + "resolved": "https://registry.npmjs.org/deep-extend/-/deep-extend-0.6.0.tgz", + "integrity": "sha512-LOHxIOaPYdHlJRtCQfDIVZtfw/ufM8+rVj649RIHzcm/vGwQRXFt6OPqIFWsm2XEMrNIEtWR64sY1LEKD2vAOA==", + "license": "MIT", + "engines": { + "node": ">=4.0.0" + } + }, + "node_modules/deep-is": { + "version": "0.1.4", + "resolved": "https://registry.npmjs.org/deep-is/-/deep-is-0.1.4.tgz", + "integrity": "sha512-oIPzksmTg4/MriiaYGO+okXDT7ztn/w3Eptv/+gSIdMdKsJo0u4CfYNFJPy+4SKMuCqGw2wxnA+URMg3t8a/bQ==", + "dev": true, + "license": "MIT" + }, + "node_modules/detect-libc": { + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/detect-libc/-/detect-libc-2.0.3.tgz", + "integrity": "sha512-bwy0MGW55bG41VqxxypOsdSdGqLwXPI/focwgTYCFMbdUiBAxLg9CFzG08sz2aqzknwiX7Hkl0bQENjg8iLByw==", + "license": "Apache-2.0", + "engines": { + "node": ">=8" + } + }, + "node_modules/diff": { + "version": "4.0.2", + "resolved": "https://registry.npmjs.org/diff/-/diff-4.0.2.tgz", + "integrity": "sha512-58lmxKSA4BNyLz+HHMUzlOEpg09FV+ev6ZMe3vJihgdxzgcwZ8VoEEPmALCZG9LmqfVoNMMKpttIYTVG6uDY7A==", + "dev": true, + "license": "BSD-3-Clause", + "engines": { + "node": ">=0.3.1" + } + }, + "node_modules/dotenv": { + "version": "10.0.0", + "resolved": "https://registry.npmjs.org/dotenv/-/dotenv-10.0.0.tgz", + "integrity": "sha512-rlBi9d8jpv9Sf1klPjNfFAuWDjKLwTIJJ/VxtoTwIR6hnZxcEOQCZg2oIL3MWBYw5GpUDKOEnND7LXTbIpQ03Q==", + "license": "BSD-2-Clause", + "engines": { + "node": ">=10" + } + }, + "node_modules/enabled": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/enabled/-/enabled-2.0.0.tgz", + "integrity": "sha512-AKrN98kuwOzMIdAizXGI86UFBoo26CL21UM763y1h/GMSJ4/OHU9k2YlsmBpyScFo/wbLzWQJBMCW4+IO3/+OQ==", + "license": "MIT" + }, + "node_modules/end-of-stream": { + "version": "1.4.4", + "resolved": "https://registry.npmjs.org/end-of-stream/-/end-of-stream-1.4.4.tgz", + "integrity": "sha512-+uw1inIHVPQoaVuHzRyXd21icM+cnt4CzD5rW+NC1wjOUSTOs+Te7FOv7AhN7vS9x/oIyhLP5PR1H+phQAHu5Q==", + "license": "MIT", + "dependencies": { + "once": "^1.4.0" + } + }, + "node_modules/escape-string-regexp": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-4.0.0.tgz", + "integrity": "sha512-TtpcNJ3XAzx3Gq8sWRzJaVajRs0uVxA2YAkdb1jm2YkPz4G6egUFAyA3n5vtEIZefPk5Wa4UXbKuS5fKkJWdgA==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/eslint": { + "version": "9.19.0", + "resolved": "https://registry.npmjs.org/eslint/-/eslint-9.19.0.tgz", + "integrity": "sha512-ug92j0LepKlbbEv6hD911THhoRHmbdXt2gX+VDABAW/Ir7D3nqKdv5Pf5vtlyY6HQMTEP2skXY43ueqTCWssEA==", + "dev": true, + "license": "MIT", + "dependencies": { + "@eslint-community/eslint-utils": "^4.2.0", + "@eslint-community/regexpp": "^4.12.1", + "@eslint/config-array": "^0.19.0", + "@eslint/core": "^0.10.0", + "@eslint/eslintrc": "^3.2.0", + "@eslint/js": "9.19.0", + "@eslint/plugin-kit": "^0.2.5", + "@humanfs/node": "^0.16.6", + "@humanwhocodes/module-importer": "^1.0.1", + "@humanwhocodes/retry": "^0.4.1", + "@types/estree": "^1.0.6", + "@types/json-schema": "^7.0.15", + "ajv": "^6.12.4", + "chalk": "^4.0.0", + "cross-spawn": "^7.0.6", + "debug": "^4.3.2", + "escape-string-regexp": "^4.0.0", + "eslint-scope": "^8.2.0", + "eslint-visitor-keys": "^4.2.0", + "espree": "^10.3.0", + "esquery": "^1.5.0", + "esutils": "^2.0.2", + "fast-deep-equal": "^3.1.3", + "file-entry-cache": "^8.0.0", + "find-up": "^5.0.0", + "glob-parent": "^6.0.2", + "ignore": "^5.2.0", + "imurmurhash": "^0.1.4", + "is-glob": "^4.0.0", + "json-stable-stringify-without-jsonify": "^1.0.1", + "lodash.merge": "^4.6.2", + "minimatch": "^3.1.2", + "natural-compare": "^1.4.0", + "optionator": "^0.9.3" + }, + "bin": { + "eslint": "bin/eslint.js" + }, + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + }, + "funding": { + "url": "https://eslint.org/donate" + }, + "peerDependencies": { + "jiti": "*" + }, + "peerDependenciesMeta": { + "jiti": { + "optional": true + } + } + }, + "node_modules/eslint-scope": { + "version": "8.2.0", + "resolved": "https://registry.npmjs.org/eslint-scope/-/eslint-scope-8.2.0.tgz", + "integrity": "sha512-PHlWUfG6lvPc3yvP5A4PNyBL1W8fkDUccmI21JUu/+GKZBoH/W5u6usENXUrWFRsyoW5ACUjFGgAFQp5gUlb/A==", + "dev": true, + "license": "BSD-2-Clause", + "dependencies": { + "esrecurse": "^4.3.0", + "estraverse": "^5.2.0" + }, + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + }, + "funding": { + "url": "https://opencollective.com/eslint" + } + }, + "node_modules/eslint-visitor-keys": { + "version": "3.4.3", + "resolved": "https://registry.npmjs.org/eslint-visitor-keys/-/eslint-visitor-keys-3.4.3.tgz", + "integrity": "sha512-wpc+LXeiyiisxPlEkUzU6svyS1frIO3Mgxj1fdy7Pm8Ygzguax2N3Fa/D/ag1WqbOprdI+uY6wMUl8/a2G+iag==", + "dev": true, + "license": "Apache-2.0", + "engines": { + "node": "^12.22.0 || ^14.17.0 || >=16.0.0" + }, + "funding": { + "url": "https://opencollective.com/eslint" + } + }, + "node_modules/eslint/node_modules/brace-expansion": { + "version": "1.1.11", + "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.11.tgz", + "integrity": "sha512-iCuPHDFgrHX7H2vEI/5xpz07zSHB00TpugqhmYtVmMO6518mCuRMoOYFldEBl0g187ufozdaHgWKcYFb61qGiA==", + "dev": true, + "license": "MIT", + "dependencies": { + "balanced-match": "^1.0.0", + "concat-map": "0.0.1" + } + }, + "node_modules/eslint/node_modules/eslint-visitor-keys": { + "version": "4.2.0", + "resolved": "https://registry.npmjs.org/eslint-visitor-keys/-/eslint-visitor-keys-4.2.0.tgz", + "integrity": "sha512-UyLnSehNt62FFhSwjZlHmeokpRK59rcz29j+F1/aDgbkbRTk7wIc9XzdoasMUbRNKDM0qQt/+BJ4BrpFeABemw==", + "dev": true, + "license": "Apache-2.0", + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + }, + "funding": { + "url": "https://opencollective.com/eslint" + } + }, + "node_modules/eslint/node_modules/minimatch": { + "version": "3.1.2", + "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.1.2.tgz", + "integrity": "sha512-J7p63hRiAjw1NDEww1W7i37+ByIrOWO5XQQAzZ3VOcL0PNybwpfmV/N05zFAzwQ9USyEcX6t3UO+K5aqBQOIHw==", + "dev": true, + "license": "ISC", + "dependencies": { + "brace-expansion": "^1.1.7" + }, + "engines": { + "node": "*" + } + }, + "node_modules/espree": { + "version": "10.3.0", + "resolved": "https://registry.npmjs.org/espree/-/espree-10.3.0.tgz", + "integrity": "sha512-0QYC8b24HWY8zjRnDTL6RiHfDbAWn63qb4LMj1Z4b076A4une81+z03Kg7l7mn/48PUTqoLptSXez8oknU8Clg==", + "dev": true, + "license": "BSD-2-Clause", + "dependencies": { + "acorn": "^8.14.0", + "acorn-jsx": "^5.3.2", + "eslint-visitor-keys": "^4.2.0" + }, + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + }, + "funding": { + "url": "https://opencollective.com/eslint" + } + }, + "node_modules/espree/node_modules/eslint-visitor-keys": { + "version": "4.2.0", + "resolved": "https://registry.npmjs.org/eslint-visitor-keys/-/eslint-visitor-keys-4.2.0.tgz", + "integrity": "sha512-UyLnSehNt62FFhSwjZlHmeokpRK59rcz29j+F1/aDgbkbRTk7wIc9XzdoasMUbRNKDM0qQt/+BJ4BrpFeABemw==", + "dev": true, + "license": "Apache-2.0", + "engines": { + "node": "^18.18.0 || ^20.9.0 || >=21.1.0" + }, + "funding": { + "url": "https://opencollective.com/eslint" + } + }, + "node_modules/esquery": { + "version": "1.6.0", + "resolved": "https://registry.npmjs.org/esquery/-/esquery-1.6.0.tgz", + "integrity": "sha512-ca9pw9fomFcKPvFLXhBKUK90ZvGibiGOvRJNbjljY7s7uq/5YO4BOzcYtJqExdx99rF6aAcnRxHmcUHcz6sQsg==", + "dev": true, + "license": "BSD-3-Clause", + "dependencies": { + "estraverse": "^5.1.0" + }, + "engines": { + "node": ">=0.10" + } + }, + "node_modules/esrecurse": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/esrecurse/-/esrecurse-4.3.0.tgz", + "integrity": "sha512-KmfKL3b6G+RXvP8N1vr3Tq1kL/oCFgn2NYXEtqP8/L3pKapUA4G8cFVaoF3SU323CD4XypR/ffioHmkti6/Tag==", + "dev": true, + "license": "BSD-2-Clause", + "dependencies": { + "estraverse": "^5.2.0" + }, + "engines": { + "node": ">=4.0" + } + }, + "node_modules/estraverse": { + "version": "5.3.0", + "resolved": "https://registry.npmjs.org/estraverse/-/estraverse-5.3.0.tgz", + "integrity": "sha512-MMdARuVEQziNTeJD8DgMqmhwR11BRQ/cBP+pLtYdSTnf3MIO8fFeiINEbX36ZdNlfU/7A9f3gUw49B3oQsvwBA==", + "dev": true, + "license": "BSD-2-Clause", + "engines": { + "node": ">=4.0" + } + }, + "node_modules/esutils": { + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/esutils/-/esutils-2.0.3.tgz", + "integrity": "sha512-kVscqXk4OCp68SZ0dkgEKVi6/8ij300KBWTJq32P/dYeWTSwK41WyTxalN1eRmA5Z9UU/LX9D7FWSmV9SAYx6g==", + "dev": true, + "license": "BSD-2-Clause", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/event-target-shim": { + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/event-target-shim/-/event-target-shim-5.0.1.tgz", + "integrity": "sha512-i/2XbnSz/uxRCU6+NdVJgKWDTM427+MqYbkQzD321DuCQJUqOuJKIA0IM2+W2xtYHdKOmZ4dR6fExsd4SXL+WQ==", + "license": "MIT", + "engines": { + "node": ">=6" + } + }, + "node_modules/expand-template": { + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/expand-template/-/expand-template-2.0.3.tgz", + "integrity": "sha512-XYfuKMvj4O35f/pOXLObndIRvyQ+/+6AhODh+OKWj9S9498pHHn/IMszH+gt0fBCRWMNfk1ZSp5x3AifmnI2vg==", + "license": "(MIT OR WTFPL)", + "engines": { + "node": ">=6" + } + }, + "node_modules/fast-deep-equal": { + "version": "3.1.3", + "resolved": "https://registry.npmjs.org/fast-deep-equal/-/fast-deep-equal-3.1.3.tgz", + "integrity": "sha512-f3qQ9oQy9j2AhBe/H9VC91wLmKBCCU/gDOnKNAYG5hswO7BLKj09Hc5HYNz9cGI++xlpDCIgDaitVs03ATR84Q==", + "dev": true, + "license": "MIT" + }, + "node_modules/fast-glob": { + "version": "3.3.3", + "resolved": "https://registry.npmjs.org/fast-glob/-/fast-glob-3.3.3.tgz", + "integrity": "sha512-7MptL8U0cqcFdzIzwOTHoilX9x5BrNqye7Z/LuC7kCMRio1EMSyqRK3BEAUD7sXRq4iT4AzTVuZdhgQ2TCvYLg==", + "dev": true, + "license": "MIT", + "dependencies": { + "@nodelib/fs.stat": "^2.0.2", + "@nodelib/fs.walk": "^1.2.3", + "glob-parent": "^5.1.2", + "merge2": "^1.3.0", + "micromatch": "^4.0.8" + }, + "engines": { + "node": ">=8.6.0" + } + }, + "node_modules/fast-glob/node_modules/glob-parent": { + "version": "5.1.2", + "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-5.1.2.tgz", + "integrity": "sha512-AOIgSQCepiJYwP3ARnGx+5VnTu2HBYdzbGP45eLw1vr3zB3vZLeyed1sC9hnbcOc9/SrMyM5RPQrkGz4aS9Zow==", + "dev": true, + "license": "ISC", + "dependencies": { + "is-glob": "^4.0.1" + }, + "engines": { + "node": ">= 6" + } + }, + "node_modules/fast-json-stable-stringify": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/fast-json-stable-stringify/-/fast-json-stable-stringify-2.1.0.tgz", + "integrity": "sha512-lhd/wF+Lk98HZoTCtlVraHtfh5XYijIjalXck7saUtuanSDyLMxnHhSXEDJqHxD7msR8D0uCmqlkwjCV8xvwHw==", + "dev": true, + "license": "MIT" + }, + "node_modules/fast-levenshtein": { + "version": "2.0.6", + "resolved": "https://registry.npmjs.org/fast-levenshtein/-/fast-levenshtein-2.0.6.tgz", + "integrity": "sha512-DCXu6Ifhqcks7TZKY3Hxp3y6qphY5SJZmrWMDrKcERSOXWQdMhU9Ig/PYrzyw/ul9jOIyh0N4M0tbC5hodg8dw==", + "dev": true, + "license": "MIT" + }, + "node_modules/fastq": { + "version": "1.19.0", + "resolved": "https://registry.npmjs.org/fastq/-/fastq-1.19.0.tgz", + "integrity": "sha512-7SFSRCNjBQIZH/xZR3iy5iQYR8aGBE0h3VG6/cwlbrpdciNYBMotQav8c1XI3HjHH+NikUpP53nPdlZSdWmFzA==", + "dev": true, + "license": "ISC", + "dependencies": { + "reusify": "^1.0.4" + } + }, + "node_modules/fecha": { + "version": "4.2.3", + "resolved": "https://registry.npmjs.org/fecha/-/fecha-4.2.3.tgz", + "integrity": "sha512-OP2IUU6HeYKJi3i0z4A19kHMQoLVs4Hc+DPqqxI2h/DPZHTm/vjsfC6P0b4jCMy14XizLBqvndQ+UilD7707Jw==", + "license": "MIT" + }, + "node_modules/fetch-blob": { + "version": "3.2.0", + "resolved": "https://registry.npmjs.org/fetch-blob/-/fetch-blob-3.2.0.tgz", + "integrity": "sha512-7yAQpD2UMJzLi1Dqv7qFYnPbaPx7ZfFK6PiIxQ4PfkGPyNyl2Ugx+a/umUonmKqjhM4DnfbMvdX6otXq83soQQ==", + "funding": [ + { + "type": "github", + "url": "https://github.com/sponsors/jimmywarting" + }, + { + "type": "paypal", + "url": "https://paypal.me/jimmywarting" + } + ], + "license": "MIT", + "dependencies": { + "node-domexception": "^1.0.0", + "web-streams-polyfill": "^3.0.3" + }, + "engines": { + "node": "^12.20 || >= 14.13" + } + }, + "node_modules/file-entry-cache": { + "version": "8.0.0", + "resolved": "https://registry.npmjs.org/file-entry-cache/-/file-entry-cache-8.0.0.tgz", + "integrity": "sha512-XXTUwCvisa5oacNGRP9SfNtYBNAMi+RPwBFmblZEF7N7swHYQS6/Zfk7SRwx4D5j3CH211YNRco1DEMNVfZCnQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "flat-cache": "^4.0.0" + }, + "engines": { + "node": ">=16.0.0" + } + }, + "node_modules/file-uri-to-path": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/file-uri-to-path/-/file-uri-to-path-1.0.0.tgz", + "integrity": "sha512-0Zt+s3L7Vf1biwWZ29aARiVYLx7iMGnEUl9x33fbB/j3jR81u/O2LbqK+Bm1CDSNDKVtJ/YjwY7TUd5SkeLQLw==", + "license": "MIT" + }, + "node_modules/fill-range": { + "version": "7.1.1", + "resolved": "https://registry.npmjs.org/fill-range/-/fill-range-7.1.1.tgz", + "integrity": "sha512-YsGpe3WHLK8ZYi4tWDg2Jy3ebRz2rXowDxnld4bkQB00cc/1Zw9AWnC0i9ztDJitivtQvaI9KaLyKrc+hBW0yg==", + "dev": true, + "license": "MIT", + "dependencies": { + "to-regex-range": "^5.0.1" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/find-up": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/find-up/-/find-up-5.0.0.tgz", + "integrity": "sha512-78/PXT1wlLLDgTzDs7sjq9hzz0vXD+zn+7wypEe4fXQxCmdmqfGsEPQxmiCSQI3ajFV91bVSsvNtrJRiW6nGng==", + "dev": true, + "license": "MIT", + "dependencies": { + "locate-path": "^6.0.0", + "path-exists": "^4.0.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/flat-cache": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/flat-cache/-/flat-cache-4.0.1.tgz", + "integrity": "sha512-f7ccFPK3SXFHpx15UIGyRJ/FJQctuKZ0zVuN3frBo4HnK3cay9VEW0R6yPYFHC0AgqhukPzKjq22t5DmAyqGyw==", + "dev": true, + "license": "MIT", + "dependencies": { + "flatted": "^3.2.9", + "keyv": "^4.5.4" + }, + "engines": { + "node": ">=16" + } + }, + "node_modules/flatted": { + "version": "3.3.2", + "resolved": "https://registry.npmjs.org/flatted/-/flatted-3.3.2.tgz", + "integrity": "sha512-AiwGJM8YcNOaobumgtng+6NHuOqC3A7MixFeDafM3X9cIUM+xUXoS5Vfgf+OihAYe20fxqNM9yPBXJzRtZ/4eA==", + "dev": true, + "license": "ISC" + }, + "node_modules/fn.name": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/fn.name/-/fn.name-1.1.0.tgz", + "integrity": "sha512-GRnmB5gPyJpAhTQdSZTSp9uaPSvl09KoYcMQtsB9rQoOmzs9dH6ffeccH+Z+cv6P68Hu5bC6JjRh4Ah/mHSNRw==", + "license": "MIT" + }, + "node_modules/formdata-polyfill": { + "version": "4.0.10", + "resolved": "https://registry.npmjs.org/formdata-polyfill/-/formdata-polyfill-4.0.10.tgz", + "integrity": "sha512-buewHzMvYL29jdeQTVILecSaZKnt/RJWjoZCF5OW60Z67/GmSLBkOFM7qh1PI3zFNtJbaZL5eQu1vLfazOwj4g==", + "license": "MIT", + "dependencies": { + "fetch-blob": "^3.1.2" + }, + "engines": { + "node": ">=12.20.0" + } + }, + "node_modules/fs-constants": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/fs-constants/-/fs-constants-1.0.0.tgz", + "integrity": "sha512-y6OAwoSIf7FyjMIv94u+b5rdheZEjzR63GTyZJm5qh4Bi+2YgwLCcI/fPFZkL5PSixOt6ZNKm+w+Hfp/Bciwow==", + "license": "MIT" + }, + "node_modules/github-from-package": { + "version": "0.0.0", + "resolved": "https://registry.npmjs.org/github-from-package/-/github-from-package-0.0.0.tgz", + "integrity": "sha512-SyHy3T1v2NUXn29OsWdxmK6RwHD+vkj3v8en8AOBZ1wBQ/hCAQ5bAQTD02kW4W9tUp/3Qh6J8r9EvntiyCmOOw==", + "license": "MIT" + }, + "node_modules/glob-parent": { + "version": "6.0.2", + "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-6.0.2.tgz", + "integrity": "sha512-XxwI8EOhVQgWp6iDL+3b0r86f4d6AX6zSU55HfB4ydCEuXLXc5FcYeOu+nnGftS4TEju/11rt4KJPTMgbfmv4A==", + "dev": true, + "license": "ISC", + "dependencies": { + "is-glob": "^4.0.3" + }, + "engines": { + "node": ">=10.13.0" + } + }, + "node_modules/globals": { + "version": "14.0.0", + "resolved": "https://registry.npmjs.org/globals/-/globals-14.0.0.tgz", + "integrity": "sha512-oahGvuMGQlPw/ivIYBjVSrWAfWLBeku5tpPE2fOPLi+WHffIWbuh2tCjhyQhTBPMf5E9jDEH4FOmTYgYwbKwtQ==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=18" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/grammy": { + "version": "1.34.1", + "resolved": "https://registry.npmjs.org/grammy/-/grammy-1.34.1.tgz", + "integrity": "sha512-0ypt5d1bna8CqSANlDCg2ce1sx87oDm9gjyPRHWYpAgvjy3WYW9oEmMllAYCWCZ3vHDULW5x14jBMXai51ar0Q==", + "license": "MIT", + "dependencies": { + "@grammyjs/types": "3.18.0", + "abort-controller": "^3.0.0", + "debug": "^4.3.4", + "node-fetch": "^2.7.0" + }, + "engines": { + "node": "^12.20.0 || >=14.13.1" + } + }, + "node_modules/grammy/node_modules/node-fetch": { + "version": "2.7.0", + "resolved": "https://registry.npmjs.org/node-fetch/-/node-fetch-2.7.0.tgz", + "integrity": "sha512-c4FRfUm/dbcWZ7U+1Wq0AwCyFL+3nt2bEw05wfxSz+DWpWsitgmSgYmy2dQdWyKC1694ELPqMs/YzUSNozLt8A==", + "license": "MIT", + "dependencies": { + "whatwg-url": "^5.0.0" + }, + "engines": { + "node": "4.x || >=6.0.0" + }, + "peerDependencies": { + "encoding": "^0.1.0" + }, + "peerDependenciesMeta": { + "encoding": { + "optional": true + } + } + }, + "node_modules/graphemer": { + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/graphemer/-/graphemer-1.4.0.tgz", + "integrity": "sha512-EtKwoO6kxCL9WO5xipiHTZlSzBm7WLT627TqC/uVRd0HKmq8NXyebnNYxDoBi7wt8eTWrUrKXCOVaFq9x1kgag==", + "dev": true, + "license": "MIT" + }, + "node_modules/has-flag": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", + "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=8" + } + }, + "node_modules/ieee754": { + "version": "1.2.1", + "resolved": "https://registry.npmjs.org/ieee754/-/ieee754-1.2.1.tgz", + "integrity": "sha512-dcyqhDvX1C46lXZcVqCpK+FtMRQVdIMN6/Df5js2zouUsqG7I6sFxitIC+7KYK29KdXOLHdu9zL4sFnoVQnqaA==", + "funding": [ + { + "type": "github", + "url": "https://github.com/sponsors/feross" + }, + { + "type": "patreon", + "url": "https://www.patreon.com/feross" + }, + { + "type": "consulting", + "url": "https://feross.org/support" + } + ], + "license": "BSD-3-Clause" + }, + "node_modules/ignore": { + "version": "5.3.2", + "resolved": "https://registry.npmjs.org/ignore/-/ignore-5.3.2.tgz", + "integrity": "sha512-hsBTNUqQTDwkWtcdYI2i06Y/nUBEsNEDJKjWdigLvegy8kDuJAS8uRlpkkcQpyEXL0Z/pjDy5HBmMjRCJ2gq+g==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">= 4" + } + }, + "node_modules/import-fresh": { + "version": "3.3.1", + "resolved": "https://registry.npmjs.org/import-fresh/-/import-fresh-3.3.1.tgz", + "integrity": "sha512-TR3KfrTZTYLPB6jUjfx6MF9WcWrHL9su5TObK4ZkYgBdWKPOFoSoQIdEuTuR82pmtxH2spWG9h6etwfr1pLBqQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "parent-module": "^1.0.0", + "resolve-from": "^4.0.0" + }, + "engines": { + "node": ">=6" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/imurmurhash": { + "version": "0.1.4", + "resolved": "https://registry.npmjs.org/imurmurhash/-/imurmurhash-0.1.4.tgz", + "integrity": "sha512-JmXMZ6wuvDmLiHEml9ykzqO6lwFbof0GG4IkcGaENdCRDDmMVnny7s5HsIgHCbaq0w2MyPhDqkhTUgS2LU2PHA==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=0.8.19" + } + }, + "node_modules/inherits": { + "version": "2.0.4", + "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.4.tgz", + "integrity": "sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ==", + "license": "ISC" + }, + "node_modules/ini": { + "version": "1.3.8", + "resolved": "https://registry.npmjs.org/ini/-/ini-1.3.8.tgz", + "integrity": "sha512-JV/yugV2uzW5iMRSiZAyDtQd+nxtUnjeLt0acNdw98kKLrvuRVyB80tsREOE7yvGVgalhZ6RNXCmEHkUKBKxew==", + "license": "ISC" + }, + "node_modules/is-arrayish": { + "version": "0.3.2", + "resolved": "https://registry.npmjs.org/is-arrayish/-/is-arrayish-0.3.2.tgz", + "integrity": "sha512-eVRqCvVlZbuw3GrM63ovNSNAeA1K16kaR/LRY/92w0zxQ5/1YzwblUX652i4Xs9RwAGjW9d9y6X88t8OaAJfWQ==", + "license": "MIT" + }, + "node_modules/is-extglob": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/is-extglob/-/is-extglob-2.1.1.tgz", + "integrity": "sha512-SbKbANkN603Vi4jEZv49LeVJMn4yGwsbzZworEoyEiutsN3nJYdbO36zfhGJ6QEDpOZIFkDtnq5JRxmvl3jsoQ==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/is-glob": { + "version": "4.0.3", + "resolved": "https://registry.npmjs.org/is-glob/-/is-glob-4.0.3.tgz", + "integrity": "sha512-xelSayHH36ZgE7ZWhli7pW34hNbNl8Ojv5KVmkJD4hBdD3th8Tfk9vYasLM+mXWOZhFkgZfxhLSnrwRr4elSSg==", + "dev": true, + "license": "MIT", + "dependencies": { + "is-extglob": "^2.1.1" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/is-number": { + "version": "7.0.0", + "resolved": "https://registry.npmjs.org/is-number/-/is-number-7.0.0.tgz", + "integrity": "sha512-41Cifkg6e8TylSpdtTpeLVMqvSBEVzTttHvERD741+pnZ8ANv0004MRL43QKPDlK9cGvNp6NZWZUBlbGXYxxng==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=0.12.0" + } + }, + "node_modules/is-stream": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/is-stream/-/is-stream-2.0.1.tgz", + "integrity": "sha512-hFoiJiTl63nn+kstHGBtewWSKnQLpyb155KHheA1l39uvtO9nWIop1p3udqPcUd/xbF1VLMO4n7OI6p7RbngDg==", + "license": "MIT", + "engines": { + "node": ">=8" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/isexe": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/isexe/-/isexe-2.0.0.tgz", + "integrity": "sha512-RHxMLp9lnKHGHRng9QFhRCMbYAcVpn69smSGcq3f36xjgVVWThj4qqLbTLlq7Ssj8B+fIQ1EuCEGI2lKsyQeIw==", + "dev": true, + "license": "ISC" + }, + "node_modules/js-yaml": { + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/js-yaml/-/js-yaml-4.1.0.tgz", + "integrity": "sha512-wpxZs9NoxZaJESJGIZTyDEaYpl0FKSA+FB9aJiyemKhMwkxQg63h4T1KJgUGHpTqPDNRcmmYLugrRjJlBtWvRA==", + "dev": true, + "license": "MIT", + "dependencies": { + "argparse": "^2.0.1" + }, + "bin": { + "js-yaml": "bin/js-yaml.js" + } + }, + "node_modules/json-buffer": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/json-buffer/-/json-buffer-3.0.1.tgz", + "integrity": "sha512-4bV5BfR2mqfQTJm+V5tPPdf+ZpuhiIvTuAB5g8kcrXOZpTT/QwwVRWBywX1ozr6lEuPdbHxwaJlm9G6mI2sfSQ==", + "dev": true, + "license": "MIT" + }, + "node_modules/json-schema-traverse": { + "version": "0.4.1", + "resolved": "https://registry.npmjs.org/json-schema-traverse/-/json-schema-traverse-0.4.1.tgz", + "integrity": "sha512-xbbCH5dCYU5T8LcEhhuh7HJ88HXuW3qsI3Y0zOZFKfZEHcpWiHU/Jxzk629Brsab/mMiHQti9wMP+845RPe3Vg==", + "dev": true, + "license": "MIT" + }, + "node_modules/json-stable-stringify-without-jsonify": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/json-stable-stringify-without-jsonify/-/json-stable-stringify-without-jsonify-1.0.1.tgz", + "integrity": "sha512-Bdboy+l7tA3OGW6FjyFHWkP5LuByj1Tk33Ljyq0axyzdk9//JSi2u3fP1QSmd1KNwq6VOKYGlAu87CisVir6Pw==", + "dev": true, + "license": "MIT" + }, + "node_modules/keyv": { + "version": "4.5.4", + "resolved": "https://registry.npmjs.org/keyv/-/keyv-4.5.4.tgz", + "integrity": "sha512-oxVHkHR/EJf2CNXnWxRLW6mg7JyCCUcG0DtEGmL2ctUo1PNTin1PUil+r/+4r5MpVgC/fn1kjsx7mjSujKqIpw==", + "dev": true, + "license": "MIT", + "dependencies": { + "json-buffer": "3.0.1" + } + }, + "node_modules/kuler": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/kuler/-/kuler-2.0.0.tgz", + "integrity": "sha512-Xq9nH7KlWZmXAtodXDDRE7vs6DU1gTU8zYDHDiWLSip45Egwq3plLHzPn27NgvzL2r1LMPC1vdqh98sQxtqj4A==", + "license": "MIT" + }, + "node_modules/levn": { + "version": "0.4.1", + "resolved": "https://registry.npmjs.org/levn/-/levn-0.4.1.tgz", + "integrity": "sha512-+bT2uH4E5LGE7h/n3evcS/sQlJXCpIp6ym8OWJ5eV6+67Dsql/LaaT7qJBAt2rzfoa/5QBGBhxDix1dMt2kQKQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "prelude-ls": "^1.2.1", + "type-check": "~0.4.0" + }, + "engines": { + "node": ">= 0.8.0" + } + }, + "node_modules/locate-path": { + "version": "6.0.0", + "resolved": "https://registry.npmjs.org/locate-path/-/locate-path-6.0.0.tgz", + "integrity": "sha512-iPZK6eYjbxRu3uB4/WZ3EsEIMJFMqAoopl3R+zuq0UjcAm/MO6KCweDgPfP3elTztoKP3KtnVHxTn2NHBSDVUw==", + "dev": true, + "license": "MIT", + "dependencies": { + "p-locate": "^5.0.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/lodash.merge": { + "version": "4.6.2", + "resolved": "https://registry.npmjs.org/lodash.merge/-/lodash.merge-4.6.2.tgz", + "integrity": "sha512-0KpjqXRVvrYyCsX1swR/XTK0va6VQkQM6MNo7PqW77ByjAhoARA8EfrP1N4+KlKj8YS0ZUCtRT/YUuhyYDujIQ==", + "dev": true, + "license": "MIT" + }, + "node_modules/logform": { + "version": "2.7.0", + "resolved": "https://registry.npmjs.org/logform/-/logform-2.7.0.tgz", + "integrity": "sha512-TFYA4jnP7PVbmlBIfhlSe+WKxs9dklXMTEGcBCIvLhE/Tn3H6Gk1norupVW7m5Cnd4bLcr08AytbyV/xj7f/kQ==", + "license": "MIT", + "dependencies": { + "@colors/colors": "1.6.0", + "@types/triple-beam": "^1.3.2", + "fecha": "^4.2.0", + "ms": "^2.1.1", + "safe-stable-stringify": "^2.3.1", + "triple-beam": "^1.3.0" + }, + "engines": { + "node": ">= 12.0.0" + } + }, + "node_modules/make-error": { + "version": "1.3.6", + "resolved": "https://registry.npmjs.org/make-error/-/make-error-1.3.6.tgz", + "integrity": "sha512-s8UhlNe7vPKomQhC1qFelMokr/Sc3AgNbso3n74mVPA5LTZwkB9NlXf4XPamLxJE8h0gh73rM94xvwRT2CVInw==", + "dev": true, + "license": "ISC" + }, + "node_modules/merge2": { + "version": "1.4.1", + "resolved": "https://registry.npmjs.org/merge2/-/merge2-1.4.1.tgz", + "integrity": "sha512-8q7VEgMJW4J8tcfVPy8g09NcQwZdbwFEqhe/WZkoIzjn/3TGDwtOCYtXGxA3O8tPzpczCCDgv+P2P5y00ZJOOg==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">= 8" + } + }, + "node_modules/micromatch": { + "version": "4.0.8", + "resolved": "https://registry.npmjs.org/micromatch/-/micromatch-4.0.8.tgz", + "integrity": "sha512-PXwfBhYu0hBCPw8Dn0E+WDYb7af3dSLVWKi3HGv84IdF4TyFoC0ysxFd0Goxw7nSv4T/PzEJQxsYsEiFCKo2BA==", + "dev": true, + "license": "MIT", + "dependencies": { + "braces": "^3.0.3", + "picomatch": "^2.3.1" + }, + "engines": { + "node": ">=8.6" + } + }, + "node_modules/mimic-response": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/mimic-response/-/mimic-response-3.1.0.tgz", + "integrity": "sha512-z0yWI+4FDrrweS8Zmt4Ej5HdJmky15+L2e6Wgn3+iK5fWzb6T3fhNFq2+MeTRb064c6Wr4N/wv0DzQTjNzHNGQ==", + "license": "MIT", + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/minimatch": { + "version": "9.0.5", + "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-9.0.5.tgz", + "integrity": "sha512-G6T0ZX48xgozx7587koeX9Ys2NYy6Gmv//P89sEte9V9whIapMNF4idKxnW2QtCcLiTWlb/wfCabAtAFWhhBow==", + "dev": true, + "license": "ISC", + "dependencies": { + "brace-expansion": "^2.0.1" + }, + "engines": { + "node": ">=16 || 14 >=14.17" + }, + "funding": { + "url": "https://github.com/sponsors/isaacs" + } + }, + "node_modules/minimist": { + "version": "1.2.8", + "resolved": "https://registry.npmjs.org/minimist/-/minimist-1.2.8.tgz", + "integrity": "sha512-2yyAR8qBkN3YuheJanUpWC5U3bb5osDywNB8RzDVlDwDHbocAJveqqj1u8+SVD7jkWT4yvsHCpWqqWqAxb0zCA==", + "license": "MIT", + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/mkdirp-classic": { + "version": "0.5.3", + "resolved": "https://registry.npmjs.org/mkdirp-classic/-/mkdirp-classic-0.5.3.tgz", + "integrity": "sha512-gKLcREMhtuZRwRAfqP3RFW+TK4JqApVBtOIftVgjuABpAtpxhPGaDcfvbhNvD0B8iD1oUr/txX35NjcaY6Ns/A==", + "license": "MIT" + }, + "node_modules/ms": { + "version": "2.1.3", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.3.tgz", + "integrity": "sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA==", + "license": "MIT" + }, + "node_modules/napi-build-utils": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/napi-build-utils/-/napi-build-utils-2.0.0.tgz", + "integrity": "sha512-GEbrYkbfF7MoNaoh2iGG84Mnf/WZfB0GdGEsM8wz7Expx/LlWf5U8t9nvJKXSp3qr5IsEbK04cBGhol/KwOsWA==", + "license": "MIT" + }, + "node_modules/natural-compare": { + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/natural-compare/-/natural-compare-1.4.0.tgz", + "integrity": "sha512-OWND8ei3VtNC9h7V60qff3SVobHr996CTwgxubgyQYEpg290h9J0buyECNNJexkFm5sOajh5G116RYA1c8ZMSw==", + "dev": true, + "license": "MIT" + }, + "node_modules/ncp": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/ncp/-/ncp-2.0.0.tgz", + "integrity": "sha512-zIdGUrPRFTUELUvr3Gmc7KZ2Sw/h1PiVM0Af/oHB6zgnV1ikqSfRk+TOufi79aHYCW3NiOXmr1BP5nWbzojLaA==", + "dev": true, + "license": "MIT", + "bin": { + "ncp": "bin/ncp" + } + }, + "node_modules/node-abi": { + "version": "3.74.0", + "resolved": "https://registry.npmjs.org/node-abi/-/node-abi-3.74.0.tgz", + "integrity": "sha512-c5XK0MjkGBrQPGYG24GBADZud0NCbznxNx0ZkS+ebUTrmV1qTDxPxSL8zEAPURXSbLRWVexxmP4986BziahL5w==", + "license": "MIT", + "dependencies": { + "semver": "^7.3.5" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/node-domexception": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/node-domexception/-/node-domexception-1.0.0.tgz", + "integrity": "sha512-/jKZoMpw0F8GRwl4/eLROPA3cfcXtLApP0QzLmUT/HuPCZWyB7IY9ZrMeKw2O/nFIqPQB3PVM9aYm0F312AXDQ==", + "funding": [ + { + "type": "github", + "url": "https://github.com/sponsors/jimmywarting" + }, + { + "type": "github", + "url": "https://paypal.me/jimmywarting" + } + ], + "license": "MIT", + "engines": { + "node": ">=10.5.0" + } + }, + "node_modules/node-fetch": { + "version": "3.3.2", + "resolved": "https://registry.npmjs.org/node-fetch/-/node-fetch-3.3.2.tgz", + "integrity": "sha512-dRB78srN/l6gqWulah9SrxeYnxeddIG30+GOqK/9OlLVyLg3HPnr6SqOWTWOXKRwC2eGYCkZ59NNuSgvSrpgOA==", + "license": "MIT", + "dependencies": { + "data-uri-to-buffer": "^4.0.0", + "fetch-blob": "^3.1.4", + "formdata-polyfill": "^4.0.10" + }, + "engines": { + "node": "^12.20.0 || ^14.13.1 || >=16.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/node-fetch" + } + }, + "node_modules/once": { + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/once/-/once-1.4.0.tgz", + "integrity": "sha512-lNaJgI+2Q5URQBkccEKHTQOPaXdUxnZZElQTZY0MFUAuaEqe1E+Nyvgdz/aIyNi6Z9MzO5dv1H8n58/GELp3+w==", + "license": "ISC", + "dependencies": { + "wrappy": "1" + } + }, + "node_modules/one-time": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/one-time/-/one-time-1.0.0.tgz", + "integrity": "sha512-5DXOiRKwuSEcQ/l0kGCF6Q3jcADFv5tSmRaJck/OqkVFcOzutB134KRSfF0xDrL39MNnqxbHBbUUcjZIhTgb2g==", + "license": "MIT", + "dependencies": { + "fn.name": "1.x.x" + } + }, + "node_modules/optionator": { + "version": "0.9.4", + "resolved": "https://registry.npmjs.org/optionator/-/optionator-0.9.4.tgz", + "integrity": "sha512-6IpQ7mKUxRcZNLIObR0hz7lxsapSSIYNZJwXPGeF0mTVqGKFIXj1DQcMoT22S3ROcLyY/rz0PWaWZ9ayWmad9g==", + "dev": true, + "license": "MIT", + "dependencies": { + "deep-is": "^0.1.3", + "fast-levenshtein": "^2.0.6", + "levn": "^0.4.1", + "prelude-ls": "^1.2.1", + "type-check": "^0.4.0", + "word-wrap": "^1.2.5" + }, + "engines": { + "node": ">= 0.8.0" + } + }, + "node_modules/p-limit": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/p-limit/-/p-limit-3.1.0.tgz", + "integrity": "sha512-TYOanM3wGwNGsZN2cVTYPArw454xnXj5qmWF1bEoAc4+cU/ol7GVh7odevjp1FNHduHc3KZMcFduxU5Xc6uJRQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "yocto-queue": "^0.1.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/p-locate": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/p-locate/-/p-locate-5.0.0.tgz", + "integrity": "sha512-LaNjtRWUBY++zB5nE/NwcaoMylSPk+S+ZHNB1TzdbMJMny6dynpAGt7X/tl/QYq3TIeE6nxHppbo2LGymrG5Pw==", + "dev": true, + "license": "MIT", + "dependencies": { + "p-limit": "^3.0.2" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/parent-module": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/parent-module/-/parent-module-1.0.1.tgz", + "integrity": "sha512-GQ2EWRpQV8/o+Aw8YqtfZZPfNRWZYkbidE9k5rpl/hC3vtHHBfGm2Ifi6qWV+coDGkrUKZAxE3Lot5kcsRlh+g==", + "dev": true, + "license": "MIT", + "dependencies": { + "callsites": "^3.0.0" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/path-exists": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/path-exists/-/path-exists-4.0.0.tgz", + "integrity": "sha512-ak9Qy5Q7jYb2Wwcey5Fpvg2KoAc/ZIhLSLOSBmRmygPsGwkVVt0fZa0qrtMz+m6tJTAHfZQ8FnmB4MG4LWy7/w==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=8" + } + }, + "node_modules/path-key": { + "version": "3.1.1", + "resolved": "https://registry.npmjs.org/path-key/-/path-key-3.1.1.tgz", + "integrity": "sha512-ojmeN0qd+y0jszEtoY48r0Peq5dwMEkIlCOu6Q5f41lfkswXuKtYrhgoTpLnyIcHm24Uhqx+5Tqm2InSwLhE6Q==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=8" + } + }, + "node_modules/picomatch": { + "version": "2.3.1", + "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-2.3.1.tgz", + "integrity": "sha512-JU3teHTNjmE2VCGFzuY8EXzCDVwEqB2a8fsIvwaStHhAWJEeVd1o1QD80CU6+ZdEXXSLbSsuLwJjkCBWqRQUVA==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=8.6" + }, + "funding": { + "url": "https://github.com/sponsors/jonschlinkert" + } + }, + "node_modules/prebuild-install": { + "version": "7.1.3", + "resolved": "https://registry.npmjs.org/prebuild-install/-/prebuild-install-7.1.3.tgz", + "integrity": "sha512-8Mf2cbV7x1cXPUILADGI3wuhfqWvtiLA1iclTDbFRZkgRQS0NqsPZphna9V+HyTEadheuPmjaJMsbzKQFOzLug==", + "license": "MIT", + "dependencies": { + "detect-libc": "^2.0.0", + "expand-template": "^2.0.3", + "github-from-package": "0.0.0", + "minimist": "^1.2.3", + "mkdirp-classic": "^0.5.3", + "napi-build-utils": "^2.0.0", + "node-abi": "^3.3.0", + "pump": "^3.0.0", + "rc": "^1.2.7", + "simple-get": "^4.0.0", + "tar-fs": "^2.0.0", + "tunnel-agent": "^0.6.0" + }, + "bin": { + "prebuild-install": "bin.js" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/prelude-ls": { + "version": "1.2.1", + "resolved": "https://registry.npmjs.org/prelude-ls/-/prelude-ls-1.2.1.tgz", + "integrity": "sha512-vkcDPrRZo1QZLbn5RLGPpg/WmIQ65qoWWhcGKf/b5eplkkarX0m9z8ppCat4mlOqUsWpyNuYgO3VRyrYHSzX5g==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">= 0.8.0" + } + }, + "node_modules/pump": { + "version": "3.0.2", + "resolved": "https://registry.npmjs.org/pump/-/pump-3.0.2.tgz", + "integrity": "sha512-tUPXtzlGM8FE3P0ZL6DVs/3P58k9nk8/jZeQCurTJylQA8qFYzHFfhBJkuqyE0FifOsQ0uKWekiZ5g8wtr28cw==", + "license": "MIT", + "dependencies": { + "end-of-stream": "^1.1.0", + "once": "^1.3.1" + } + }, + "node_modules/punycode": { + "version": "2.3.1", + "resolved": "https://registry.npmjs.org/punycode/-/punycode-2.3.1.tgz", + "integrity": "sha512-vYt7UD1U9Wg6138shLtLOvdAu+8DsC/ilFtEVHcH+wydcSpNE20AfSOduf6MkRFahL5FY7X1oU7nKVZFtfq8Fg==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6" + } + }, + "node_modules/queue-microtask": { + "version": "1.2.3", + "resolved": "https://registry.npmjs.org/queue-microtask/-/queue-microtask-1.2.3.tgz", + "integrity": "sha512-NuaNSa6flKT5JaSYQzJok04JzTL1CA6aGhv5rfLW3PgqA+M2ChpZQnAC8h8i4ZFkBS8X5RqkDBHA7r4hej3K9A==", + "dev": true, + "funding": [ + { + "type": "github", + "url": "https://github.com/sponsors/feross" + }, + { + "type": "patreon", + "url": "https://www.patreon.com/feross" + }, + { + "type": "consulting", + "url": "https://feross.org/support" + } + ], + "license": "MIT" + }, + "node_modules/rc": { + "version": "1.2.8", + "resolved": "https://registry.npmjs.org/rc/-/rc-1.2.8.tgz", + "integrity": "sha512-y3bGgqKj3QBdxLbLkomlohkvsA8gdAiUQlSBJnBhfn+BPxg4bc62d8TcBW15wavDfgexCgccckhcZvywyQYPOw==", + "license": "(BSD-2-Clause OR MIT OR Apache-2.0)", + "dependencies": { + "deep-extend": "^0.6.0", + "ini": "~1.3.0", + "minimist": "^1.2.0", + "strip-json-comments": "~2.0.1" + }, + "bin": { + "rc": "cli.js" + } + }, + "node_modules/rc/node_modules/strip-json-comments": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/strip-json-comments/-/strip-json-comments-2.0.1.tgz", + "integrity": "sha512-4gB8na07fecVVkOI6Rs4e7T6NOTki5EmL7TUduTs6bu3EdnSycntVJ4re8kgZA+wx9IueI2Y11bfbgwtzuE0KQ==", + "license": "MIT", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/readable-stream": { + "version": "3.6.2", + "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-3.6.2.tgz", + "integrity": "sha512-9u/sniCrY3D5WdsERHzHE4G2YCXqoG5FTHUiCC4SIbr6XcLZBY05ya9EKjYek9O5xOAwjGq+1JdGBAS7Q9ScoA==", + "license": "MIT", + "dependencies": { + "inherits": "^2.0.3", + "string_decoder": "^1.1.1", + "util-deprecate": "^1.0.1" + }, + "engines": { + "node": ">= 6" + } + }, + "node_modules/resolve-from": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/resolve-from/-/resolve-from-4.0.0.tgz", + "integrity": "sha512-pb/MYmXstAkysRFx8piNI1tGFNQIFA3vkE3Gq4EuA1dF6gHp/+vgZqsCGJapvy8N3Q+4o7FwvquPJcnZ7RYy4g==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=4" + } + }, + "node_modules/reusify": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/reusify/-/reusify-1.0.4.tgz", + "integrity": "sha512-U9nH88a3fc/ekCF1l0/UP1IosiuIjyTh7hBvXVMHYgVcfGvt897Xguj2UOLDeI5BG2m7/uwyaLVT6fbtCwTyzw==", + "dev": true, + "license": "MIT", + "engines": { + "iojs": ">=1.0.0", + "node": ">=0.10.0" + } + }, + "node_modules/run-parallel": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/run-parallel/-/run-parallel-1.2.0.tgz", + "integrity": "sha512-5l4VyZR86LZ/lDxZTR6jqL8AFE2S0IFLMP26AbjsLVADxHdhB/c0GUsH+y39UfCi3dzz8OlQuPmnaJOMoDHQBA==", + "dev": true, + "funding": [ + { + "type": "github", + "url": "https://github.com/sponsors/feross" + }, + { + "type": "patreon", + "url": "https://www.patreon.com/feross" + }, + { + "type": "consulting", + "url": "https://feross.org/support" + } + ], + "license": "MIT", + "dependencies": { + "queue-microtask": "^1.2.2" + } + }, + "node_modules/safe-buffer": { + "version": "5.2.1", + "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.2.1.tgz", + "integrity": "sha512-rp3So07KcdmmKbGvgaNxQSJr7bGVSVk5S9Eq1F+ppbRo70+YeaDxkw5Dd8NPN+GD6bjnYm2VuPuCXmpuYvmCXQ==", + "funding": [ + { + "type": "github", + "url": "https://github.com/sponsors/feross" + }, + { + "type": "patreon", + "url": "https://www.patreon.com/feross" + }, + { + "type": "consulting", + "url": "https://feross.org/support" + } + ], + "license": "MIT" + }, + "node_modules/safe-stable-stringify": { + "version": "2.5.0", + "resolved": "https://registry.npmjs.org/safe-stable-stringify/-/safe-stable-stringify-2.5.0.tgz", + "integrity": "sha512-b3rppTKm9T+PsVCBEOUR46GWI7fdOs00VKZ1+9c1EWDaDMvjQc6tUwuFyIprgGgTcWoVHSKrU8H31ZHA2e0RHA==", + "license": "MIT", + "engines": { + "node": ">=10" + } + }, + "node_modules/semver": { + "version": "7.7.1", + "resolved": "https://registry.npmjs.org/semver/-/semver-7.7.1.tgz", + "integrity": "sha512-hlq8tAfn0m/61p4BVRcPzIGr6LKiMwo4VM6dGi6pt4qcRkmNzTcWq6eCEjEh+qXjkMDvPlOFFSGwQjoEa6gyMA==", + "license": "ISC", + "bin": { + "semver": "bin/semver.js" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/shebang-command": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/shebang-command/-/shebang-command-2.0.0.tgz", + "integrity": "sha512-kHxr2zZpYtdmrN1qDjrrX/Z1rR1kG8Dx+gkpK1G4eXmvXswmcE1hTWBWYUzlraYw1/yZp6YuDY77YtvbN0dmDA==", + "dev": true, + "license": "MIT", + "dependencies": { + "shebang-regex": "^3.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/shebang-regex": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/shebang-regex/-/shebang-regex-3.0.0.tgz", + "integrity": "sha512-7++dFhtcx3353uBaq8DDR4NuxBetBzC7ZQOhmTQInHEd6bSrXdiEyzCvG07Z44UYdLShWUyXt5M/yhz8ekcb1A==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=8" + } + }, + "node_modules/simple-concat": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/simple-concat/-/simple-concat-1.0.1.tgz", + "integrity": "sha512-cSFtAPtRhljv69IK0hTVZQ+OfE9nePi/rtJmw5UjHeVyVroEqJXP1sFztKUy1qU+xvz3u/sfYJLa947b7nAN2Q==", + "funding": [ + { + "type": "github", + "url": "https://github.com/sponsors/feross" + }, + { + "type": "patreon", + "url": "https://www.patreon.com/feross" + }, + { + "type": "consulting", + "url": "https://feross.org/support" + } + ], + "license": "MIT" + }, + "node_modules/simple-get": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/simple-get/-/simple-get-4.0.1.tgz", + "integrity": "sha512-brv7p5WgH0jmQJr1ZDDfKDOSeWWg+OVypG99A/5vYGPqJ6pxiaHLy8nxtFjBA7oMa01ebA9gfh1uMCFqOuXxvA==", + "funding": [ + { + "type": "github", + "url": "https://github.com/sponsors/feross" + }, + { + "type": "patreon", + "url": "https://www.patreon.com/feross" + }, + { + "type": "consulting", + "url": "https://feross.org/support" + } + ], + "license": "MIT", + "dependencies": { + "decompress-response": "^6.0.0", + "once": "^1.3.1", + "simple-concat": "^1.0.0" + } + }, + "node_modules/simple-swizzle": { + "version": "0.2.2", + "resolved": "https://registry.npmjs.org/simple-swizzle/-/simple-swizzle-0.2.2.tgz", + "integrity": "sha512-JA//kQgZtbuY83m+xT+tXJkmJncGMTFT+C+g2h2R9uxkYIrE2yy9sgmcLhCnw57/WSD+Eh3J97FPEDFnbXnDUg==", + "license": "MIT", + "dependencies": { + "is-arrayish": "^0.3.1" + } + }, + "node_modules/stack-trace": { + "version": "0.0.10", + "resolved": "https://registry.npmjs.org/stack-trace/-/stack-trace-0.0.10.tgz", + "integrity": "sha512-KGzahc7puUKkzyMt+IqAep+TVNbKP+k2Lmwhub39m1AsTSkaDutx56aDCo+HLDzf/D26BIHTJWNiTG1KAJiQCg==", + "license": "MIT", + "engines": { + "node": "*" + } + }, + "node_modules/string_decoder": { + "version": "1.3.0", + "resolved": "https://registry.npmjs.org/string_decoder/-/string_decoder-1.3.0.tgz", + "integrity": "sha512-hkRX8U1WjJFd8LsDJ2yQ/wWWxaopEsABU1XfkM8A+j0+85JAGppt16cr1Whg6KIbb4okU6Mql6BOj+uup/wKeA==", + "license": "MIT", + "dependencies": { + "safe-buffer": "~5.2.0" + } + }, + "node_modules/strip-json-comments": { + "version": "3.1.1", + "resolved": "https://registry.npmjs.org/strip-json-comments/-/strip-json-comments-3.1.1.tgz", + "integrity": "sha512-6fPc+R4ihwqP6N/aIv2f1gMH8lOVtWQHoqC4yK6oSDVVocumAsfCqjkXnqiYMhmMwS/mEHLp7Vehlt3ql6lEig==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=8" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/supports-color": { + "version": "7.2.0", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", + "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", + "dev": true, + "license": "MIT", + "dependencies": { + "has-flag": "^4.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/tar-fs": { + "version": "2.1.2", + "resolved": "https://registry.npmjs.org/tar-fs/-/tar-fs-2.1.2.tgz", + "integrity": "sha512-EsaAXwxmx8UB7FRKqeozqEPop69DXcmYwTQwXvyAPF352HJsPdkVhvTaDPYqfNgruveJIJy3TA2l+2zj8LJIJA==", + "license": "MIT", + "dependencies": { + "chownr": "^1.1.1", + "mkdirp-classic": "^0.5.2", + "pump": "^3.0.0", + "tar-stream": "^2.1.4" + } + }, + "node_modules/tar-stream": { + "version": "2.2.0", + "resolved": "https://registry.npmjs.org/tar-stream/-/tar-stream-2.2.0.tgz", + "integrity": "sha512-ujeqbceABgwMZxEJnk2HDY2DlnUZ+9oEcb1KzTVfYHio0UE6dG71n60d8D2I4qNvleWrrXpmjpt7vZeF1LnMZQ==", + "license": "MIT", + "dependencies": { + "bl": "^4.0.3", + "end-of-stream": "^1.4.1", + "fs-constants": "^1.0.0", + "inherits": "^2.0.3", + "readable-stream": "^3.1.1" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/text-hex": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/text-hex/-/text-hex-1.0.0.tgz", + "integrity": "sha512-uuVGNWzgJ4yhRaNSiubPY7OjISw4sw4E5Uv0wbjp+OzcbmVU/rsT8ujgcXJhn9ypzsgr5vlzpPqP+MBBKcGvbg==", + "license": "MIT" + }, + "node_modules/to-regex-range": { + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/to-regex-range/-/to-regex-range-5.0.1.tgz", + "integrity": "sha512-65P7iz6X5yEr1cwcgvQxbbIw7Uk3gOy5dIdtZ4rDveLqhrdJP+Li/Hx6tyK0NEb+2GCyneCMJiGqrADCSNk8sQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "is-number": "^7.0.0" + }, + "engines": { + "node": ">=8.0" + } + }, + "node_modules/tr46": { + "version": "0.0.3", + "resolved": "https://registry.npmjs.org/tr46/-/tr46-0.0.3.tgz", + "integrity": "sha512-N3WMsuqV66lT30CrXNbEjx4GEwlow3v6rr4mCcv6prnfwhS01rkgyFdjPNBYd9br7LpXV1+Emh01fHnq2Gdgrw==", + "license": "MIT" + }, + "node_modules/triple-beam": { + "version": "1.4.1", + "resolved": "https://registry.npmjs.org/triple-beam/-/triple-beam-1.4.1.tgz", + "integrity": "sha512-aZbgViZrg1QNcG+LULa7nhZpJTZSLm/mXnHXnbAbjmN5aSa0y7V+wvv6+4WaBtpISJzThKy+PIPxc1Nq1EJ9mg==", + "license": "MIT", + "engines": { + "node": ">= 14.0.0" + } + }, + "node_modules/ts-api-utils": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/ts-api-utils/-/ts-api-utils-2.0.1.tgz", + "integrity": "sha512-dnlgjFSVetynI8nzgJ+qF62efpglpWRk8isUEWZGWlJYySCTD6aKvbUDu+zbPeDakk3bg5H4XpitHukgfL1m9w==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=18.12" + }, + "peerDependencies": { + "typescript": ">=4.8.4" + } + }, + "node_modules/ts-node": { + "version": "10.9.2", + "resolved": "https://registry.npmjs.org/ts-node/-/ts-node-10.9.2.tgz", + "integrity": "sha512-f0FFpIdcHgn8zcPSbf1dRevwt047YMnaiJM3u2w2RewrB+fob/zePZcrOyQoLMMO7aBIddLcQIEK5dYjkLnGrQ==", + "dev": true, + "license": "MIT", + "dependencies": { + "@cspotcode/source-map-support": "^0.8.0", + "@tsconfig/node10": "^1.0.7", + "@tsconfig/node12": "^1.0.7", + "@tsconfig/node14": "^1.0.0", + "@tsconfig/node16": "^1.0.2", + "acorn": "^8.4.1", + "acorn-walk": "^8.1.1", + "arg": "^4.1.0", + "create-require": "^1.1.0", + "diff": "^4.0.1", + "make-error": "^1.1.1", + "v8-compile-cache-lib": "^3.0.1", + "yn": "3.1.1" + }, + "bin": { + "ts-node": "dist/bin.js", + "ts-node-cwd": "dist/bin-cwd.js", + "ts-node-esm": "dist/bin-esm.js", + "ts-node-script": "dist/bin-script.js", + "ts-node-transpile-only": "dist/bin-transpile.js", + "ts-script": "dist/bin-script-deprecated.js" + }, + "peerDependencies": { + "@swc/core": ">=1.2.50", + "@swc/wasm": ">=1.2.50", + "@types/node": "*", + "typescript": ">=2.7" + }, + "peerDependenciesMeta": { + "@swc/core": { + "optional": true + }, + "@swc/wasm": { + "optional": true + } + } + }, + "node_modules/tunnel-agent": { + "version": "0.6.0", + "resolved": "https://registry.npmjs.org/tunnel-agent/-/tunnel-agent-0.6.0.tgz", + "integrity": "sha512-McnNiV1l8RYeY8tBgEpuodCC1mLUdbSN+CYBL7kJsJNInOP8UjDDEwdk6Mw60vdLLrr5NHKZhMAOSrR2NZuQ+w==", + "license": "Apache-2.0", + "dependencies": { + "safe-buffer": "^5.0.1" + }, + "engines": { + "node": "*" + } + }, + "node_modules/type-check": { + "version": "0.4.0", + "resolved": "https://registry.npmjs.org/type-check/-/type-check-0.4.0.tgz", + "integrity": "sha512-XleUoc9uwGXqjWwXaUTZAmzMcFZ5858QA2vvx1Ur5xIcixXIP+8LnFDgRplU30us6teqdlskFfu+ae4K79Ooew==", + "dev": true, + "license": "MIT", + "dependencies": { + "prelude-ls": "^1.2.1" + }, + "engines": { + "node": ">= 0.8.0" + } + }, + "node_modules/typescript": { + "version": "4.9.5", + "resolved": "https://registry.npmjs.org/typescript/-/typescript-4.9.5.tgz", + "integrity": "sha512-1FXk9E2Hm+QzZQ7z+McJiHL4NW1F2EzMu9Nq9i3zAaGqibafqYwCVU6WyWAuyQRRzOlxou8xZSyXLEN8oKj24g==", + "dev": true, + "license": "Apache-2.0", + "bin": { + "tsc": "bin/tsc", + "tsserver": "bin/tsserver" + }, + "engines": { + "node": ">=4.2.0" + } + }, + "node_modules/uri-js": { + "version": "4.4.1", + "resolved": "https://registry.npmjs.org/uri-js/-/uri-js-4.4.1.tgz", + "integrity": "sha512-7rKUyy33Q1yc98pQ1DAmLtwX109F7TIfWlW1Ydo8Wl1ii1SeHieeh0HHfPeL2fMXK6z0s8ecKs9frCuLJvndBg==", + "dev": true, + "license": "BSD-2-Clause", + "dependencies": { + "punycode": "^2.1.0" + } + }, + "node_modules/util-deprecate": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/util-deprecate/-/util-deprecate-1.0.2.tgz", + "integrity": "sha512-EPD5q1uXyFxJpCrLnCc1nHnq3gOa6DZBocAIiI2TaSCA7VCJ1UJDMagCzIkXNsUYfD1daK//LTEQ8xiIbrHtcw==", + "license": "MIT" + }, + "node_modules/v8-compile-cache-lib": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/v8-compile-cache-lib/-/v8-compile-cache-lib-3.0.1.tgz", + "integrity": "sha512-wa7YjyUGfNZngI/vtK0UHAN+lgDCxBPCylVXGp0zu59Fz5aiGtNXaq3DhIov063MorB+VfufLh3JlF2KdTK3xg==", + "dev": true, + "license": "MIT" + }, + "node_modules/web-streams-polyfill": { + "version": "3.3.3", + "resolved": "https://registry.npmjs.org/web-streams-polyfill/-/web-streams-polyfill-3.3.3.tgz", + "integrity": "sha512-d2JWLCivmZYTSIoge9MsgFCZrt571BikcWGYkjC1khllbTeDlGqZ2D8vD8E/lJa8WGWbb7Plm8/XJYV7IJHZZw==", + "license": "MIT", + "engines": { + "node": ">= 8" + } + }, + "node_modules/webidl-conversions": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/webidl-conversions/-/webidl-conversions-3.0.1.tgz", + "integrity": "sha512-2JAn3z8AR6rjK8Sm8orRC0h/bcl/DqL7tRPdGZ4I1CjdF+EaMLmYxBHyXuKL849eucPFhvBoxMsflfOb8kxaeQ==", + "license": "BSD-2-Clause" + }, + "node_modules/whatwg-url": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/whatwg-url/-/whatwg-url-5.0.0.tgz", + "integrity": "sha512-saE57nupxk6v3HY35+jzBwYa0rKSy0XR8JSxZPwgLr7ys0IBzhGviA1/TUGJLmSVqs8pb9AnvICXEuOHLprYTw==", + "license": "MIT", + "dependencies": { + "tr46": "~0.0.3", + "webidl-conversions": "^3.0.0" + } + }, + "node_modules/which": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/which/-/which-2.0.2.tgz", + "integrity": "sha512-BLI3Tl1TW3Pvl70l3yq3Y64i+awpwXqsGBYWkkqMtnbXgrMD+yj7rhW0kuEDxzJaYXGjEW5ogapKNMEKNMjibA==", + "dev": true, + "license": "ISC", + "dependencies": { + "isexe": "^2.0.0" + }, + "bin": { + "node-which": "bin/node-which" + }, + "engines": { + "node": ">= 8" + } + }, + "node_modules/winston": { + "version": "3.17.0", + "resolved": "https://registry.npmjs.org/winston/-/winston-3.17.0.tgz", + "integrity": "sha512-DLiFIXYC5fMPxaRg832S6F5mJYvePtmO5G9v9IgUFPhXm9/GkXarH/TUrBAVzhTCzAj9anE/+GjrgXp/54nOgw==", + "license": "MIT", + "dependencies": { + "@colors/colors": "^1.6.0", + "@dabh/diagnostics": "^2.0.2", + "async": "^3.2.3", + "is-stream": "^2.0.0", + "logform": "^2.7.0", + "one-time": "^1.0.0", + "readable-stream": "^3.4.0", + "safe-stable-stringify": "^2.3.1", + "stack-trace": "0.0.x", + "triple-beam": "^1.3.0", + "winston-transport": "^4.9.0" + }, + "engines": { + "node": ">= 12.0.0" + } + }, + "node_modules/winston-transport": { + "version": "4.9.0", + "resolved": "https://registry.npmjs.org/winston-transport/-/winston-transport-4.9.0.tgz", + "integrity": "sha512-8drMJ4rkgaPo1Me4zD/3WLfI/zPdA9o2IipKODunnGDcuqbHwjsbB79ylv04LCGGzU0xQ6vTznOMpQGaLhhm6A==", + "license": "MIT", + "dependencies": { + "logform": "^2.7.0", + "readable-stream": "^3.6.2", + "triple-beam": "^1.3.0" + }, + "engines": { + "node": ">= 12.0.0" + } + }, + "node_modules/word-wrap": { + "version": "1.2.5", + "resolved": "https://registry.npmjs.org/word-wrap/-/word-wrap-1.2.5.tgz", + "integrity": "sha512-BN22B5eaMMI9UMtjrGd5g5eCYPpCPDUy0FJXbYsaT5zYxjFOckS53SQDE3pWkVoWpHXVb3BrYcEN4Twa55B5cA==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/wrappy": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/wrappy/-/wrappy-1.0.2.tgz", + "integrity": "sha512-l4Sp/DRseor9wL6EvV2+TuQn63dMkPjZ/sp9XkghTEbV9KlPS1xUsZ3u7/IQO4wxtcFB4bgpQPRcR3QCvezPcQ==", + "license": "ISC" + }, + "node_modules/yn": { + "version": "3.1.1", + "resolved": "https://registry.npmjs.org/yn/-/yn-3.1.1.tgz", + "integrity": "sha512-Ux4ygGWsu2c7isFWe8Yu1YluJmqVhxqK2cLXNQA5AcC3QfbGNpM7fu0Y8b/z16pXLnFxZYvWhd3fhBY9DLmC6Q==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=6" + } + }, + "node_modules/yocto-queue": { + "version": "0.1.0", + "resolved": "https://registry.npmjs.org/yocto-queue/-/yocto-queue-0.1.0.tgz", + "integrity": "sha512-rVksvsnNCdJ/ohGc6xgPwyN8eheCxsiLM8mxuE/t/mOVqJewPuO1miLpTHQiRgTKCLexL4MeAFVagts7HmNZ2Q==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + } } } diff --git a/package.json b/package.json index 5c9ed1f..18001d2 100644 --- a/package.json +++ b/package.json @@ -1,45 +1,42 @@ { - - "name": "tgbot", - + "name": "tgstorebot", "version": "1.0.0", - - "description": "", - - "main": "index.js", - - "devDependencies": { - - - "@types/node": "^22.13.0", - - - "@types/node-fetch": "^2.6.12" - - }, - + "description": "Telegram bot for product store", + "main": "dist/bot.js", "scripts": { - - - "test": "echo \"Error: no test specified\" && exit 1" - + "start": "node dist/bot.js", + "dev": "ts-node src/bot.ts", + "build": "tsc && npm run copy-languages", + "copy-languages": "ncp src/languages dist/languages", + "test": "echo \"No test specified\" && exit 0" + }, + "repository": { + "type": "git", + "url": "https://github.com/MAmamiyaAhmad/TGStoreBot.git" }, - - "keywords": [], - - "author": "", - - "license": "ISC", - - "type": "commonjs", - + "keywords": [ + "telegram", + "bot", + "typescript" + ], + "author": "M Amamiya Ahmad - https://github.com/MAmamiyaAhmad/", + "license": "MIT", "dependencies": { - - - "dotenv": "^16.4.7", - - - "grammy": "^1.34.1" - + "better-sqlite3": "^11.8.1", + "dotenv": "^10.0.0", + "grammy": "^1.34.1", + "node-fetch": "^3.0.0", + "winston": "^3.17.0" + }, + "devDependencies": { + "@grammyjs/types": "^3.18.0", + "@types/better-sqlite3": "^7.6.12", + "@types/node": "^16.18.126", + "@typescript-eslint/eslint-plugin": "^8.23.0", + "@typescript-eslint/parser": "^8.23.0", + "eslint": "^9.19.0", + "ncp": "^2.0.0", + "ts-node": "^10.9.2", + "typescript": "^4.9.5" } } diff --git a/src/bot.ts b/src/bot.ts index 474c745..9fe46f3 100644 --- a/src/bot.ts +++ b/src/bot.ts @@ -1,16 +1,28 @@ import { Bot } from "grammy"; -import { BOT_TOKEN } from "./config"; +import { BOT_TOKEN } from "./config/config"; import { startHandler } from "./handlers/start"; import { languageHandler } from "./handlers/language"; +import { logger } from "./utils/logger"; // Import the logger const bot = new Bot(BOT_TOKEN); -// Command /start dengan auto-detect bot username -bot.command("start", async (ctx) => startHandler(ctx, bot)); +// Command /start with auto-detect bot username +bot.command("start", async (ctx) => { + logger.info(`Received /start command from ${ctx.from?.id}`); + await startHandler(ctx, bot); +}); -// Handle pilihan bahasa -bot.callbackQuery(/.*/, languageHandler); +// Handle language selection +bot.callbackQuery(/.*/, (ctx) => { + logger.info(`Received language selection from ${ctx.from?.id}: ${ctx.match}`); + languageHandler(ctx); +}); + +// Run the bot +bot.start().then(() => { + logger.info("Bot started successfully."); +}).catch((error) => { + logger.error(`Error starting bot: ${error.message}`); +}); -// Jalankan bot -bot.start(); console.log("🤖 Bot is running..."); diff --git a/src/config.ts b/src/config.ts deleted file mode 100644 index ab33cb5..0000000 --- a/src/config.ts +++ /dev/null @@ -1,4 +0,0 @@ -import * as dotenv from "dotenv"; -dotenv.config(); - -export const BOT_TOKEN = process.env.BOT_TOKEN as string; diff --git a/src/config/config.ts b/src/config/config.ts new file mode 100644 index 0000000..0b7858d --- /dev/null +++ b/src/config/config.ts @@ -0,0 +1,6 @@ +import * as dotenv from "dotenv"; +dotenv.config(); + +export const BOT_TOKEN = process.env.BOT_TOKEN as string; +export const SUPER_ADMIN_ID = parseInt(process.env.SUPER_ADMIN_ID as string); +export const ADMIN_IDS = process.env.ADMIN_IDS?.split(",").map(id => parseInt(id)) || []; \ No newline at end of file diff --git a/src/handlers/language.ts b/src/handlers/language.ts index bec38b7..2a2d094 100644 --- a/src/handlers/language.ts +++ b/src/handlers/language.ts @@ -1,6 +1,23 @@ import { Context } from "grammy"; +import { logger } from "../utils/logger"; // Import the logger export async function languageHandler(ctx: Context) { const selectedLanguage = ctx.match; - await ctx.reply(`✅ You have selected: ${selectedLanguage}`); + try { + // Log the language selection + logger.info(`User ${ctx.from?.id} selected language: ${selectedLanguage}`); + + await ctx.reply(`✅ You have selected: ${selectedLanguage}`); + + // Log success + logger.info(`Successfully sent language selection reply to ${ctx.from?.id}`); + } catch (error: unknown) { + // Type assertion to Error + if (error instanceof Error) { + logger.error(`Error in languageHandler: ${error.message}`); + } else { + logger.error("Unknown error in languageHandler"); + } + await ctx.reply("⚠️ Something went wrong with language selection."); + } } diff --git a/src/handlers/start.ts b/src/handlers/start.ts index 4eec29c..cc93b94 100644 --- a/src/handlers/start.ts +++ b/src/handlers/start.ts @@ -1,26 +1,43 @@ import { Context, Bot } from "grammy"; -import { getLanguageKeyboard, getBotUsername } from "../utils"; +import { getLanguageKeyboard, getBotUsername } from "../utils/language"; +import { logger } from "../utils/logger"; // Import the logger export async function startHandler(ctx: Context, bot: Bot) { const userId = ctx.from?.id; const firstName = ctx.from?.first_name; const username = ctx.from?.username || "Tidak Ada"; - // Ambil nama bot secara otomatis - const botUsername = await getBotUsername(bot); + // Log user info + logger.info(`User ${firstName} (${userId}) started the bot`); - const message = `🌏 Choose Language\n` - + `•❂•─•─•❂•─•❂••❂•─•❂•─•─•❂•\n\n` - + `🆔 Name: ${firstName}\n` - + ` \u251c Username: @${username}\n` - + ` \u251c ID Telegram: ${userId}\n` - + ` \u2514 Link: Click Here\n\n` - + `👁‍🗨 Hello ${firstName}, Welcome to Bot Monitoring\n🌏 Please Choose your Language\n\n` - + `•❂•─•─•❂•─•❂••❂•─•❂•─•─•❂•\n\n`; + try { + // Get bot username + const botUsername = await getBotUsername(bot); - await ctx.replyWithPhoto("https://imgur.com/a/mXrMZsI", { - caption: message, - parse_mode: "HTML", - reply_markup: getLanguageKeyboard(), - }); + const message = `🌏 Choose Language\n` + + `•❂•─•─•❂•─•❂••❂•─•❂•─•─•❂•\n\n` + + `🆔 Name: ${firstName}\n` + + ` \u251c Username: @${username}\n` + + ` \u251c ID Telegram: ${userId}\n` + + ` \u2514 Link: Click Here\n\n` + + `👁‍🗨 Hello ${firstName}, Welcome to Bot Monitoring\n🌏 Please Choose your Language\n\n` + + `•❂•─•─•❂•─•❂••❂•─•❂•─•─•❂•\n\n`; + + await ctx.replyWithPhoto("https://imgur.com/a/mXrMZsI", { + caption: message, + parse_mode: "HTML", + reply_markup: getLanguageKeyboard(), + }); + + // Log success + logger.info(`Sent start message to ${firstName} (${userId})`); + } catch (error: unknown) { + // Type assertion to Error + if (error instanceof Error) { + logger.error(`Error in startHandler: ${error.message}`); + } else { + logger.error("Unknown error in startHandler"); + } + await ctx.reply("⚠️ Something went wrong. Please try again later."); + } } diff --git a/src/languages/arabic.json b/src/languages/arabic.json new file mode 100644 index 0000000..452bd01 --- /dev/null +++ b/src/languages/arabic.json @@ -0,0 +1,12 @@ +{ + "languageSetMessage": "Bahasa telah berhasil diatur!", + "dashboard": "📋 لوحة القيادة", + "buyProduct": "🛒 شراء المنتج", + "referral": "🔗 الإحالة", + "bonus": "🎁 مكافأة", + "support": "📞 الدعم", + "aboutUs": "ℹ️ حولنا", + "extraMenu": "⚙️ القائمة الإضافية", + "changeLanguage": "🗣️ تغيير اللغة" + } + \ No newline at end of file diff --git a/src/languages/chinese.json b/src/languages/chinese.json new file mode 100644 index 0000000..ab4fc91 --- /dev/null +++ b/src/languages/chinese.json @@ -0,0 +1,12 @@ +{ + "languageSetMessage": "Language has been set successfully!", + "dashboard": "📋 仪表板", + "buyProduct": "🛒 购买产品", + "referral": "🔗 推荐", + "bonus": "🎁 奖金", + "support": "📞 支持", + "aboutUs": "ℹ️ 关于我们", + "extraMenu": "⚙️ 额外菜单", + "changeLanguage": "🗣️ 更改语言" + } + \ No newline at end of file diff --git a/src/languages/english.json b/src/languages/english.json new file mode 100644 index 0000000..1b1e301 --- /dev/null +++ b/src/languages/english.json @@ -0,0 +1,12 @@ +{ + "languageSetMessage": "Language has been set successfully!", + "dashboard": "📋 Dashboard", + "buyProduct": "🛒 Buy Product", + "referral": "🔗 Referral", + "bonus": "🎁 Bonus", + "support": "📞 Support", + "aboutUs": "ℹ️ About Us", + "extraMenu": "⚙️ Extra Menu", + "changeLanguage": "🗣️ Change Language" + } + \ No newline at end of file diff --git a/src/languages/french.json b/src/languages/french.json new file mode 100644 index 0000000..e735204 --- /dev/null +++ b/src/languages/french.json @@ -0,0 +1,12 @@ +{ + "languageSetMessage": "Language has been set successfully!", + "dashboard": "📋 Tableau de bord", + "buyProduct": "🛒 Acheter produit", + "referral": "🔗 Parrainage", + "bonus": "🎁 Bonus", + "support": "📞 Support", + "aboutUs": "ℹ️ À propos de nous", + "extraMenu": "⚙️ Menu supplémentaire", + "changeLanguage": "🗣️ Changer de langue" + } + \ No newline at end of file diff --git a/src/languages/german.json b/src/languages/german.json new file mode 100644 index 0000000..ddc9f2a --- /dev/null +++ b/src/languages/german.json @@ -0,0 +1,12 @@ +{ + "languageSetMessage": "Bahasa telah berhasil diatur!", + "dashboard": "📋 Dashboard", + "buyProduct": "🛒 Produkt kaufen", + "referral": "🔗 Referral", + "bonus": "🎁 Bonus", + "support": "📞 Unterstützung", + "aboutUs": "ℹ️ Über uns", + "extraMenu": "⚙️ Extras", + "changeLanguage": "🗣️ Sprache ändern" + } + \ No newline at end of file diff --git a/src/languages/hindi.json b/src/languages/hindi.json new file mode 100644 index 0000000..b2e71e5 --- /dev/null +++ b/src/languages/hindi.json @@ -0,0 +1,12 @@ +{ + "languageSetMessage": "Language has been set successfully!", + "dashboard": "📋 डैशबोर्ड", + "buyProduct": "🛒 उत्पाद खरीदें", + "referral": "🔗 रेफरल", + "bonus": "🎁 बोनस", + "support": "📞 सहायता", + "aboutUs": "ℹ️ हमारे बारे में", + "extraMenu": "⚙️ अतिरिक्त मेनू", + "changeLanguage": "🗣️ भाषा बदलें" + } + \ No newline at end of file diff --git a/src/languages/indonesian.json b/src/languages/indonesian.json new file mode 100644 index 0000000..14187b6 --- /dev/null +++ b/src/languages/indonesian.json @@ -0,0 +1,12 @@ +{ + "languageSetMessage": "Language has been set successfully!", + "dashboard": "📋 Dasbor", + "buyProduct": "🛒 Beli Produk", + "referral": "🔗 Referensi", + "bonus": "🎁 Hadiah", + "support": "📞 Bantuan", + "aboutUs": "ℹ️ Tentang Kami", + "extraMenu": "⚙️ Menu Tambahan", + "changeLanguage": "🗣️ Ubah Bahasa" + } + \ No newline at end of file diff --git a/src/languages/italian.json b/src/languages/italian.json new file mode 100644 index 0000000..0485d92 --- /dev/null +++ b/src/languages/italian.json @@ -0,0 +1,12 @@ +{ + "languageSetMessage": "Language has been set successfully!", + "dashboard": "📋 Cruscotto", + "buyProduct": "🛒 Acquista prodotto", + "referral": "🔗 Referral", + "bonus": "🎁 Bonus", + "support": "📞 Supporto", + "aboutUs": "ℹ️ Chi siamo", + "extraMenu": "⚙️ Menu extra", + "changeLanguage": "🗣️ Cambia lingua" + } + \ No newline at end of file diff --git a/src/languages/japanese.json b/src/languages/japanese.json new file mode 100644 index 0000000..de1952f --- /dev/null +++ b/src/languages/japanese.json @@ -0,0 +1,12 @@ +{ + "languageSetMessage": "Language has been set successfully!", + "dashboard": "📋 ダッシュボード", + "buyProduct": "🛒 製品を購入", + "referral": "🔗 紹介", + "bonus": "🎁 ボーナス", + "support": "📞 サポート", + "aboutUs": "ℹ️ 私たちについて", + "extraMenu": "⚙️ 追加メニュー", + "changeLanguage": "🗣️ 言語を変更" + } + \ No newline at end of file diff --git a/src/languages/korean.json b/src/languages/korean.json new file mode 100644 index 0000000..b42549f --- /dev/null +++ b/src/languages/korean.json @@ -0,0 +1,12 @@ +{ + "languageSetMessage": "Language has been set successfully!", + "dashboard": "📋 대시보드", + "buyProduct": "🛒 제품 구매", + "referral": "🔗 추천인", + "bonus": "🎁 보너스", + "support": "📞 지원", + "aboutUs": "ℹ️ 회사 소개", + "extraMenu": "⚙️ 추가 메뉴", + "changeLanguage": "🗣️ 언어 변경" + } + \ No newline at end of file diff --git a/src/languages/malaysian.json b/src/languages/malaysian.json new file mode 100644 index 0000000..889219b --- /dev/null +++ b/src/languages/malaysian.json @@ -0,0 +1,12 @@ +{ + "languageSetMessage": "Language has been set successfully!", + "dashboard": "📋 Papan Pemuka", + "buyProduct": "🛒 Beli Produk", + "referral": "🔗 Rujukan", + "bonus": "🎁 Bonus", + "support": "📞 Sokongan", + "aboutUs": "ℹ️ Tentang Kami", + "extraMenu": "⚙️ Menu Tambahan", + "changeLanguage": "🗣️ Tukar Bahasa" + } + \ No newline at end of file diff --git a/src/languages/philippine.json b/src/languages/philippine.json new file mode 100644 index 0000000..9d086bb --- /dev/null +++ b/src/languages/philippine.json @@ -0,0 +1,12 @@ +{ + "languageSetMessage": "Language has been set successfully!", + "dashboard": "📋 Dashboard", + "buyProduct": "🛒 Bumili ng Produkto", + "referral": "🔗 Referral", + "bonus": "🎁 Bonus", + "support": "📞 Suporta", + "aboutUs": "ℹ️ Tungkol sa Amin", + "extraMenu": "⚙️ Karagdagang Menu", + "changeLanguage": "🗣️ Palitan ang Wika" + } + \ No newline at end of file diff --git a/src/languages/polish.json b/src/languages/polish.json new file mode 100644 index 0000000..d164895 --- /dev/null +++ b/src/languages/polish.json @@ -0,0 +1,12 @@ +{ + "languageSetMessage": "Language has been set successfully!", + "dashboard": "📋 Panel", + "buyProduct": "🛒 Kup produkt", + "referral": "🔗 Polecenie", + "bonus": "🎁 Bonus", + "support": "📞 Wsparcie", + "aboutUs": "ℹ️ O nas", + "extraMenu": "⚙️ Dodatkowe menu", + "changeLanguage": "🗣️ Zmień język" + } + \ No newline at end of file diff --git a/src/languages/portuguese.json b/src/languages/portuguese.json new file mode 100644 index 0000000..1d30de7 --- /dev/null +++ b/src/languages/portuguese.json @@ -0,0 +1,12 @@ +{ + "languageSetMessage": "Language has been set successfully!", + "dashboard": "📋 Painel", + "buyProduct": "🛒 Comprar Produto", + "referral": "🔗 Referência", + "bonus": "🎁 Bônus", + "support": "📞 Suporte", + "aboutUs": "ℹ️ Sobre nós", + "extraMenu": "⚙️ Menu extra", + "changeLanguage": "🗣️ Mudar idioma" + } + \ No newline at end of file diff --git a/src/languages/russian.json b/src/languages/russian.json new file mode 100644 index 0000000..084e9e8 --- /dev/null +++ b/src/languages/russian.json @@ -0,0 +1,12 @@ +{ + "languageSetMessage": "Language has been set successfully!", + "dashboard": "📋 Панель управления", + "buyProduct": "🛒 Купить продукт", + "referral": "🔗 Реферальная программа", + "bonus": "🎁 Бонус", + "support": "📞 Поддержка", + "aboutUs": "ℹ️ О нас", + "extraMenu": "⚙️ Дополнительное меню", + "changeLanguage": "🗣️ Изменить язык" + } + \ No newline at end of file diff --git a/src/languages/spanish.json b/src/languages/spanish.json new file mode 100644 index 0000000..63f7fdf --- /dev/null +++ b/src/languages/spanish.json @@ -0,0 +1,12 @@ +{ + "languageSetMessage": "Language has been set successfully!", + "dashboard": "📋 Tablero", + "buyProduct": "🛒 Comprar Producto", + "referral": "🔗 Referido", + "bonus": "🎁 Bono", + "support": "📞 Soporte", + "aboutUs": "ℹ️ Sobre nosotros", + "extraMenu": "⚙️ Menú extra", + "changeLanguage": "🗣️ Cambiar idioma" + } + \ No newline at end of file diff --git a/src/middlewares/middleware.ts b/src/middlewares/middleware.ts deleted file mode 100644 index e492b4a..0000000 --- a/src/middlewares/middleware.ts +++ /dev/null @@ -1 +0,0 @@ -import { Context, NextFunction } from "grammy"; diff --git a/src/utils.ts b/src/utils/language.ts similarity index 100% rename from src/utils.ts rename to src/utils/language.ts diff --git a/src/utils/logger.ts b/src/utils/logger.ts new file mode 100644 index 0000000..23a3ddc --- /dev/null +++ b/src/utils/logger.ts @@ -0,0 +1,31 @@ +import winston from "winston"; + +// Create a custom logger +export const logger = winston.createLogger({ + level: "info", // Set the default log level + transports: [ + // Log to the console + new winston.transports.Console({ + format: winston.format.combine( + winston.format.colorize(), + winston.format.simple() + ), + }), + // Log to a file (error logs and general logs) + new winston.transports.File({ + filename: "logs/error.log", + level: "error", + format: winston.format.combine( + winston.format.timestamp(), + winston.format.json() + ), + }), + new winston.transports.File({ + filename: "logs/combined.log", + format: winston.format.combine( + winston.format.timestamp(), + winston.format.json() + ), + }), + ], +}); diff --git a/tsconfig.json b/tsconfig.json index 95c89cd..c79927a 100644 --- a/tsconfig.json +++ b/tsconfig.json @@ -1,12 +1,16 @@ { - "compilerOptions": { - "target": "ESNext", - "module": "CommonJS", - "outDir": "./dist", - "rootDir": "./src", - "strict": true, - "types": ["node"] - }, - "include": ["src/**/*.ts", "node_modules/grammy/types/**/*.d.ts"] - } - \ No newline at end of file + "compilerOptions": { + "target": "ESNext", + "module": "CommonJS", + "outDir": "./dist", + "rootDir": "./src", + "strict": true, + "types": ["node"], + "esModuleInterop": true, + "resolveJsonModule": true, + "moduleResolution": "node", + "skipLibCheck": true, + "forceConsistentCasingInFileNames": true + }, + "include": ["src/**/*.ts", "node_modules/grammy/types/**/*.d.ts", "src/languages/*.json"] +} From e2a4e0129ae5ba7a044b91922ddbf94c995bcea9 Mon Sep 17 00:00:00 2001 From: MAmamiyaAhmad Date: Tue, 4 Feb 2025 22:52:11 +0700 Subject: [PATCH 06/11] Update --- src/bot.ts | 9 ++-- src/database/db.ts | 53 +++++++++++++++++++ src/handlers/language.ts | 23 -------- src/handlers/languageHandler.ts | 31 +++++++++++ src/handlers/{start.ts => startHandler.ts} | 40 ++++++++------ src/languages/{arabic.json => arabian.json} | 4 +- .../{portuguese.json => brazilian.json} | 4 +- src/languages/chinese.json | 4 +- src/languages/english.json | 4 +- src/languages/{french.json => france.json} | 4 +- src/languages/{german.json => germany.json} | 4 +- src/languages/hindi.json | 4 +- src/languages/indonesian.json | 4 +- src/languages/italian.json | 4 +- src/languages/{japanese.json => japan.json} | 4 +- src/languages/korean.json | 4 +- src/languages/malaysian.json | 4 +- .../{philippine.json => philippines.json} | 4 +- src/languages/{polish.json => poland.json} | 4 +- src/languages/russian.json | 4 +- src/languages/spanish.json | 4 +- src/types/types.ts | 5 ++ src/utils/{language.ts => languageUtil.ts} | 16 ++++++ src/utils/{logger.ts => loggerUtil.ts} | 0 src/utils/mainMenuUtil.ts | 0 src/utils/menuUtil.ts | 36 +++++++++++++ src/utils/rolesUtil.ts | 9 ++++ 27 files changed, 196 insertions(+), 90 deletions(-) create mode 100644 src/database/db.ts delete mode 100644 src/handlers/language.ts create mode 100644 src/handlers/languageHandler.ts rename src/handlers/{start.ts => startHandler.ts} (58%) rename src/languages/{arabic.json => arabian.json} (59%) rename src/languages/{portuguese.json => brazilian.json} (59%) rename src/languages/{french.json => france.json} (58%) rename src/languages/{german.json => germany.json} (60%) rename src/languages/{japanese.json => japan.json} (60%) rename src/languages/{philippine.json => philippines.json} (58%) rename src/languages/{polish.json => poland.json} (56%) create mode 100644 src/types/types.ts rename src/utils/{language.ts => languageUtil.ts} (62%) rename src/utils/{logger.ts => loggerUtil.ts} (100%) create mode 100644 src/utils/mainMenuUtil.ts create mode 100644 src/utils/menuUtil.ts create mode 100644 src/utils/rolesUtil.ts diff --git a/src/bot.ts b/src/bot.ts index 9fe46f3..7fef0af 100644 --- a/src/bot.ts +++ b/src/bot.ts @@ -1,17 +1,18 @@ import { Bot } from "grammy"; import { BOT_TOKEN } from "./config/config"; -import { startHandler } from "./handlers/start"; -import { languageHandler } from "./handlers/language"; -import { logger } from "./utils/logger"; // Import the logger +import { startHandler } from "./handlers/startHandler"; +import { languageHandler } from "./handlers/languageHandler"; +import { logger } from "./utils/loggerUtil"; // Import the logger const bot = new Bot(BOT_TOKEN); // Command /start with auto-detect bot username bot.command("start", async (ctx) => { logger.info(`Received /start command from ${ctx.from?.id}`); - await startHandler(ctx, bot); + await startHandler(ctx, bot); // Kirim `bot` sebagai argumen }); + // Handle language selection bot.callbackQuery(/.*/, (ctx) => { logger.info(`Received language selection from ${ctx.from?.id}: ${ctx.match}`); diff --git a/src/database/db.ts b/src/database/db.ts new file mode 100644 index 0000000..83da570 --- /dev/null +++ b/src/database/db.ts @@ -0,0 +1,53 @@ +import Database from "better-sqlite3"; +import { User } from "../types/types"; +// Initialize the database (or open if exists) +const db = new Database("users.db", { verbose: console.log }); + +// Create the users table if it doesn't exist +db.prepare(` + CREATE TABLE IF NOT EXISTS users ( + user_id INTEGER PRIMARY KEY, + role TEXT, + language TEXT + ) +`).run(); + +// Function to get a user by ID +// Contoh penyimpanan user di database +const users: Map = new Map(); + +export function getUser(userId: number): User | undefined { + return users.get(userId); +} + +export function saveUser(userId: number, user: Partial) { + let existingUser = users.get(userId); + + if (existingUser) { + // Update existing user + existingUser = { ...existingUser, ...user }; + } else { + // Ensure `existingUser` is fully typed + existingUser = { + userId, + role: user.role || "user", // Default role jika tidak ada + language: user.language || "english", // Default language jika tidak ada + }; + } + + users.set(userId, existingUser); +} + +// Function to set or update the user's language +export function setUserLanguage(userId: number, language: string) { + const stmt = db.prepare("INSERT OR REPLACE INTO users (user_id, language) VALUES (?, ?)"); + stmt.run(userId, language); +} + +// Function to set the user role +export function setUserRole(userId: number, role: string) { + const stmt = db.prepare("INSERT OR REPLACE INTO users (user_id, role) VALUES (?, ?)"); + stmt.run(userId, role); +} + +export default db; diff --git a/src/handlers/language.ts b/src/handlers/language.ts deleted file mode 100644 index 2a2d094..0000000 --- a/src/handlers/language.ts +++ /dev/null @@ -1,23 +0,0 @@ -import { Context } from "grammy"; -import { logger } from "../utils/logger"; // Import the logger - -export async function languageHandler(ctx: Context) { - const selectedLanguage = ctx.match; - try { - // Log the language selection - logger.info(`User ${ctx.from?.id} selected language: ${selectedLanguage}`); - - await ctx.reply(`✅ You have selected: ${selectedLanguage}`); - - // Log success - logger.info(`Successfully sent language selection reply to ${ctx.from?.id}`); - } catch (error: unknown) { - // Type assertion to Error - if (error instanceof Error) { - logger.error(`Error in languageHandler: ${error.message}`); - } else { - logger.error("Unknown error in languageHandler"); - } - await ctx.reply("⚠️ Something went wrong with language selection."); - } -} diff --git a/src/handlers/languageHandler.ts b/src/handlers/languageHandler.ts new file mode 100644 index 0000000..8f5deb1 --- /dev/null +++ b/src/handlers/languageHandler.ts @@ -0,0 +1,31 @@ +import { Context } from "grammy"; +import { getMainMenuKeyboard } from "../utils/menuUtil"; +import { saveUser } from "../database/db"; +import { logger } from "../utils/loggerUtil"; + +export async function languageHandler(ctx: Context) { + const userId = ctx.from?.id; + const language = ctx.callbackQuery?.data; + + if (!userId || !language) { + logger.error("User ID or language selection is missing."); + await ctx.answerCallbackQuery("⚠️ Language selection failed."); + return; + } + + try { + saveUser(userId, { role: "user", language }); + logger.info(`User ${userId} selected language: ${language}`); + + const mainMenu = await getMainMenuKeyboard(language, userId); + await ctx.reply(`✅ Language set to ${language}! Here is your main menu:`, { + reply_markup: mainMenu, + parse_mode: "HTML" + }); + + await ctx.answerCallbackQuery(); + } catch (error) { + logger.error(`Error in languageHandler: ${error}`); + await ctx.reply("⚠️ Failed to update language. Please try again."); + } +} diff --git a/src/handlers/start.ts b/src/handlers/startHandler.ts similarity index 58% rename from src/handlers/start.ts rename to src/handlers/startHandler.ts index cc93b94..e19343c 100644 --- a/src/handlers/start.ts +++ b/src/handlers/startHandler.ts @@ -1,19 +1,35 @@ import { Context, Bot } from "grammy"; -import { getLanguageKeyboard, getBotUsername } from "../utils/language"; -import { logger } from "../utils/logger"; // Import the logger +import { getLanguageKeyboard, getBotUsername } from "../utils/languageUtil"; +import { logger } from "../utils/loggerUtil"; +import { saveUser, getUser } from "../database/db"; +import { getMainMenuKeyboard } from "../utils/menuUtil"; export async function startHandler(ctx: Context, bot: Bot) { const userId = ctx.from?.id; const firstName = ctx.from?.first_name; const username = ctx.from?.username || "Tidak Ada"; - // Log user info - logger.info(`User ${firstName} (${userId}) started the bot`); + if (!userId) { + logger.error("User ID not found."); + return; + } try { - // Get bot username - const botUsername = await getBotUsername(bot); + logger.info(`User ${firstName} (${userId}) started the bot`); + + // Periksa apakah user sudah memiliki bahasa yang tersimpan + const user = getUser(userId); + if (user?.language) { + const mainMenu = await getMainMenuKeyboard(user.language, userId); + await ctx.reply("✅ Welcome back! Here is your main menu:", { + reply_markup: mainMenu, + parse_mode: "HTML" + }); + return; + } + // Jika bahasa belum dipilih, tampilkan opsi bahasa + const botUsername = await getBotUsername(bot); const message = `🌏 Choose Language\n` + `•❂•─•─•❂•─•❂••❂•─•❂•─•─•❂•\n\n` + `🆔 Name: ${firstName}\n` @@ -29,15 +45,9 @@ export async function startHandler(ctx: Context, bot: Bot) { reply_markup: getLanguageKeyboard(), }); - // Log success - logger.info(`Sent start message to ${firstName} (${userId})`); - } catch (error: unknown) { - // Type assertion to Error - if (error instanceof Error) { - logger.error(`Error in startHandler: ${error.message}`); - } else { - logger.error("Unknown error in startHandler"); - } + logger.info(`Sent language selection to ${firstName} (${userId})`); + } catch (error) { + logger.error(`Error in startHandler: ${error}`); await ctx.reply("⚠️ Something went wrong. Please try again later."); } } diff --git a/src/languages/arabic.json b/src/languages/arabian.json similarity index 59% rename from src/languages/arabic.json rename to src/languages/arabian.json index 452bd01..cdf315e 100644 --- a/src/languages/arabic.json +++ b/src/languages/arabian.json @@ -1,12 +1,10 @@ { - "languageSetMessage": "Bahasa telah berhasil diatur!", "dashboard": "📋 لوحة القيادة", "buyProduct": "🛒 شراء المنتج", "referral": "🔗 الإحالة", "bonus": "🎁 مكافأة", "support": "📞 الدعم", "aboutUs": "ℹ️ حولنا", - "extraMenu": "⚙️ القائمة الإضافية", - "changeLanguage": "🗣️ تغيير اللغة" + "extraMenu": "⚙️ القائمة الإضافية" } \ No newline at end of file diff --git a/src/languages/portuguese.json b/src/languages/brazilian.json similarity index 59% rename from src/languages/portuguese.json rename to src/languages/brazilian.json index 1d30de7..bcfe80d 100644 --- a/src/languages/portuguese.json +++ b/src/languages/brazilian.json @@ -1,12 +1,10 @@ { - "languageSetMessage": "Language has been set successfully!", "dashboard": "📋 Painel", "buyProduct": "🛒 Comprar Produto", "referral": "🔗 Referência", "bonus": "🎁 Bônus", "support": "📞 Suporte", "aboutUs": "ℹ️ Sobre nós", - "extraMenu": "⚙️ Menu extra", - "changeLanguage": "🗣️ Mudar idioma" + "extraMenu": "⚙️ Menu extra" } \ No newline at end of file diff --git a/src/languages/chinese.json b/src/languages/chinese.json index ab4fc91..90ab585 100644 --- a/src/languages/chinese.json +++ b/src/languages/chinese.json @@ -1,12 +1,10 @@ { - "languageSetMessage": "Language has been set successfully!", "dashboard": "📋 仪表板", "buyProduct": "🛒 购买产品", "referral": "🔗 推荐", "bonus": "🎁 奖金", "support": "📞 支持", "aboutUs": "ℹ️ 关于我们", - "extraMenu": "⚙️ 额外菜单", - "changeLanguage": "🗣️ 更改语言" + "extraMenu": "⚙️ 额外菜单" } \ No newline at end of file diff --git a/src/languages/english.json b/src/languages/english.json index 1b1e301..b66ce92 100644 --- a/src/languages/english.json +++ b/src/languages/english.json @@ -1,12 +1,10 @@ { - "languageSetMessage": "Language has been set successfully!", "dashboard": "📋 Dashboard", "buyProduct": "🛒 Buy Product", "referral": "🔗 Referral", "bonus": "🎁 Bonus", "support": "📞 Support", "aboutUs": "ℹ️ About Us", - "extraMenu": "⚙️ Extra Menu", - "changeLanguage": "🗣️ Change Language" + "extraMenu": "⚙️ Extra Menu" } \ No newline at end of file diff --git a/src/languages/french.json b/src/languages/france.json similarity index 58% rename from src/languages/french.json rename to src/languages/france.json index e735204..332794f 100644 --- a/src/languages/french.json +++ b/src/languages/france.json @@ -1,12 +1,10 @@ { - "languageSetMessage": "Language has been set successfully!", "dashboard": "📋 Tableau de bord", "buyProduct": "🛒 Acheter produit", "referral": "🔗 Parrainage", "bonus": "🎁 Bonus", "support": "📞 Support", "aboutUs": "ℹ️ À propos de nous", - "extraMenu": "⚙️ Menu supplémentaire", - "changeLanguage": "🗣️ Changer de langue" + "extraMenu": "⚙️ Menu supplémentaire" } \ No newline at end of file diff --git a/src/languages/german.json b/src/languages/germany.json similarity index 60% rename from src/languages/german.json rename to src/languages/germany.json index ddc9f2a..7c03106 100644 --- a/src/languages/german.json +++ b/src/languages/germany.json @@ -1,12 +1,10 @@ { - "languageSetMessage": "Bahasa telah berhasil diatur!", "dashboard": "📋 Dashboard", "buyProduct": "🛒 Produkt kaufen", "referral": "🔗 Referral", "bonus": "🎁 Bonus", "support": "📞 Unterstützung", "aboutUs": "ℹ️ Über uns", - "extraMenu": "⚙️ Extras", - "changeLanguage": "🗣️ Sprache ändern" + "extraMenu": "⚙️ Extras" } \ No newline at end of file diff --git a/src/languages/hindi.json b/src/languages/hindi.json index b2e71e5..8bdeb94 100644 --- a/src/languages/hindi.json +++ b/src/languages/hindi.json @@ -1,12 +1,10 @@ { - "languageSetMessage": "Language has been set successfully!", "dashboard": "📋 डैशबोर्ड", "buyProduct": "🛒 उत्पाद खरीदें", "referral": "🔗 रेफरल", "bonus": "🎁 बोनस", "support": "📞 सहायता", "aboutUs": "ℹ️ हमारे बारे में", - "extraMenu": "⚙️ अतिरिक्त मेनू", - "changeLanguage": "🗣️ भाषा बदलें" + "extraMenu": "⚙️ अतिरिक्त मेनू" } \ No newline at end of file diff --git a/src/languages/indonesian.json b/src/languages/indonesian.json index 14187b6..ca256b7 100644 --- a/src/languages/indonesian.json +++ b/src/languages/indonesian.json @@ -1,12 +1,10 @@ { - "languageSetMessage": "Language has been set successfully!", "dashboard": "📋 Dasbor", "buyProduct": "🛒 Beli Produk", "referral": "🔗 Referensi", "bonus": "🎁 Hadiah", "support": "📞 Bantuan", "aboutUs": "ℹ️ Tentang Kami", - "extraMenu": "⚙️ Menu Tambahan", - "changeLanguage": "🗣️ Ubah Bahasa" + "extraMenu": "⚙️ Menu Tambahan" } \ No newline at end of file diff --git a/src/languages/italian.json b/src/languages/italian.json index 0485d92..a8698e1 100644 --- a/src/languages/italian.json +++ b/src/languages/italian.json @@ -1,12 +1,10 @@ { - "languageSetMessage": "Language has been set successfully!", "dashboard": "📋 Cruscotto", "buyProduct": "🛒 Acquista prodotto", "referral": "🔗 Referral", "bonus": "🎁 Bonus", "support": "📞 Supporto", "aboutUs": "ℹ️ Chi siamo", - "extraMenu": "⚙️ Menu extra", - "changeLanguage": "🗣️ Cambia lingua" + "extraMenu": "⚙️ Menu extra" } \ No newline at end of file diff --git a/src/languages/japanese.json b/src/languages/japan.json similarity index 60% rename from src/languages/japanese.json rename to src/languages/japan.json index de1952f..65af024 100644 --- a/src/languages/japanese.json +++ b/src/languages/japan.json @@ -1,12 +1,10 @@ { - "languageSetMessage": "Language has been set successfully!", "dashboard": "📋 ダッシュボード", "buyProduct": "🛒 製品を購入", "referral": "🔗 紹介", "bonus": "🎁 ボーナス", "support": "📞 サポート", "aboutUs": "ℹ️ 私たちについて", - "extraMenu": "⚙️ 追加メニュー", - "changeLanguage": "🗣️ 言語を変更" + "extraMenu": "⚙️ 追加メニュー" } \ No newline at end of file diff --git a/src/languages/korean.json b/src/languages/korean.json index b42549f..22eb276 100644 --- a/src/languages/korean.json +++ b/src/languages/korean.json @@ -1,12 +1,10 @@ { - "languageSetMessage": "Language has been set successfully!", "dashboard": "📋 대시보드", "buyProduct": "🛒 제품 구매", "referral": "🔗 추천인", "bonus": "🎁 보너스", "support": "📞 지원", "aboutUs": "ℹ️ 회사 소개", - "extraMenu": "⚙️ 추가 메뉴", - "changeLanguage": "🗣️ 언어 변경" + "extraMenu": "⚙️ 추가 메뉴" } \ No newline at end of file diff --git a/src/languages/malaysian.json b/src/languages/malaysian.json index 889219b..920aaf2 100644 --- a/src/languages/malaysian.json +++ b/src/languages/malaysian.json @@ -1,12 +1,10 @@ { - "languageSetMessage": "Language has been set successfully!", "dashboard": "📋 Papan Pemuka", "buyProduct": "🛒 Beli Produk", "referral": "🔗 Rujukan", "bonus": "🎁 Bonus", "support": "📞 Sokongan", "aboutUs": "ℹ️ Tentang Kami", - "extraMenu": "⚙️ Menu Tambahan", - "changeLanguage": "🗣️ Tukar Bahasa" + "extraMenu": "⚙️ Menu Tambahan" } \ No newline at end of file diff --git a/src/languages/philippine.json b/src/languages/philippines.json similarity index 58% rename from src/languages/philippine.json rename to src/languages/philippines.json index 9d086bb..47abea5 100644 --- a/src/languages/philippine.json +++ b/src/languages/philippines.json @@ -1,12 +1,10 @@ { - "languageSetMessage": "Language has been set successfully!", "dashboard": "📋 Dashboard", "buyProduct": "🛒 Bumili ng Produkto", "referral": "🔗 Referral", "bonus": "🎁 Bonus", "support": "📞 Suporta", "aboutUs": "ℹ️ Tungkol sa Amin", - "extraMenu": "⚙️ Karagdagang Menu", - "changeLanguage": "🗣️ Palitan ang Wika" + "extraMenu": "⚙️ Karagdagang Menu" } \ No newline at end of file diff --git a/src/languages/polish.json b/src/languages/poland.json similarity index 56% rename from src/languages/polish.json rename to src/languages/poland.json index d164895..033d099 100644 --- a/src/languages/polish.json +++ b/src/languages/poland.json @@ -1,12 +1,10 @@ { - "languageSetMessage": "Language has been set successfully!", "dashboard": "📋 Panel", "buyProduct": "🛒 Kup produkt", "referral": "🔗 Polecenie", "bonus": "🎁 Bonus", "support": "📞 Wsparcie", "aboutUs": "ℹ️ O nas", - "extraMenu": "⚙️ Dodatkowe menu", - "changeLanguage": "🗣️ Zmień język" + "extraMenu": "⚙️ Dodatkowe menu" } \ No newline at end of file diff --git a/src/languages/russian.json b/src/languages/russian.json index 084e9e8..851f408 100644 --- a/src/languages/russian.json +++ b/src/languages/russian.json @@ -1,12 +1,10 @@ { - "languageSetMessage": "Language has been set successfully!", "dashboard": "📋 Панель управления", "buyProduct": "🛒 Купить продукт", "referral": "🔗 Реферальная программа", "bonus": "🎁 Бонус", "support": "📞 Поддержка", "aboutUs": "ℹ️ О нас", - "extraMenu": "⚙️ Дополнительное меню", - "changeLanguage": "🗣️ Изменить язык" + "extraMenu": "⚙️ Дополнительное меню" } \ No newline at end of file diff --git a/src/languages/spanish.json b/src/languages/spanish.json index 63f7fdf..e36ccd3 100644 --- a/src/languages/spanish.json +++ b/src/languages/spanish.json @@ -1,12 +1,10 @@ { - "languageSetMessage": "Language has been set successfully!", "dashboard": "📋 Tablero", "buyProduct": "🛒 Comprar Producto", "referral": "🔗 Referido", "bonus": "🎁 Bono", "support": "📞 Soporte", "aboutUs": "ℹ️ Sobre nosotros", - "extraMenu": "⚙️ Menú extra", - "changeLanguage": "🗣️ Cambiar idioma" + "extraMenu": "⚙️ Menú extra" } \ No newline at end of file diff --git a/src/types/types.ts b/src/types/types.ts new file mode 100644 index 0000000..bc6eff3 --- /dev/null +++ b/src/types/types.ts @@ -0,0 +1,5 @@ +export interface User { + userId: number; + language: string; + role?: "admin" | "super_admin" | "user"; // Jika ada role +} diff --git a/src/utils/language.ts b/src/utils/languageUtil.ts similarity index 62% rename from src/utils/language.ts rename to src/utils/languageUtil.ts index 5b33aa6..bf88a51 100644 --- a/src/utils/language.ts +++ b/src/utils/languageUtil.ts @@ -1,5 +1,21 @@ import { InlineKeyboard } from "grammy"; import { Bot } from "grammy"; +import * as fs from "fs"; +import path from "path"; + +// Fungsi untuk memuat file terjemahan berdasarkan bahasa yang dipilih +export function loadTranslations(language: string) { + const filePath = path.join(__dirname, "..", "languages", `${language}.json`); + + try { + const translations = JSON.parse(fs.readFileSync(filePath, "utf8")); + return translations; + } catch (error: unknown) { + console.error(`Failed to load translations for ${language}:`, (error as Error).message); + const defaultPath = path.join(__dirname, "..", "languages", "english.json"); + return JSON.parse(fs.readFileSync(defaultPath, "utf8")); + } +} export function getLanguageKeyboard() { return new InlineKeyboard() diff --git a/src/utils/logger.ts b/src/utils/loggerUtil.ts similarity index 100% rename from src/utils/logger.ts rename to src/utils/loggerUtil.ts diff --git a/src/utils/mainMenuUtil.ts b/src/utils/mainMenuUtil.ts new file mode 100644 index 0000000..e69de29 diff --git a/src/utils/menuUtil.ts b/src/utils/menuUtil.ts new file mode 100644 index 0000000..349aa64 --- /dev/null +++ b/src/utils/menuUtil.ts @@ -0,0 +1,36 @@ +import { Keyboard } from "grammy"; +import { getUser } from "../database/db"; +import { User } from "../types/types"; +import { loadTranslations } from "./languageUtil"; +import dotenv from "dotenv"; + +dotenv.config(); + +const SUPER_ADMIN_ID = process.env.SUPER_ADMIN_ID ? Number(process.env.SUPER_ADMIN_ID) : 0; +const ADMIN_IDS = process.env.ADMIN_IDS ? process.env.ADMIN_IDS.split(",").map(Number) : []; + +export async function getMainMenuKeyboard(language: string, userId: number) { + try { + const languageData = loadTranslations(language); + const userData = getUser(userId); + const user: User | undefined = typeof userData === "object" ? (userData as User) : undefined; + + const keyboard = new Keyboard() + .text(languageData.dashboard) + .text(languageData.buyProduct).row() + .text(languageData.referral) + .text(languageData.bonus).row() + .text(languageData.support) + .text(languageData.aboutUs).row() + .text(languageData.extraMenu); + + if (user && (user.role === "admin" || user.role === "super_admin")) { + keyboard.row().text(languageData.adminMenu); + } + + return keyboard.resized(); + } catch (error) { + console.error(`Failed to generate menu keyboard: ${error}`); + throw new Error("Failed to generate main menu."); + } +} diff --git a/src/utils/rolesUtil.ts b/src/utils/rolesUtil.ts new file mode 100644 index 0000000..b4bbd02 --- /dev/null +++ b/src/utils/rolesUtil.ts @@ -0,0 +1,9 @@ +const userRoles = new Map(); + +export function getUserRole(userId: number): string { + return userRoles.get(userId) || "user"; +} + +export function setUserRole(userId: number, role: string): void { + userRoles.set(userId, role); +} From 45572200c31b0f600f2d593cca6d25c0dfbb457b Mon Sep 17 00:00:00 2001 From: MAmamiyaAhmad Date: Tue, 4 Feb 2025 23:57:38 +0700 Subject: [PATCH 07/11] Update --- src/config/config.ts | 4 ++-- src/database/db.ts | 1 - src/handlers/languageHandler.ts | 12 ++++++---- src/handlers/startHandler.ts | 6 ++--- src/languages/arabian.json | 3 ++- src/languages/brazilian.json | 3 ++- src/languages/chinese.json | 3 ++- src/languages/english.json | 3 ++- src/languages/france.json | 3 ++- src/languages/germany.json | 3 ++- src/languages/hindi.json | 3 ++- src/languages/indonesian.json | 3 ++- src/languages/{italian.json => italy.json} | 3 ++- src/languages/japan.json | 3 ++- src/languages/korean.json | 3 ++- src/languages/malaysian.json | 3 ++- src/languages/philippines.json | 3 ++- src/languages/poland.json | 3 ++- src/languages/russian.json | 3 ++- src/languages/spanish.json | 3 ++- src/utils/menuUtil.ts | 27 +++++++++++----------- 21 files changed, 58 insertions(+), 40 deletions(-) rename src/languages/{italian.json => italy.json} (74%) diff --git a/src/config/config.ts b/src/config/config.ts index 0b7858d..a8a0c0d 100644 --- a/src/config/config.ts +++ b/src/config/config.ts @@ -2,5 +2,5 @@ import * as dotenv from "dotenv"; dotenv.config(); export const BOT_TOKEN = process.env.BOT_TOKEN as string; -export const SUPER_ADMIN_ID = parseInt(process.env.SUPER_ADMIN_ID as string); -export const ADMIN_IDS = process.env.ADMIN_IDS?.split(",").map(id => parseInt(id)) || []; \ No newline at end of file +export const SUPER_ADMIN_ID = parseInt(process.env.SUPER_ADMIN_ID as string); // Ensure this exists in your .env +export const ADMIN_IDS = process.env.ADMIN_IDS?.split(",").map(id => parseInt(id)) || []; diff --git a/src/database/db.ts b/src/database/db.ts index 83da570..8b271ca 100644 --- a/src/database/db.ts +++ b/src/database/db.ts @@ -13,7 +13,6 @@ db.prepare(` `).run(); // Function to get a user by ID -// Contoh penyimpanan user di database const users: Map = new Map(); export function getUser(userId: number): User | undefined { diff --git a/src/handlers/languageHandler.ts b/src/handlers/languageHandler.ts index 8f5deb1..bde573a 100644 --- a/src/handlers/languageHandler.ts +++ b/src/handlers/languageHandler.ts @@ -2,6 +2,7 @@ import { Context } from "grammy"; import { getMainMenuKeyboard } from "../utils/menuUtil"; import { saveUser } from "../database/db"; import { logger } from "../utils/loggerUtil"; +import { SUPER_ADMIN_ID } from "../config/config"; // Import SUPER_ADMIN_ID export async function languageHandler(ctx: Context) { const userId = ctx.from?.id; @@ -14,11 +15,14 @@ export async function languageHandler(ctx: Context) { } try { - saveUser(userId, { role: "user", language }); - logger.info(`User ${userId} selected language: ${language}`); + // Check if user is super admin or regular user + const role = userId === SUPER_ADMIN_ID ? "super_admin" : "user"; // Use SUPER_ADMIN_ID from config - const mainMenu = await getMainMenuKeyboard(language, userId); - await ctx.reply(`✅ Language set to ${language}! Here is your main menu:`, { + saveUser(userId, { role, language }); + logger.info(`User ${userId} selected language: ${language}, Role: ${role}`); + + const mainMenu = await getMainMenuKeyboard(language, role, userId); + await ctx.reply(`✅ Language set to ${language}!\n Here is your main menu:`, { reply_markup: mainMenu, parse_mode: "HTML" }); diff --git a/src/handlers/startHandler.ts b/src/handlers/startHandler.ts index e19343c..243c1e3 100644 --- a/src/handlers/startHandler.ts +++ b/src/handlers/startHandler.ts @@ -20,8 +20,8 @@ export async function startHandler(ctx: Context, bot: Bot) { // Periksa apakah user sudah memiliki bahasa yang tersimpan const user = getUser(userId); if (user?.language) { - const mainMenu = await getMainMenuKeyboard(user.language, userId); - await ctx.reply("✅ Welcome back! Here is your main menu:", { + const mainMenu = await getMainMenuKeyboard(user.language || 'english', user.role || 'user', userId); // Fix: Added role and fallback for language + await ctx.reply(`✅ Welcome back! ${firstName}\nHere is your main menu:`, { reply_markup: mainMenu, parse_mode: "HTML" }); @@ -36,7 +36,7 @@ export async function startHandler(ctx: Context, bot: Bot) { + ` \u251c Username: @${username}\n` + ` \u251c ID Telegram: ${userId}\n` + ` \u2514 Link: Click Here\n\n` - + `👁‍🗨 Hello ${firstName}, Welcome to Bot Monitoring\n🌏 Please Choose your Language\n\n` + + `👁‍🗨 Hello ${firstName}, Welcome to ${botUsername}\n🌏 Please Choose your Language\n\n` + `•❂•─•─•❂•─•❂••❂•─•❂•─•─•❂•\n\n`; await ctx.replyWithPhoto("https://imgur.com/a/mXrMZsI", { diff --git a/src/languages/arabian.json b/src/languages/arabian.json index cdf315e..ddf503a 100644 --- a/src/languages/arabian.json +++ b/src/languages/arabian.json @@ -5,6 +5,7 @@ "bonus": "🎁 مكافأة", "support": "📞 الدعم", "aboutUs": "ℹ️ حولنا", - "extraMenu": "⚙️ القائمة الإضافية" + "extraMenu": "⚙️ القائمة الإضافية", + "adminMenu": "👑 Admin Menu" } \ No newline at end of file diff --git a/src/languages/brazilian.json b/src/languages/brazilian.json index bcfe80d..f2ddc90 100644 --- a/src/languages/brazilian.json +++ b/src/languages/brazilian.json @@ -5,6 +5,7 @@ "bonus": "🎁 Bônus", "support": "📞 Suporte", "aboutUs": "ℹ️ Sobre nós", - "extraMenu": "⚙️ Menu extra" + "extraMenu": "⚙️ Menu extra", + "adminMenu": "👑 Admin Menu" } \ No newline at end of file diff --git a/src/languages/chinese.json b/src/languages/chinese.json index 90ab585..0ee574e 100644 --- a/src/languages/chinese.json +++ b/src/languages/chinese.json @@ -5,6 +5,7 @@ "bonus": "🎁 奖金", "support": "📞 支持", "aboutUs": "ℹ️ 关于我们", - "extraMenu": "⚙️ 额外菜单" + "extraMenu": "⚙️ 额外菜单", + "adminMenu": "👑 Admin Menu" } \ No newline at end of file diff --git a/src/languages/english.json b/src/languages/english.json index b66ce92..98b903e 100644 --- a/src/languages/english.json +++ b/src/languages/english.json @@ -5,6 +5,7 @@ "bonus": "🎁 Bonus", "support": "📞 Support", "aboutUs": "ℹ️ About Us", - "extraMenu": "⚙️ Extra Menu" + "extraMenu": "⚙️ Extra Menu", + "adminMenu": "👑 Admin Menu" } \ No newline at end of file diff --git a/src/languages/france.json b/src/languages/france.json index 332794f..87e06eb 100644 --- a/src/languages/france.json +++ b/src/languages/france.json @@ -5,6 +5,7 @@ "bonus": "🎁 Bonus", "support": "📞 Support", "aboutUs": "ℹ️ À propos de nous", - "extraMenu": "⚙️ Menu supplémentaire" + "extraMenu": "⚙️ Menu supplémentaire", + "adminMenu": "👑 Admin Menu" } \ No newline at end of file diff --git a/src/languages/germany.json b/src/languages/germany.json index 7c03106..a93702f 100644 --- a/src/languages/germany.json +++ b/src/languages/germany.json @@ -5,6 +5,7 @@ "bonus": "🎁 Bonus", "support": "📞 Unterstützung", "aboutUs": "ℹ️ Über uns", - "extraMenu": "⚙️ Extras" + "extraMenu": "⚙️ Extras", + "adminMenu": "👑 Admin Menu" } \ No newline at end of file diff --git a/src/languages/hindi.json b/src/languages/hindi.json index 8bdeb94..49a02c0 100644 --- a/src/languages/hindi.json +++ b/src/languages/hindi.json @@ -5,6 +5,7 @@ "bonus": "🎁 बोनस", "support": "📞 सहायता", "aboutUs": "ℹ️ हमारे बारे में", - "extraMenu": "⚙️ अतिरिक्त मेनू" + "extraMenu": "⚙️ अतिरिक्त मेनू", + "adminMenu": "👑 Admin Menu" } \ No newline at end of file diff --git a/src/languages/indonesian.json b/src/languages/indonesian.json index ca256b7..9382ab2 100644 --- a/src/languages/indonesian.json +++ b/src/languages/indonesian.json @@ -5,6 +5,7 @@ "bonus": "🎁 Hadiah", "support": "📞 Bantuan", "aboutUs": "ℹ️ Tentang Kami", - "extraMenu": "⚙️ Menu Tambahan" + "extraMenu": "⚙️ Menu Tambahan", + "adminMenu": "👑 Admin Menu" } \ No newline at end of file diff --git a/src/languages/italian.json b/src/languages/italy.json similarity index 74% rename from src/languages/italian.json rename to src/languages/italy.json index a8698e1..1a83223 100644 --- a/src/languages/italian.json +++ b/src/languages/italy.json @@ -5,6 +5,7 @@ "bonus": "🎁 Bonus", "support": "📞 Supporto", "aboutUs": "ℹ️ Chi siamo", - "extraMenu": "⚙️ Menu extra" + "extraMenu": "⚙️ Menu extra", + "adminMenu": "👑 Admin Menu" } \ No newline at end of file diff --git a/src/languages/japan.json b/src/languages/japan.json index 65af024..79ee7e8 100644 --- a/src/languages/japan.json +++ b/src/languages/japan.json @@ -5,6 +5,7 @@ "bonus": "🎁 ボーナス", "support": "📞 サポート", "aboutUs": "ℹ️ 私たちについて", - "extraMenu": "⚙️ 追加メニュー" + "extraMenu": "⚙️ 追加メニュー", + "adminMenu": "👑 Admin Menu" } \ No newline at end of file diff --git a/src/languages/korean.json b/src/languages/korean.json index 22eb276..327008f 100644 --- a/src/languages/korean.json +++ b/src/languages/korean.json @@ -5,6 +5,7 @@ "bonus": "🎁 보너스", "support": "📞 지원", "aboutUs": "ℹ️ 회사 소개", - "extraMenu": "⚙️ 추가 메뉴" + "extraMenu": "⚙️ 추가 메뉴", + "adminMenu": "👑 Admin Menu" } \ No newline at end of file diff --git a/src/languages/malaysian.json b/src/languages/malaysian.json index 920aaf2..584f642 100644 --- a/src/languages/malaysian.json +++ b/src/languages/malaysian.json @@ -5,6 +5,7 @@ "bonus": "🎁 Bonus", "support": "📞 Sokongan", "aboutUs": "ℹ️ Tentang Kami", - "extraMenu": "⚙️ Menu Tambahan" + "extraMenu": "⚙️ Menu Tambahan", + "adminMenu": "👑 Admin Menu" } \ No newline at end of file diff --git a/src/languages/philippines.json b/src/languages/philippines.json index 47abea5..1642105 100644 --- a/src/languages/philippines.json +++ b/src/languages/philippines.json @@ -5,6 +5,7 @@ "bonus": "🎁 Bonus", "support": "📞 Suporta", "aboutUs": "ℹ️ Tungkol sa Amin", - "extraMenu": "⚙️ Karagdagang Menu" + "extraMenu": "⚙️ Karagdagang Menu", + "adminMenu": "👑 Admin Menu" } \ No newline at end of file diff --git a/src/languages/poland.json b/src/languages/poland.json index 033d099..6773fa7 100644 --- a/src/languages/poland.json +++ b/src/languages/poland.json @@ -5,6 +5,7 @@ "bonus": "🎁 Bonus", "support": "📞 Wsparcie", "aboutUs": "ℹ️ O nas", - "extraMenu": "⚙️ Dodatkowe menu" + "extraMenu": "⚙️ Dodatkowe menu", + "adminMenu": "👑 Admin Menu" } \ No newline at end of file diff --git a/src/languages/russian.json b/src/languages/russian.json index 851f408..2fd12e2 100644 --- a/src/languages/russian.json +++ b/src/languages/russian.json @@ -5,6 +5,7 @@ "bonus": "🎁 Бонус", "support": "📞 Поддержка", "aboutUs": "ℹ️ О нас", - "extraMenu": "⚙️ Дополнительное меню" + "extraMenu": "⚙️ Дополнительное меню", + "adminMenu": "👑 Admin Menu" } \ No newline at end of file diff --git a/src/languages/spanish.json b/src/languages/spanish.json index e36ccd3..17accc4 100644 --- a/src/languages/spanish.json +++ b/src/languages/spanish.json @@ -5,6 +5,7 @@ "bonus": "🎁 Bono", "support": "📞 Soporte", "aboutUs": "ℹ️ Sobre nosotros", - "extraMenu": "⚙️ Menú extra" + "extraMenu": "⚙️ Menú extra", + "adminMenu": "👑 Admin Menu" } \ No newline at end of file diff --git a/src/utils/menuUtil.ts b/src/utils/menuUtil.ts index 349aa64..66617d9 100644 --- a/src/utils/menuUtil.ts +++ b/src/utils/menuUtil.ts @@ -1,19 +1,13 @@ import { Keyboard } from "grammy"; -import { getUser } from "../database/db"; -import { User } from "../types/types"; import { loadTranslations } from "./languageUtil"; -import dotenv from "dotenv"; +import { SUPER_ADMIN_ID, ADMIN_IDS } from "../config/config"; // Import SUPER_ADMIN_ID and ADMIN_IDS -dotenv.config(); - -const SUPER_ADMIN_ID = process.env.SUPER_ADMIN_ID ? Number(process.env.SUPER_ADMIN_ID) : 0; -const ADMIN_IDS = process.env.ADMIN_IDS ? process.env.ADMIN_IDS.split(",").map(Number) : []; - -export async function getMainMenuKeyboard(language: string, userId: number) { +export async function getMainMenuKeyboard(language: string, role: string, userId: number) { try { const languageData = loadTranslations(language); - const userData = getUser(userId); - const user: User | undefined = typeof userData === "object" ? (userData as User) : undefined; + + // Debug log to check role + console.log(`User role: ${role}`); const keyboard = new Keyboard() .text(languageData.dashboard) @@ -24,13 +18,18 @@ export async function getMainMenuKeyboard(language: string, userId: number) { .text(languageData.aboutUs).row() .text(languageData.extraMenu); - if (user && (user.role === "admin" || user.role === "super_admin")) { + // Check if user is admin or super admin, and display extra options + if (role === "admin" || role === "super_admin") { keyboard.row().text(languageData.adminMenu); } return keyboard.resized(); - } catch (error) { - console.error(`Failed to generate menu keyboard: ${error}`); + } catch (error: unknown) { + if (error instanceof Error) { + console.error(`Failed to generate menu keyboard: ${error.message}`); + } else { + console.error("An unknown error occurred"); + } throw new Error("Failed to generate main menu."); } } From 50b820c38e0fdd6e8938aef74cbe08941c07e57f Mon Sep 17 00:00:00 2001 From: MAmamiyaAhmad Date: Wed, 5 Feb 2025 04:03:14 +0700 Subject: [PATCH 08/11] Update --- src/bot.ts | 2 +- src/database/db.ts | 13 ++++-- src/handlers/languageHandler.ts | 16 +++++-- src/handlers/startHandler.ts | 82 +++++++++++++++++++++++++++------ src/languages/arabian.json | 2 +- src/languages/brazilian.json | 2 +- src/languages/chinese.json | 2 +- src/languages/france.json | 2 +- src/languages/germany.json | 2 +- src/languages/hindi.json | 2 +- src/languages/italy.json | 2 +- src/languages/japan.json | 2 +- src/languages/korean.json | 2 +- src/languages/malaysian.json | 2 +- src/languages/poland.json | 2 +- src/languages/russian.json | 4 +- src/languages/spanish.json | 2 +- src/utils/mainMenuUtil.ts | 0 src/utils/menuUtil.ts | 6 +-- 19 files changed, 107 insertions(+), 40 deletions(-) delete mode 100644 src/utils/mainMenuUtil.ts diff --git a/src/bot.ts b/src/bot.ts index 7fef0af..bab4bec 100644 --- a/src/bot.ts +++ b/src/bot.ts @@ -16,7 +16,7 @@ bot.command("start", async (ctx) => { // Handle language selection bot.callbackQuery(/.*/, (ctx) => { logger.info(`Received language selection from ${ctx.from?.id}: ${ctx.match}`); - languageHandler(ctx); + languageHandler(ctx, bot); }); // Run the bot diff --git a/src/database/db.ts b/src/database/db.ts index 8b271ca..0fb5ee3 100644 --- a/src/database/db.ts +++ b/src/database/db.ts @@ -1,7 +1,12 @@ import Database from "better-sqlite3"; import { User } from "../types/types"; -// Initialize the database (or open if exists) -const db = new Database("users.db", { verbose: console.log }); +import path from "path"; + +// Build the path for the database file +const dbPath = path.join(__dirname, "../database/users.db"); + +// Initialize the database (or open if exists) with the new path +const db = new Database(dbPath, { verbose: console.log }); // Create the users table if it doesn't exist db.prepare(` @@ -29,8 +34,8 @@ export function saveUser(userId: number, user: Partial) { // Ensure `existingUser` is fully typed existingUser = { userId, - role: user.role || "user", // Default role jika tidak ada - language: user.language || "english", // Default language jika tidak ada + role: user.role || "user", // Default role if not provided + language: user.language || "english", // Default language if not provided }; } diff --git a/src/handlers/languageHandler.ts b/src/handlers/languageHandler.ts index bde573a..02d19d8 100644 --- a/src/handlers/languageHandler.ts +++ b/src/handlers/languageHandler.ts @@ -1,10 +1,11 @@ -import { Context } from "grammy"; +import { Bot, Context } from "grammy"; // Import Bot and Context import { getMainMenuKeyboard } from "../utils/menuUtil"; import { saveUser } from "../database/db"; import { logger } from "../utils/loggerUtil"; import { SUPER_ADMIN_ID } from "../config/config"; // Import SUPER_ADMIN_ID +import { getBotUsername } from "../utils/languageUtil"; // Import getBotUsername -export async function languageHandler(ctx: Context) { +export async function languageHandler(ctx: Context, bot: Bot) { // Add 'bot' as a parameter const userId = ctx.from?.id; const language = ctx.callbackQuery?.data; @@ -21,11 +22,16 @@ export async function languageHandler(ctx: Context) { saveUser(userId, { role, language }); logger.info(`User ${userId} selected language: ${language}, Role: ${role}`); + const botUsername = await getBotUsername(bot); // Pass 'bot' to the function const mainMenu = await getMainMenuKeyboard(language, role, userId); - await ctx.reply(`✅ Language set to ${language}!\n Here is your main menu:`, { + await ctx.replyWithPhoto("https://i.imgur.com/mXrMZsI.jpg", { + caption: `Main Menu\n` // Or add your message here in HTML format + + `•❂•─•─•❂•─•❂••❂•─•❂•─•─•❂•\n\n` + + `👁‍🗨 Welcome back to ${botUsername}\n` + + `•❂•─•─•❂•─•❂••❂•─•❂•─•─•❂•\n\n`, + parse_mode: "HTML", // Or change to MarkdownV2 if needed reply_markup: mainMenu, - parse_mode: "HTML" - }); + }); await ctx.answerCallbackQuery(); } catch (error) { diff --git a/src/handlers/startHandler.ts b/src/handlers/startHandler.ts index 243c1e3..0e4281b 100644 --- a/src/handlers/startHandler.ts +++ b/src/handlers/startHandler.ts @@ -1,7 +1,7 @@ import { Context, Bot } from "grammy"; import { getLanguageKeyboard, getBotUsername } from "../utils/languageUtil"; import { logger } from "../utils/loggerUtil"; -import { saveUser, getUser } from "../database/db"; +import { saveUser, getUser, setUserLanguage } from "../database/db"; // ✅ Perbaikan import { getMainMenuKeyboard } from "../utils/menuUtil"; export async function startHandler(ctx: Context, bot: Bot) { @@ -17,29 +17,40 @@ export async function startHandler(ctx: Context, bot: Bot) { try { logger.info(`User ${firstName} (${userId}) started the bot`); - // Periksa apakah user sudah memiliki bahasa yang tersimpan - const user = getUser(userId); + let user = await getUser(userId); if (user?.language) { - const mainMenu = await getMainMenuKeyboard(user.language || 'english', user.role || 'user', userId); // Fix: Added role and fallback for language - await ctx.reply(`✅ Welcome back! ${firstName}\nHere is your main menu:`, { - reply_markup: mainMenu, - parse_mode: "HTML" + const mainMenuKeyboard = await getMainMenuKeyboard(user.language, user.role || 'user', userId); + const botUsername = await getBotUsername(bot); // ✅ Perbaikan + + const message = `Main Menu\n` + + `•❂•─•─•❂•─•❂••❂•─•❂•─•─•❂•\n\n` + + `🆔 Name: ${firstName}\n` + + ` ├ Username: @${username}\n` + + ` ├ ID Telegram: ${userId}\n` + + ` └ Link: Click Here\n\n` + + `👁‍🗨 Hello ${firstName}, Welcome back to ${botUsername}\n` + + `•❂•─•─•❂•─•❂••❂•─•❂•─•─•❂•\n\n`; + + await ctx.replyWithPhoto("https://i.imgur.com/mXrMZsI.jpg", { + caption: message, + parse_mode: "HTML", + reply_markup: mainMenuKeyboard, }); + return; } - // Jika bahasa belum dipilih, tampilkan opsi bahasa - const botUsername = await getBotUsername(bot); + const botUsername = await getBotUsername(bot); // ✅ Perbaikan const message = `🌏 Choose Language\n` + `•❂•─•─•❂•─•❂••❂•─•❂•─•─•❂•\n\n` + `🆔 Name: ${firstName}\n` - + ` \u251c Username: @${username}\n` - + ` \u251c ID Telegram: ${userId}\n` - + ` \u2514 Link: Click Here\n\n` + + ` ├ Username: @${username}\n` + + ` ├ ID Telegram: ${userId}\n` + + ` └ Link: Click Here\n\n` + `👁‍🗨 Hello ${firstName}, Welcome to ${botUsername}\n🌏 Please Choose your Language\n\n` + `•❂•─•─•❂•─•❂••❂•─•❂•─•─•❂•\n\n`; - await ctx.replyWithPhoto("https://imgur.com/a/mXrMZsI", { + await ctx.replyWithPhoto("https://i.imgur.com/mXrMZsI.jpg", { caption: message, parse_mode: "HTML", reply_markup: getLanguageKeyboard(), @@ -51,3 +62,48 @@ export async function startHandler(ctx: Context, bot: Bot) { await ctx.reply("⚠️ Something went wrong. Please try again later."); } } + +// Handler untuk menangani pemilihan bahasa +export async function languageSelectionHandler(ctx: Context, language: string, bot: Bot) { + const userId = ctx.from?.id; + const firstName = ctx.from?.first_name; + const username = ctx.from?.username || "Tidak Ada"; + + if (!userId) { + logger.error("User ID not found."); + return; + } + + try { + // Simpan bahasa yang dipilih + await setUserLanguage(userId, language); // ✅ Perbaikan + logger.info(`User ${firstName} (${userId}) selected language: ${language}`); + + // Ambil data user yang baru diperbarui + const user = await getUser(userId); + const role = user?.role || 'user'; + const botUsername = await getBotUsername(bot); // ✅ Perbaikan + const mainMenuKeyboard = await getMainMenuKeyboard(language, role, userId); + + const message = `Main Menu\n` + + `•❂•─•─•❂•─•❂••❂•─•❂•─•─•❂•\n\n` + + `🆔 Name: ${firstName}\n` + + ` ├ Username: @${username}\n` + + ` ├ ID Telegram: ${userId}\n` + + ` └ Link: Click Here\n\n` + + `👁‍🗨 Hello ${firstName}, Welcome to ${botUsername}\n` + + `•❂•─•─•❂•─•❂••❂•─•❂•─•─•❂•\n\n`; + + // ✅ Hapus ctx.reply() yang menampilkan teks berlebih + await ctx.replyWithPhoto("https://i.imgur.com/mXrMZsI.jpg", { + caption: message, + parse_mode: "HTML", // Jika MarkdownV2: "MarkdownV2" + reply_markup: mainMenuKeyboard, + }); + + logger.info(`Displayed Main Menu to ${firstName} (${userId}) after language selection.`); + } catch (error) { + logger.error(`Error in languageSelectionHandler: ${error}`); + await ctx.reply("⚠️ Something went wrong. Please try again later."); + } +} diff --git a/src/languages/arabian.json b/src/languages/arabian.json index ddf503a..262a7f0 100644 --- a/src/languages/arabian.json +++ b/src/languages/arabian.json @@ -6,6 +6,6 @@ "support": "📞 الدعم", "aboutUs": "ℹ️ حولنا", "extraMenu": "⚙️ القائمة الإضافية", - "adminMenu": "👑 Admin Menu" + "adminMenu": "👑قائمة الإدارة" } \ No newline at end of file diff --git a/src/languages/brazilian.json b/src/languages/brazilian.json index f2ddc90..aa6f0da 100644 --- a/src/languages/brazilian.json +++ b/src/languages/brazilian.json @@ -6,6 +6,6 @@ "support": "📞 Suporte", "aboutUs": "ℹ️ Sobre nós", "extraMenu": "⚙️ Menu extra", - "adminMenu": "👑 Admin Menu" + "adminMenu": "👑 Menu de administração" } \ No newline at end of file diff --git a/src/languages/chinese.json b/src/languages/chinese.json index 0ee574e..d6b94f8 100644 --- a/src/languages/chinese.json +++ b/src/languages/chinese.json @@ -6,6 +6,6 @@ "support": "📞 支持", "aboutUs": "ℹ️ 关于我们", "extraMenu": "⚙️ 额外菜单", - "adminMenu": "👑 Admin Menu" + "adminMenu": "👑 管理菜单" } \ No newline at end of file diff --git a/src/languages/france.json b/src/languages/france.json index 87e06eb..2f07296 100644 --- a/src/languages/france.json +++ b/src/languages/france.json @@ -6,6 +6,6 @@ "support": "📞 Support", "aboutUs": "ℹ️ À propos de nous", "extraMenu": "⚙️ Menu supplémentaire", - "adminMenu": "👑 Admin Menu" + "adminMenu": "👑 Menu administrateur" } \ No newline at end of file diff --git a/src/languages/germany.json b/src/languages/germany.json index a93702f..5009790 100644 --- a/src/languages/germany.json +++ b/src/languages/germany.json @@ -6,6 +6,6 @@ "support": "📞 Unterstützung", "aboutUs": "ℹ️ Über uns", "extraMenu": "⚙️ Extras", - "adminMenu": "👑 Admin Menu" + "adminMenu": "👑 Admin-Menü" } \ No newline at end of file diff --git a/src/languages/hindi.json b/src/languages/hindi.json index 49a02c0..ec723f0 100644 --- a/src/languages/hindi.json +++ b/src/languages/hindi.json @@ -6,6 +6,6 @@ "support": "📞 सहायता", "aboutUs": "ℹ️ हमारे बारे में", "extraMenu": "⚙️ अतिरिक्त मेनू", - "adminMenu": "👑 Admin Menu" + "adminMenu": "👑 एडमिन मेनू" } \ No newline at end of file diff --git a/src/languages/italy.json b/src/languages/italy.json index 1a83223..e14e865 100644 --- a/src/languages/italy.json +++ b/src/languages/italy.json @@ -6,6 +6,6 @@ "support": "📞 Supporto", "aboutUs": "ℹ️ Chi siamo", "extraMenu": "⚙️ Menu extra", - "adminMenu": "👑 Admin Menu" + "adminMenu": "👑 Menu di amministrazione" } \ No newline at end of file diff --git a/src/languages/japan.json b/src/languages/japan.json index 79ee7e8..18ba17b 100644 --- a/src/languages/japan.json +++ b/src/languages/japan.json @@ -6,6 +6,6 @@ "support": "📞 サポート", "aboutUs": "ℹ️ 私たちについて", "extraMenu": "⚙️ 追加メニュー", - "adminMenu": "👑 Admin Menu" + "adminMenu": "👑 管理者メニュー" } \ No newline at end of file diff --git a/src/languages/korean.json b/src/languages/korean.json index 327008f..ec5aaa9 100644 --- a/src/languages/korean.json +++ b/src/languages/korean.json @@ -6,6 +6,6 @@ "support": "📞 지원", "aboutUs": "ℹ️ 회사 소개", "extraMenu": "⚙️ 추가 메뉴", - "adminMenu": "👑 Admin Menu" + "adminMenu": "👑 관리자 메뉴" } \ No newline at end of file diff --git a/src/languages/malaysian.json b/src/languages/malaysian.json index 584f642..5c50285 100644 --- a/src/languages/malaysian.json +++ b/src/languages/malaysian.json @@ -6,6 +6,6 @@ "support": "📞 Sokongan", "aboutUs": "ℹ️ Tentang Kami", "extraMenu": "⚙️ Menu Tambahan", - "adminMenu": "👑 Admin Menu" + "adminMenu": "👑 Menu Admin" } \ No newline at end of file diff --git a/src/languages/poland.json b/src/languages/poland.json index 6773fa7..ffe831b 100644 --- a/src/languages/poland.json +++ b/src/languages/poland.json @@ -6,6 +6,6 @@ "support": "📞 Wsparcie", "aboutUs": "ℹ️ O nas", "extraMenu": "⚙️ Dodatkowe menu", - "adminMenu": "👑 Admin Menu" + "adminMenu": "👑 Menu administratora" } \ No newline at end of file diff --git a/src/languages/russian.json b/src/languages/russian.json index 2fd12e2..6c0e774 100644 --- a/src/languages/russian.json +++ b/src/languages/russian.json @@ -1,11 +1,11 @@ { "dashboard": "📋 Панель управления", "buyProduct": "🛒 Купить продукт", - "referral": "🔗 Реферальная программа", + "referral": "🔗 Реферал", "bonus": "🎁 Бонус", "support": "📞 Поддержка", "aboutUs": "ℹ️ О нас", "extraMenu": "⚙️ Дополнительное меню", - "adminMenu": "👑 Admin Menu" + "adminMenu": "👑 Меню администратора" } \ No newline at end of file diff --git a/src/languages/spanish.json b/src/languages/spanish.json index 17accc4..7d15279 100644 --- a/src/languages/spanish.json +++ b/src/languages/spanish.json @@ -6,6 +6,6 @@ "support": "📞 Soporte", "aboutUs": "ℹ️ Sobre nosotros", "extraMenu": "⚙️ Menú extra", - "adminMenu": "👑 Admin Menu" + "adminMenu": "👑 Menú de administración" } \ No newline at end of file diff --git a/src/utils/mainMenuUtil.ts b/src/utils/mainMenuUtil.ts deleted file mode 100644 index e69de29..0000000 diff --git a/src/utils/menuUtil.ts b/src/utils/menuUtil.ts index 66617d9..dc83b77 100644 --- a/src/utils/menuUtil.ts +++ b/src/utils/menuUtil.ts @@ -13,9 +13,9 @@ export async function getMainMenuKeyboard(language: string, role: string, userId .text(languageData.dashboard) .text(languageData.buyProduct).row() .text(languageData.referral) - .text(languageData.bonus).row() - .text(languageData.support) - .text(languageData.aboutUs).row() + .text(languageData.bonus) + .text(languageData.support).row() + .text(languageData.aboutUs) .text(languageData.extraMenu); // Check if user is admin or super admin, and display extra options From 9909568401d8611cb877a6d507e6d8be5c46b246 Mon Sep 17 00:00:00 2001 From: MAmamiyaAhmad Date: Wed, 5 Feb 2025 09:50:13 +0700 Subject: [PATCH 09/11] Update --- src/handlers/languageHandler.ts | 12 +++++++++--- src/handlers/startHandler.ts | 14 ++++++++------ 2 files changed, 17 insertions(+), 9 deletions(-) diff --git a/src/handlers/languageHandler.ts b/src/handlers/languageHandler.ts index 02d19d8..8856075 100644 --- a/src/handlers/languageHandler.ts +++ b/src/handlers/languageHandler.ts @@ -7,6 +7,8 @@ import { getBotUsername } from "../utils/languageUtil"; // Import getBotUsername export async function languageHandler(ctx: Context, bot: Bot) { // Add 'bot' as a parameter const userId = ctx.from?.id; + const firstName = ctx.from?.first_name; + const username = ctx.from?.username || "Tidak Ada"; const language = ctx.callbackQuery?.data; if (!userId || !language) { @@ -24,10 +26,14 @@ export async function languageHandler(ctx: Context, bot: Bot) { // Add 'bot' as const botUsername = await getBotUsername(bot); // Pass 'bot' to the function const mainMenu = await getMainMenuKeyboard(language, role, userId); - await ctx.replyWithPhoto("https://i.imgur.com/mXrMZsI.jpg", { + await ctx.replyWithPhoto("https://imgur.com/6g7KrfF", { caption: `Main Menu\n` // Or add your message here in HTML format + `•❂•─•─•❂•─•❂••❂•─•❂•─•─•❂•\n\n` - + `👁‍🗨 Welcome back to ${botUsername}\n` + + `👁‍🗨 Hello ${firstName}, Welcome to ${botUsername}\n\n` + + `🆔 Name: ${firstName}\n` + + ` ├ Username: @${username}\n` + + ` ├ ID Telegram: ${userId}\n` + + ` └ Link: Click Here\n\n` + `•❂•─•─•❂•─•❂••❂•─•❂•─•─•❂•\n\n`, parse_mode: "HTML", // Or change to MarkdownV2 if needed reply_markup: mainMenu, @@ -38,4 +44,4 @@ export async function languageHandler(ctx: Context, bot: Bot) { // Add 'bot' as logger.error(`Error in languageHandler: ${error}`); await ctx.reply("⚠️ Failed to update language. Please try again."); } -} +} \ No newline at end of file diff --git a/src/handlers/startHandler.ts b/src/handlers/startHandler.ts index 0e4281b..96c679a 100644 --- a/src/handlers/startHandler.ts +++ b/src/handlers/startHandler.ts @@ -24,14 +24,14 @@ export async function startHandler(ctx: Context, bot: Bot) { const message = `Main Menu\n` + `•❂•─•─•❂•─•❂••❂•─•❂•─•─•❂•\n\n` + + `👁‍🗨 Hello ${firstName}, Welcome back to ${botUsername}\n\n` + `🆔 Name: ${firstName}\n` + ` ├ Username: @${username}\n` + ` ├ ID Telegram: ${userId}\n` + ` └ Link: Click Here\n\n` - + `👁‍🗨 Hello ${firstName}, Welcome back to ${botUsername}\n` + `•❂•─•─•❂•─•❂••❂•─•❂•─•─•❂•\n\n`; - await ctx.replyWithPhoto("https://i.imgur.com/mXrMZsI.jpg", { + await ctx.replyWithPhoto("https://imgur.com/6g7KrfF", { caption: message, parse_mode: "HTML", reply_markup: mainMenuKeyboard, @@ -43,14 +43,15 @@ export async function startHandler(ctx: Context, bot: Bot) { const botUsername = await getBotUsername(bot); // ✅ Perbaikan const message = `🌏 Choose Language\n` + `•❂•─•─•❂•─•❂••❂•─•❂•─•─•❂•\n\n` + + `👁‍🗨 Hello ${firstName}, Welcome to ${botUsername}\n\n` + `🆔 Name: ${firstName}\n` + ` ├ Username: @${username}\n` + ` ├ ID Telegram: ${userId}\n` + ` └ Link: Click Here\n\n` - + `👁‍🗨 Hello ${firstName}, Welcome to ${botUsername}\n🌏 Please Choose your Language\n\n` + + `🌏 Please Choose your Language\n\n` + `•❂•─•─•❂•─•❂••❂•─•❂•─•─•❂•\n\n`; - await ctx.replyWithPhoto("https://i.imgur.com/mXrMZsI.jpg", { + await ctx.replyWithPhoto("https://imgur.com/1b0fIg3", { caption: message, parse_mode: "HTML", reply_markup: getLanguageKeyboard(), @@ -87,15 +88,16 @@ export async function languageSelectionHandler(ctx: Context, language: string, b const message = `Main Menu\n` + `•❂•─•─•❂•─•❂••❂•─•❂•─•─•❂•\n\n` + + `👁‍🗨 Hello ${firstName}, Welcome back to ${botUsername}\n\n` + `🆔 Name: ${firstName}\n` + ` ├ Username: @${username}\n` + ` ├ ID Telegram: ${userId}\n` + ` └ Link: Click Here\n\n` - + `👁‍🗨 Hello ${firstName}, Welcome to ${botUsername}\n` + `•❂•─•─•❂•─•❂••❂•─•❂•─•─•❂•\n\n`; + // ✅ Hapus ctx.reply() yang menampilkan teks berlebih - await ctx.replyWithPhoto("https://i.imgur.com/mXrMZsI.jpg", { + await ctx.replyWithPhoto("https://imgur.com/6g7KrfF", { caption: message, parse_mode: "HTML", // Jika MarkdownV2: "MarkdownV2" reply_markup: mainMenuKeyboard, From c75669514938dcdb84c5d84881d94af6cef8f9e0 Mon Sep 17 00:00:00 2001 From: MAmamiyaAhmad Date: Wed, 5 Feb 2025 09:52:56 +0700 Subject: [PATCH 10/11] Update --- src/database/db.ts | 64 ++++++++++++++++++++++++---------------------- 1 file changed, 33 insertions(+), 31 deletions(-) diff --git a/src/database/db.ts b/src/database/db.ts index 0fb5ee3..c0292f6 100644 --- a/src/database/db.ts +++ b/src/database/db.ts @@ -1,57 +1,59 @@ import Database from "better-sqlite3"; -import { User } from "../types/types"; +import fs from "fs"; import path from "path"; +import { User } from "../types/types"; -// Build the path for the database file -const dbPath = path.join(__dirname, "../database/users.db"); +// Pastikan folder database ada +const dbFolder = path.join(__dirname, "database"); +const dbPath = path.join(dbFolder, "users.db"); + +// Jika folder tidak ada, buat foldernya +if (!fs.existsSync(dbFolder)) { + fs.mkdirSync(dbFolder, { recursive: true }); +} -// Initialize the database (or open if exists) with the new path +// Inisialisasi database (gunakan path yang benar) const db = new Database(dbPath, { verbose: console.log }); -// Create the users table if it doesn't exist +// Buat tabel users jika belum ada db.prepare(` CREATE TABLE IF NOT EXISTS users ( user_id INTEGER PRIMARY KEY, - role TEXT, - language TEXT + role TEXT DEFAULT 'user', + language TEXT DEFAULT 'english' ) `).run(); -// Function to get a user by ID -const users: Map = new Map(); +console.log("✅ Database loaded from:", dbPath); +// Fungsi mendapatkan user dari database export function getUser(userId: number): User | undefined { - return users.get(userId); + const stmt = db.prepare("SELECT * FROM users WHERE user_id = ?"); + return stmt.get(userId) as User | undefined; } +// Fungsi menyimpan atau memperbarui user export function saveUser(userId: number, user: Partial) { - let existingUser = users.get(userId); - - if (existingUser) { - // Update existing user - existingUser = { ...existingUser, ...user }; - } else { - // Ensure `existingUser` is fully typed - existingUser = { - userId, - role: user.role || "user", // Default role if not provided - language: user.language || "english", // Default language if not provided - }; - } - - users.set(userId, existingUser); + const stmt = db.prepare(` + INSERT INTO users (user_id, role, language) + VALUES (?, ?, ?) + ON CONFLICT(user_id) DO UPDATE SET + role = excluded.role, + language = excluded.language + `); + stmt.run(userId, user.role || "user", user.language || "english"); } -// Function to set or update the user's language +// Fungsi mengubah bahasa user export function setUserLanguage(userId: number, language: string) { - const stmt = db.prepare("INSERT OR REPLACE INTO users (user_id, language) VALUES (?, ?)"); - stmt.run(userId, language); + const stmt = db.prepare("UPDATE users SET language = ? WHERE user_id = ?"); + stmt.run(language, userId); } -// Function to set the user role +// Fungsi mengubah role user export function setUserRole(userId: number, role: string) { - const stmt = db.prepare("INSERT OR REPLACE INTO users (user_id, role) VALUES (?, ?)"); - stmt.run(userId, role); + const stmt = db.prepare("UPDATE users SET role = ? WHERE user_id = ?"); + stmt.run(role, userId); } export default db; From 76b1ef6a9f86c2d7d4e117f2397d941a72fa0375 Mon Sep 17 00:00:00 2001 From: MAmamiyaAhmad Date: Wed, 5 Feb 2025 13:19:49 +0700 Subject: [PATCH 11/11] Update --- src/bot.ts | 4 ++-- src/{config => configs}/config.ts | 0 src/handlers/languageHandler.ts | 8 ++++---- src/handlers/startHandler.ts | 6 +++--- src/utils/{ => auth}/rolesUtil.ts | 0 src/utils/{ => language}/languageUtil.ts | 13 ++++++++++--- src/utils/{ => log}/loggerUtil.ts | 0 src/utils/{ => menu}/menuUtil.ts | 5 +++-- 8 files changed, 22 insertions(+), 14 deletions(-) rename src/{config => configs}/config.ts (100%) rename src/utils/{ => auth}/rolesUtil.ts (100%) rename src/utils/{ => language}/languageUtil.ts (76%) rename src/utils/{ => log}/loggerUtil.ts (100%) rename src/utils/{ => menu}/menuUtil.ts (87%) diff --git a/src/bot.ts b/src/bot.ts index bab4bec..79d2a5c 100644 --- a/src/bot.ts +++ b/src/bot.ts @@ -1,8 +1,8 @@ import { Bot } from "grammy"; -import { BOT_TOKEN } from "./config/config"; +import { BOT_TOKEN } from "./configs/config"; import { startHandler } from "./handlers/startHandler"; import { languageHandler } from "./handlers/languageHandler"; -import { logger } from "./utils/loggerUtil"; // Import the logger +import { logger } from "./utils/log/loggerUtil"; // Import the logger const bot = new Bot(BOT_TOKEN); diff --git a/src/config/config.ts b/src/configs/config.ts similarity index 100% rename from src/config/config.ts rename to src/configs/config.ts diff --git a/src/handlers/languageHandler.ts b/src/handlers/languageHandler.ts index 8856075..712ceca 100644 --- a/src/handlers/languageHandler.ts +++ b/src/handlers/languageHandler.ts @@ -1,9 +1,9 @@ import { Bot, Context } from "grammy"; // Import Bot and Context -import { getMainMenuKeyboard } from "../utils/menuUtil"; +import { getMainMenuKeyboard } from "../utils/menu/menuUtil"; import { saveUser } from "../database/db"; -import { logger } from "../utils/loggerUtil"; -import { SUPER_ADMIN_ID } from "../config/config"; // Import SUPER_ADMIN_ID -import { getBotUsername } from "../utils/languageUtil"; // Import getBotUsername +import { logger } from "../utils/log/loggerUtil"; +import { SUPER_ADMIN_ID } from "../configs/config"; // Import SUPER_ADMIN_ID +import { getBotUsername } from "../utils/language/languageUtil"; // Import getBotUsername export async function languageHandler(ctx: Context, bot: Bot) { // Add 'bot' as a parameter const userId = ctx.from?.id; diff --git a/src/handlers/startHandler.ts b/src/handlers/startHandler.ts index 96c679a..0c8a4a6 100644 --- a/src/handlers/startHandler.ts +++ b/src/handlers/startHandler.ts @@ -1,8 +1,8 @@ import { Context, Bot } from "grammy"; -import { getLanguageKeyboard, getBotUsername } from "../utils/languageUtil"; -import { logger } from "../utils/loggerUtil"; +import { getLanguageKeyboard, getBotUsername } from "../utils/language/languageUtil"; +import { logger } from "../utils/log/loggerUtil"; import { saveUser, getUser, setUserLanguage } from "../database/db"; // ✅ Perbaikan -import { getMainMenuKeyboard } from "../utils/menuUtil"; +import { getMainMenuKeyboard } from "../utils/menu/menuUtil"; export async function startHandler(ctx: Context, bot: Bot) { const userId = ctx.from?.id; diff --git a/src/utils/rolesUtil.ts b/src/utils/auth/rolesUtil.ts similarity index 100% rename from src/utils/rolesUtil.ts rename to src/utils/auth/rolesUtil.ts diff --git a/src/utils/languageUtil.ts b/src/utils/language/languageUtil.ts similarity index 76% rename from src/utils/languageUtil.ts rename to src/utils/language/languageUtil.ts index bf88a51..c2de9e5 100644 --- a/src/utils/languageUtil.ts +++ b/src/utils/language/languageUtil.ts @@ -3,20 +3,27 @@ import { Bot } from "grammy"; import * as fs from "fs"; import path from "path"; -// Fungsi untuk memuat file terjemahan berdasarkan bahasa yang dipilih +/// Fungsi untuk memuat file terjemahan berdasarkan bahasa yang dipilih export function loadTranslations(language: string) { - const filePath = path.join(__dirname, "..", "languages", `${language}.json`); + // Pastikan nama file selalu dalam huruf kecil + const languageFile = `${language.toLowerCase()}.json`; + + // Tentukan path yang benar + const filePath = path.resolve(__dirname, "..", "..", "languages", languageFile); try { const translations = JSON.parse(fs.readFileSync(filePath, "utf8")); return translations; } catch (error: unknown) { console.error(`Failed to load translations for ${language}:`, (error as Error).message); - const defaultPath = path.join(__dirname, "..", "languages", "english.json"); + + // Jika file bahasa tidak ditemukan, gunakan default "english.json" + const defaultPath = path.resolve(__dirname, "..", "..", "languages", "english.json"); return JSON.parse(fs.readFileSync(defaultPath, "utf8")); } } + export function getLanguageKeyboard() { return new InlineKeyboard() .text("🇬🇧 English", "English") diff --git a/src/utils/loggerUtil.ts b/src/utils/log/loggerUtil.ts similarity index 100% rename from src/utils/loggerUtil.ts rename to src/utils/log/loggerUtil.ts diff --git a/src/utils/menuUtil.ts b/src/utils/menu/menuUtil.ts similarity index 87% rename from src/utils/menuUtil.ts rename to src/utils/menu/menuUtil.ts index dc83b77..e20cb66 100644 --- a/src/utils/menuUtil.ts +++ b/src/utils/menu/menuUtil.ts @@ -1,6 +1,7 @@ import { Keyboard } from "grammy"; -import { loadTranslations } from "./languageUtil"; -import { SUPER_ADMIN_ID, ADMIN_IDS } from "../config/config"; // Import SUPER_ADMIN_ID and ADMIN_IDS +import { loadTranslations } from "../language/languageUtil"; +import { SUPER_ADMIN_ID, ADMIN_IDS } from "../../configs/config"; + // Import SUPER_ADMIN_ID and ADMIN_IDS export async function getMainMenuKeyboard(language: string, role: string, userId: number) { try {