Add axios, refactor parseStats

This commit is contained in:
maxswa
2019-06-10 11:57:24 -04:00
parent 3c0e8143c3
commit f776e5e936
3 changed files with 187 additions and 230 deletions

3
.gitignore vendored
View File

@@ -1 +1,2 @@
.idea/ .idea/
node_modules

6
.prettierrc Normal file
View File

@@ -0,0 +1,6 @@
{
"trailingComma": "es5",
"tabWidth": 2,
"semi": true,
"singleQuote": true
}

View File

@@ -1,3 +1,5 @@
const axios = require('axios');
const URLs = { const URLs = {
main: 'http://services.runescape.com/m=hiscore_oldschool/', main: 'http://services.runescape.com/m=hiscore_oldschool/',
iron: 'http://services.runescape.com/m=hiscore_oldschool_ironman/', iron: 'http://services.runescape.com/m=hiscore_oldschool_ironman/',
@@ -7,7 +9,7 @@ const URLs = {
sdmm: 'http://services.runescape.com/m=hiscore_oldschool_seasonal/', sdmm: 'http://services.runescape.com/m=hiscore_oldschool_seasonal/',
dmmt: 'http://services.runescape.com/m=hiscore_oldschool_tournament/', dmmt: 'http://services.runescape.com/m=hiscore_oldschool_tournament/',
stats: 'index_lite.ws?player=', stats: 'index_lite.ws?player=',
scores: 'overall.ws?' scores: 'overall.ws?',
}, },
hiscores = { hiscores = {
skills: [ skills: [
@@ -34,20 +36,12 @@ const URLs = {
'farming', 'farming',
'runecraft', 'runecraft',
'hunter', 'hunter',
'construction'], 'construction',
other: [ ],
'easyclues', clues: ['all', 'beginner', 'easy', 'medium', 'hard', 'elite', 'master'],
'mediumclues', bh: ['rouge', 'hunter'],
'allclues',
'roguebh',
'hunterbh',
'hardclues',
'lms',
'eliteclues',
'masterclues'
]
}, },
validModes = ['full', 'main', 'iron', 'hc', 'ult', 'dmm', 'sdmm', 'dmmt'] validModes = ['full', 'main', 'iron', 'hc', 'ult', 'dmm', 'sdmm', 'dmmt'];
/** /**
* Gets a player's stats. * Gets a player's stats.
@@ -61,21 +55,17 @@ const URLs = {
* *
* @returns {Object} A player object. * @returns {Object} A player object.
*/ */
async function getStats (rsn, mode = 'full') { async function getStats(rsn, mode = 'full') {
if(typeof rsn !== 'string') { if (typeof rsn !== 'string') {
throw Error('RSN must be a string') throw Error('RSN must be a string');
} } else if (!/^[a-zA-Z0-9 _]+$/.test(rsn)) {
else if(!/^[a-zA-Z0-9 _]+$/.test(rsn)) { throw Error('RSN contains invalid character');
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(rsn.length > 12 || rsn.length < 1) { } else if (!validModes.includes(mode.toLowerCase())) {
throw Error('RSN must be between 1 and 12 characters') throw Error('Invalid game mode');
} } else {
else if(!validModes.includes(mode.toLowerCase())) { return await getPlayerStats(rsn, mode.toLowerCase());
throw Error('Invalid game mode')
}
else {
return await getPlayerStats(rsn, mode.toLowerCase())
} }
} }
@@ -91,107 +81,97 @@ async function getStats (rsn, mode = 'full') {
* *
* @returns {Object} A player object. * @returns {Object} A player object.
*/ */
async function getPlayerStats (rsn, mode) { async function getPlayerStats(rsn, mode) {
let player = { let player = {
rsn: rsn, rsn: rsn,
mode: mode, mode: mode,
dead: false, dead: false,
deironed: false deironed: false,
} };
if(mode === 'full') { if (mode === 'full') {
const responses = [] const responses = [];
let csv
responses[0] = await fetch(URLs.main + URLs.stats + encodeURIComponent(rsn)) responses[0] = await axios(
if (responses[0].ok) { URLs.main + URLs.stats + encodeURIComponent(rsn)
);
if (responses[0].status === 200) {
const otherResponses = await Promise.all([ const otherResponses = await Promise.all([
fetch(URLs.iron + URLs.stats + encodeURIComponent(rsn)), axios(URLs.iron + URLs.stats + encodeURIComponent(rsn)).catch(
fetch(URLs.hc + URLs.stats + encodeURIComponent(rsn)), res => res
fetch(URLs.ult + URLs.stats + encodeURIComponent(rsn)), ),
getRSNFormat(rsn) axios(URLs.hc + URLs.stats + encodeURIComponent(rsn)).catch(res => res),
]) axios(URLs.ult + URLs.stats + encodeURIComponent(rsn)).catch(
res => res
player.rsn = otherResponses.pop() ),
]);
for (let res of otherResponses) { for (let res of otherResponses) {
responses.push(res) responses.push(res);
} }
if (responses[1].ok) { if (responses[1].status === 200) {
if (responses[2].ok) { if (responses[2].status === 200) {
player.mode = 'hc' player.mode = 'hc';
} } else if (responses[3].status === 200) {
else if (responses[3].ok) { player.mode = 'ult';
player.mode = 'ult' } else {
} player.mode = 'iron';
else {
player.mode = 'iron'
} }
} else {
player.mode = 'main';
} }
else { } else {
player.mode = 'main' throw Error('Player not found');
}
}
else {
throw Error('Player not found')
} }
switch (player.mode) { switch (player.mode) {
case 'main': case 'main':
csv = await responses[0].text() player.main = parseStats(responses[0].data);
player.main = parseStats(csv) break;
break
case 'iron': case 'iron':
csv = await responses[0].text() player.main = parseStats(responses[0].data);
player.main = parseStats(csv) player.iron = parseStats(responses[1].data);
csv = await responses[1].text() if (player.main.stats.overall.xp !== player.iron.stats.overall.xp) {
player.iron = parseStats(csv) player.deironed = true;
if(player.main.stats.overall.xp !== player.iron.stats.overall.xp) { player.mode = 'main';
player.deironed = true
player.mode = 'main'
} }
break break;
case 'hc': case 'hc':
csv = await responses[0].text() player.main = parseStats(responses[0].data);
player.main = parseStats(csv) player.iron = parseStats(responses[1].data);
csv = await responses[1].text() player.hc = parseStats(responses[2].data);
player.iron = parseStats(csv) if (player.iron.stats.overall.xp !== player.hc.stats.overall.xp) {
csv = await responses[2].text() player.dead = true;
player.hc = parseStats(csv) player.mode = 'iron';
if(player.iron.stats.overall.xp !== player.hc.stats.overall.xp) {
player.dead = true
player.mode = 'iron'
} }
if(player.main.stats.overall.xp !== player.iron.stats.overall.xp) { if (player.main.stats.overall.xp !== player.iron.stats.overall.xp) {
player.deironed = true player.deironed = true;
player.mode = 'main' player.mode = 'main';
} }
break break;
case 'ult': case 'ult':
csv = await responses[0].text() player.main = parseStats(responses[0].data);
player.main = parseStats(csv) player.iron = parseStats(responses[1].data);
csv = await responses[1].text() player.ult = parseStats(responses[3].data);
player.iron = parseStats(csv) if (player.main.stats.overall.xp !== player.iron.stats.overall.xp) {
csv = await responses[3].text() player.deironed = true;
player.ult = parseStats(csv) player.mode = 'main';
if(player.main.stats.overall.xp !== player.iron.stats.overall.xp) {
player.deironed = true
player.mode = 'main'
} }
break break;
} }
return player return player;
} } else {
else { const response = await axios(
const response = await fetch(URLs[mode] + URLs.stats + encodeURIComponent(rsn)) URLs[mode] + URLs.stats + encodeURIComponent(rsn)
if(!response.ok) { );
throw Error('Player not found') if (response.status !== 200) {
throw Error('Player not found');
} }
const csv = await response.text() player[mode] = parseStats(response.data);
player[mode] = parseStats(csv) return player;
return player
} }
} }
@@ -208,18 +188,25 @@ async function getPlayerStats (rsn, mode) {
* *
* @returns {Object[]} Array of player objects. * @returns {Object[]} Array of player objects.
*/ */
async function getHiscores (mode, category = 'overall', page = 1) { async function getHiscores(mode, category = 'overall', page = 1) {
if(!validModes.includes(mode.toLowerCase()) || mode.toLowerCase() === 'full') { if (
throw Error('Invalid game mode') !validModes.includes(mode.toLowerCase()) ||
} mode.toLowerCase() === 'full'
else if(!Number.isInteger(page) || page < 1) { ) {
throw Error('Page must be an integer greater than 0') throw Error('Invalid game mode');
} } else if (!Number.isInteger(page) || page < 1) {
else if(!hiscores.skills.includes(category.toLowerCase()) && !hiscores.other.includes(category.toLowerCase())) { throw Error('Page must be an integer greater than 0');
throw Error('Invalid category') } else if (
} !hiscores.skills.includes(category.toLowerCase()) &&
else { !hiscores.other.includes(category.toLowerCase())
return await getHiscoresPage(mode.toLowerCase(), category.toLowerCase(), page) ) {
throw Error('Invalid category');
} else {
return await getHiscoresPage(
mode.toLowerCase(),
category.toLowerCase(),
page
);
} }
} }
@@ -237,40 +224,48 @@ async function getHiscores (mode, category = 'overall', page = 1) {
* @returns {Object[]} Array of player objects. * @returns {Object[]} Array of player objects.
*/ */
async function getHiscoresPage(mode, category, page) { async function getHiscoresPage(mode, category, page) {
const url = URLs[mode] + URLs.scores + const url =
(hiscores.skills.includes(category) ? URLs[mode] +
'table=' + hiscores.skills.indexOf(category) : URLs.scores +
'category_type=1' + '&table=' + hiscores.other.indexOf(category)) + (hiscores.skills.includes(category)
'&page=' + page ? 'table=' + hiscores.skills.indexOf(category)
: 'category_type=1' + '&table=' + hiscores.other.indexOf(category)) +
'&page=' +
page;
const response = await fetch(url) const response = await axios(url);
let players = [], element = document.createElement('html') let players = [],
element.innerHTML = await response.text() element = document.createElement('html');
const playersHTML = element.querySelectorAll('.personal-hiscores__row') element.innerHTML = await response.text();
const playersHTML = element.querySelectorAll('.personal-hiscores__row');
for(let player of playersHTML) { for (let player of playersHTML) {
const attributes = player.querySelectorAll('td') const attributes = player.querySelectorAll('td');
let playerInfo = { let playerInfo = {
mode: mode, mode: mode,
category: category, category: category,
rank: attributes[0].innerHTML.slice(1, -1), rank: attributes[0].innerHTML.slice(1, -1),
rsn: attributes[1].childNodes[1].innerHTML.replace(/\uFFFD/g, ' ') rsn: attributes[1].childNodes[1].innerHTML.replace(/\uFFFD/g, ' '),
};
hiscores.skills.includes(category.toLowerCase())
? (playerInfo = Object.assign(
{
level: attributes[2].innerHTML.slice(1, -1),
xp: attributes[3].innerHTML.slice(1, -1),
},
playerInfo
))
: (playerInfo.score = attributes[2].innerHTML.slice(1, -1));
if (mode === 'hc') {
playerInfo.dead = attributes[1].childElementCount > 1;
} }
hiscores.skills.includes(category.toLowerCase()) ? players.push(playerInfo);
playerInfo = Object.assign(
{level: attributes[2].innerHTML.slice(1, -1),
xp: attributes[3].innerHTML.slice(1, -1)}, playerInfo) :
playerInfo.score = attributes[2].innerHTML.slice(1, -1)
if(mode === 'hc') {
playerInfo.dead = attributes[1].childElementCount > 1
}
players.push(playerInfo)
} }
return players return players;
} }
/** /**
@@ -285,102 +280,57 @@ async function getHiscoresPage(mode, category, page) {
* @returns {string} The player's formatted username. * @returns {string} The player's formatted username.
*/ */
async function getRSNFormat(rsn) { async function getRSNFormat(rsn) {
const url = URLs.main + URLs.scores + 'table=0&user=' + rsn const url = URLs.main + URLs.scores + 'table=0&user=' + rsn;
const response = await fetch(url) const response = await axios(url);
let element = document.createElement('html') let element = document.createElement('html');
element.innerHTML = await response.text() element.innerHTML = await response.text();
const cells = element.querySelectorAll('[style="color:#AA0022;"]') const cells = element.querySelectorAll('[style="color:#AA0022;"]');
if(cells.length >= 2) { if (cells.length >= 2) {
return cells[1].innerHTML.replace(/\uFFFD/g, ' ') return cells[1].innerHTML.replace(/\uFFFD/g, ' ');
} } else {
else { throw Error('Player not found');
throw Error('Player not found')
} }
} }
let parseStats = (csv) => { let parseStats = csv => {
let stats = { const stats = {
stats:{ stats: {},
overall:{rank:0,level:0,xp:0}, clues: {},
attack:{rank:0,level:0,xp:0}, bh: {},
defence:{rank:0,level:0,xp:0}, lms: {},
strength:{rank:0,level:0,xp:0}, };
hitpoints:{rank:0,level:0,xp:0}, const splitCSV = csv.split('\n');
ranged:{rank:0,level:0,xp:0},
prayer:{rank:0,level:0,xp:0},
magic:{rank:0,level:0,xp:0},
cooking:{rank:0,level:0,xp:0},
woodcutting:{rank:0,level:0,xp:0},
fletching:{rank:0,level:0,xp:0},
fishing:{rank:0,level:0,xp:0},
firemaking:{rank:0,level:0,xp:0},
crafting:{rank:0,level:0,xp:0},
smithing:{rank:0,level:0,xp:0},
mining:{rank:0,level:0,xp:0},
herblore:{rank:0,level:0,xp:0},
agility:{rank:0,level:0,xp:0},
thieving:{rank:0,level:0,xp:0},
slayer:{rank:0,level:0,xp:0},
farming:{rank:0,level:0,xp:0},
runecraft:{rank:0,level:0,xp:0},
hunter:{rank:0,level:0,xp:0},
construction:{rank:0,level:0,xp:0}
},
clues:{
all:{rank:0,score:0},
easy:{rank:0,score:0},
medium:{rank:0,score:0},
hard:{rank:0,score:0},
elite:{rank:0,score:0},
master:{rank:0,score:0}
},
bh:{
rogue:{rank:0,score:0},
hunter:{rank:0,score:0}
},
lms:{rank:0,score:0}
}, splitCSV = csv.split('\n'), i = 0, skillInfo
for (let skill of hiscores.skills) {
skillInfo = splitCSV[i].split(',')
stats.stats[skill].rank = skillInfo[0]
stats.stats[skill].level = skillInfo[1]
stats.stats[skill].xp = skillInfo[2]
i++
}
skillInfo = splitCSV[26].split(',') const statObjects = splitCSV
stats.clues.all.rank = skillInfo[0] .filter(stat => !!stat)
stats.clues.all.score = skillInfo[1] .map(stat => {
skillInfo = splitCSV[24].split(',') const splitStat = stat.split(',');
stats.clues.easy.rank = skillInfo[0] const obj = {};
stats.clues.easy.score = skillInfo[1] if (splitStat.length === 3) {
skillInfo = splitCSV[25].split(',') [obj.rank, obj.level, obj.xp] = splitStat;
stats.clues.medium.rank = skillInfo[0] } else {
stats.clues.medium.score = skillInfo[1] [obj.rank, obj.score] = splitStat;
skillInfo = splitCSV[29].split(',') }
stats.clues.hard.rank = skillInfo[0] return obj;
stats.clues.hard.score = skillInfo[1] });
skillInfo = splitCSV[31].split(',')
stats.clues.elite.rank = skillInfo[0]
stats.clues.elite.score = skillInfo[1]
skillInfo = splitCSV[32].split(',')
stats.clues.master.rank = skillInfo[0]
stats.clues.master.score = skillInfo[1]
skillInfo = splitCSV[27].split(',') statObjects.forEach((obj, index) => {
stats.bh.rogue.rank = skillInfo[0] if (index < hiscores.skills.length) {
stats.bh.rogue.score = skillInfo[1] stats.stats[hiscores.skills[index]] = obj;
skillInfo = splitCSV[28].split(',') } else if (index < hiscores.skills.length + hiscores.bh.length) {
stats.bh.hunter.rank = skillInfo[0] stats.bh[hiscores.bh[index - hiscores.skills.length]] = obj;
stats.bh.hunter.score = skillInfo[1] } else if (index < hiscores.skills.length + hiscores.bh.length + 1) {
stats.lms = obj;
} else {
stats.clues[
hiscores.clues[index - hiscores.skills.length - hiscores.bh.length - 1]
] = obj;
}
});
skillInfo = splitCSV[30].split(',') return stats;
stats.lms.rank = skillInfo[0] };
stats.lms.score = skillInfo[1]
return stats module.exports = { getStats, getHiscores, getRSNFormat };
}
export default {getStats, getHiscores, getRSNFormat}