mirror of
https://github.com/maxswa/osrs-json-hiscores.git
synced 2025-10-15 10:19:04 +00:00
249 lines
6.5 KiB
TypeScript
249 lines
6.5 KiB
TypeScript
import axios from 'axios';
|
|
import * as cheerio from 'cheerio';
|
|
import {
|
|
Player,
|
|
Mode,
|
|
Activity,
|
|
Skill,
|
|
Stats,
|
|
Skills,
|
|
BH as BHStats,
|
|
Clues,
|
|
Gamemode,
|
|
Category,
|
|
} from './types';
|
|
import {
|
|
getStatsURL,
|
|
SKILLS,
|
|
BH,
|
|
CLUES,
|
|
MODES,
|
|
getPlayerTableURL,
|
|
getHiscoresPageURL,
|
|
GAMEMODES,
|
|
OTHER,
|
|
} from './utils';
|
|
|
|
export async function getStats(
|
|
rsn: string,
|
|
mode: Mode = 'full'
|
|
): Promise<Player> {
|
|
if (typeof rsn !== 'string') {
|
|
throw Error('RSN must be a string');
|
|
} else if (!/^[a-zA-Z0-9 _]+$/.test(rsn)) {
|
|
throw Error('RSN contains invalid character');
|
|
} else if (rsn.length > 12 || rsn.length < 1) {
|
|
throw Error('RSN must be between 1 and 12 characters');
|
|
} else if (!MODES.includes(mode)) {
|
|
throw Error('Invalid game mode');
|
|
}
|
|
if (mode === 'full') {
|
|
const mainRes = await axios(getStatsURL('main', rsn));
|
|
if (mainRes.status === 200) {
|
|
const otherResponses = await Promise.all([
|
|
axios(getStatsURL('iron', rsn)).catch(err => err),
|
|
axios(getStatsURL('hc', rsn)).catch(err => err),
|
|
axios(getStatsURL('ult', rsn)).catch(err => err),
|
|
getRSNFormat(rsn),
|
|
]);
|
|
|
|
const [ironRes, hcRes, ultRes, formattedName] = otherResponses;
|
|
|
|
const player: Player = {
|
|
rsn: formattedName,
|
|
mode: 'main',
|
|
dead: false,
|
|
deulted: false,
|
|
deironed: false,
|
|
};
|
|
player.main = parseStats(mainRes.data);
|
|
|
|
if (ironRes.status === 200) {
|
|
player.iron = parseStats(ironRes.data);
|
|
if (hcRes.status === 200) {
|
|
player.mode = 'hc';
|
|
player.hc = parseStats(hcRes.data);
|
|
if (player.iron.skills.overall.xp !== player.hc.skills.overall.xp) {
|
|
player.dead = true;
|
|
player.mode = 'iron';
|
|
}
|
|
if (player.main.skills.overall.xp !== player.iron.skills.overall.xp) {
|
|
player.deironed = true;
|
|
player.mode = 'main';
|
|
}
|
|
} else if (ultRes.status === 200) {
|
|
player.mode = 'ult';
|
|
player.ult = parseStats(ultRes.data);
|
|
if (player.iron.skills.overall.xp !== player.ult.skills.overall.xp) {
|
|
player.deulted = true;
|
|
player.mode = 'iron';
|
|
}
|
|
if (player.main.skills.overall.xp !== player.iron.skills.overall.xp) {
|
|
player.deironed = true;
|
|
player.mode = 'main';
|
|
}
|
|
} else {
|
|
player.mode = 'iron';
|
|
if (player.main.skills.overall.xp !== player.iron.skills.overall.xp) {
|
|
player.deironed = true;
|
|
player.mode = 'main';
|
|
}
|
|
}
|
|
}
|
|
|
|
return player;
|
|
}
|
|
throw Error('Player not found');
|
|
} else {
|
|
const response = await axios(getStatsURL(mode, rsn));
|
|
if (response.status !== 200) {
|
|
throw Error('Player not found');
|
|
}
|
|
const player: Player = {
|
|
rsn,
|
|
mode,
|
|
dead: false,
|
|
deulted: false,
|
|
deironed: false,
|
|
[mode]: parseStats(response.data),
|
|
};
|
|
return player;
|
|
}
|
|
}
|
|
|
|
// export async function getHiscores(
|
|
// mode: Gamemode,
|
|
// category: Category,
|
|
// page: number
|
|
// ) {
|
|
// if (GAMEMODES.includes(mode) || mode.toLowerCase() === 'full') {
|
|
// throw Error('Invalid game mode');
|
|
// } else if (!Number.isInteger(page) || page < 1) {
|
|
// throw Error('Page must be an integer greater than 0');
|
|
// } else if ([...SKILLS, ...OTHER].includes(category)) {
|
|
// throw Error('Invalid category');
|
|
// }
|
|
// const url = getHiscoresPageURL(mode, category, page);
|
|
|
|
// const players = [];
|
|
// const response = await axios(url);
|
|
// const $ = cheerio.load(response.data);
|
|
// const playersHTML = $('.personal-hiscores__row').toArray();
|
|
|
|
// for (const player of playersHTML) {
|
|
// const attributes = player.children.filter(node => node.name === 'td');
|
|
|
|
// let playerInfo = {
|
|
// mode,
|
|
// category,
|
|
// rank: (attributes[0].children[0].data || '').slice(1, -1),
|
|
// rsn: (attributes[1].children[1].children[0].data || '').replace(
|
|
// /\uFFFD/g,
|
|
// ' '
|
|
// ),
|
|
// };
|
|
|
|
// SKILLS.includes(category)
|
|
// ? (playerInfo = Object.assign(
|
|
// {
|
|
// level: attributes[2].children[0].data.slice(1, -1),
|
|
// xp: attributes[3].children[0].data.slice(1, -1),
|
|
// },
|
|
// playerInfo
|
|
// ))
|
|
// : (playerInfo.score = attributes[2].children[0].data.slice(1, -1));
|
|
|
|
// if (mode === 'hc') {
|
|
// playerInfo.dead = attributes[1].children.length > 1;
|
|
// }
|
|
|
|
// players.push(playerInfo);
|
|
// }
|
|
|
|
// return players;
|
|
// }
|
|
|
|
export const getRSNFormat = async (rsn: string) => {
|
|
const url = getPlayerTableURL('main', rsn);
|
|
|
|
try {
|
|
const response = await axios(url);
|
|
const $ = cheerio.load(response.data);
|
|
const rawName = $('[style="color:#AA0022;"]')[1].children[0].data;
|
|
if (rawName) {
|
|
return rawName.replace(/\uFFFD/g, ' ');
|
|
}
|
|
throw Error('Player not found');
|
|
} catch {
|
|
throw Error('Player not found');
|
|
}
|
|
};
|
|
|
|
export const parseStats = (csv: string): Stats => {
|
|
const splitCSV = csv
|
|
.split('\n')
|
|
.filter(entry => !!entry)
|
|
.map(stat => stat.split(','));
|
|
|
|
const skillObjects: Skill[] = splitCSV
|
|
.filter(stat => stat.length === 3)
|
|
.map(stat => {
|
|
const skill: Skill = {
|
|
rank: parseInt(stat[0], 10),
|
|
level: parseInt(stat[1], 10),
|
|
xp: parseInt(stat[2], 10),
|
|
};
|
|
return skill;
|
|
});
|
|
|
|
const activityObjects: Activity[] = splitCSV
|
|
.filter(stat => stat.length === 2)
|
|
.map(stat => {
|
|
const activity: Activity = {
|
|
rank: parseInt(stat[0], 10),
|
|
score: parseInt(stat[1], 10),
|
|
};
|
|
return activity;
|
|
});
|
|
|
|
const bhObjects = activityObjects.splice(0, BH.length);
|
|
const [lms] = activityObjects.splice(0, 1);
|
|
const clueObjects = activityObjects.splice(0, CLUES.length);
|
|
|
|
const skills: Skills = skillObjects.reduce<Skills>(
|
|
(prev, curr, index) => {
|
|
const newSkills = { ...prev };
|
|
newSkills[SKILLS[index]] = curr;
|
|
return newSkills;
|
|
},
|
|
{} as Skills
|
|
);
|
|
|
|
const bh: BHStats = bhObjects.reduce<BHStats>(
|
|
(prev, curr, index) => {
|
|
const newBH = { ...prev };
|
|
newBH[BH[index]] = curr;
|
|
return newBH;
|
|
},
|
|
{} as BHStats
|
|
);
|
|
|
|
const clues: Clues = clueObjects.reduce<Clues>(
|
|
(prev, curr, index) => {
|
|
const newClues = { ...prev };
|
|
newClues[CLUES[index]] = curr;
|
|
return newClues;
|
|
},
|
|
{} as Clues
|
|
);
|
|
|
|
const stats: Stats = {
|
|
skills,
|
|
bh,
|
|
lms,
|
|
clues,
|
|
};
|
|
|
|
return stats;
|
|
};
|