metacritic-ts 1.0.0 → 1.0.2
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +160 -0
- package/dist/index.cjs +3 -3
- package/dist/index.cjs.map +1 -1
- package/dist/index.d.cts +1 -1
- package/dist/index.d.ts +1 -1
- package/dist/index.js +3 -3
- package/dist/index.js.map +1 -1
- package/package.json +1 -1
package/README.md
ADDED
|
@@ -0,0 +1,160 @@
|
|
|
1
|
+
# metacritic-ts
|
|
2
|
+
[](https://github.com/Deadlock-too/metacritic-ts)
|
|
3
|
+
[](https://www.npmjs.com/package/metacritic-ts)
|
|
4
|
+
[](https://www.npmjs.com/package/metacritic-ts)
|
|
5
|
+
|
|
6
|
+
A TypeScript library for interacting with the Metacritic website API. Easily search for games, movies and tv shows and retrieve their ratings Metacritic rating.
|
|
7
|
+
|
|
8
|
+
This library takes inspiration from another library of mine [howlongtobeat-ts](https://github.com/Deadlock-too/howlongtobeat-ts).
|
|
9
|
+
|
|
10
|
+
> ⚠️ **Disclaimer:** This library is not an official API and is not affiliated nor endorsed with Metacritic.com or Fandom Inc
|
|
11
|
+
> in any way. Please use this library responsibly and do not abuse or overload the Metacritic servers. Use at your own risk.
|
|
12
|
+
|
|
13
|
+
## Features
|
|
14
|
+
|
|
15
|
+
- Search for games, movies and tv shows on Metacritic
|
|
16
|
+
- Retrieve rating data for games, movies and tv shows
|
|
17
|
+
- TypeScript support with full type definitions
|
|
18
|
+
|
|
19
|
+
## Installation
|
|
20
|
+
|
|
21
|
+
Install the library via npm:
|
|
22
|
+
```bash
|
|
23
|
+
npm install metacritic-ts
|
|
24
|
+
```
|
|
25
|
+
|
|
26
|
+
## Usage
|
|
27
|
+
|
|
28
|
+
```typescript
|
|
29
|
+
import { MetacriticService, RecordType } from 'metacritic-ts';
|
|
30
|
+
|
|
31
|
+
/**
|
|
32
|
+
* Example usage of the MetacriticService to search for a keyword
|
|
33
|
+
*/
|
|
34
|
+
async function search() {
|
|
35
|
+
const metacriticService = new MetacriticService();
|
|
36
|
+
|
|
37
|
+
// Search for a game
|
|
38
|
+
const results = await metacriticService.search('The Last of Us');
|
|
39
|
+
|
|
40
|
+
// This will result in a list of games, movies or tv shows
|
|
41
|
+
if (results) {
|
|
42
|
+
console.log('Search results:', results);
|
|
43
|
+
}
|
|
44
|
+
}
|
|
45
|
+
|
|
46
|
+
/**
|
|
47
|
+
* Example usage of the MetacriticService to get the details of a game
|
|
48
|
+
*/
|
|
49
|
+
async function getDetail() {
|
|
50
|
+
const metacriticService = new MetacriticService();
|
|
51
|
+
|
|
52
|
+
// Get the details of a game
|
|
53
|
+
const details = await metacriticService.getDetail('The Last of Us Part II', RecordType.GAME);
|
|
54
|
+
|
|
55
|
+
// This will result in the details of the game, including the rating
|
|
56
|
+
if (details) {
|
|
57
|
+
console.log('Game details:', details);
|
|
58
|
+
}
|
|
59
|
+
}
|
|
60
|
+
|
|
61
|
+
await search();
|
|
62
|
+
await getDetails();
|
|
63
|
+
```
|
|
64
|
+
|
|
65
|
+
## API
|
|
66
|
+
|
|
67
|
+
### `MetacriticService`
|
|
68
|
+
|
|
69
|
+
The main service class for interacting with the Metacritic website.
|
|
70
|
+
|
|
71
|
+
#### Constructor
|
|
72
|
+
|
|
73
|
+
- `constructor(minSimilarity: number = 0.5)`: Creates an instance of the MetacriticService class.
|
|
74
|
+
- `minSimilarity`: Optional parameter to set the minimum similarity threshold for search results to not be filtered out (Default: 0.5).
|
|
75
|
+
|
|
76
|
+
#### Methods
|
|
77
|
+
- `async search(searchKey: string): Promise<MetacriticSearchEntry[]>`: Searches for games, movies or tv shows matching the provided search key.
|
|
78
|
+
- `searchKey`: The title to search for
|
|
79
|
+
- `recordType`: Optional record type to adjust search behavior, it hasn't a default value and will search for all types.
|
|
80
|
+
- `sortBySimilarity`: Optional boolean to sort the results by similarity to the search key (Default: true). If set to false, the results will leave to the order returned by the Metacritic API.
|
|
81
|
+
|
|
82
|
+
- `async getDetail(title: string, recordType: RecordType): Promise<MetacriticEntry | null>`: Retrieves the details of a game, movie or tv show matching the provided title.
|
|
83
|
+
- `title`: The title to search for
|
|
84
|
+
- `recordType`: The type of record to retrieve (game, movie or tv show)
|
|
85
|
+
- `sortBySimilarity`: Optional boolean to sort the results by similarity to the search key (Default: true). Pay attention that this parameter is very important for the `getDetail` method, if set to false, the results will leave to the order returned by the Metacritic API and the first result may not be the one you are looking for.
|
|
86
|
+
|
|
87
|
+
### `RecordType`
|
|
88
|
+
An enum representing the type of record to retrieve.
|
|
89
|
+
- `TVShow`: Record type for TV shows
|
|
90
|
+
- `Movie`: Record type for movies
|
|
91
|
+
- `Game`: Record type for games
|
|
92
|
+
|
|
93
|
+
### `MetacriticSearchEntry`
|
|
94
|
+
An interface representing a search entry returned by the MetacriticService.
|
|
95
|
+
- `id`: The unique identifier for the record.
|
|
96
|
+
- `recordType`: The type of record (game, movie or tv show).
|
|
97
|
+
- `title`: The title of the record.
|
|
98
|
+
- `slug`: The slug of the record.
|
|
99
|
+
- `must`: A boolean indicating if the record is a must-see, must-play or must-watch.
|
|
100
|
+
- `criticScore`: The critic score of the record.
|
|
101
|
+
- `similarity`: A computed value that indicates how similar the record is to the search term.
|
|
102
|
+
|
|
103
|
+
### `MetacriticEntry`
|
|
104
|
+
An interface representing a record entry returned by the MetacriticService.
|
|
105
|
+
- `id`: The unique identifier for the record.
|
|
106
|
+
- `recordType`: The type of record (game, movie or tv show).
|
|
107
|
+
- `title`: The title of the record.
|
|
108
|
+
- `slug`: The slug of the record.
|
|
109
|
+
- `must`: A boolean indicating if the record is a must-see, must-play or must-watch.
|
|
110
|
+
- `criticScore`:
|
|
111
|
+
- `score`: The critic score of the record.
|
|
112
|
+
- `maxScore`: The maximum critic score.
|
|
113
|
+
- `count`: The number of critic reviews.
|
|
114
|
+
- `sentiment`: The sentiment of the critic reviews.
|
|
115
|
+
- `count`:
|
|
116
|
+
- `positive`: The number of positive reviews.
|
|
117
|
+
- `negative`: The number of negative reviews.
|
|
118
|
+
- `neutral`: The number of neutral reviews.
|
|
119
|
+
- `total`: The total number of reviews.
|
|
120
|
+
- `userScore`:
|
|
121
|
+
- `score`: The user score of the record.
|
|
122
|
+
- `maxScore`: The maximum user score.
|
|
123
|
+
- `count`: The number of user reviews.
|
|
124
|
+
- `sentiment`: The sentiment of the user reviews.
|
|
125
|
+
- `count`:
|
|
126
|
+
- `positive`: The number of positive reviews.
|
|
127
|
+
- `negative`: The number of negative reviews.
|
|
128
|
+
- `neutral`: The number of neutral reviews.
|
|
129
|
+
- `total`: The total number of reviews.
|
|
130
|
+
|
|
131
|
+
## Development
|
|
132
|
+
|
|
133
|
+
### Prerequisites
|
|
134
|
+
|
|
135
|
+
- Node.js
|
|
136
|
+
- npm or yarn
|
|
137
|
+
|
|
138
|
+
### Setup
|
|
139
|
+
|
|
140
|
+
```bash
|
|
141
|
+
# Clone the repository
|
|
142
|
+
git clone https://github.com/Deadlock-too/metacritic-ts.git
|
|
143
|
+
|
|
144
|
+
# Install dependencies
|
|
145
|
+
cd metacritic-ts
|
|
146
|
+
npm install
|
|
147
|
+
|
|
148
|
+
# Build the project
|
|
149
|
+
npm run build
|
|
150
|
+
|
|
151
|
+
# Run tests
|
|
152
|
+
npm test
|
|
153
|
+
```
|
|
154
|
+
|
|
155
|
+
## Issues, Questions & Discussions
|
|
156
|
+
If you found a bug, report it as soon as possible creating an [issue](https://github.com/Deadlock-too/metacritic-ts/issues/new), the code is not perfect for sure, and I will be happy to fix it.
|
|
157
|
+
If you need any new feature, or want to discuss the current implementation/features, consider opening a [discussion](https://github.com/Deadlock-too/metacritic-ts/discussions/) or even propose a change with a [Pull Request](https://github.com/Deadlock-too/metacritic-ts/pulls).
|
|
158
|
+
|
|
159
|
+
## License
|
|
160
|
+
This project is licensed under the MIT License - see the <a href="https://github.com/Deadlock-too/metacritic-ts/blob/main/LICENSE" target="_blank">LICENSE</a> file for details.
|
package/dist/index.cjs
CHANGED
|
@@ -81,7 +81,7 @@ function parseSearchJsonResult(jsonString, searchKey, minSimilarity, sortBySimil
|
|
|
81
81
|
}
|
|
82
82
|
entries.push(entry);
|
|
83
83
|
}
|
|
84
|
-
return entries.sort((a, b) => b.similarity - a.similarity);
|
|
84
|
+
return sortBySimilarity ? entries.sort((a, b) => b.similarity - a.similarity) : entries;
|
|
85
85
|
} catch (error) {
|
|
86
86
|
console.error("Error parsing JSON:", error);
|
|
87
87
|
return [];
|
|
@@ -185,7 +185,7 @@ var _MetacriticService = class _MetacriticService {
|
|
|
185
185
|
}
|
|
186
186
|
const jsonResult = await _MetacriticService.sendSearchWebRequest(searchKey, this.apiKey, recordType);
|
|
187
187
|
if (jsonResult) {
|
|
188
|
-
return parseSearchJsonResult(jsonResult, searchKey, this.minSimilarity);
|
|
188
|
+
return parseSearchJsonResult(jsonResult, searchKey, this.minSimilarity, sortBySimilarity);
|
|
189
189
|
}
|
|
190
190
|
return [];
|
|
191
191
|
}
|
|
@@ -202,7 +202,7 @@ var _MetacriticService = class _MetacriticService {
|
|
|
202
202
|
return null;
|
|
203
203
|
}
|
|
204
204
|
}
|
|
205
|
-
const searchResult = await this.search(searchKey, recordType);
|
|
205
|
+
const searchResult = await this.search(searchKey, recordType, sortBySimilarity);
|
|
206
206
|
if (searchResult && searchResult.length > 0) {
|
|
207
207
|
const detailResult = await _MetacriticService.sendDetailWebRequest(searchResult[0].slug, this.apiKey, recordType);
|
|
208
208
|
if (detailResult) {
|
package/dist/index.cjs.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"sources":["../src/index.ts","../src/lib/utils.ts","../src/lib/parser.ts","../src/lib/service.ts","../src/lib/types.ts"],"sourcesContent":["export { MetacriticService } from './lib/service'\r\nexport * from './lib/types'\r\n","export function getSimilarity(a: string, b: string): number {\r\n if (a === b) return 1\r\n if (!a || !b) return 0\r\n\r\n const str1 = a.toLowerCase()\r\n const str2 = b.toLowerCase()\r\n const distance = levenshteinDistance(str1, str2)\r\n const maxLength = Math.max(str1.length, str2.length)\r\n return (maxLength - distance) / maxLength\r\n}\r\n\r\nfunction levenshteinDistance(a: string, b: string): number {\r\n const matrix: number[][] = []\r\n for (let i = 0; i <= b.length; i++) {\r\n matrix[i] = [i]\r\n }\r\n for (let j = 0; j <= a.length; j++) {\r\n matrix[0][j] = j\r\n }\r\n for (let i = 1; i <= b.length; i++) {\r\n for (let j = 1; j <= a.length; j++) {\r\n if (b.charAt(i - 1) === a.charAt(j - 1)) {\r\n matrix[i][j] = matrix[i - 1][j - 1]\r\n } else {\r\n matrix[i][j] = Math.min(\r\n matrix[i - 1][j - 1] + 1,\r\n Math.min(matrix[i][j - 1] + 1, matrix[i - 1][j] + 1)\r\n )\r\n }\r\n }\r\n }\r\n return matrix[b.length][a.length]\r\n}\r\n","import { getSimilarity } from './utils'\r\nimport { MetacriticEntry, RecordType, MetacriticSearchEntry } from './types'\r\n\r\nexport function parseSearchJsonResult(jsonString: string, searchKey: string, minSimilarity: number, sortBySimilarity: boolean = true): MetacriticSearchEntry[] {\r\n try {\r\n const parsedData = JSON.parse(jsonString)\r\n const entries: MetacriticSearchEntry[] = []\r\n\r\n const itemsList = parsedData.components.find((component: any) => component.meta.componentName === 'search').data.items\r\n for (const item of itemsList) {\r\n const entry = mapSearchEntry(item, searchKey)\r\n\r\n if (entry.similarity < minSimilarity) {\r\n continue\r\n }\r\n entries.push(entry)\r\n }\r\n\r\n return entries.sort((a, b) => b.similarity - a.similarity)\r\n } catch (error) {\r\n console.error('Error parsing JSON:', error)\r\n return []\r\n }\r\n}\r\n\r\nexport function parseDetailJsonResult(jsonString: string, must: boolean): MetacriticEntry | null {\r\n try {\r\n const parsedData = JSON.parse(jsonString)\r\n return mapDetailEntry(parsedData, must)\r\n } catch (error) {\r\n console.error('Error parsing JSON:', error)\r\n return null\r\n }\r\n}\r\n\r\nfunction mapSearchEntry(data: any, searchKey: string): MetacriticSearchEntry {\r\n return {\r\n id: data.id,\r\n recordType: data.typeId as RecordType,\r\n title: data.title,\r\n slug: data.slug,\r\n criticScore: data.criticScoreSummary.score,\r\n must: data.mustSee || data.mustWatch || data.mustPlay,\r\n similarity: getSimilarity(data.title, searchKey)\r\n }\r\n}\r\n\r\nfunction mapDetailEntry(data: any, must: boolean): MetacriticEntry {\r\n const product = data.components.find((component: any) => component.meta.componentName === 'product').data.item\r\n const criticScoreSummary = data.components.find((component: any) => component.meta.componentName === 'critic-score-summary').data.item\r\n const userScoreSummary = data.components.find((component: any) => component.meta.componentName === 'user-score-summary').data.item\r\n\r\n return {\r\n id: product.id,\r\n recordType: product.typeId as RecordType,\r\n title: product.title,\r\n slug: product.slug,\r\n must: must,\r\n userScore: {\r\n score: userScoreSummary.score,\r\n maxScore: userScoreSummary.max,\r\n sentiment: userScoreSummary.sentiment,\r\n count: {\r\n positive: userScoreSummary.positiveCount,\r\n neutral: userScoreSummary.neutralCount,\r\n negative: userScoreSummary.negativeCount,\r\n total: userScoreSummary.reviewCount,\r\n }\r\n },\r\n criticScore: {\r\n score: criticScoreSummary.score,\r\n maxScore: criticScoreSummary.max,\r\n sentiment: criticScoreSummary.sentiment,\r\n count: {\r\n positive: criticScoreSummary.positiveCount,\r\n neutral: criticScoreSummary.neutralCount,\r\n negative: criticScoreSummary.negativeCount,\r\n total: criticScoreSummary.reviewCount,\r\n }\r\n },\r\n }\r\n}\r\n","import { parseDetailJsonResult, parseSearchJsonResult } from './parser'\r\nimport UserAgent from 'user-agents'\r\nimport { MetacriticEntry, RecordType, MetacriticSearchEntry } from './types'\r\n\r\nclass SearchInformations {\r\n apiKey: string\r\n\r\n constructor(scriptContent: string) {\r\n this.apiKey = this.extractApiFromScript(scriptContent)\r\n }\r\n\r\n extractApiFromScript(scriptContent: any) {\r\n const apiKeyPattern = /apiKey=([^\"&]+)/\r\n let matches = scriptContent.match(apiKeyPattern)\r\n if (matches) {\r\n return matches[1]\r\n }\r\n }\r\n}\r\n\r\nexport class MetacriticService {\r\n private minSimilarity: number\r\n private apiKey?: string\r\n\r\n static REFERER_HEADER = 'https://www.metacritic.com/'\r\n static HOMEPAGE_URL = 'https://www.metacritic.com/'\r\n static BASE_URL = 'https://backend.metacritic.com/composer/metacritic/pages/'\r\n static SEARCH_URL = MetacriticService.BASE_URL + 'search/'\r\n\r\n constructor(minSimilarity: number = 0.5) {\r\n this.minSimilarity = minSimilarity\r\n }\r\n\r\n async search(searchKey: string, recordType?: RecordType, sortBySimilarity: boolean = true): Promise<MetacriticSearchEntry[]> {\r\n if (!searchKey) {\r\n return []\r\n }\r\n\r\n if (!this.apiKey) {\r\n const searchInfo = await MetacriticService.sendApiKeyRetrievalWebsiteRequest()\r\n if (searchInfo) {\r\n this.apiKey = searchInfo.apiKey\r\n } else {\r\n console.error('Failed to retrieve API key')\r\n return []\r\n }\r\n }\r\n\r\n const jsonResult = await MetacriticService.sendSearchWebRequest(searchKey, this.apiKey, recordType)\r\n if (jsonResult) {\r\n return parseSearchJsonResult(jsonResult, searchKey, this.minSimilarity)\r\n }\r\n\r\n return []\r\n }\r\n\r\n async getDetail(searchKey: string, recordType: RecordType, sortBySimilarity: boolean = true): Promise<MetacriticEntry | null> {\r\n if (!searchKey) {\r\n return null\r\n }\r\n\r\n if (!this.apiKey) {\r\n const searchInfo = await MetacriticService.sendApiKeyRetrievalWebsiteRequest()\r\n if (searchInfo) {\r\n this.apiKey = searchInfo.apiKey\r\n } else {\r\n console.error('Failed to retrieve API key')\r\n return null\r\n }\r\n }\r\n\r\n const searchResult = await this.search(searchKey, recordType)\r\n if (searchResult && searchResult.length > 0) {\r\n const detailResult = await MetacriticService.sendDetailWebRequest(searchResult[0].slug, this.apiKey, recordType)\r\n\r\n if (detailResult) {\r\n return parseDetailJsonResult(detailResult, searchResult[0].must)\r\n }\r\n }\r\n\r\n return null\r\n }\r\n\r\n static async sendSearchWebRequest(searchKey: string, apiKey: string, recordType?: RecordType) {\r\n const headers = this.getRequestHeaders()\r\n\r\n const searchUrl = new URL(MetacriticService.SEARCH_URL + encodeURI(searchKey) + '/web')\r\n\r\n searchUrl.searchParams.append('apiKey', apiKey)\r\n\r\n if (recordType) {\r\n searchUrl.searchParams.append('mcoTypeId', recordType.toString())\r\n }\r\n\r\n try {\r\n const response = await fetch(searchUrl, {\r\n headers: headers,\r\n method: 'GET',\r\n signal: AbortSignal.timeout(60000)\r\n })\r\n\r\n if (response.ok) {\r\n return await response.text()\r\n }\r\n } catch (error) {\r\n console.error('Error fetching search results:', error)\r\n }\r\n }\r\n\r\n static async sendDetailWebRequest(slug: string, apiKey: string, recordType: RecordType) {\r\n const headers = this.getRequestHeaders()\r\n\r\n let baseUrl = MetacriticService.BASE_URL\r\n switch (recordType) {\r\n case RecordType.Game:\r\n baseUrl = baseUrl + 'games/'\r\n break\r\n case RecordType.Movie:\r\n baseUrl = baseUrl + 'movies/'\r\n break\r\n case RecordType.TVShow:\r\n baseUrl = baseUrl + 'shows/'\r\n break\r\n default:\r\n console.error('Unsupported record type for detail request:', recordType)\r\n return\r\n }\r\n\r\n const detailUrl = new URL(baseUrl + encodeURI(slug) + '/web')\r\n\r\n detailUrl.searchParams.append('apiKey', apiKey)\r\n\r\n try {\r\n const response = await fetch(detailUrl, {\r\n headers: headers,\r\n method: 'GET',\r\n signal: AbortSignal.timeout(60000)\r\n })\r\n\r\n if (response.ok) {\r\n return await response.text()\r\n }\r\n } catch (error) {\r\n console.error('Error fetching detail result:', error)\r\n }\r\n }\r\n\r\n static getRequestHeaders() {\r\n const minimumHeaders = this.getMinimalRequestHeaders()\r\n return {\r\n ...minimumHeaders,\r\n 'Content-Type': 'application/json',\r\n 'Accept': '*/*',\r\n 'Referer': MetacriticService.REFERER_HEADER,\r\n }\r\n }\r\n\r\n static getMinimalRequestHeaders() {\r\n const userAgent = new UserAgent()\r\n return {\r\n 'User-Agent': userAgent.toString(),\r\n }\r\n }\r\n\r\n static async sendApiKeyRetrievalWebsiteRequest(): Promise<SearchInformations | null> {\r\n const headers = this.getMinimalRequestHeaders()\r\n try {\r\n const response = await fetch(MetacriticService.HOMEPAGE_URL, {\r\n headers: headers,\r\n signal: AbortSignal.timeout(60000)\r\n })\r\n\r\n if (response.ok) {\r\n const html = await response.text()\r\n\r\n const scriptPattern = /<script[^>]*>([\\s\\S]*?)<\\/script>/g\r\n const scripts: string[] = []\r\n let match\r\n while ((match = scriptPattern.exec(html)) !== null) {\r\n scripts.push(match[1])\r\n }\r\n\r\n for (const script of scripts) {\r\n const searchInfo = new SearchInformations(script)\r\n if (searchInfo.apiKey) {\r\n return searchInfo\r\n }\r\n }\r\n }\r\n\r\n return null\r\n } catch (error) {\r\n console.error('Error fetching website:', error)\r\n return null\r\n }\r\n }\r\n}\r\n","export enum RecordType {\r\n TVShow = 1,\r\n Movie = 2,\r\n Game = 13,\r\n}\r\n\r\nexport type MetacriticSearchEntry = {\r\n id: number\r\n recordType: RecordType\r\n title: string\r\n slug: string\r\n must: boolean\r\n criticScore: number\r\n similarity: number\r\n}\r\n\r\nexport type MetacriticEntry = {\r\n id: number\r\n recordType: RecordType\r\n title: string\r\n slug: string\r\n must: boolean\r\n userScore: Score\r\n criticScore: Score\r\n}\r\n\r\ntype Score = {\r\n score: number\r\n maxScore: number\r\n sentiment: string\r\n count: {\r\n positive: number\r\n neutral: number\r\n negative: number\r\n total: number\r\n }\r\n}\r\n"],"mappings":"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","names":["RecordType","UserAgent"]}
|
|
1
|
+
{"version":3,"sources":["../src/index.ts","../src/lib/utils.ts","../src/lib/parser.ts","../src/lib/service.ts","../src/lib/types.ts"],"sourcesContent":["export { MetacriticService } from './lib/service'\r\nexport * from './lib/types'\r\n","export function getSimilarity(a: string, b: string): number {\r\n if (a === b) return 1\r\n if (!a || !b) return 0\r\n\r\n const str1 = a.toLowerCase()\r\n const str2 = b.toLowerCase()\r\n const distance = levenshteinDistance(str1, str2)\r\n const maxLength = Math.max(str1.length, str2.length)\r\n return (maxLength - distance) / maxLength\r\n}\r\n\r\nfunction levenshteinDistance(a: string, b: string): number {\r\n const matrix: number[][] = []\r\n for (let i = 0; i <= b.length; i++) {\r\n matrix[i] = [i]\r\n }\r\n for (let j = 0; j <= a.length; j++) {\r\n matrix[0][j] = j\r\n }\r\n for (let i = 1; i <= b.length; i++) {\r\n for (let j = 1; j <= a.length; j++) {\r\n if (b.charAt(i - 1) === a.charAt(j - 1)) {\r\n matrix[i][j] = matrix[i - 1][j - 1]\r\n } else {\r\n matrix[i][j] = Math.min(\r\n matrix[i - 1][j - 1] + 1,\r\n Math.min(matrix[i][j - 1] + 1, matrix[i - 1][j] + 1)\r\n )\r\n }\r\n }\r\n }\r\n return matrix[b.length][a.length]\r\n}\r\n","import { getSimilarity } from './utils'\r\nimport { MetacriticEntry, RecordType, MetacriticSearchEntry } from './types'\r\n\r\nexport function parseSearchJsonResult(jsonString: string, searchKey: string, minSimilarity: number, sortBySimilarity: boolean = true): MetacriticSearchEntry[] {\r\n try {\r\n const parsedData = JSON.parse(jsonString)\r\n const entries: MetacriticSearchEntry[] = []\r\n\r\n const itemsList = parsedData.components.find((component: any) => component.meta.componentName === 'search').data.items\r\n for (const item of itemsList) {\r\n const entry = mapSearchEntry(item, searchKey)\r\n\r\n if (entry.similarity < minSimilarity) {\r\n continue\r\n }\r\n entries.push(entry)\r\n }\r\n\r\n return sortBySimilarity ? entries.sort((a, b) => b.similarity - a.similarity) : entries\r\n } catch (error) {\r\n console.error('Error parsing JSON:', error)\r\n return []\r\n }\r\n}\r\n\r\nexport function parseDetailJsonResult(jsonString: string, must: boolean): MetacriticEntry | null {\r\n try {\r\n const parsedData = JSON.parse(jsonString)\r\n return mapDetailEntry(parsedData, must)\r\n } catch (error) {\r\n console.error('Error parsing JSON:', error)\r\n return null\r\n }\r\n}\r\n\r\nfunction mapSearchEntry(data: any, searchKey: string): MetacriticSearchEntry {\r\n return {\r\n id: data.id,\r\n recordType: data.typeId as RecordType,\r\n title: data.title,\r\n slug: data.slug,\r\n criticScore: data.criticScoreSummary.score,\r\n must: data.mustSee || data.mustWatch || data.mustPlay,\r\n similarity: getSimilarity(data.title, searchKey)\r\n }\r\n}\r\n\r\nfunction mapDetailEntry(data: any, must: boolean): MetacriticEntry {\r\n const product = data.components.find((component: any) => component.meta.componentName === 'product').data.item\r\n const criticScoreSummary = data.components.find((component: any) => component.meta.componentName === 'critic-score-summary').data.item\r\n const userScoreSummary = data.components.find((component: any) => component.meta.componentName === 'user-score-summary').data.item\r\n\r\n return {\r\n id: product.id,\r\n recordType: product.typeId as RecordType,\r\n title: product.title,\r\n slug: product.slug,\r\n must: must,\r\n userScore: {\r\n score: userScoreSummary.score,\r\n maxScore: userScoreSummary.max,\r\n sentiment: userScoreSummary.sentiment,\r\n count: {\r\n positive: userScoreSummary.positiveCount,\r\n neutral: userScoreSummary.neutralCount,\r\n negative: userScoreSummary.negativeCount,\r\n total: userScoreSummary.reviewCount,\r\n }\r\n },\r\n criticScore: {\r\n score: criticScoreSummary.score,\r\n maxScore: criticScoreSummary.max,\r\n sentiment: criticScoreSummary.sentiment,\r\n count: {\r\n positive: criticScoreSummary.positiveCount,\r\n neutral: criticScoreSummary.neutralCount,\r\n negative: criticScoreSummary.negativeCount,\r\n total: criticScoreSummary.reviewCount,\r\n }\r\n },\r\n }\r\n}\r\n","import { parseDetailJsonResult, parseSearchJsonResult } from './parser'\r\nimport UserAgent from 'user-agents'\r\nimport { MetacriticEntry, RecordType, MetacriticSearchEntry } from './types'\r\n\r\nclass SearchInformations {\r\n apiKey: string\r\n\r\n constructor(scriptContent: string) {\r\n this.apiKey = this.extractApiFromScript(scriptContent)\r\n }\r\n\r\n extractApiFromScript(scriptContent: any) {\r\n const apiKeyPattern = /apiKey=([^\"&]+)/\r\n let matches = scriptContent.match(apiKeyPattern)\r\n if (matches) {\r\n return matches[1]\r\n }\r\n }\r\n}\r\n\r\nexport class MetacriticService {\r\n private minSimilarity: number\r\n private apiKey?: string\r\n\r\n static REFERER_HEADER = 'https://www.metacritic.com/'\r\n static HOMEPAGE_URL = 'https://www.metacritic.com/'\r\n static BASE_URL = 'https://backend.metacritic.com/composer/metacritic/pages/'\r\n static SEARCH_URL = MetacriticService.BASE_URL + 'search/'\r\n\r\n constructor(minSimilarity: number = 0.5) {\r\n this.minSimilarity = minSimilarity\r\n }\r\n\r\n async search(searchKey: string, recordType?: RecordType, sortBySimilarity: boolean = true): Promise<MetacriticSearchEntry[]> {\r\n if (!searchKey) {\r\n return []\r\n }\r\n\r\n if (!this.apiKey) {\r\n const searchInfo = await MetacriticService.sendApiKeyRetrievalWebsiteRequest()\r\n if (searchInfo) {\r\n this.apiKey = searchInfo.apiKey\r\n } else {\r\n console.error('Failed to retrieve API key')\r\n return []\r\n }\r\n }\r\n\r\n const jsonResult = await MetacriticService.sendSearchWebRequest(searchKey, this.apiKey, recordType)\r\n if (jsonResult) {\r\n return parseSearchJsonResult(jsonResult, searchKey, this.minSimilarity, sortBySimilarity)\r\n }\r\n\r\n return []\r\n }\r\n\r\n async getDetail(searchKey: string, recordType: RecordType, sortBySimilarity: boolean = true): Promise<MetacriticEntry | null> {\r\n if (!searchKey) {\r\n return null\r\n }\r\n\r\n if (!this.apiKey) {\r\n const searchInfo = await MetacriticService.sendApiKeyRetrievalWebsiteRequest()\r\n if (searchInfo) {\r\n this.apiKey = searchInfo.apiKey\r\n } else {\r\n console.error('Failed to retrieve API key')\r\n return null\r\n }\r\n }\r\n\r\n const searchResult = await this.search(searchKey, recordType, sortBySimilarity)\r\n if (searchResult && searchResult.length > 0) {\r\n const detailResult = await MetacriticService.sendDetailWebRequest(searchResult[0].slug, this.apiKey, recordType)\r\n\r\n if (detailResult) {\r\n return parseDetailJsonResult(detailResult, searchResult[0].must)\r\n }\r\n }\r\n\r\n return null\r\n }\r\n\r\n static async sendSearchWebRequest(searchKey: string, apiKey: string, recordType?: RecordType) {\r\n const headers = this.getRequestHeaders()\r\n\r\n const searchUrl = new URL(MetacriticService.SEARCH_URL + encodeURI(searchKey) + '/web')\r\n\r\n searchUrl.searchParams.append('apiKey', apiKey)\r\n\r\n if (recordType) {\r\n searchUrl.searchParams.append('mcoTypeId', recordType.toString())\r\n }\r\n\r\n try {\r\n const response = await fetch(searchUrl, {\r\n headers: headers,\r\n method: 'GET',\r\n signal: AbortSignal.timeout(60000)\r\n })\r\n\r\n if (response.ok) {\r\n return await response.text()\r\n }\r\n } catch (error) {\r\n console.error('Error fetching search results:', error)\r\n }\r\n }\r\n\r\n static async sendDetailWebRequest(slug: string, apiKey: string, recordType: RecordType) {\r\n const headers = this.getRequestHeaders()\r\n\r\n let baseUrl = MetacriticService.BASE_URL\r\n switch (recordType) {\r\n case RecordType.Game:\r\n baseUrl = baseUrl + 'games/'\r\n break\r\n case RecordType.Movie:\r\n baseUrl = baseUrl + 'movies/'\r\n break\r\n case RecordType.TVShow:\r\n baseUrl = baseUrl + 'shows/'\r\n break\r\n default:\r\n console.error('Unsupported record type for detail request:', recordType)\r\n return\r\n }\r\n\r\n const detailUrl = new URL(baseUrl + encodeURI(slug) + '/web')\r\n\r\n detailUrl.searchParams.append('apiKey', apiKey)\r\n\r\n try {\r\n const response = await fetch(detailUrl, {\r\n headers: headers,\r\n method: 'GET',\r\n signal: AbortSignal.timeout(60000)\r\n })\r\n\r\n if (response.ok) {\r\n return await response.text()\r\n }\r\n } catch (error) {\r\n console.error('Error fetching detail result:', error)\r\n }\r\n }\r\n\r\n static getRequestHeaders() {\r\n const minimumHeaders = this.getMinimalRequestHeaders()\r\n return {\r\n ...minimumHeaders,\r\n 'Content-Type': 'application/json',\r\n 'Accept': '*/*',\r\n 'Referer': MetacriticService.REFERER_HEADER,\r\n }\r\n }\r\n\r\n static getMinimalRequestHeaders() {\r\n const userAgent = new UserAgent()\r\n return {\r\n 'User-Agent': userAgent.toString(),\r\n }\r\n }\r\n\r\n static async sendApiKeyRetrievalWebsiteRequest(): Promise<SearchInformations | null> {\r\n const headers = this.getMinimalRequestHeaders()\r\n try {\r\n const response = await fetch(MetacriticService.HOMEPAGE_URL, {\r\n headers: headers,\r\n signal: AbortSignal.timeout(60000)\r\n })\r\n\r\n if (response.ok) {\r\n const html = await response.text()\r\n\r\n const scriptPattern = /<script[^>]*>([\\s\\S]*?)<\\/script>/g\r\n const scripts: string[] = []\r\n let match\r\n while ((match = scriptPattern.exec(html)) !== null) {\r\n scripts.push(match[1])\r\n }\r\n\r\n for (const script of scripts) {\r\n const searchInfo = new SearchInformations(script)\r\n if (searchInfo.apiKey) {\r\n return searchInfo\r\n }\r\n }\r\n }\r\n\r\n return null\r\n } catch (error) {\r\n console.error('Error fetching website:', error)\r\n return null\r\n }\r\n }\r\n}\r\n","export enum RecordType {\r\n TVShow = 1,\r\n Movie = 2,\r\n Game = 13,\r\n}\r\n\r\nexport type MetacriticSearchEntry = {\r\n id: number\r\n recordType: RecordType\r\n title: string\r\n slug: string\r\n must: boolean\r\n criticScore: number\r\n similarity: number\r\n}\r\n\r\nexport type MetacriticEntry = {\r\n id: number\r\n recordType: RecordType\r\n title: string\r\n slug: string\r\n must: boolean\r\n userScore: Score\r\n criticScore: Score\r\n}\r\n\r\nexport type Score = {\r\n score: number\r\n maxScore: number\r\n sentiment: string\r\n count: {\r\n positive: number\r\n neutral: number\r\n negative: number\r\n total: number\r\n }\r\n}\r\n"],"mappings":"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","names":["RecordType","UserAgent"]}
|
package/dist/index.d.cts
CHANGED
|
@@ -62,4 +62,4 @@ declare class MetacriticService {
|
|
|
62
62
|
static sendApiKeyRetrievalWebsiteRequest(): Promise<SearchInformations | null>;
|
|
63
63
|
}
|
|
64
64
|
|
|
65
|
-
export { type MetacriticEntry, type MetacriticSearchEntry, MetacriticService, RecordType };
|
|
65
|
+
export { type MetacriticEntry, type MetacriticSearchEntry, MetacriticService, RecordType, type Score };
|
package/dist/index.d.ts
CHANGED
|
@@ -62,4 +62,4 @@ declare class MetacriticService {
|
|
|
62
62
|
static sendApiKeyRetrievalWebsiteRequest(): Promise<SearchInformations | null>;
|
|
63
63
|
}
|
|
64
64
|
|
|
65
|
-
export { type MetacriticEntry, type MetacriticSearchEntry, MetacriticService, RecordType };
|
|
65
|
+
export { type MetacriticEntry, type MetacriticSearchEntry, MetacriticService, RecordType, type Score };
|
package/dist/index.js
CHANGED
|
@@ -44,7 +44,7 @@ function parseSearchJsonResult(jsonString, searchKey, minSimilarity, sortBySimil
|
|
|
44
44
|
}
|
|
45
45
|
entries.push(entry);
|
|
46
46
|
}
|
|
47
|
-
return entries.sort((a, b) => b.similarity - a.similarity);
|
|
47
|
+
return sortBySimilarity ? entries.sort((a, b) => b.similarity - a.similarity) : entries;
|
|
48
48
|
} catch (error) {
|
|
49
49
|
console.error("Error parsing JSON:", error);
|
|
50
50
|
return [];
|
|
@@ -148,7 +148,7 @@ var _MetacriticService = class _MetacriticService {
|
|
|
148
148
|
}
|
|
149
149
|
const jsonResult = await _MetacriticService.sendSearchWebRequest(searchKey, this.apiKey, recordType);
|
|
150
150
|
if (jsonResult) {
|
|
151
|
-
return parseSearchJsonResult(jsonResult, searchKey, this.minSimilarity);
|
|
151
|
+
return parseSearchJsonResult(jsonResult, searchKey, this.minSimilarity, sortBySimilarity);
|
|
152
152
|
}
|
|
153
153
|
return [];
|
|
154
154
|
}
|
|
@@ -165,7 +165,7 @@ var _MetacriticService = class _MetacriticService {
|
|
|
165
165
|
return null;
|
|
166
166
|
}
|
|
167
167
|
}
|
|
168
|
-
const searchResult = await this.search(searchKey, recordType);
|
|
168
|
+
const searchResult = await this.search(searchKey, recordType, sortBySimilarity);
|
|
169
169
|
if (searchResult && searchResult.length > 0) {
|
|
170
170
|
const detailResult = await _MetacriticService.sendDetailWebRequest(searchResult[0].slug, this.apiKey, recordType);
|
|
171
171
|
if (detailResult) {
|
package/dist/index.js.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"sources":["../src/lib/utils.ts","../src/lib/parser.ts","../src/lib/service.ts","../src/lib/types.ts"],"sourcesContent":["export function getSimilarity(a: string, b: string): number {\r\n if (a === b) return 1\r\n if (!a || !b) return 0\r\n\r\n const str1 = a.toLowerCase()\r\n const str2 = b.toLowerCase()\r\n const distance = levenshteinDistance(str1, str2)\r\n const maxLength = Math.max(str1.length, str2.length)\r\n return (maxLength - distance) / maxLength\r\n}\r\n\r\nfunction levenshteinDistance(a: string, b: string): number {\r\n const matrix: number[][] = []\r\n for (let i = 0; i <= b.length; i++) {\r\n matrix[i] = [i]\r\n }\r\n for (let j = 0; j <= a.length; j++) {\r\n matrix[0][j] = j\r\n }\r\n for (let i = 1; i <= b.length; i++) {\r\n for (let j = 1; j <= a.length; j++) {\r\n if (b.charAt(i - 1) === a.charAt(j - 1)) {\r\n matrix[i][j] = matrix[i - 1][j - 1]\r\n } else {\r\n matrix[i][j] = Math.min(\r\n matrix[i - 1][j - 1] + 1,\r\n Math.min(matrix[i][j - 1] + 1, matrix[i - 1][j] + 1)\r\n )\r\n }\r\n }\r\n }\r\n return matrix[b.length][a.length]\r\n}\r\n","import { getSimilarity } from './utils'\r\nimport { MetacriticEntry, RecordType, MetacriticSearchEntry } from './types'\r\n\r\nexport function parseSearchJsonResult(jsonString: string, searchKey: string, minSimilarity: number, sortBySimilarity: boolean = true): MetacriticSearchEntry[] {\r\n try {\r\n const parsedData = JSON.parse(jsonString)\r\n const entries: MetacriticSearchEntry[] = []\r\n\r\n const itemsList = parsedData.components.find((component: any) => component.meta.componentName === 'search').data.items\r\n for (const item of itemsList) {\r\n const entry = mapSearchEntry(item, searchKey)\r\n\r\n if (entry.similarity < minSimilarity) {\r\n continue\r\n }\r\n entries.push(entry)\r\n }\r\n\r\n return entries.sort((a, b) => b.similarity - a.similarity)\r\n } catch (error) {\r\n console.error('Error parsing JSON:', error)\r\n return []\r\n }\r\n}\r\n\r\nexport function parseDetailJsonResult(jsonString: string, must: boolean): MetacriticEntry | null {\r\n try {\r\n const parsedData = JSON.parse(jsonString)\r\n return mapDetailEntry(parsedData, must)\r\n } catch (error) {\r\n console.error('Error parsing JSON:', error)\r\n return null\r\n }\r\n}\r\n\r\nfunction mapSearchEntry(data: any, searchKey: string): MetacriticSearchEntry {\r\n return {\r\n id: data.id,\r\n recordType: data.typeId as RecordType,\r\n title: data.title,\r\n slug: data.slug,\r\n criticScore: data.criticScoreSummary.score,\r\n must: data.mustSee || data.mustWatch || data.mustPlay,\r\n similarity: getSimilarity(data.title, searchKey)\r\n }\r\n}\r\n\r\nfunction mapDetailEntry(data: any, must: boolean): MetacriticEntry {\r\n const product = data.components.find((component: any) => component.meta.componentName === 'product').data.item\r\n const criticScoreSummary = data.components.find((component: any) => component.meta.componentName === 'critic-score-summary').data.item\r\n const userScoreSummary = data.components.find((component: any) => component.meta.componentName === 'user-score-summary').data.item\r\n\r\n return {\r\n id: product.id,\r\n recordType: product.typeId as RecordType,\r\n title: product.title,\r\n slug: product.slug,\r\n must: must,\r\n userScore: {\r\n score: userScoreSummary.score,\r\n maxScore: userScoreSummary.max,\r\n sentiment: userScoreSummary.sentiment,\r\n count: {\r\n positive: userScoreSummary.positiveCount,\r\n neutral: userScoreSummary.neutralCount,\r\n negative: userScoreSummary.negativeCount,\r\n total: userScoreSummary.reviewCount,\r\n }\r\n },\r\n criticScore: {\r\n score: criticScoreSummary.score,\r\n maxScore: criticScoreSummary.max,\r\n sentiment: criticScoreSummary.sentiment,\r\n count: {\r\n positive: criticScoreSummary.positiveCount,\r\n neutral: criticScoreSummary.neutralCount,\r\n negative: criticScoreSummary.negativeCount,\r\n total: criticScoreSummary.reviewCount,\r\n }\r\n },\r\n }\r\n}\r\n","import { parseDetailJsonResult, parseSearchJsonResult } from './parser'\r\nimport UserAgent from 'user-agents'\r\nimport { MetacriticEntry, RecordType, MetacriticSearchEntry } from './types'\r\n\r\nclass SearchInformations {\r\n apiKey: string\r\n\r\n constructor(scriptContent: string) {\r\n this.apiKey = this.extractApiFromScript(scriptContent)\r\n }\r\n\r\n extractApiFromScript(scriptContent: any) {\r\n const apiKeyPattern = /apiKey=([^\"&]+)/\r\n let matches = scriptContent.match(apiKeyPattern)\r\n if (matches) {\r\n return matches[1]\r\n }\r\n }\r\n}\r\n\r\nexport class MetacriticService {\r\n private minSimilarity: number\r\n private apiKey?: string\r\n\r\n static REFERER_HEADER = 'https://www.metacritic.com/'\r\n static HOMEPAGE_URL = 'https://www.metacritic.com/'\r\n static BASE_URL = 'https://backend.metacritic.com/composer/metacritic/pages/'\r\n static SEARCH_URL = MetacriticService.BASE_URL + 'search/'\r\n\r\n constructor(minSimilarity: number = 0.5) {\r\n this.minSimilarity = minSimilarity\r\n }\r\n\r\n async search(searchKey: string, recordType?: RecordType, sortBySimilarity: boolean = true): Promise<MetacriticSearchEntry[]> {\r\n if (!searchKey) {\r\n return []\r\n }\r\n\r\n if (!this.apiKey) {\r\n const searchInfo = await MetacriticService.sendApiKeyRetrievalWebsiteRequest()\r\n if (searchInfo) {\r\n this.apiKey = searchInfo.apiKey\r\n } else {\r\n console.error('Failed to retrieve API key')\r\n return []\r\n }\r\n }\r\n\r\n const jsonResult = await MetacriticService.sendSearchWebRequest(searchKey, this.apiKey, recordType)\r\n if (jsonResult) {\r\n return parseSearchJsonResult(jsonResult, searchKey, this.minSimilarity)\r\n }\r\n\r\n return []\r\n }\r\n\r\n async getDetail(searchKey: string, recordType: RecordType, sortBySimilarity: boolean = true): Promise<MetacriticEntry | null> {\r\n if (!searchKey) {\r\n return null\r\n }\r\n\r\n if (!this.apiKey) {\r\n const searchInfo = await MetacriticService.sendApiKeyRetrievalWebsiteRequest()\r\n if (searchInfo) {\r\n this.apiKey = searchInfo.apiKey\r\n } else {\r\n console.error('Failed to retrieve API key')\r\n return null\r\n }\r\n }\r\n\r\n const searchResult = await this.search(searchKey, recordType)\r\n if (searchResult && searchResult.length > 0) {\r\n const detailResult = await MetacriticService.sendDetailWebRequest(searchResult[0].slug, this.apiKey, recordType)\r\n\r\n if (detailResult) {\r\n return parseDetailJsonResult(detailResult, searchResult[0].must)\r\n }\r\n }\r\n\r\n return null\r\n }\r\n\r\n static async sendSearchWebRequest(searchKey: string, apiKey: string, recordType?: RecordType) {\r\n const headers = this.getRequestHeaders()\r\n\r\n const searchUrl = new URL(MetacriticService.SEARCH_URL + encodeURI(searchKey) + '/web')\r\n\r\n searchUrl.searchParams.append('apiKey', apiKey)\r\n\r\n if (recordType) {\r\n searchUrl.searchParams.append('mcoTypeId', recordType.toString())\r\n }\r\n\r\n try {\r\n const response = await fetch(searchUrl, {\r\n headers: headers,\r\n method: 'GET',\r\n signal: AbortSignal.timeout(60000)\r\n })\r\n\r\n if (response.ok) {\r\n return await response.text()\r\n }\r\n } catch (error) {\r\n console.error('Error fetching search results:', error)\r\n }\r\n }\r\n\r\n static async sendDetailWebRequest(slug: string, apiKey: string, recordType: RecordType) {\r\n const headers = this.getRequestHeaders()\r\n\r\n let baseUrl = MetacriticService.BASE_URL\r\n switch (recordType) {\r\n case RecordType.Game:\r\n baseUrl = baseUrl + 'games/'\r\n break\r\n case RecordType.Movie:\r\n baseUrl = baseUrl + 'movies/'\r\n break\r\n case RecordType.TVShow:\r\n baseUrl = baseUrl + 'shows/'\r\n break\r\n default:\r\n console.error('Unsupported record type for detail request:', recordType)\r\n return\r\n }\r\n\r\n const detailUrl = new URL(baseUrl + encodeURI(slug) + '/web')\r\n\r\n detailUrl.searchParams.append('apiKey', apiKey)\r\n\r\n try {\r\n const response = await fetch(detailUrl, {\r\n headers: headers,\r\n method: 'GET',\r\n signal: AbortSignal.timeout(60000)\r\n })\r\n\r\n if (response.ok) {\r\n return await response.text()\r\n }\r\n } catch (error) {\r\n console.error('Error fetching detail result:', error)\r\n }\r\n }\r\n\r\n static getRequestHeaders() {\r\n const minimumHeaders = this.getMinimalRequestHeaders()\r\n return {\r\n ...minimumHeaders,\r\n 'Content-Type': 'application/json',\r\n 'Accept': '*/*',\r\n 'Referer': MetacriticService.REFERER_HEADER,\r\n }\r\n }\r\n\r\n static getMinimalRequestHeaders() {\r\n const userAgent = new UserAgent()\r\n return {\r\n 'User-Agent': userAgent.toString(),\r\n }\r\n }\r\n\r\n static async sendApiKeyRetrievalWebsiteRequest(): Promise<SearchInformations | null> {\r\n const headers = this.getMinimalRequestHeaders()\r\n try {\r\n const response = await fetch(MetacriticService.HOMEPAGE_URL, {\r\n headers: headers,\r\n signal: AbortSignal.timeout(60000)\r\n })\r\n\r\n if (response.ok) {\r\n const html = await response.text()\r\n\r\n const scriptPattern = /<script[^>]*>([\\s\\S]*?)<\\/script>/g\r\n const scripts: string[] = []\r\n let match\r\n while ((match = scriptPattern.exec(html)) !== null) {\r\n scripts.push(match[1])\r\n }\r\n\r\n for (const script of scripts) {\r\n const searchInfo = new SearchInformations(script)\r\n if (searchInfo.apiKey) {\r\n return searchInfo\r\n }\r\n }\r\n }\r\n\r\n return null\r\n } catch (error) {\r\n console.error('Error fetching website:', error)\r\n return null\r\n }\r\n }\r\n}\r\n","export enum RecordType {\r\n TVShow = 1,\r\n Movie = 2,\r\n Game = 13,\r\n}\r\n\r\nexport type MetacriticSearchEntry = {\r\n id: number\r\n recordType: RecordType\r\n title: string\r\n slug: string\r\n must: boolean\r\n criticScore: number\r\n similarity: number\r\n}\r\n\r\nexport type MetacriticEntry = {\r\n id: number\r\n recordType: RecordType\r\n title: string\r\n slug: string\r\n must: boolean\r\n userScore: Score\r\n criticScore: Score\r\n}\r\n\r\ntype Score = {\r\n score: number\r\n maxScore: number\r\n sentiment: string\r\n count: {\r\n positive: number\r\n neutral: number\r\n negative: number\r\n total: number\r\n }\r\n}\r\n"],"mappings":"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","names":["RecordType"]}
|
|
1
|
+
{"version":3,"sources":["../src/lib/utils.ts","../src/lib/parser.ts","../src/lib/service.ts","../src/lib/types.ts"],"sourcesContent":["export function getSimilarity(a: string, b: string): number {\r\n if (a === b) return 1\r\n if (!a || !b) return 0\r\n\r\n const str1 = a.toLowerCase()\r\n const str2 = b.toLowerCase()\r\n const distance = levenshteinDistance(str1, str2)\r\n const maxLength = Math.max(str1.length, str2.length)\r\n return (maxLength - distance) / maxLength\r\n}\r\n\r\nfunction levenshteinDistance(a: string, b: string): number {\r\n const matrix: number[][] = []\r\n for (let i = 0; i <= b.length; i++) {\r\n matrix[i] = [i]\r\n }\r\n for (let j = 0; j <= a.length; j++) {\r\n matrix[0][j] = j\r\n }\r\n for (let i = 1; i <= b.length; i++) {\r\n for (let j = 1; j <= a.length; j++) {\r\n if (b.charAt(i - 1) === a.charAt(j - 1)) {\r\n matrix[i][j] = matrix[i - 1][j - 1]\r\n } else {\r\n matrix[i][j] = Math.min(\r\n matrix[i - 1][j - 1] + 1,\r\n Math.min(matrix[i][j - 1] + 1, matrix[i - 1][j] + 1)\r\n )\r\n }\r\n }\r\n }\r\n return matrix[b.length][a.length]\r\n}\r\n","import { getSimilarity } from './utils'\r\nimport { MetacriticEntry, RecordType, MetacriticSearchEntry } from './types'\r\n\r\nexport function parseSearchJsonResult(jsonString: string, searchKey: string, minSimilarity: number, sortBySimilarity: boolean = true): MetacriticSearchEntry[] {\r\n try {\r\n const parsedData = JSON.parse(jsonString)\r\n const entries: MetacriticSearchEntry[] = []\r\n\r\n const itemsList = parsedData.components.find((component: any) => component.meta.componentName === 'search').data.items\r\n for (const item of itemsList) {\r\n const entry = mapSearchEntry(item, searchKey)\r\n\r\n if (entry.similarity < minSimilarity) {\r\n continue\r\n }\r\n entries.push(entry)\r\n }\r\n\r\n return sortBySimilarity ? entries.sort((a, b) => b.similarity - a.similarity) : entries\r\n } catch (error) {\r\n console.error('Error parsing JSON:', error)\r\n return []\r\n }\r\n}\r\n\r\nexport function parseDetailJsonResult(jsonString: string, must: boolean): MetacriticEntry | null {\r\n try {\r\n const parsedData = JSON.parse(jsonString)\r\n return mapDetailEntry(parsedData, must)\r\n } catch (error) {\r\n console.error('Error parsing JSON:', error)\r\n return null\r\n }\r\n}\r\n\r\nfunction mapSearchEntry(data: any, searchKey: string): MetacriticSearchEntry {\r\n return {\r\n id: data.id,\r\n recordType: data.typeId as RecordType,\r\n title: data.title,\r\n slug: data.slug,\r\n criticScore: data.criticScoreSummary.score,\r\n must: data.mustSee || data.mustWatch || data.mustPlay,\r\n similarity: getSimilarity(data.title, searchKey)\r\n }\r\n}\r\n\r\nfunction mapDetailEntry(data: any, must: boolean): MetacriticEntry {\r\n const product = data.components.find((component: any) => component.meta.componentName === 'product').data.item\r\n const criticScoreSummary = data.components.find((component: any) => component.meta.componentName === 'critic-score-summary').data.item\r\n const userScoreSummary = data.components.find((component: any) => component.meta.componentName === 'user-score-summary').data.item\r\n\r\n return {\r\n id: product.id,\r\n recordType: product.typeId as RecordType,\r\n title: product.title,\r\n slug: product.slug,\r\n must: must,\r\n userScore: {\r\n score: userScoreSummary.score,\r\n maxScore: userScoreSummary.max,\r\n sentiment: userScoreSummary.sentiment,\r\n count: {\r\n positive: userScoreSummary.positiveCount,\r\n neutral: userScoreSummary.neutralCount,\r\n negative: userScoreSummary.negativeCount,\r\n total: userScoreSummary.reviewCount,\r\n }\r\n },\r\n criticScore: {\r\n score: criticScoreSummary.score,\r\n maxScore: criticScoreSummary.max,\r\n sentiment: criticScoreSummary.sentiment,\r\n count: {\r\n positive: criticScoreSummary.positiveCount,\r\n neutral: criticScoreSummary.neutralCount,\r\n negative: criticScoreSummary.negativeCount,\r\n total: criticScoreSummary.reviewCount,\r\n }\r\n },\r\n }\r\n}\r\n","import { parseDetailJsonResult, parseSearchJsonResult } from './parser'\r\nimport UserAgent from 'user-agents'\r\nimport { MetacriticEntry, RecordType, MetacriticSearchEntry } from './types'\r\n\r\nclass SearchInformations {\r\n apiKey: string\r\n\r\n constructor(scriptContent: string) {\r\n this.apiKey = this.extractApiFromScript(scriptContent)\r\n }\r\n\r\n extractApiFromScript(scriptContent: any) {\r\n const apiKeyPattern = /apiKey=([^\"&]+)/\r\n let matches = scriptContent.match(apiKeyPattern)\r\n if (matches) {\r\n return matches[1]\r\n }\r\n }\r\n}\r\n\r\nexport class MetacriticService {\r\n private minSimilarity: number\r\n private apiKey?: string\r\n\r\n static REFERER_HEADER = 'https://www.metacritic.com/'\r\n static HOMEPAGE_URL = 'https://www.metacritic.com/'\r\n static BASE_URL = 'https://backend.metacritic.com/composer/metacritic/pages/'\r\n static SEARCH_URL = MetacriticService.BASE_URL + 'search/'\r\n\r\n constructor(minSimilarity: number = 0.5) {\r\n this.minSimilarity = minSimilarity\r\n }\r\n\r\n async search(searchKey: string, recordType?: RecordType, sortBySimilarity: boolean = true): Promise<MetacriticSearchEntry[]> {\r\n if (!searchKey) {\r\n return []\r\n }\r\n\r\n if (!this.apiKey) {\r\n const searchInfo = await MetacriticService.sendApiKeyRetrievalWebsiteRequest()\r\n if (searchInfo) {\r\n this.apiKey = searchInfo.apiKey\r\n } else {\r\n console.error('Failed to retrieve API key')\r\n return []\r\n }\r\n }\r\n\r\n const jsonResult = await MetacriticService.sendSearchWebRequest(searchKey, this.apiKey, recordType)\r\n if (jsonResult) {\r\n return parseSearchJsonResult(jsonResult, searchKey, this.minSimilarity, sortBySimilarity)\r\n }\r\n\r\n return []\r\n }\r\n\r\n async getDetail(searchKey: string, recordType: RecordType, sortBySimilarity: boolean = true): Promise<MetacriticEntry | null> {\r\n if (!searchKey) {\r\n return null\r\n }\r\n\r\n if (!this.apiKey) {\r\n const searchInfo = await MetacriticService.sendApiKeyRetrievalWebsiteRequest()\r\n if (searchInfo) {\r\n this.apiKey = searchInfo.apiKey\r\n } else {\r\n console.error('Failed to retrieve API key')\r\n return null\r\n }\r\n }\r\n\r\n const searchResult = await this.search(searchKey, recordType, sortBySimilarity)\r\n if (searchResult && searchResult.length > 0) {\r\n const detailResult = await MetacriticService.sendDetailWebRequest(searchResult[0].slug, this.apiKey, recordType)\r\n\r\n if (detailResult) {\r\n return parseDetailJsonResult(detailResult, searchResult[0].must)\r\n }\r\n }\r\n\r\n return null\r\n }\r\n\r\n static async sendSearchWebRequest(searchKey: string, apiKey: string, recordType?: RecordType) {\r\n const headers = this.getRequestHeaders()\r\n\r\n const searchUrl = new URL(MetacriticService.SEARCH_URL + encodeURI(searchKey) + '/web')\r\n\r\n searchUrl.searchParams.append('apiKey', apiKey)\r\n\r\n if (recordType) {\r\n searchUrl.searchParams.append('mcoTypeId', recordType.toString())\r\n }\r\n\r\n try {\r\n const response = await fetch(searchUrl, {\r\n headers: headers,\r\n method: 'GET',\r\n signal: AbortSignal.timeout(60000)\r\n })\r\n\r\n if (response.ok) {\r\n return await response.text()\r\n }\r\n } catch (error) {\r\n console.error('Error fetching search results:', error)\r\n }\r\n }\r\n\r\n static async sendDetailWebRequest(slug: string, apiKey: string, recordType: RecordType) {\r\n const headers = this.getRequestHeaders()\r\n\r\n let baseUrl = MetacriticService.BASE_URL\r\n switch (recordType) {\r\n case RecordType.Game:\r\n baseUrl = baseUrl + 'games/'\r\n break\r\n case RecordType.Movie:\r\n baseUrl = baseUrl + 'movies/'\r\n break\r\n case RecordType.TVShow:\r\n baseUrl = baseUrl + 'shows/'\r\n break\r\n default:\r\n console.error('Unsupported record type for detail request:', recordType)\r\n return\r\n }\r\n\r\n const detailUrl = new URL(baseUrl + encodeURI(slug) + '/web')\r\n\r\n detailUrl.searchParams.append('apiKey', apiKey)\r\n\r\n try {\r\n const response = await fetch(detailUrl, {\r\n headers: headers,\r\n method: 'GET',\r\n signal: AbortSignal.timeout(60000)\r\n })\r\n\r\n if (response.ok) {\r\n return await response.text()\r\n }\r\n } catch (error) {\r\n console.error('Error fetching detail result:', error)\r\n }\r\n }\r\n\r\n static getRequestHeaders() {\r\n const minimumHeaders = this.getMinimalRequestHeaders()\r\n return {\r\n ...minimumHeaders,\r\n 'Content-Type': 'application/json',\r\n 'Accept': '*/*',\r\n 'Referer': MetacriticService.REFERER_HEADER,\r\n }\r\n }\r\n\r\n static getMinimalRequestHeaders() {\r\n const userAgent = new UserAgent()\r\n return {\r\n 'User-Agent': userAgent.toString(),\r\n }\r\n }\r\n\r\n static async sendApiKeyRetrievalWebsiteRequest(): Promise<SearchInformations | null> {\r\n const headers = this.getMinimalRequestHeaders()\r\n try {\r\n const response = await fetch(MetacriticService.HOMEPAGE_URL, {\r\n headers: headers,\r\n signal: AbortSignal.timeout(60000)\r\n })\r\n\r\n if (response.ok) {\r\n const html = await response.text()\r\n\r\n const scriptPattern = /<script[^>]*>([\\s\\S]*?)<\\/script>/g\r\n const scripts: string[] = []\r\n let match\r\n while ((match = scriptPattern.exec(html)) !== null) {\r\n scripts.push(match[1])\r\n }\r\n\r\n for (const script of scripts) {\r\n const searchInfo = new SearchInformations(script)\r\n if (searchInfo.apiKey) {\r\n return searchInfo\r\n }\r\n }\r\n }\r\n\r\n return null\r\n } catch (error) {\r\n console.error('Error fetching website:', error)\r\n return null\r\n }\r\n }\r\n}\r\n","export enum RecordType {\r\n TVShow = 1,\r\n Movie = 2,\r\n Game = 13,\r\n}\r\n\r\nexport type MetacriticSearchEntry = {\r\n id: number\r\n recordType: RecordType\r\n title: string\r\n slug: string\r\n must: boolean\r\n criticScore: number\r\n similarity: number\r\n}\r\n\r\nexport type MetacriticEntry = {\r\n id: number\r\n recordType: RecordType\r\n title: string\r\n slug: string\r\n must: boolean\r\n userScore: Score\r\n criticScore: Score\r\n}\r\n\r\nexport type Score = {\r\n score: number\r\n maxScore: number\r\n sentiment: string\r\n count: {\r\n positive: number\r\n neutral: number\r\n negative: number\r\n total: number\r\n }\r\n}\r\n"],"mappings":"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","names":["RecordType"]}
|