pushing for transit

This commit is contained in:
luke-hagar-sp
2022-07-25 19:29:36 -05:00
parent 480884abcd
commit 0498642e1b
2 changed files with 273 additions and 155 deletions

426
index.mjs
View File

@@ -4,33 +4,29 @@ import axios from "axios";
import qs from "qs"; import qs from "qs";
export class PlexAPIOAuth { export class PlexAPIOAuth {
constructor( constructor(
clientId = "", clientId,
product = "Plex-API-OAuth", product = "Plex-API-OAuth",
device = "Web-Client", device = "Web-Client",
version = "1", version = "1",
forwardUrl = "", forwardUrl = "",
platform = "Web", platform = "Web",
plexTVAuthToken = "", plexTVAuthToken,
plexTVUserData = {}, plexTVUserData,
plexServers = {}, plexServers = [],
plexLibraries = {}, plexDevices = []
plexMovies = {},
plexMusic = {},
plexTVShows = {}
) { ) {
this.plexTVAuthToken = plexTVAuthToken;
this.plexTVUserData = plexTVUserData;
this.clientId = clientId; this.clientId = clientId;
this.product = product; this.product = product;
this.device = device; this.device = device;
this.version = version; this.version = version;
this.forwardUrl = forwardUrl; this.forwardUrl = forwardUrl;
this.platform = platform; this.platform = platform;
this.plexTVAuthToken = plexTVAuthToken;
this.plexTVUserData = plexTVUserData;
this.plexServers = plexServers; this.plexServers = plexServers;
this.plexLibraries = plexLibraries; this.plexDevices = plexDevices;
this.plexMovies = plexMovies;
this.plexMusic = plexMusic;
this.plexTVShows = plexTVShows;
this.plexClientInformation = { 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 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 product: this.product, // Name of your application - Defaults to Plex-API-OAuth
@@ -41,145 +37,144 @@ export class PlexAPIOAuth {
}; };
} }
get ClientId() { SetPlexSession({
return this.clientId; clientId,
} product = "Plex-API-OAuth",
set ClientId(clientId) { device = "Web-Client",
this.clientId = clientId; version = "1",
} forwardUrl = "",
get PlexTVAuthToken() { platform = "Web",
return this.plexTVAuthToken; plexTVAuthToken,
} plexTVUserData,
set PlexTVAuthToken(plexTVAuthToken) { plexServers = [],
plexDevices = [],
selectedPlexServer,
}) {
this.plexTVAuthToken = plexTVAuthToken; this.plexTVAuthToken = plexTVAuthToken;
}
get PlexTVUserData() {
return this.plexTVUserData;
}
set PlexTVUserData(plexTVUserData) {
this.plexTVUserData = plexTVUserData; this.plexTVUserData = plexTVUserData;
} this.selectedPlexServer = selectedPlexServer;
get Product() { this.clientId = clientId;
return this.product;
}
set Product(product) {
this.product = product; this.product = product;
}
get Device() {
return this.device;
}
set Device(device) {
this.device = device; this.device = device;
}
get Version() {
return this.version;
}
set Version(version) {
this.version = version; this.version = version;
}
get ForwardUrl() {
return this.forwardUrl;
}
set ForwardUrl(forwardUrl) {
this.forwardUrl = forwardUrl; this.forwardUrl = forwardUrl;
}
get Platform() {
return this.platform;
}
set Platform(platform) {
this.platform = platform; this.platform = platform;
}
get PlexServers() {
return this.plexServers;
}
set PlexServers(plexServers) {
this.plexServers = plexServers; 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'
};
} }
get PlexLibraries() {
return this.plexLibraries; 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,
})
);
} }
set PlexLibraries(plexLibraries) {
this.plexLibraries = plexLibraries; LoadPlexSession() {
} console.log("Loading State:");
get PlexMovies() { this.SetPlexSession(
return this.plexMovies; JSON.parse(window.localStorage.getItem("plexSessionData"))
} );
set PlexMovies(plexMovies) { console.log(JSON.parse(window.localStorage.getItem("plexSessionData")));
this.plexMovies = plexMovies;
}
get PlexMusic() {
return this.plexMusic;
}
set PlexMusic(plexMusic) {
this.plexMusic = plexMusic;
}
get PlexTVShows() {
return this.plexTVShows;
}
set PlexTVShows(plexTVShows) {
this.plexTVShows = plexTVShows;
}
get PlexClientInformation() {
return this.plexClientInformation;
}
set PlexClientInformation(plexClientInformation) {
this.plexClientInformation = plexClientInformation;
} }
GenerateClientId() { GenerateClientId() {
this.clientId = v4(); this.clientId = v4();
this.plexClientInformation.clientId = this.clientId; this.plexClientInformation.clientId = this.clientId;
console.log("Generated ClientId");
} }
PlexLogin() { async PlexLogin() {
if (this.ClientId === null || this.clientId === undefined) {
this.GenerateClientId();
}
var plexOauth = new PlexOauth(this.plexClientInformation); var plexOauth = new PlexOauth(this.plexClientInformation);
plexOauth let data = await plexOauth.requestHostedLoginURL().catch((err) => {
.requestHostedLoginURL() throw err;
.then((data) => { });
let [hostedUILink, pinId] = data;
console.log("Plex Auth URL:"); let [hostedUILink, pinId] = data;
console.log(hostedUILink); // UI URL used to log into Plex
console.log("Plex Pin ID:");
console.log(pinId);
window.open(hostedUILink, "_blank"); console.log("Plex Auth URL:");
window.focus(); console.log(hostedUILink); // UI URL used to log into Plex
console.log("Plex Pin ID:");
console.log(pinId);
/* window.open(hostedUILink, "_blank");
* You can now navigate the user's browser to the 'hostedUILink'. This will include the forward URL window.focus();
* 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 /*
* 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)
*/
plexOauth // Check for the auth token, once returning to the application
.checkForAuthToken(pinId, 1000, 10)
.then((authToken) => {
if (authToken !== null) {
this.plexTVAuthToken = authToken;
console.log("Plex Authentication Successful");
return true;
} else {
console.log("Plex Authentication Failed");
return false;
}
// 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
})
.catch((err) => { let authToken = await plexOauth
throw err; .checkForAuthToken(pinId, 1000, 10)
});
})
.catch((err) => { .catch((err) => {
throw 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
} }
GetPlexUserData() { async GetPlexUserData() {
axios({ let response = await axios({
method: "GET", method: "GET",
url: url:
"https://plex.tv/api/v2/user?" + "https://plex.tv/api/v2/user?" +
@@ -189,36 +184,159 @@ export class PlexAPIOAuth {
"X-Plex-Token": this.plexTVAuthToken, "X-Plex-Token": this.plexTVAuthToken,
}), }),
headers: { accept: "application/json" }, headers: { accept: "application/json" },
}) }).catch(function (error) {
.then((response) => { if (error.response) {
console.log(response); // The request was made and the server responded with a status code
console.log(response.status); // that falls out of the range of 2xx
if (response.status === 200) { console.log(error.response.data);
this.plexUserData = response.data; console.log(error.response.status);
return true; console.log(error.response.headers);
} } else if (error.request) {
if (response.status === 401) { // The request was made but no response was received
console.log("Authentican Token Failed Validation"); // `error.request` is an instance of XMLHttpRequest in the browser and an instance of
return false; // http.ClientRequest in node.js
} console.log(error.request);
}) } else {
.catch(function (error) { // Something happened in setting up the request that triggered an Error
if (error.response) { console.log("Error", error.message);
// The request was made and the server responded with a status code }
// that falls out of the range of 2xx console.log(error.config);
console.log(error.response.data); });
console.log(error.response.status);
console.log(error.response.headers); console.log(response);
} else if (error.request) { console.log(response.status);
// The request was made but no response was received if (response.status === 200) {
// `error.request` is an instance of XMLHttpRequest in the browser and an instance of console.log("Authentican Token Validated Successfully");
// http.ClientRequest in node.js this.plexTVUserData = response.data;
console.log(error.request); console.log("Populated User Data Successfully");
} else { return response.data;
// Something happened in setting up the request that triggered an Error }
console.log("Error", error.message); if (response.status === 401) {
} console.log("Authentican Token Failed Validation");
console.log(error.config); 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(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[this.plexServers.indexOf(server)]);
server.libraries = {
plexLibraries: plexLibraries,
plexMusicLibraries: plexMusicLibraries,
plexMovieLibraries: plexMovieLibraries,
plexTVShowLibraries: plexTVShowLibraries,
};
this.plexServers[this.plexServers.indexOf(server)] = server;
return {
plexLibraries: plexLibraries,
plexMusicLibraries: plexMusicLibraries,
plexMovieLibraries: plexMovieLibraries,
plexTVShowLibraries: plexTVShowLibraries,
};
}
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);
} }
} }

View File

@@ -1,6 +1,6 @@
{ {
"name": "plex-api-oauth", "name": "plex-api-oauth",
"version": "1.0.37", "version": "1.0.124",
"description": "An NPM Module designed to make Plex Media Server and plex.tv API calls easier to implement in JavaScript and React projects", "description": "An NPM Module designed to make Plex Media Server and plex.tv API calls easier to implement in JavaScript and React projects",
"main": "index.mjs", "main": "index.mjs",
"type": "module", "type": "module",