hypixel-auction-notifier/index.ts

194 lines
6.1 KiB
TypeScript
Raw Normal View History

2024-12-27 21:17:59 +01:00
import axios from 'axios';
import { WebhookClient, EmbedBuilder, Embed } from 'discord.js';
import { Worker } from 'worker_threads';
import { asyncInterval, addNotation } from './src/helperFunctions';
import { string } from 'prismarine-nbt';
2024-12-28 13:41:48 +01:00
import { SqlSystem } from './src/sqlFunctions';
2024-12-28 14:05:03 +01:00
import { setupErrorHandlers } from './src/errorHandler';
import { ItemCompactData } from './src/Types';
2024-12-27 21:17:59 +01:00
2024-12-28 14:05:03 +01:00
setupErrorHandlers();
2024-12-27 22:58:39 +01:00
2024-12-28 14:05:03 +01:00
import { loadConfig } from './src/configLoader';
const config = loadConfig();
2024-12-27 22:58:39 +01:00
let worker_count = config.data.worker_count ?? 1;
let lastUpdated = 0;
let doneWorkers = 0;
2024-12-27 22:58:39 +01:00
let startingTime: number;
let maxPrice = 0;
2024-12-28 14:05:03 +01:00
let itemDatas: Record<string, ItemCompactData> = {};
2024-12-27 21:17:59 +01:00
const workers: Worker[] = [];
const webhookRegex = /https:\/\/discord.com\/api\/webhooks\/(.+)\/(.+)/;
2022-08-06 19:02:15 +02:00
const bazaarPrice = {
RECOMBOBULATOR_3000: 0,
HOT_POTATO_BOOK: 0,
FUMING_POTATO_BOOK: 0,
};
2022-08-06 19:02:15 +02:00
async function initialize() {
2024-12-28 13:41:48 +01:00
await SqlSystem.InitTable();
2024-12-28 11:25:31 +01:00
const matches = process.env.WEBHOOK_URL.match(webhookRegex);
if (!matches) return console.log(`[Main thread] Couldn't parse Webhook URL`);
const webhook = new WebhookClient({ id: matches[1], token: matches[2] });
await getBzData();
await getMoulberry();
await getLBINs();
2024-12-27 22:58:39 +01:00
for (let j = 0; j < worker_count; j++) {
workers[j] = new Worker('/app/dist/AuctionHandler.worker.js', {
workerData: {
itemDatas: itemDatas,
bazaarData: bazaarPrice,
workerNumber: j,
maxPrice: maxPrice,
},
});
workers[j].on('message', async (result) => {
if (result.itemData !== undefined) {
2024-12-27 22:58:39 +01:00
let averagePrice: number | null = itemDatas[result.itemData.id]?.cleanPrice || null;
if (
result.auctionData.lbin - result.auctionData.price >=
2024-12-27 21:17:59 +01:00
config.data.minSnipeProfit &&
2024-12-27 22:58:39 +01:00
(averagePrice || averagePrice! - result.auctionData.price >= config.data.minAvgProfit)
) {
let mustBuyMessage = '';
const embed = new EmbedBuilder()
.setTitle(`**${result.itemData.name.replace(/§./g, '')}**`)
.setColor('#2e3137')
.setThumbnail(`https://sky.shiiyu.moe/item/${result.itemData.id}`)
.setDescription(
`${mustBuyMessage}\nAuction:
\`\`\`/viewauction ${result.auctionData.auctionID}\`\`\`
\nProfit: \`${addNotation(
'oneLetters',
result.auctionData.profit
)} (${result.auctionData.percentProfit}%)\`
\nCost: \`${addNotation('oneLetters', result.auctionData.price)}\`
\nLBIN: \`${addNotation('oneLetters', result.auctionData.lbin)}\`
\nSales/Day: \`${addNotation(
'oneLetters',
result.auctionData.sales
)}\`
\nType: \`${result.auctionData.ahType}\`
2024-12-27 22:58:39 +01:00
\nAverage Price: \`${averagePrice ? addNotation('oneLetters', averagePrice) : 'N/A'}\``
);
await webhook.send({
2024-12-28 11:25:31 +01:00
username: process.env.WEBHOOK_NAME,
avatarURL: process.env.WEBHOOK_PROFILE_PICTURE,
embeds: [embed],
});
}
} else if (result === 'finished') {
doneWorkers++;
2024-12-27 22:58:39 +01:00
if (doneWorkers === worker_count) {
doneWorkers = 0;
console.log(
`Completed in ${(Date.now() - startingTime) / 1000} seconds`
);
startingTime = 0;
workers[0].emit('done');
}
}
});
}
asyncInterval(
async () => {
await getLBINs();
workers.forEach((worker) => {
worker.postMessage({ type: 'moulberry', data: itemDatas });
});
},
'lbin',
60000
);
asyncInterval(
async () => {
await getMoulberry();
workers.forEach((worker) => {
worker.postMessage({ type: 'moulberry', data: itemDatas });
});
},
'avg',
60e5
);
asyncInterval(
async () => {
return new Promise(async (resolve) => {
const ahFirstPage = await axios.get(
'https://api.hypixel.net/skyblock/auctions?page=0'
);
const totalPages = ahFirstPage.data.totalPages;
if (ahFirstPage.data.lastUpdated === lastUpdated) {
resolve();
} else {
lastUpdated = ahFirstPage.data.lastUpdated;
startingTime = Date.now();
console.log('Getting auctions..');
workers.forEach((worker) => {
worker.postMessage({ type: 'pageCount', data: totalPages });
});
workers[0].once('done', () => {
resolve();
});
}
});
},
'check',
0
);
2022-08-06 19:02:15 +02:00
}
async function getLBINs(): Promise<void> {
const lbins = await axios.get('https://moulberry.codes/lowestbin.json');
const lbinData = lbins.data;
for (const item of Object.keys(lbinData)) {
2024-12-28 14:05:03 +01:00
if (!itemDatas[item]) itemDatas[item] = new ItemCompactData();
itemDatas[item].lbin = lbinData[item];
}
2022-08-06 19:02:15 +02:00
}
async function getMoulberry(): Promise<void> {
const moulberryAvgs = await axios.get(
'https://moulberry.codes/auction_averages/3day.json'
);
const avgData = moulberryAvgs.data;
const cleanPriceAvgs = await axios.get(
'https://moulberry.codes/auction_averages_lbin/1day.json'
);
const cleanPriceData = cleanPriceAvgs.data;
for (const item of Object.keys(avgData)) {
2024-12-28 14:05:03 +01:00
if (!itemDatas[item]) itemDatas[item] = new ItemCompactData();
const itemInfo = avgData[item];
itemDatas[item].sales = itemInfo.sales !== undefined ? itemInfo.sales : 0;
itemDatas[item].cleanPrice =
cleanPriceData[item] !== undefined
? Math.round(cleanPriceData[item])
: itemInfo.clean_price !== undefined
2024-12-27 21:17:59 +01:00
? itemInfo.clean_price
: itemInfo.price;
}
2022-08-06 19:02:15 +02:00
}
async function getBzData(): Promise<void> {
const bzData = await axios.get('https://api.hypixel.net/skyblock/bazaar');
bazaarPrice['RECOMBOBULATOR_3000'] =
bzData.data.products.RECOMBOBULATOR_3000.quick_status.buyPrice;
bazaarPrice['HOT_POTATO_BOOK'] =
bzData.data.products.HOT_POTATO_BOOK.quick_status.buyPrice;
bazaarPrice['FUMING_POTATO_BOOK'] =
bzData.data.products.FUMING_POTATO_BOOK.quick_status.buyPrice;
2022-08-06 19:02:15 +02:00
}
initialize();