|
|
|
|
#region Copyright
|
|
|
|
|
// /************************************************************************
|
|
|
|
|
// Copyright (c) 2016 Jamie Rees
|
|
|
|
|
// File: PlexApi.cs
|
|
|
|
|
// Created By: Jamie Rees
|
|
|
|
|
//
|
|
|
|
|
// Permission is hereby granted, free of charge, to any person obtaining
|
|
|
|
|
// a copy of this software and associated documentation files (the
|
|
|
|
|
// "Software"), to deal in the Software without restriction, including
|
|
|
|
|
// without limitation the rights to use, copy, modify, merge, publish,
|
|
|
|
|
// distribute, sublicense, and/or sell copies of the Software, and to
|
|
|
|
|
// permit persons to whom the Software is furnished to do so, subject to
|
|
|
|
|
// the following conditions:
|
|
|
|
|
//
|
|
|
|
|
// The above copyright notice and this permission notice shall be
|
|
|
|
|
// included in all copies or substantial portions of the Software.
|
|
|
|
|
//
|
|
|
|
|
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
|
|
|
|
// EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
|
|
|
|
// MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
|
|
|
|
|
// NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
|
|
|
|
|
// LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
|
|
|
|
|
// OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
|
|
|
|
|
// WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
|
|
|
|
// ************************************************************************/
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
#endregion
|
|
|
|
|
|
|
|
|
|
using System;
|
|
|
|
|
using NLog;
|
|
|
|
|
using Ombi.Api.Interfaces;
|
|
|
|
|
using Ombi.Api.Models.Plex;
|
|
|
|
|
using Ombi.Helpers;
|
|
|
|
|
using RestSharp;
|
|
|
|
|
|
|
|
|
|
namespace Ombi.Api
|
|
|
|
|
{
|
|
|
|
|
public class PlexApi : IPlexApi
|
|
|
|
|
{
|
|
|
|
|
static PlexApi()
|
|
|
|
|
{
|
|
|
|
|
Version = AssemblyHelper.GetAssemblyVersion();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public PlexApi (IApiRequest api)
|
|
|
|
|
{
|
|
|
|
|
Api = api;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private IApiRequest Api { get; }
|
|
|
|
|
|
|
|
|
|
private const string SignInUri = "https://plex.tv/users/sign_in.json";
|
|
|
|
|
private const string FriendsUri = "https://plex.tv/pms/friends/all";
|
|
|
|
|
private const string GetAccountUri = "https://plex.tv/users/account";
|
|
|
|
|
private const string ServerUri = "https://plex.tv/pms/servers.xml";
|
|
|
|
|
|
|
|
|
|
private static Logger Log = LogManager.GetCurrentClassLogger();
|
|
|
|
|
private static string Version { get; }
|
|
|
|
|
|
|
|
|
|
public PlexAuthentication SignIn(string username, string password)
|
|
|
|
|
{
|
|
|
|
|
var userModel = new PlexUserRequest
|
|
|
|
|
{
|
|
|
|
|
user = new UserRequest
|
|
|
|
|
{
|
|
|
|
|
password = password,
|
|
|
|
|
login = username
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
var request = new RestRequest
|
|
|
|
|
{
|
|
|
|
|
Method = Method.POST
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
AddHeaders(ref request, false);
|
|
|
|
|
|
|
|
|
|
request.AddJsonBody(userModel);
|
|
|
|
|
|
|
|
|
|
var obj = Api.Execute<PlexAuthentication> (request, new Uri(SignInUri));
|
|
|
|
|
|
|
|
|
|
return obj;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public PlexFriends GetUsers(string authToken)
|
|
|
|
|
{
|
|
|
|
|
var request = new RestRequest
|
|
|
|
|
{
|
|
|
|
|
Method = Method.GET,
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
AddHeaders(ref request, authToken, false);
|
|
|
|
|
|
|
|
|
|
var users = RetryHandler.Execute(() => Api.ExecuteXml<PlexFriends> (request, new Uri(FriendsUri)),
|
|
|
|
|
(exception, timespan) => Log.Error (exception, "Exception when calling GetUsers for Plex, Retrying {0}", timespan), null);
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
return users;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// Gets the users.
|
|
|
|
|
/// </summary>
|
|
|
|
|
/// <param name="authToken">The authentication token.</param>
|
|
|
|
|
/// <param name="searchTerm">The search term.</param>
|
|
|
|
|
/// <param name="plexFullHost">The full plex host.</param>
|
|
|
|
|
/// <returns></returns>
|
|
|
|
|
public PlexSearch SearchContent(string authToken, string searchTerm, Uri plexFullHost)
|
|
|
|
|
{
|
|
|
|
|
var request = new RestRequest
|
|
|
|
|
{
|
|
|
|
|
Method = Method.GET,
|
|
|
|
|
Resource = "search?query={searchTerm}"
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
request.AddUrlSegment("searchTerm", searchTerm);
|
|
|
|
|
AddHeaders(ref request, authToken, false);
|
|
|
|
|
|
|
|
|
|
var search = RetryHandler.Execute(() => Api.ExecuteXml<PlexSearch> (request, plexFullHost),
|
|
|
|
|
(exception, timespan) => Log.Error (exception, "Exception when calling SearchContent for Plex, Retrying {0}", timespan), null);
|
|
|
|
|
|
|
|
|
|
return search;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public PlexStatus GetStatus(string authToken, Uri uri)
|
|
|
|
|
{
|
|
|
|
|
var request = new RestRequest
|
|
|
|
|
{
|
|
|
|
|
Method = Method.GET,
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
AddHeaders(ref request, authToken, false);
|
|
|
|
|
|
|
|
|
|
var users = RetryHandler.Execute(() => Api.ExecuteXml<PlexStatus> (request, uri),
|
|
|
|
|
(exception, timespan) => Log.Error (exception, "Exception when calling GetStatus for Plex, Retrying {0}", timespan), null);
|
|
|
|
|
|
|
|
|
|
return users;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public PlexAccount GetAccount(string authToken)
|
|
|
|
|
{
|
|
|
|
|
var request = new RestRequest
|
|
|
|
|
{
|
|
|
|
|
Method = Method.GET,
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
AddHeaders(ref request, authToken, false);
|
|
|
|
|
|
|
|
|
|
var account = RetryHandler.Execute(() => Api.ExecuteXml<PlexAccount> (request, new Uri(GetAccountUri)),
|
|
|
|
|
(exception, timespan) => Log.Error (exception, "Exception when calling GetAccount for Plex, Retrying {0}", timespan), null);
|
|
|
|
|
|
|
|
|
|
return account;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public PlexLibraries GetLibrarySections(string authToken, Uri plexFullHost)
|
|
|
|
|
{
|
|
|
|
|
var request = new RestRequest
|
|
|
|
|
{
|
|
|
|
|
Method = Method.GET,
|
|
|
|
|
Resource = "library/sections"
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
AddHeaders(ref request, authToken, false);
|
|
|
|
|
|
|
|
|
|
try
|
|
|
|
|
{
|
|
|
|
|
var lib = RetryHandler.Execute(() => Api.ExecuteXml<PlexLibraries> (request, plexFullHost),
|
|
|
|
|
(exception, timespan) => Log.Error (exception, "Exception when calling GetLibrarySections for Plex, Retrying {0}", timespan), new TimeSpan[] {
|
|
|
|
|
TimeSpan.FromSeconds (5),
|
|
|
|
|
TimeSpan.FromSeconds(10),
|
|
|
|
|
TimeSpan.FromSeconds(30)
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
return lib;
|
|
|
|
|
}
|
|
|
|
|
catch (Exception e)
|
|
|
|
|
{
|
|
|
|
|
Log.Error(e,"There has been a API Exception when attempting to get the Plex Libraries");
|
|
|
|
|
return new PlexLibraries();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public PlexSearch GetLibrary(string authToken, Uri plexFullHost, string libraryId)
|
|
|
|
|
{
|
|
|
|
|
var request = new RestRequest
|
|
|
|
|
{
|
|
|
|
|
Method = Method.GET,
|
|
|
|
|
Resource = "library/sections/{libraryId}/all"
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
request.AddUrlSegment("libraryId", libraryId);
|
|
|
|
|
AddHeaders(ref request, authToken, false);
|
|
|
|
|
|
|
|
|
|
try
|
|
|
|
|
{
|
|
|
|
|
var lib = RetryHandler.Execute(() => Api.ExecuteXml<PlexSearch> (request, plexFullHost),
|
|
|
|
|
(exception, timespan) => Log.Error (exception, "Exception when calling GetLibrary for Plex, Retrying {0}", timespan), new TimeSpan[] {
|
|
|
|
|
TimeSpan.FromSeconds (5),
|
|
|
|
|
TimeSpan.FromSeconds(10),
|
|
|
|
|
TimeSpan.FromSeconds(30)
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
return lib;
|
|
|
|
|
}
|
|
|
|
|
catch (Exception e)
|
|
|
|
|
{
|
|
|
|
|
Log.Error(e,"There has been a API Exception when attempting to get the Plex Library");
|
|
|
|
|
return new PlexSearch();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public PlexEpisodeMetadata GetEpisodeMetaData(string authToken, Uri host, string ratingKey)
|
|
|
|
|
{
|
|
|
|
|
|
|
|
|
|
// 192.168.1.69:32400/library/metadata/3662/allLeaves
|
|
|
|
|
// The metadata ratingkey should be in the Cache
|
|
|
|
|
// Search for it and then call the above with the Directory.RatingKey
|
|
|
|
|
// THEN! We need the episode metadata using result.Vide.Key ("/library/metadata/3664")
|
|
|
|
|
// We then have the GUID which contains the TVDB ID plus the season and episode number: guid="com.plexapp.agents.thetvdb://269586/2/8?lang=en"
|
|
|
|
|
var request = new RestRequest
|
|
|
|
|
{
|
|
|
|
|
Method = Method.GET,
|
|
|
|
|
Resource = "/library/metadata/{ratingKey}"
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
request.AddUrlSegment("ratingKey", ratingKey);
|
|
|
|
|
AddHeaders(ref request, authToken, false);
|
|
|
|
|
|
|
|
|
|
try
|
|
|
|
|
{
|
|
|
|
|
var lib = RetryHandler.Execute(() => Api.ExecuteXml<PlexEpisodeMetadata>(request, host),
|
|
|
|
|
(exception, timespan) => Log.Error(exception, "Exception when calling GetEpisodeMetaData for Plex, Retrying {0}", timespan));
|
|
|
|
|
|
|
|
|
|
return lib;
|
|
|
|
|
}
|
|
|
|
|
catch (Exception e)
|
|
|
|
|
{
|
|
|
|
|
Log.Error(e, "There has been a API Exception when attempting to get GetEpisodeMetaData");
|
|
|
|
|
return new PlexEpisodeMetadata();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public PlexSearch GetAllEpisodes(string authToken, Uri host, string section, int startPage, int returnCount)
|
|
|
|
|
{
|
|
|
|
|
var request = new RestRequest
|
|
|
|
|
{
|
|
|
|
|
Method = Method.GET,
|
|
|
|
|
Resource = "/library/sections/{section}/all"
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
request.AddQueryParameter("type", 4.ToString());
|
|
|
|
|
AddLimitHeaders(ref request, startPage, returnCount);
|
|
|
|
|
request.AddUrlSegment("section", section);
|
|
|
|
|
AddHeaders(ref request, authToken, false);
|
|
|
|
|
|
|
|
|
|
try
|
|
|
|
|
{
|
|
|
|
|
var lib = RetryHandler.Execute(() => Api.ExecuteXml<PlexSearch>(request, host),
|
|
|
|
|
(exception, timespan) => Log.Error(exception, "Exception when calling GetAllEpisodes for Plex, Retrying {0}", timespan));
|
|
|
|
|
|
|
|
|
|
return lib;
|
|
|
|
|
}
|
|
|
|
|
catch (Exception e)
|
|
|
|
|
{
|
|
|
|
|
Log.Error(e, "There has been a API Exception when attempting to get GetAllEpisodes");
|
|
|
|
|
return new PlexSearch();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public PlexMetadata GetMetadata(string authToken, Uri plexFullHost, string itemId)
|
|
|
|
|
{
|
|
|
|
|
var request = new RestRequest
|
|
|
|
|
{
|
|
|
|
|
Method = Method.GET,
|
|
|
|
|
Resource = "library/metadata/{itemId}"
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
request.AddUrlSegment("itemId", itemId);
|
|
|
|
|
AddHeaders(ref request, authToken, false);
|
|
|
|
|
|
|
|
|
|
try
|
|
|
|
|
{
|
|
|
|
|
var lib = RetryHandler.Execute(() => Api.ExecuteXml<PlexMetadata>(request, plexFullHost),
|
|
|
|
|
(exception, timespan) => Log.Error(exception, "Exception when calling GetMetadata for Plex, Retrying {0}", timespan), new[] {
|
|
|
|
|
TimeSpan.FromSeconds (5),
|
|
|
|
|
TimeSpan.FromSeconds(10),
|
|
|
|
|
TimeSpan.FromSeconds(30)
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
return lib;
|
|
|
|
|
}
|
|
|
|
|
catch (Exception e)
|
|
|
|
|
{
|
|
|
|
|
Log.Error(e, "There has been a API Exception when attempting to get the Plex GetMetadata");
|
|
|
|
|
return new PlexMetadata();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
public PlexSeasonMetadata GetSeasons(string authToken, Uri plexFullHost, string ratingKey)
|
|
|
|
|
{
|
|
|
|
|
var request = new RestRequest
|
|
|
|
|
{
|
|
|
|
|
Method = Method.GET,
|
|
|
|
|
Resource = "library/metadata/{ratingKey}/children"
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
request.AddUrlSegment("ratingKey", ratingKey);
|
|
|
|
|
AddHeaders(ref request, authToken, false);
|
|
|
|
|
|
|
|
|
|
try
|
|
|
|
|
{
|
|
|
|
|
var lib = RetryHandler.Execute(() => Api.ExecuteXml<PlexSeasonMetadata>(request, plexFullHost),
|
|
|
|
|
(exception, timespan) => Log.Error(exception, "Exception when calling GetMetadata for Plex, Retrying {0}", timespan), new[] {
|
|
|
|
|
TimeSpan.FromSeconds (5),
|
|
|
|
|
TimeSpan.FromSeconds(10),
|
|
|
|
|
TimeSpan.FromSeconds(30)
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
return lib;
|
|
|
|
|
}
|
|
|
|
|
catch (Exception e)
|
|
|
|
|
{
|
|
|
|
|
Log.Error(e, "There has been a API Exception when attempting to get the Plex GetMetadata");
|
|
|
|
|
return new PlexSeasonMetadata();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public PlexServer GetServer(string authToken)
|
|
|
|
|
{
|
|
|
|
|
var request = new RestRequest
|
|
|
|
|
{
|
|
|
|
|
Method = Method.GET,
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
AddHeaders(ref request, authToken, false);
|
|
|
|
|
|
|
|
|
|
var servers = RetryHandler.Execute(() => Api.ExecuteXml<PlexServer>(request, new Uri(ServerUri)),
|
|
|
|
|
(exception, timespan) => Log.Error(exception, "Exception when calling GetServer for Plex, Retrying {0}", timespan));
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
return servers;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public RecentlyAddedModelOld RecentlyAddedOld(string authToken, Uri plexFullHost, string sectionId)
|
|
|
|
|
{
|
|
|
|
|
var request = new RestRequest
|
|
|
|
|
{
|
|
|
|
|
Method = Method.GET,
|
|
|
|
|
Resource = "library/sections/{sectionId}/recentlyAdded"
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
request.AddUrlSegment("sectionId", sectionId);
|
|
|
|
|
AddHeaders(ref request, authToken, true);
|
|
|
|
|
AddLimitHeaders(ref request, 0, 25);
|
|
|
|
|
|
|
|
|
|
try
|
|
|
|
|
{
|
|
|
|
|
var lib = RetryHandler.Execute(() => Api.ExecuteJson<RecentlyAddedModelOld>(request, plexFullHost),
|
|
|
|
|
(exception, timespan) => Log.Error(exception, "Exception when calling RecentlyAddedModel for Plex, Retrying {0}", timespan), new[] {
|
|
|
|
|
TimeSpan.FromSeconds (5),
|
|
|
|
|
TimeSpan.FromSeconds(10),
|
|
|
|
|
TimeSpan.FromSeconds(30)
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
return lib;
|
|
|
|
|
}
|
|
|
|
|
catch (Exception e)
|
|
|
|
|
{
|
|
|
|
|
Log.Error(e, "There has been a API Exception when attempting to get the Plex RecentlyAddedModel");
|
|
|
|
|
return new RecentlyAddedModelOld();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public PlexRecentlyAddedModel RecentlyAdded(string authToken, Uri plexFullHost, string sectionId)
|
|
|
|
|
{
|
|
|
|
|
var request = new RestRequest
|
|
|
|
|
{
|
|
|
|
|
Method = Method.GET,
|
|
|
|
|
Resource = "library/sections/{sectionId}/recentlyAdded"
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
request.AddUrlSegment("sectionId", sectionId);
|
|
|
|
|
AddHeaders(ref request, authToken, true);
|
|
|
|
|
AddLimitHeaders(ref request, 0, 25);
|
|
|
|
|
|
|
|
|
|
try
|
|
|
|
|
{
|
|
|
|
|
var lib = RetryHandler.Execute(() => Api.ExecuteJson<PlexRecentlyAddedModel>(request, plexFullHost),
|
|
|
|
|
(exception, timespan) => Log.Error(exception, "Exception when calling PlexRecentlyAddedModel for Plex, Retrying {0}", timespan), new[] {
|
|
|
|
|
TimeSpan.FromSeconds (5),
|
|
|
|
|
TimeSpan.FromSeconds(10),
|
|
|
|
|
TimeSpan.FromSeconds(30)
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
return lib;
|
|
|
|
|
}
|
|
|
|
|
catch (Exception e)
|
|
|
|
|
{
|
|
|
|
|
Log.Error(e, "There has been a API Exception when attempting to get the Plex RecentlyAddedModel");
|
|
|
|
|
return new PlexRecentlyAddedModel();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private void AddLimitHeaders(ref RestRequest request, int from, int to)
|
|
|
|
|
{
|
|
|
|
|
request.AddHeader("X-Plex-Container-Start", from.ToString());
|
|
|
|
|
request.AddHeader("X-Plex-Container-Size", to.ToString());
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private void AddHeaders(ref RestRequest request, string authToken, bool json)
|
|
|
|
|
{
|
|
|
|
|
request.AddHeader("X-Plex-Token", authToken);
|
|
|
|
|
AddHeaders(ref request, json);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private void AddHeaders(ref RestRequest request, bool json)
|
|
|
|
|
{
|
|
|
|
|
request.AddHeader("X-Plex-Client-Identifier", $"Ombi{Version}");
|
|
|
|
|
request.AddHeader("X-Plex-Product", "Ombi");
|
|
|
|
|
request.AddHeader("X-Plex-Version", Version);
|
|
|
|
|
request.AddHeader("Content-Type", json ? "application/json" : "application/xml");
|
|
|
|
|
request.AddHeader("Accept", json ? "application/json" : "application/xml");
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|