mirror of
https://github.com/maxswa/osrs-json-hiscores.git
synced 2025-10-15 10:19:04 +00:00
Compare commits
24 Commits
Author | SHA1 | Date | |
---|---|---|---|
![]() |
d097567a3c | ||
![]() |
be5c093cf7 | ||
![]() |
db72f0c57d | ||
![]() |
bd2ea21980 | ||
![]() |
d67837c161 | ||
![]() |
4aa6ad2752 | ||
![]() |
2f788fabf6 | ||
![]() |
0f5aea795c | ||
![]() |
7acb84e310 | ||
![]() |
9619bd2a2b | ||
![]() |
55abd9f800 | ||
![]() |
cf8a4cc26e | ||
![]() |
0f68c1995c | ||
![]() |
ae47df31f1 | ||
![]() |
7a3f3a0800 | ||
![]() |
b16e8ace6e | ||
![]() |
09f7805fd4 | ||
![]() |
939f4d2721 | ||
![]() |
a4577ffb2c | ||
![]() |
19ba7e2916 | ||
![]() |
7de2d9a95a | ||
![]() |
a143306519 | ||
![]() |
86b81abfd8 | ||
![]() |
8d065742d0 |
18
.github/workflows/main.yml
vendored
Normal file
18
.github/workflows/main.yml
vendored
Normal file
@@ -0,0 +1,18 @@
|
|||||||
|
name: CI
|
||||||
|
|
||||||
|
on: [push, pull_request]
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
build:
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
steps:
|
||||||
|
- uses: actions/checkout@v2
|
||||||
|
- uses: borales/actions-yarn@v2.0.0
|
||||||
|
with:
|
||||||
|
cmd: install
|
||||||
|
- uses: borales/actions-yarn@v2.0.0
|
||||||
|
with:
|
||||||
|
cmd: build
|
||||||
|
- uses: borales/actions-yarn@v2.0.0
|
||||||
|
with:
|
||||||
|
cmd: test
|
@@ -108,6 +108,7 @@ Activities consist of all levels of clue scrolls as well as minigames and bosses
|
|||||||
| Bounty Hunter (Rogue) | `rogueBH` |
|
| Bounty Hunter (Rogue) | `rogueBH` |
|
||||||
| Bounty Hunter (Hunter) | `hunterBH` |
|
| Bounty Hunter (Hunter) | `hunterBH` |
|
||||||
| Last Man Standing | `lastManStanding` |
|
| Last Man Standing | `lastManStanding` |
|
||||||
|
| Soul Wars Zeal | `soulWarsZeal` |
|
||||||
|
|
||||||
### Leagues
|
### Leagues
|
||||||
|
|
||||||
@@ -186,6 +187,7 @@ Activities consist of all levels of clue scrolls as well as minigames and bosses
|
|||||||
leaguePoints: {},
|
leaguePoints: {},
|
||||||
bountyHunter: {},
|
bountyHunter: {},
|
||||||
lastManStanding: {},
|
lastManStanding: {},
|
||||||
|
soulWarsZeal: {},
|
||||||
bosses: {}
|
bosses: {}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@@ -32,9 +32,9 @@ test('Parse CSV to json', () => {
|
|||||||
169,99,43127930
|
169,99,43127930
|
||||||
810,99,37688883
|
810,99,37688883
|
||||||
92,99,32005622
|
92,99,32005622
|
||||||
-1,-1
|
23423,478
|
||||||
-1,-1
|
99831,23
|
||||||
-1,-1
|
89912,37
|
||||||
32,12148
|
32,12148
|
||||||
3105,76
|
3105,76
|
||||||
1997,505
|
1997,505
|
||||||
@@ -43,6 +43,7 @@ test('Parse CSV to json', () => {
|
|||||||
392,250
|
392,250
|
||||||
1,6143
|
1,6143
|
||||||
4814,898
|
4814,898
|
||||||
|
37,225
|
||||||
382,2780
|
382,2780
|
||||||
944,3000
|
944,3000
|
||||||
1981,1452
|
1981,1452
|
||||||
@@ -115,12 +116,13 @@ test('Parse CSV to json', () => {
|
|||||||
hunter: { rank: 810, level: 99, xp: 37688883 },
|
hunter: { rank: 810, level: 99, xp: 37688883 },
|
||||||
construction: { rank: 92, level: 99, xp: 32005622 },
|
construction: { rank: 92, level: 99, xp: 32005622 },
|
||||||
},
|
},
|
||||||
leaguePoints: { rank: -1, score: -1 },
|
leaguePoints: { rank: 23423, score: 478 },
|
||||||
bountyHunter: {
|
bountyHunter: {
|
||||||
rogue: { rank: -1, score: -1 },
|
hunter: { rank: 99831, score: 23 },
|
||||||
hunter: { rank: -1, score: -1 },
|
rogue: { rank: 89912, score: 37 },
|
||||||
},
|
},
|
||||||
lastManStanding: { rank: 4814, score: 898 },
|
lastManStanding: { rank: 4814, score: 898 },
|
||||||
|
soulWarsZeal: { rank: 37, score: 225 },
|
||||||
clues: {
|
clues: {
|
||||||
all: { rank: 32, score: 12148 },
|
all: { rank: 32, score: 12148 },
|
||||||
beginner: { rank: 3105, score: 76 },
|
beginner: { rank: 3105, score: 76 },
|
||||||
|
17
package.json
17
package.json
@@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "osrs-json-hiscores",
|
"name": "osrs-json-hiscores",
|
||||||
"version": "2.1.0",
|
"version": "2.3.2",
|
||||||
"description": "The Oldschool Runescape API wrapper that does more!",
|
"description": "The Oldschool Runescape API wrapper that does more!",
|
||||||
"main": "lib/index.js",
|
"main": "lib/index.js",
|
||||||
"types": "lib/index.d.ts",
|
"types": "lib/index.d.ts",
|
||||||
@@ -36,16 +36,17 @@
|
|||||||
},
|
},
|
||||||
"homepage": "https://github.com/maxswa/osrs-json-hiscores#readme",
|
"homepage": "https://github.com/maxswa/osrs-json-hiscores#readme",
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"axios": "^0.19.0",
|
"axios": "^0.21.1",
|
||||||
"cheerio": "^1.0.0-rc.3"
|
"jsdom": "^16.3.0",
|
||||||
|
"useragent-generator": "^1.1.0"
|
||||||
},
|
},
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"@types/cheerio": "^0.22.11",
|
"@types/jest": "^26.0.20",
|
||||||
"@types/jest": "^24.0.14",
|
"@types/jsdom": "^16.2.3",
|
||||||
"jest": "^24.8.0",
|
"jest": "^26.6.3",
|
||||||
"np": "^5.0.3",
|
"np": "6.5.0",
|
||||||
"prettier": "^1.19.1",
|
"prettier": "^1.19.1",
|
||||||
"ts-jest": "^24.0.2",
|
"ts-jest": "^26.4.4",
|
||||||
"tslint": "^5.17.0",
|
"tslint": "^5.17.0",
|
||||||
"tslint-config-airbnb": "^5.11.1",
|
"tslint-config-airbnb": "^5.11.1",
|
||||||
"tslint-config-prettier": "^1.18.0",
|
"tslint-config-prettier": "^1.18.0",
|
||||||
|
51
src/@types/useragent-generator.d.ts
vendored
Normal file
51
src/@types/useragent-generator.d.ts
vendored
Normal file
@@ -0,0 +1,51 @@
|
|||||||
|
declare module 'useragent-generator' {
|
||||||
|
/********************
|
||||||
|
* Google Chrome *
|
||||||
|
/*******************/
|
||||||
|
export function chrome(opt: number | string | { version: string, os?: string }): string;
|
||||||
|
export namespace chrome {
|
||||||
|
function androidPhone(opt: number | string | { version: string, androidVersion?: string, device?: string }): string;
|
||||||
|
function androidTablet(opt: number | string | { version: string, androidVersion?: string, device?: string })
|
||||||
|
: string;
|
||||||
|
function androidWebview(opt: number | string | { androidVersion: string, chromeVersion?: string, device?: string })
|
||||||
|
: string;
|
||||||
|
function chromecast(opt: number | string | { version: string }): string;
|
||||||
|
function iOS(opt: number | string | { iOSVersion: string, chromeVersion?: string, device?: string }): string;
|
||||||
|
}
|
||||||
|
export function chromium(opt: number | string | { version: string, os?: string }): string;
|
||||||
|
/***************
|
||||||
|
* Firefox *
|
||||||
|
/*************/
|
||||||
|
export function firefox(opt: number | string | { version: string, os?: string }): string;
|
||||||
|
export namespace firefox {
|
||||||
|
function androidPhone(opt: number | string | { version: string, androidVersion?: string, device?: string }): string;
|
||||||
|
function androidTablet(opt: number | string | { version: string, androidVersion?: string, device?: string })
|
||||||
|
: string;
|
||||||
|
function iOS(opt: number | string | { iOSVersion: string, device?: string }): string;
|
||||||
|
}
|
||||||
|
/**************
|
||||||
|
* Safari *
|
||||||
|
/************/
|
||||||
|
export function safari(opt: number | string | { version: string, os?: string }): string;
|
||||||
|
export namespace safari {
|
||||||
|
function iOS(opt: number | string | { iOSVersion: string, safariVersion?: string, device?: string }): string;
|
||||||
|
function iOSWebview(opt: number | string | { iOSVersion: string, safariVersion?: string, device?: string }): string;
|
||||||
|
}
|
||||||
|
/***********************
|
||||||
|
* Internet Explorer *
|
||||||
|
/*********************/
|
||||||
|
export function ie(opt: number | string | { version: string, os?: string }): string;
|
||||||
|
export namespace ie {
|
||||||
|
function windowsPhone(opt: number | string | { version: string, device?: string }): string;
|
||||||
|
}
|
||||||
|
/**********************
|
||||||
|
* Microsoft Edge *
|
||||||
|
/********************/
|
||||||
|
export function edge(opt: number | string | { version: string, chromeVersion?: string, os?: string }): string;
|
||||||
|
/************************
|
||||||
|
* Search Engine Bots *
|
||||||
|
/**********************/
|
||||||
|
export function googleBot(opt?: number | string | { version?: string }): string;
|
||||||
|
export function bingBot(opt?: number | string | { version?: string }): string;
|
||||||
|
export function yahooBot(): string;
|
||||||
|
}
|
@@ -1,5 +1,3 @@
|
|||||||
import axios from 'axios';
|
|
||||||
import * as cheerio from 'cheerio';
|
|
||||||
import {
|
import {
|
||||||
Player,
|
Player,
|
||||||
Activity,
|
Activity,
|
||||||
@@ -14,7 +12,6 @@ import {
|
|||||||
ActivityName,
|
ActivityName,
|
||||||
PlayerActivityRow,
|
PlayerActivityRow,
|
||||||
Bosses,
|
Bosses,
|
||||||
Boss,
|
|
||||||
} from './types';
|
} from './types';
|
||||||
import {
|
import {
|
||||||
getStatsURL,
|
getStatsURL,
|
||||||
@@ -28,8 +25,10 @@ import {
|
|||||||
numberFromElement,
|
numberFromElement,
|
||||||
rsnFromElement,
|
rsnFromElement,
|
||||||
getActivityPageURL,
|
getActivityPageURL,
|
||||||
|
httpGet,
|
||||||
BOSSES,
|
BOSSES,
|
||||||
} from './utils';
|
} from './utils';
|
||||||
|
import { JSDOM } from 'jsdom';
|
||||||
|
|
||||||
export async function getStats(rsn: string): Promise<Player> {
|
export async function getStats(rsn: string): Promise<Player> {
|
||||||
if (typeof rsn !== 'string') {
|
if (typeof rsn !== 'string') {
|
||||||
@@ -40,12 +39,12 @@ export async function getStats(rsn: string): Promise<Player> {
|
|||||||
throw Error('RSN must be between 1 and 12 characters');
|
throw Error('RSN must be between 1 and 12 characters');
|
||||||
}
|
}
|
||||||
|
|
||||||
const mainRes = await axios(getStatsURL('main', rsn));
|
const mainRes = await httpGet(getStatsURL('main', rsn));
|
||||||
if (mainRes.status === 200) {
|
if (mainRes.status === 200) {
|
||||||
const otherResponses = await Promise.all([
|
const otherResponses = await Promise.all([
|
||||||
axios(getStatsURL('ironman', rsn)).catch(err => err),
|
httpGet(getStatsURL('ironman', rsn)).catch(err => err),
|
||||||
axios(getStatsURL('hardcore', rsn)).catch(err => err),
|
httpGet(getStatsURL('hardcore', rsn)).catch(err => err),
|
||||||
axios(getStatsURL('ultimate', rsn)).catch(err => err),
|
httpGet(getStatsURL('ultimate', rsn)).catch(err => err),
|
||||||
getRSNFormat(rsn).catch(() => undefined),
|
getRSNFormat(rsn).catch(() => undefined),
|
||||||
]);
|
]);
|
||||||
|
|
||||||
@@ -121,7 +120,7 @@ export async function getStatsByGamemode(
|
|||||||
} else if (!GAMEMODES.includes(mode)) {
|
} else if (!GAMEMODES.includes(mode)) {
|
||||||
throw Error('Invalid game mode');
|
throw Error('Invalid game mode');
|
||||||
}
|
}
|
||||||
const response = await axios(getStatsURL(mode, rsn));
|
const response = await httpGet(getStatsURL(mode, rsn));
|
||||||
if (response.status !== 200) {
|
if (response.status !== 200) {
|
||||||
throw Error('Player not found');
|
throw Error('Player not found');
|
||||||
}
|
}
|
||||||
@@ -144,23 +143,27 @@ export async function getSkillPage(
|
|||||||
}
|
}
|
||||||
const url = getSkillPageURL(mode, skill, page);
|
const url = getSkillPageURL(mode, skill, page);
|
||||||
|
|
||||||
const response = await axios(url);
|
const response = await httpGet(url);
|
||||||
const $ = cheerio.load(response.data);
|
const dom = new JSDOM(response.data);
|
||||||
const playersHTML = $('.personal-hiscores__row').toArray();
|
const playersHTML = dom.window.document.querySelectorAll(
|
||||||
|
'.personal-hiscores__row'
|
||||||
|
);
|
||||||
|
|
||||||
const players: PlayerSkillRow[] = playersHTML.map(row => {
|
const players: PlayerSkillRow[] = [];
|
||||||
const cells = row.children.filter(el => el.name === 'td');
|
playersHTML.forEach(row => {
|
||||||
const [rankEl, nameCell, levelEl, xpEl] = cells;
|
const rankEl = row.querySelector('td');
|
||||||
const nameEl = nameCell.children.find(el => el.name === 'a');
|
const nameEl = row.querySelector('td a');
|
||||||
const isDead = !!nameCell.children.find(el => el.name === 'img');
|
const levelEl = row.querySelector('td.left + td');
|
||||||
|
const xpEl = row.querySelector('td.left + td + td');
|
||||||
|
const isDead = !!row.querySelector('td img');
|
||||||
|
|
||||||
return {
|
players.push({
|
||||||
name: rsnFromElement(nameEl),
|
name: rsnFromElement(nameEl),
|
||||||
rank: numberFromElement(rankEl),
|
rank: numberFromElement(rankEl),
|
||||||
level: numberFromElement(levelEl),
|
level: numberFromElement(levelEl),
|
||||||
xp: numberFromElement(xpEl),
|
xp: numberFromElement(xpEl),
|
||||||
dead: isDead,
|
dead: isDead,
|
||||||
};
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
return players;
|
return players;
|
||||||
@@ -180,22 +183,25 @@ export async function getActivityPage(
|
|||||||
}
|
}
|
||||||
const url = getActivityPageURL(mode, activity, page);
|
const url = getActivityPageURL(mode, activity, page);
|
||||||
|
|
||||||
const response = await axios(url);
|
const response = await httpGet(url);
|
||||||
const $ = cheerio.load(response.data);
|
const dom = new JSDOM(response.data);
|
||||||
const playersHTML = $('.personal-hiscores__row').toArray();
|
const playersHTML = dom.window.document.querySelectorAll(
|
||||||
|
'.personal-hiscores__row'
|
||||||
|
);
|
||||||
|
|
||||||
const players: PlayerActivityRow[] = playersHTML.map(row => {
|
const players: PlayerActivityRow[] = [];
|
||||||
const cells = row.children.filter(el => el.name === 'td');
|
playersHTML.forEach(row => {
|
||||||
const [rankEl, nameCell, scoreEl] = cells;
|
const rankEl = row.querySelector('td');
|
||||||
const nameEl = nameCell.children.find(el => el.name === 'a');
|
const nameEl = row.querySelector('td a');
|
||||||
const isDead = !!nameCell.children.find(el => el.name === 'img');
|
const scoreEl = row.querySelector('td.left + td');
|
||||||
|
const isDead = !!row.querySelector('td img');
|
||||||
|
|
||||||
return {
|
players.push({
|
||||||
name: rsnFromElement(nameEl),
|
name: rsnFromElement(nameEl),
|
||||||
rank: numberFromElement(rankEl),
|
rank: numberFromElement(rankEl),
|
||||||
score: numberFromElement(scoreEl),
|
score: numberFromElement(scoreEl),
|
||||||
dead: isDead,
|
dead: isDead,
|
||||||
};
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
return players;
|
return players;
|
||||||
@@ -212,11 +218,14 @@ export async function getRSNFormat(rsn: string): Promise<string> {
|
|||||||
|
|
||||||
const url = getPlayerTableURL('main', rsn);
|
const url = getPlayerTableURL('main', rsn);
|
||||||
try {
|
try {
|
||||||
const response = await axios(url);
|
const response = await httpGet(url);
|
||||||
const $ = cheerio.load(response.data);
|
const dom = new JSDOM(response.data);
|
||||||
const rawName = $('[style="color:#AA0022;"]')[1].children[0].data;
|
const spans = dom.window.document.querySelectorAll(
|
||||||
if (rawName) {
|
'span[style="color:#AA0022;"]'
|
||||||
return rawName.replace(/\uFFFD/g, ' ');
|
);
|
||||||
|
if (spans.length >= 2) {
|
||||||
|
const nameSpan = spans[1];
|
||||||
|
return rsnFromElement(nameSpan);
|
||||||
}
|
}
|
||||||
throw Error('Player not found');
|
throw Error('Player not found');
|
||||||
} catch {
|
} catch {
|
||||||
@@ -256,7 +265,7 @@ export function parseStats(csv: string): Stats {
|
|||||||
const [leaguePoints] = activityObjects.splice(0, 1);
|
const [leaguePoints] = activityObjects.splice(0, 1);
|
||||||
const bhObjects = activityObjects.splice(0, BH_MODES.length);
|
const bhObjects = activityObjects.splice(0, BH_MODES.length);
|
||||||
const clueObjects = activityObjects.splice(0, CLUES.length);
|
const clueObjects = activityObjects.splice(0, CLUES.length);
|
||||||
const [lastManStanding] = activityObjects.splice(0, 1);
|
const [lastManStanding, soulWarsZeal] = activityObjects.splice(0, 2);
|
||||||
const bossObjects = activityObjects.splice(0, BOSSES.length);
|
const bossObjects = activityObjects.splice(0, BOSSES.length);
|
||||||
|
|
||||||
const skills: Skills = skillObjects.reduce<Skills>((prev, curr, index) => {
|
const skills: Skills = skillObjects.reduce<Skills>((prev, curr, index) => {
|
||||||
@@ -288,6 +297,7 @@ export function parseStats(csv: string): Stats {
|
|||||||
leaguePoints,
|
leaguePoints,
|
||||||
bountyHunter,
|
bountyHunter,
|
||||||
lastManStanding,
|
lastManStanding,
|
||||||
|
soulWarsZeal,
|
||||||
clues,
|
clues,
|
||||||
bosses,
|
bosses,
|
||||||
};
|
};
|
||||||
|
@@ -114,6 +114,7 @@ export type ActivityName =
|
|||||||
| 'hunterBH'
|
| 'hunterBH'
|
||||||
| 'rogueBH'
|
| 'rogueBH'
|
||||||
| 'lastManStanding'
|
| 'lastManStanding'
|
||||||
|
| 'soulWarsZeal'
|
||||||
| 'allClues'
|
| 'allClues'
|
||||||
| 'beginnerClues'
|
| 'beginnerClues'
|
||||||
| 'easyClues'
|
| 'easyClues'
|
||||||
@@ -129,6 +130,7 @@ export interface Stats {
|
|||||||
leaguePoints: Activity;
|
leaguePoints: Activity;
|
||||||
bountyHunter: BH;
|
bountyHunter: BH;
|
||||||
lastManStanding: Activity;
|
lastManStanding: Activity;
|
||||||
|
soulWarsZeal: Activity;
|
||||||
bosses: Bosses;
|
bosses: Bosses;
|
||||||
}
|
}
|
||||||
export type Modes = { [M in Gamemode]?: Stats };
|
export type Modes = { [M in Gamemode]?: Stats };
|
||||||
|
@@ -59,7 +59,7 @@ export const CLUES: ClueType[] = [
|
|||||||
'elite',
|
'elite',
|
||||||
'master',
|
'master',
|
||||||
];
|
];
|
||||||
export const BH_MODES: BHType[] = ['rogue', 'hunter'];
|
export const BH_MODES: BHType[] = ['hunter', 'rogue'];
|
||||||
export const GAMEMODES: Gamemode[] = [
|
export const GAMEMODES: Gamemode[] = [
|
||||||
'main',
|
'main',
|
||||||
'ironman',
|
'ironman',
|
||||||
@@ -127,6 +127,7 @@ export const ACTIVITIES: ActivityName[] = [
|
|||||||
'eliteClues',
|
'eliteClues',
|
||||||
'masterClues',
|
'masterClues',
|
||||||
'lastManStanding',
|
'lastManStanding',
|
||||||
|
'soulWarsZeal',
|
||||||
...BOSSES,
|
...BOSSES,
|
||||||
];
|
];
|
||||||
|
|
||||||
@@ -236,4 +237,5 @@ export const FORMATTED_BH_NAMES: FormattedBHNames = {
|
|||||||
};
|
};
|
||||||
|
|
||||||
export const FORMATTED_LMS = 'Last Man Standing';
|
export const FORMATTED_LMS = 'Last Man Standing';
|
||||||
|
export const FORMATTED_SOUL_WARS = 'Soul Wars Zeal';
|
||||||
export const FORMATTED_LEAGUE_POINTS = 'League Points';
|
export const FORMATTED_LEAGUE_POINTS = 'League Points';
|
||||||
|
@@ -1,3 +1,5 @@
|
|||||||
|
import axios from 'axios';
|
||||||
|
import * as ua from 'useragent-generator';
|
||||||
import { Gamemode, SkillName, ActivityName } from '../types';
|
import { Gamemode, SkillName, ActivityName } from '../types';
|
||||||
import {
|
import {
|
||||||
GAMEMODE_URL,
|
GAMEMODE_URL,
|
||||||
@@ -35,13 +37,22 @@ export const getActivityPageURL = (
|
|||||||
activity
|
activity
|
||||||
)}&page=${page}`;
|
)}&page=${page}`;
|
||||||
|
|
||||||
export const numberFromElement = (el: CheerioElement) => {
|
export const numberFromElement = (el: Element | null) => {
|
||||||
const innerText = el.firstChild.data;
|
const { innerHTML } = el || {};
|
||||||
const number = innerText ? innerText.replace(/[\n|,]/g, '') : '-1';
|
const number = innerHTML?.replace(/[\n|,]/g, '') ?? '-1';
|
||||||
return parseInt(number, 10);
|
return parseInt(number, 10);
|
||||||
};
|
};
|
||||||
|
|
||||||
export const rsnFromElement = (el: CheerioElement | undefined) => {
|
export const rsnFromElement = (el: Element | null) => {
|
||||||
const innerText = el?.firstChild.data;
|
const { innerHTML } = el || {};
|
||||||
return innerText ? innerText.replace(/\uFFFD/g, ' ') : '';
|
return innerHTML?.replace(/\uFFFD/g, ' ') || '';
|
||||||
|
};
|
||||||
|
|
||||||
|
export const httpGet = (url: string) => {
|
||||||
|
return axios.get(url, {
|
||||||
|
headers: {
|
||||||
|
// without User-Agent header requests may be rejected by DDoS protection mechanism
|
||||||
|
'User-Agent': ua.firefox(80)
|
||||||
|
}
|
||||||
|
});
|
||||||
};
|
};
|
||||||
|
Reference in New Issue
Block a user