mirror of
https://github.com/LukeHagar/plex-api-oauth.git
synced 2025-12-06 12:37:50 +00:00
344 lines
11 KiB
JavaScript
344 lines
11 KiB
JavaScript
import { PlexOauth } from "plex-oauth";
|
|
import { v4 } from "uuid";
|
|
import axios from "axios";
|
|
import qs from "qs";
|
|
export class PlexAPIOAuth {
|
|
constructor(
|
|
clientId,
|
|
product = "Plex-API-OAuth",
|
|
device = "Web-Client",
|
|
version = "1",
|
|
forwardUrl = "",
|
|
platform = "Web",
|
|
plexTVAuthToken,
|
|
plexTVUserData,
|
|
plexServers = [],
|
|
plexDevices = []
|
|
) {
|
|
this.clientId = clientId;
|
|
this.product = product;
|
|
this.device = device;
|
|
this.version = version;
|
|
this.forwardUrl = forwardUrl;
|
|
this.platform = platform;
|
|
|
|
this.plexTVAuthToken = plexTVAuthToken;
|
|
this.plexTVUserData = plexTVUserData;
|
|
this.plexServers = plexServers;
|
|
this.plexDevices = plexDevices;
|
|
|
|
this.plexClientInformation = {
|
|
clientId: this.clientId, // This is a unique identifier used to identify your app with Plex. - If none is provided a new one is generated and saved locally
|
|
product: this.product, // Name of your application - Defaults to Plex-API-OAuth
|
|
device: this.device, // The type of device your application is running on - Defaults to "Web Client"
|
|
version: this.version, // Version of your application - Defaults to 1
|
|
forwardUrl: this.forwardUrl, // Url to forward back to after signing in - Defaults to an empty string
|
|
platform: this.platform, // Platform your application runs on - Defaults to 'Web'
|
|
};
|
|
}
|
|
|
|
SetPlexSession({
|
|
clientId,
|
|
product = "Plex-API-OAuth",
|
|
device = "Web-Client",
|
|
version = "1",
|
|
forwardUrl = "",
|
|
platform = "Web",
|
|
plexTVAuthToken,
|
|
plexTVUserData,
|
|
plexServers = [],
|
|
plexDevices = [],
|
|
selectedPlexServer,
|
|
}) {
|
|
this.plexTVAuthToken = plexTVAuthToken;
|
|
this.plexTVUserData = plexTVUserData;
|
|
this.selectedPlexServer = selectedPlexServer;
|
|
this.clientId = clientId;
|
|
this.product = product;
|
|
this.device = device;
|
|
this.version = version;
|
|
this.forwardUrl = forwardUrl;
|
|
this.platform = platform;
|
|
this.plexServers = plexServers;
|
|
this.plexDevices = plexDevices;
|
|
|
|
this.plexClientInformation = {
|
|
clientId: this.clientId, // This is a unique identifier used to identify your app with Plex. - If none is provided a new one is generated and saved locally
|
|
product: this.product, // Name of your application - Defaults to Plex-API-OAuth
|
|
device: this.device, // The type of device your application is running on - Defaults to "Web Client"
|
|
version: this.version, // Version of your application - Defaults to 1
|
|
forwardUrl: this.forwardUrl, // Url to forward back to after signing in - Defaults to an empty string
|
|
platform: this.platform, // Platform your application runs on - Defaults to 'Web'
|
|
};
|
|
}
|
|
|
|
SavePlexSession() {
|
|
console.log("Saving State:");
|
|
console.log({
|
|
plexTVAuthToken: this.plexTVAuthToken,
|
|
plexTVUserData: this.plexTVUserData,
|
|
selectedPlexServer: this.selectedPlexServer,
|
|
clientId: this.clientId,
|
|
product: this.product,
|
|
device: this.device,
|
|
version: this.version,
|
|
forwardUrl: this.forwardUrl,
|
|
platform: this.platform,
|
|
plexServers: this.plexServers,
|
|
plexDevices: this.plexDevices,
|
|
plexClientInformation: this.plexClientInformation,
|
|
});
|
|
window.localStorage.setItem(
|
|
"plexSessionData",
|
|
JSON.stringify({
|
|
plexTVAuthToken: this.plexTVAuthToken,
|
|
plexTVUserData: this.plexTVUserData,
|
|
selectedPlexServer: this.selectedPlexServer,
|
|
clientId: this.clientId,
|
|
product: this.product,
|
|
device: this.device,
|
|
version: this.version,
|
|
forwardUrl: this.forwardUrl,
|
|
platform: this.platform,
|
|
plexServers: this.plexServers,
|
|
plexDevices: this.plexDevices,
|
|
plexLibraries: this.plexLibraries,
|
|
plexMovieLibraries: this.plexMovieLibraries,
|
|
plexMusicLibraries: this.plexMusicLibraries,
|
|
plexTVShowLibraries: this.plexTVShowLibraries,
|
|
plexMovies: this.plexMovies,
|
|
plexMusic: this.plexMusic,
|
|
plexTVShows: this.plexTVShows,
|
|
plexClientInformation: this.plexClientInformation,
|
|
})
|
|
);
|
|
}
|
|
|
|
LoadPlexSession() {
|
|
console.log("Loading State:");
|
|
this.SetPlexSession(
|
|
JSON.parse(window.localStorage.getItem("plexSessionData"))
|
|
);
|
|
console.log(JSON.parse(window.localStorage.getItem("plexSessionData")));
|
|
}
|
|
|
|
GenerateClientId() {
|
|
this.clientId = v4();
|
|
this.plexClientInformation.clientId = this.clientId;
|
|
console.log("Generated ClientId");
|
|
}
|
|
|
|
async PlexLogin() {
|
|
if (this.ClientId === null || this.clientId === undefined) {
|
|
this.GenerateClientId();
|
|
}
|
|
var plexOauth = new PlexOauth(this.plexClientInformation);
|
|
let data = await plexOauth.requestHostedLoginURL().catch((err) => {
|
|
throw err;
|
|
});
|
|
|
|
let [hostedUILink, pinId] = data;
|
|
|
|
console.log("Plex Auth URL:");
|
|
console.log(hostedUILink); // UI URL used to log into Plex
|
|
console.log("Plex Pin ID:");
|
|
console.log(pinId);
|
|
|
|
window.open(hostedUILink, "_blank");
|
|
window.focus();
|
|
|
|
/*
|
|
* You can now navigate the user's browser to the 'hostedUILink'. This will include the forward URL
|
|
* for your application, so when they have finished signing into Plex, they will be redirected back
|
|
* to the specified URL. From there, you just need to perform a query to check for the auth token.
|
|
* (See Below)
|
|
*/
|
|
|
|
// Check for the auth token, once returning to the application
|
|
|
|
let authToken = await plexOauth
|
|
.checkForAuthToken(pinId, 1000, 10)
|
|
.catch((err) => {
|
|
throw err;
|
|
});
|
|
|
|
if (authToken !== null) {
|
|
this.plexTVAuthToken = authToken;
|
|
console.log("Plex Authentication Successful");
|
|
return authToken;
|
|
} else {
|
|
console.log("Plex Authentication Failed");
|
|
return null;
|
|
}
|
|
// An auth token will only be null if the user never signs into the hosted UI, or you stop checking for a new one before they can log in
|
|
}
|
|
|
|
async GetPlexUserData() {
|
|
let response = await axios({
|
|
method: "GET",
|
|
url:
|
|
"https://plex.tv/api/v2/user?" +
|
|
qs.stringify({
|
|
"X-Plex-Product": this.plexClientInformation.product,
|
|
"X-Plex-Client-Identifier": this.plexClientInformation.clientId,
|
|
"X-Plex-Token": this.plexTVAuthToken,
|
|
}),
|
|
headers: { accept: "application/json" },
|
|
}).catch(function (error) {
|
|
if (error.response) {
|
|
// The request was made and the server responded with a status code
|
|
// that falls out of the range of 2xx
|
|
console.log(error.response.data);
|
|
console.log(error.response.status);
|
|
console.log(error.response.headers);
|
|
} else if (error.request) {
|
|
// The request was made but no response was received
|
|
// `error.request` is an instance of XMLHttpRequest in the browser and an instance of
|
|
// http.ClientRequest in node.js
|
|
console.log(error.request);
|
|
} else {
|
|
// Something happened in setting up the request that triggered an Error
|
|
console.log("Error", error.message);
|
|
}
|
|
console.log(error.config);
|
|
});
|
|
|
|
console.log(response);
|
|
console.log(response.status);
|
|
if (response.status === 200) {
|
|
console.log("Authentican Token Validated Successfully");
|
|
this.plexTVUserData = response.data;
|
|
console.log("Populated User Data Successfully");
|
|
return response.data;
|
|
}
|
|
if (response.status === 401) {
|
|
console.log("Authentican Token Failed Validation");
|
|
return null;
|
|
}
|
|
}
|
|
|
|
async GetPlexServers() {
|
|
let response = await axios({
|
|
method: "GET",
|
|
url:
|
|
"https://plex.tv/api/v2/resources?" +
|
|
qs.stringify({
|
|
includeHttps: 1,
|
|
includeRelay: 1,
|
|
includeIPv6: 1,
|
|
"X-Plex-Product": this.plexClientInformation.product,
|
|
"X-Plex-Client-Identifier": this.plexClientInformation.clientId,
|
|
"X-Plex-Token": this.plexTVAuthToken,
|
|
}),
|
|
headers: { accept: "application/json" },
|
|
}).catch((err) => {
|
|
throw err;
|
|
});
|
|
console.log("Plex Devices:");
|
|
console.log(response);
|
|
this.plexDevices = response.data;
|
|
this.plexServers = response.data
|
|
?.filter((obj) => obj?.product?.includes("Plex Media Server"))
|
|
?.map((obj) => {
|
|
return {
|
|
accessToken: obj?.accessToken,
|
|
clientIdentifier: obj?.clientIdentifier,
|
|
connections: obj?.connections,
|
|
localConnections: obj?.connections?.filter(
|
|
(obj) => obj?.local == true
|
|
),
|
|
relayConnections: obj?.connections?.filter(
|
|
(obj) => obj?.relay == true
|
|
),
|
|
createdAt: obj?.createdAt,
|
|
device: obj?.device,
|
|
dnsRebindingProtection: obj?.dnsRebindingProtection,
|
|
home: obj?.home,
|
|
httpsRequired: obj?.httpsRequired,
|
|
lastSeenAt: obj?.lastSeenAt,
|
|
libraries: [],
|
|
name: obj?.name,
|
|
natLoopbackSupported: obj?.natLoopbackSupported,
|
|
owned: obj?.owned,
|
|
ownerId: obj?.ownerId,
|
|
platform: obj?.platform,
|
|
platformVersion: obj?.platformVersion,
|
|
presence: obj?.presence,
|
|
product: obj?.product,
|
|
productVersion: obj?.productVersion,
|
|
provides: obj?.provides,
|
|
publicAddress: obj?.publicAddress,
|
|
publicAddressMatches: obj?.publicAddressMatches,
|
|
relay: obj?.relay,
|
|
sourceTitle: obj?.sourceTitle,
|
|
synced: obj?.accessTokensynced,
|
|
};
|
|
});
|
|
return response.data;
|
|
}
|
|
|
|
async GetPlexLibraries() {
|
|
await this.plexServers?.forEach(async (server) => {
|
|
let response = await axios({
|
|
method: "GET",
|
|
url:
|
|
server?.relayConnections[0].uri +
|
|
"/library/sections/?" +
|
|
qs.stringify({
|
|
"X-Plex-Token": server?.accessToken,
|
|
}),
|
|
headers: { accept: "application/json" },
|
|
}).catch((err) => {
|
|
throw err;
|
|
});
|
|
|
|
let plexLibraries = response?.data?.MediaContainer?.Directory;
|
|
let plexMusicLibraries =
|
|
response?.data?.MediaContainer?.Directory?.filter(
|
|
(obj) => obj.type === "artist"
|
|
);
|
|
let plexMovieLibraries =
|
|
response?.data?.MediaContainer?.Directory?.filter(
|
|
(obj) => obj.type === "movie"
|
|
);
|
|
let plexTVShowLibraries =
|
|
response?.data?.MediaContainer?.Directory?.filter(
|
|
(obj) => obj.type === "show"
|
|
);
|
|
|
|
console.log(this.plexServers[server]);
|
|
server.libraries = {
|
|
plexLibraries: plexLibraries,
|
|
plexMusicLibraries: plexMusicLibraries,
|
|
plexMovieLibraries: plexMovieLibraries,
|
|
plexTVShowLibraries: plexTVShowLibraries,
|
|
};
|
|
console.log(server);
|
|
this.plexServers[server] = server;
|
|
});
|
|
return true;
|
|
}
|
|
|
|
async GetPlexLibraryContent(server, library) {
|
|
let response = await axios({
|
|
method: "GET",
|
|
url:
|
|
server?.relayConnections[0].uri +
|
|
"/library/sections/" +
|
|
library.uuid +
|
|
"?" +
|
|
qs.stringify({
|
|
"X-Plex-Token": server?.accessToken,
|
|
}),
|
|
headers: { accept: "application/json" },
|
|
}).catch((err) => {
|
|
throw err;
|
|
});
|
|
console.log(response);
|
|
return true;
|
|
// this.plexMusic = response?.data?.MediaContainer?.Directory;
|
|
|
|
// console.log(this.plexLibraries);
|
|
}
|
|
}
|