#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. // ************************************************************************/ using Polly; #endregion using System; using NLog; using PlexRequests.Api.Interfaces; using PlexRequests.Api.Models.Plex; using PlexRequests.Helpers; using PlexRequests.Helpers.Exceptions; using RestSharp; namespace PlexRequests.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 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); request.AddJsonBody(userModel); var obj = RetryHandler.Execute(() => Api.Execute (request, new Uri(SignInUri)), null, (exception, timespan) => Log.Error (exception, "Exception when calling SignIn for Plex, Retrying {0}", timespan)); return obj; } public PlexFriends GetUsers(string authToken) { var request = new RestRequest { Method = Method.GET, }; AddHeaders(ref request, authToken); var users = RetryHandler.Execute(() => Api.ExecuteXml (request, new Uri(FriendsUri)), null, (exception, timespan) => Log.Error (exception, "Exception when calling GetUsers for Plex, Retrying {0}", timespan)); return users; } /// /// Gets the users. /// /// The authentication token. /// The search term. /// The full plex host. /// 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); var search = RetryHandler.Execute(() => Api.ExecuteXml (request, plexFullHost), null, (exception, timespan) => Log.Error (exception, "Exception when calling SearchContent for Plex, Retrying {0}", timespan)); return search; } public PlexStatus GetStatus(string authToken, Uri uri) { var request = new RestRequest { Method = Method.GET, }; AddHeaders(ref request, authToken); var users = RetryHandler.Execute(() => Api.ExecuteXml (request, uri), null, (exception, timespan) => Log.Error (exception, "Exception when calling GetStatus for Plex, Retrying {0}", timespan)); return users; } public PlexAccount GetAccount(string authToken) { var request = new RestRequest { Method = Method.GET, }; AddHeaders(ref request, authToken); var account = RetryHandler.Execute(() => Api.ExecuteXml (request, new Uri(GetAccountUri)), null, (exception, timespan) => Log.Error (exception, "Exception when calling GetAccount for Plex, Retrying {0}", timespan)); return account; } public PlexLibraries GetLibrarySections(string authToken, Uri plexFullHost) { var request = new RestRequest { Method = Method.GET, Resource = "library/sections" }; AddHeaders(ref request, authToken); try { var lib = RetryHandler.Execute(() => Api.ExecuteXml (request, plexFullHost), new TimeSpan[] { TimeSpan.FromSeconds (5), TimeSpan.FromSeconds(10), TimeSpan.FromSeconds(30) }, (exception, timespan) => Log.Error (exception, "Exception when calling GetLibrarySections for Plex, Retrying {0}", timespan)); 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); try { var lib = RetryHandler.Execute(() => Api.ExecuteXml (request, plexFullHost), new TimeSpan[] { TimeSpan.FromSeconds (5), TimeSpan.FromSeconds(10), TimeSpan.FromSeconds(30) }, (exception, timespan) => Log.Error (exception, "Exception when calling GetLibrary for Plex, Retrying {0}", timespan)); 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 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); try { var lib = RetryHandler.Execute(() => Api.ExecuteXml(request, plexFullHost), new[] { TimeSpan.FromSeconds (5), TimeSpan.FromSeconds(10), TimeSpan.FromSeconds(30) }, (exception, timespan) => Log.Error(exception, "Exception when calling GetMetadata for Plex, Retrying {0}", timespan)); return lib; } catch (Exception e) { Log.Error(e, "There has been a API Exception when attempting to get the Plex GetMetadata"); return new PlexMetadata(); } } private void AddHeaders(ref RestRequest request, string authToken) { request.AddHeader("X-Plex-Token", authToken); AddHeaders(ref request); } private void AddHeaders(ref RestRequest request) { request.AddHeader("X-Plex-Client-Identifier", "Test213"); request.AddHeader("X-Plex-Product", "Request Plex"); request.AddHeader("X-Plex-Version", Version); request.AddHeader("Content-Type", "application/xml"); } } }