mirror of
				https://github.com/maxswa/osrs-json-hiscores.git
				synced 2025-10-15 10:19:04 +00:00 
			
		
		
		
	Compare commits
	
		
			20 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 | 
							
								
								
									
										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 (Hunter) |    `hunterBH`     | | ||||
| | Last Man Standing      | `lastManStanding` | | ||||
| | Soul Wars Zeal         |  `soulWarsZeal`   | | ||||
|  | ||||
| ### Leagues | ||||
|  | ||||
| @@ -186,6 +187,7 @@ Activities consist of all levels of clue scrolls as well as minigames and bosses | ||||
|     leaguePoints: {}, | ||||
|     bountyHunter: {}, | ||||
|     lastManStanding: {}, | ||||
|     soulWarsZeal: {}, | ||||
|     bosses: {} | ||||
|   } | ||||
| } | ||||
|   | ||||
| @@ -32,9 +32,9 @@ test('Parse CSV to json', () => { | ||||
|   169,99,43127930 | ||||
|   810,99,37688883 | ||||
|   92,99,32005622 | ||||
|   -1,-1 | ||||
|   -1,-1 | ||||
|   -1,-1 | ||||
|   23423,478 | ||||
|   99831,23 | ||||
|   89912,37 | ||||
|   32,12148 | ||||
|   3105,76 | ||||
|   1997,505 | ||||
| @@ -43,6 +43,7 @@ test('Parse CSV to json', () => { | ||||
|   392,250 | ||||
|   1,6143 | ||||
|   4814,898 | ||||
|   37,225 | ||||
|   382,2780 | ||||
|   944,3000 | ||||
|   1981,1452 | ||||
| @@ -115,12 +116,13 @@ test('Parse CSV to json', () => { | ||||
|       hunter: { rank: 810, level: 99, xp: 37688883 }, | ||||
|       construction: { rank: 92, level: 99, xp: 32005622 }, | ||||
|     }, | ||||
|     leaguePoints: { rank: -1, score: -1 }, | ||||
|     leaguePoints: { rank: 23423, score: 478 }, | ||||
|     bountyHunter: { | ||||
|       rogue: { rank: -1, score: -1 }, | ||||
|       hunter: { rank: -1, score: -1 }, | ||||
|       hunter: { rank: 99831, score: 23 }, | ||||
|       rogue: { rank: 89912, score: 37 }, | ||||
|     }, | ||||
|     lastManStanding: { rank: 4814, score: 898 }, | ||||
|     soulWarsZeal: { rank: 37, score: 225 }, | ||||
|     clues: { | ||||
|       all: { rank: 32, score: 12148 }, | ||||
|       beginner: { rank: 3105, score: 76 }, | ||||
|   | ||||
							
								
								
									
										15
									
								
								package.json
									
									
									
									
									
								
							
							
						
						
									
										15
									
								
								package.json
									
									
									
									
									
								
							| @@ -1,6 +1,6 @@ | ||||
| { | ||||
|   "name": "osrs-json-hiscores", | ||||
|   "version": "2.2.0", | ||||
|   "version": "2.3.2", | ||||
|   "description": "The Oldschool Runescape API wrapper that does more!", | ||||
|   "main": "lib/index.js", | ||||
|   "types": "lib/index.d.ts", | ||||
| @@ -36,16 +36,17 @@ | ||||
|   }, | ||||
|   "homepage": "https://github.com/maxswa/osrs-json-hiscores#readme", | ||||
|   "dependencies": { | ||||
|     "axios": "^0.19.0", | ||||
|     "jsdom": "^16.3.0" | ||||
|     "axios": "^0.21.1", | ||||
|     "jsdom": "^16.3.0", | ||||
|     "useragent-generator": "^1.1.0" | ||||
|   }, | ||||
|   "devDependencies": { | ||||
|     "@types/jest": "^24.0.14", | ||||
|     "@types/jest": "^26.0.20", | ||||
|     "@types/jsdom": "^16.2.3", | ||||
|     "jest": "^24.8.0", | ||||
|     "np": "^5.0.3", | ||||
|     "jest": "^26.6.3", | ||||
|     "np": "6.5.0", | ||||
|     "prettier": "^1.19.1", | ||||
|     "ts-jest": "^24.0.2", | ||||
|     "ts-jest": "^26.4.4", | ||||
|     "tslint": "^5.17.0", | ||||
|     "tslint-config-airbnb": "^5.11.1", | ||||
|     "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,4 +1,3 @@ | ||||
| import axios from 'axios'; | ||||
| import { | ||||
|   Player, | ||||
|   Activity, | ||||
| @@ -26,6 +25,7 @@ import { | ||||
|   numberFromElement, | ||||
|   rsnFromElement, | ||||
|   getActivityPageURL, | ||||
|   httpGet, | ||||
|   BOSSES, | ||||
| } from './utils'; | ||||
| import { JSDOM } from 'jsdom'; | ||||
| @@ -39,12 +39,12 @@ export async function getStats(rsn: string): Promise<Player> { | ||||
|     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) { | ||||
|     const otherResponses = await Promise.all([ | ||||
|       axios(getStatsURL('ironman', rsn)).catch(err => err), | ||||
|       axios(getStatsURL('hardcore', rsn)).catch(err => err), | ||||
|       axios(getStatsURL('ultimate', rsn)).catch(err => err), | ||||
|       httpGet(getStatsURL('ironman', rsn)).catch(err => err), | ||||
|       httpGet(getStatsURL('hardcore', rsn)).catch(err => err), | ||||
|       httpGet(getStatsURL('ultimate', rsn)).catch(err => err), | ||||
|       getRSNFormat(rsn).catch(() => undefined), | ||||
|     ]); | ||||
|  | ||||
| @@ -120,7 +120,7 @@ export async function getStatsByGamemode( | ||||
|   } else if (!GAMEMODES.includes(mode)) { | ||||
|     throw Error('Invalid game mode'); | ||||
|   } | ||||
|   const response = await axios(getStatsURL(mode, rsn)); | ||||
|   const response = await httpGet(getStatsURL(mode, rsn)); | ||||
|   if (response.status !== 200) { | ||||
|     throw Error('Player not found'); | ||||
|   } | ||||
| @@ -143,7 +143,7 @@ export async function getSkillPage( | ||||
|   } | ||||
|   const url = getSkillPageURL(mode, skill, page); | ||||
|  | ||||
|   const response = await axios(url); | ||||
|   const response = await httpGet(url); | ||||
|   const dom = new JSDOM(response.data); | ||||
|   const playersHTML = dom.window.document.querySelectorAll( | ||||
|     '.personal-hiscores__row' | ||||
| @@ -183,7 +183,7 @@ export async function getActivityPage( | ||||
|   } | ||||
|   const url = getActivityPageURL(mode, activity, page); | ||||
|  | ||||
|   const response = await axios(url); | ||||
|   const response = await httpGet(url); | ||||
|   const dom = new JSDOM(response.data); | ||||
|   const playersHTML = dom.window.document.querySelectorAll( | ||||
|     '.personal-hiscores__row' | ||||
| @@ -218,7 +218,7 @@ export async function getRSNFormat(rsn: string): Promise<string> { | ||||
|  | ||||
|   const url = getPlayerTableURL('main', rsn); | ||||
|   try { | ||||
|     const response = await axios(url); | ||||
|     const response = await httpGet(url); | ||||
|     const dom = new JSDOM(response.data); | ||||
|     const spans = dom.window.document.querySelectorAll( | ||||
|       'span[style="color:#AA0022;"]' | ||||
| @@ -265,7 +265,7 @@ export function parseStats(csv: string): Stats { | ||||
|   const [leaguePoints] = activityObjects.splice(0, 1); | ||||
|   const bhObjects = activityObjects.splice(0, BH_MODES.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 skills: Skills = skillObjects.reduce<Skills>((prev, curr, index) => { | ||||
| @@ -297,6 +297,7 @@ export function parseStats(csv: string): Stats { | ||||
|     leaguePoints, | ||||
|     bountyHunter, | ||||
|     lastManStanding, | ||||
|     soulWarsZeal, | ||||
|     clues, | ||||
|     bosses, | ||||
|   }; | ||||
|   | ||||
| @@ -114,6 +114,7 @@ export type ActivityName = | ||||
|   | 'hunterBH' | ||||
|   | 'rogueBH' | ||||
|   | 'lastManStanding' | ||||
|   | 'soulWarsZeal' | ||||
|   | 'allClues' | ||||
|   | 'beginnerClues' | ||||
|   | 'easyClues' | ||||
| @@ -129,6 +130,7 @@ export interface Stats { | ||||
|   leaguePoints: Activity; | ||||
|   bountyHunter: BH; | ||||
|   lastManStanding: Activity; | ||||
|   soulWarsZeal: Activity; | ||||
|   bosses: Bosses; | ||||
| } | ||||
| export type Modes = { [M in Gamemode]?: Stats }; | ||||
|   | ||||
| @@ -59,7 +59,7 @@ export const CLUES: ClueType[] = [ | ||||
|   'elite', | ||||
|   'master', | ||||
| ]; | ||||
| export const BH_MODES: BHType[] = ['rogue', 'hunter']; | ||||
| export const BH_MODES: BHType[] = ['hunter', 'rogue']; | ||||
| export const GAMEMODES: Gamemode[] = [ | ||||
|   'main', | ||||
|   'ironman', | ||||
| @@ -127,6 +127,7 @@ export const ACTIVITIES: ActivityName[] = [ | ||||
|   'eliteClues', | ||||
|   'masterClues', | ||||
|   'lastManStanding', | ||||
|   'soulWarsZeal', | ||||
|   ...BOSSES, | ||||
| ]; | ||||
|  | ||||
| @@ -236,4 +237,5 @@ export const FORMATTED_BH_NAMES: FormattedBHNames = { | ||||
| }; | ||||
|  | ||||
| export const FORMATTED_LMS = 'Last Man Standing'; | ||||
| export const FORMATTED_SOUL_WARS = 'Soul Wars Zeal'; | ||||
| 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_URL, | ||||
| @@ -45,3 +47,12 @@ export const rsnFromElement = (el: Element | null) => { | ||||
|   const { innerHTML } = el || {}; | ||||
|   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