poke/src/libpoketube/libpoketube-dislikes.js

112 lines
3.1 KiB
JavaScript
Raw Normal View History

2023-05-21 22:24:24 +02:00
/**
* PokeTube is a Free/Libre youtube front-end !
*
* This file is Licensed under LGPL-3.0-or-later. Poketube itself is GPL, Only this file is LGPL.
* See a copy here: https://www.gnu.org/licenses/lgpl-3.0.txt
* Please don't remove this comment while sharing this code.
*/
2023-05-01 21:20:13 +02:00
/**
* A class representing a PokeTube API instance for a specific video.
*/
class PokeTubeDislikesAPIManager {
2023-05-01 21:20:13 +02:00
/**
* Creates a new PokeTube API instance for the given video ID.
* @param {string} videoId - The ID of the YouTube video.
*/
constructor(videoId) {
this.videoId = videoId;
this.engagement = null;
this.videoData = null;
this.headers = {};
2023-03-04 15:08:54 +01:00
}
2022-12-21 16:42:06 +01:00
2023-05-01 21:20:13 +02:00
/**
* Parses a JSON string and returns the resulting object.
* @param {string} str - The JSON string to parse.
* @returns {object|null} The parsed JSON object, or null if an error occurs.
* @private
*/
_getJson(str) {
try {
return JSON.parse(str);
} catch {
return null;
}
}
2023-09-30 07:09:06 +02:00
2023-05-01 21:20:13 +02:00
/**
* Retrieves engagement data for the YouTube video.
* @returns {Promise<object|null>} A Promise that resolves with the engagement data, or null if an error occurs.
* @private
*/
async _getEngagementData() {
2023-10-21 07:12:57 +02:00
const apiUrl = `https://p.poketube.fun/api?v=${this.videoId}&hash=d0550b6e28c8f93533a569c314d5b4e2`;
2023-10-21 07:14:20 +02:00
const fallbackUrl = `https://returnyoutubedislikeapi.com/votes?videoId=${this.videoId}`;
2023-09-30 07:15:04 +02:00
2023-10-21 07:10:02 +02:00
const { fetch } = await import("undici");
2023-10-21 07:10:02 +02:00
try {
2023-10-21 07:12:57 +02:00
// Set a timeout of 2 seconds.
const timeoutMilliseconds = 2000; // 2 seconds
2023-10-21 07:14:20 +02:00
var engagementP = await fetch(fallbackUrl, { timeout: timeoutMilliseconds })
2023-10-21 07:12:57 +02:00
.then((res) => {
if (res.statusCode === 504) {
throw new Error("Request timed out.");
}
return res.json();
});
2023-10-21 06:57:10 +02:00
2023-10-21 07:10:02 +02:00
if (typeof engagementP.dislikes === 'number') {
return engagementP;
} else {
throw new Error("API response doesn't contain valid dislikes count. Using fallback URL.");
2023-02-28 15:12:57 +01:00
}
2023-10-21 07:10:02 +02:00
} catch (error) {
2023-10-21 07:12:57 +02:00
console.error(error);
2023-10-21 07:14:20 +02:00
var engagement = await fetch(apiUrl).then((res) => res.json());
2023-10-21 07:10:02 +02:00
return engagement;
}
2023-10-21 07:12:57 +02:00
2023-10-21 07:10:02 +02:00
}
2022-12-20 14:38:25 +01:00
2023-05-01 21:20:13 +02:00
/**
* Retrieves data about the YouTube video and its engagement.
* @returns {Promise<object>} A Promise that resolves with an object containing video and engagement data.
*/
async getData() {
this.engagement = await this._getEngagementData();
return {
engagement: this.engagement,
2023-09-30 07:15:04 +02:00
};
2023-05-01 21:20:13 +02:00
}
2022-12-20 14:38:25 +01:00
2023-05-01 21:20:13 +02:00
/**
* Logs an error message to the console.
* @param {string} args - The error message to log.
* @private
*/
_handleError(args) {
2023-09-30 07:15:04 +02:00
console.error(`[LIBPT DISLIKES ERROR] ${args}`);
2023-05-01 21:20:13 +02:00
}
}
/*
Returns basic data about a given YouTube video using PokeTubeDislikesAPIManager.
2023-05-01 21:20:13 +02:00
@async
@function
@param {string} videoId - The YouTube video ID to get data for.
2023-09-09 18:53:04 +02:00
@returns {Promise<Object>} An object containing the engagement data, as well as the YouTube URL for the video.
2023-05-01 21:20:13 +02:00
@throws {Error} If the video ID is invalid or the request fails.
*/
const getDislikesData = async (videoId) => {
const pokeTubeAPI = new PokeTubeDislikesAPIManager(videoId);
return await PokeTubeDislikesAPIManager.getData();
};
2023-02-25 18:47:28 +01:00
module.exports = getDislikesData;