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 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | ae47df31f1 | ||
|   | 7a3f3a0800 | ||
|   | b16e8ace6e | ||
|   | 09f7805fd4 | ||
|   | 939f4d2721 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | a4577ffb2c | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 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