mirror of
https://github.com/maxswa/osrs-json-hiscores.git
synced 2025-10-15 10:19:04 +00:00
Add axios, refactor parseStats
This commit is contained in:
3
.gitignore
vendored
3
.gitignore
vendored
@@ -1 +1,2 @@
|
|||||||
.idea/
|
.idea/
|
||||||
|
node_modules
|
6
.prettierrc
Normal file
6
.prettierrc
Normal file
@@ -0,0 +1,6 @@
|
|||||||
|
{
|
||||||
|
"trailingComma": "es5",
|
||||||
|
"tabWidth": 2,
|
||||||
|
"semi": true,
|
||||||
|
"singleQuote": true
|
||||||
|
}
|
408
hiscores.js
408
hiscores.js
@@ -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}
|
|
||||||
|
Reference in New Issue
Block a user