mirror of
https://github.com/Radarr/Radarr
synced 2025-01-01 12:54:21 +00:00
Merge pull request #778 from geogolem/traktAuthentication
fully functional traktAuthentication
This commit is contained in:
commit
f49d68ad6a
8 changed files with 228 additions and 35 deletions
|
@ -40,6 +40,7 @@
|
|||
"run-sequence": "1.1.1",
|
||||
"streamqueue": "1.1.0",
|
||||
"tar.gz": "0.1.1",
|
||||
"url-search-params": "^0.6.1",
|
||||
"webpack": "1.12.0",
|
||||
"webpack-stream": "2.1.0"
|
||||
}
|
||||
|
|
|
@ -8,6 +8,9 @@ public class NetImportConfigResource : RestResource
|
|||
public int NetImportSyncInterval { get; set; }
|
||||
public string ListSyncLevel { get; set; }
|
||||
public string ImportExclusions { get; set; }
|
||||
public string TraktAuthToken { get; set; }
|
||||
public string TraktRefreshToken { get; set; }
|
||||
public int TraktTokenExpiry { get; set; }
|
||||
}
|
||||
|
||||
public static class NetImportConfigResourceMapper
|
||||
|
@ -19,6 +22,9 @@ public static NetImportConfigResource ToResource(IConfigService model)
|
|||
NetImportSyncInterval = model.NetImportSyncInterval,
|
||||
ListSyncLevel = model.ListSyncLevel,
|
||||
ImportExclusions = model.ImportExclusions,
|
||||
TraktAuthToken = model.TraktAuthToken,
|
||||
TraktRefreshToken = model.TraktRefreshToken,
|
||||
TraktTokenExpiry = model.TraktTokenExpiry,
|
||||
};
|
||||
}
|
||||
}
|
||||
|
|
|
@ -117,6 +117,45 @@ public int NetImportSyncInterval
|
|||
|
||||
set { SetValue("NetImportSyncInterval", value); }
|
||||
}
|
||||
|
||||
public string TraktAuthToken
|
||||
{
|
||||
get { return GetValue("TraktAuthToken", string.Empty); }
|
||||
|
||||
set { SetValue("TraktAuthToken", value); }
|
||||
}
|
||||
|
||||
public string TraktRefreshToken
|
||||
{
|
||||
get { return GetValue("TraktRefreshToken", string.Empty); }
|
||||
|
||||
set {SetValue("TraktRefreshToken", value); }
|
||||
}
|
||||
|
||||
public int TraktTokenExpiry
|
||||
{
|
||||
get { return GetValueInt("TraktTokenExpiry", 0); }
|
||||
|
||||
set { SetValue("TraktTokenExpiry", value); }
|
||||
}
|
||||
|
||||
public string NewTraktAuthToken
|
||||
{
|
||||
get {return GetValue("NewTraktAuthToken", string.Empty); }
|
||||
set { SetValue("NewTraktAuthToken", value); }
|
||||
}
|
||||
|
||||
public string NewTraktRefreshToken
|
||||
{
|
||||
get {return GetValue("NewTraktRefreshToken", string.Empty); }
|
||||
set { SetValue("NewTraktRefreshToken", value); }
|
||||
}
|
||||
|
||||
public int NewTraktTokenExpiry
|
||||
{
|
||||
get {return GetValueInt("NewTraktTokenExpiry", 0); }
|
||||
set { SetValue("NewTraktTokenExpiry", value); }
|
||||
}
|
||||
|
||||
public string ListSyncLevel
|
||||
{
|
||||
|
|
|
@ -51,6 +51,12 @@ public interface IConfigService
|
|||
int NetImportSyncInterval { get; set; }
|
||||
string ListSyncLevel { get; set; }
|
||||
string ImportExclusions { get; set; }
|
||||
string TraktAuthToken { get; set; }
|
||||
string TraktRefreshToken { get; set; }
|
||||
int TraktTokenExpiry { get; set; }
|
||||
string NewTraktAuthToken { get; set; }
|
||||
string NewTraktRefreshToken {get; set; }
|
||||
int NewTraktTokenExpiry { get; set; }
|
||||
|
||||
//UI
|
||||
int FirstDayOfWeek { get; set; }
|
||||
|
@ -60,6 +66,7 @@ public interface IConfigService
|
|||
string LongDateFormat { get; set; }
|
||||
string TimeFormat { get; set; }
|
||||
bool ShowRelativeDates { get; set; }
|
||||
|
||||
bool EnableColorImpairedMode { get; set; }
|
||||
|
||||
//Internal
|
||||
|
|
|
@ -10,14 +10,15 @@ public class TraktImport : HttpNetImportBase<TraktSettings>
|
|||
public override string Name => "Trakt List";
|
||||
public override bool Enabled => true;
|
||||
public override bool EnableAuto => false;
|
||||
public IConfigService _configService;
|
||||
|
||||
public TraktImport(IHttpClient httpClient, IConfigService configService, IParsingService parsingService, Logger logger)
|
||||
: base(httpClient, configService, parsingService, logger)
|
||||
{ }
|
||||
{ _configService = configService; }
|
||||
|
||||
public override INetImportRequestGenerator GetRequestGenerator()
|
||||
{
|
||||
return new TraktRequestGenerator() { Settings = Settings };
|
||||
return new TraktRequestGenerator() { Settings = Settings, _configService=_configService };
|
||||
}
|
||||
|
||||
public override IParseNetImportResponse GetParser()
|
||||
|
|
|
@ -1,10 +1,26 @@
|
|||
using NzbDrone.Common.Http;
|
||||
using System;
|
||||
using System.Collections.Generic;
|
||||
using System.Net;
|
||||
using System.IO;
|
||||
using Newtonsoft.Json.Linq;
|
||||
using NzbDrone.Core.Configuration;
|
||||
|
||||
|
||||
namespace NzbDrone.Core.NetImport.Trakt
|
||||
{
|
||||
public class refreshRequestResponse
|
||||
{
|
||||
public string access_token { get; set; }
|
||||
public string token_type { get; set; }
|
||||
public int expires_in { get; set; }
|
||||
public string refresh_token { get; set; }
|
||||
public string scope { get; set; }
|
||||
}
|
||||
|
||||
public class TraktRequestGenerator : INetImportRequestGenerator
|
||||
{
|
||||
public IConfigService _configService;
|
||||
public TraktSettings Settings { get; set; }
|
||||
|
||||
public virtual NetImportPageableRequestChain GetMovies()
|
||||
|
@ -58,10 +74,52 @@ private IEnumerable<NetImportRequest> GetMovies(string searchParameters)
|
|||
link = link + "/movies/watched/all" + filters;
|
||||
break;
|
||||
}
|
||||
if (_configService.TraktRefreshToken != string.Empty)
|
||||
{
|
||||
//tokens were overwritten with something other than nothing
|
||||
if (_configService.NewTraktTokenExpiry > _configService.TraktTokenExpiry)
|
||||
{
|
||||
//but our refreshedTokens are more current
|
||||
_configService.TraktAuthToken = _configService.NewTraktAuthToken;
|
||||
_configService.TraktRefreshToken = _configService.NewTraktRefreshToken;
|
||||
_configService.TraktTokenExpiry = _configService.NewTraktTokenExpiry;
|
||||
}
|
||||
|
||||
Int32 unixTime= (Int32)(DateTime.UtcNow.Subtract(new DateTime(1970, 1, 1))).TotalSeconds;
|
||||
|
||||
if ( unixTime > _configService.TraktTokenExpiry)
|
||||
{
|
||||
var url = "http://radarr.aeonlucid.com/v1/trakt/refresh?refresh="+_configService.TraktRefreshToken;
|
||||
|
||||
HttpWebRequest rquest = (HttpWebRequest)WebRequest.Create(url);
|
||||
string rsponseString = string.Empty;
|
||||
using (HttpWebResponse rsponse = (HttpWebResponse)rquest.GetResponse())
|
||||
using (Stream stream = rsponse.GetResponseStream())
|
||||
using (StreamReader reader = new StreamReader(stream))
|
||||
{
|
||||
rsponseString = reader.ReadToEnd();
|
||||
}
|
||||
refreshRequestResponse j1 = Newtonsoft.Json.JsonConvert.DeserializeObject<refreshRequestResponse>(rsponseString);
|
||||
_configService.TraktAuthToken = j1.access_token;
|
||||
_configService.TraktRefreshToken = j1.refresh_token;
|
||||
|
||||
//lets have it expire in 8 weeks (4838400 seconds)
|
||||
_configService.TraktTokenExpiry = unixTime + 4838400;
|
||||
|
||||
//store the refreshed tokens in case they get overwritten by an old set of tokens
|
||||
_configService.NewTraktAuthToken = _configService.TraktAuthToken;
|
||||
_configService.NewTraktRefreshToken = _configService.TraktRefreshToken;
|
||||
_configService.NewTraktTokenExpiry = _configService.TraktTokenExpiry;
|
||||
}
|
||||
}
|
||||
|
||||
var request = new NetImportRequest($"{link}", HttpAccept.Json);
|
||||
request.HttpRequest.Headers.Add("trakt-api-version", "2");
|
||||
request.HttpRequest.Headers.Add("trakt-api-key", "657bb899dcb81ec8ee838ff09f6e013ff7c740bf0ccfa54dd41e791b9a70b2f0");
|
||||
request.HttpRequest.Headers.Add("trakt-api-key", "964f67b126ade0112c4ae1f0aea3a8fb03190f71117bd83af6a0560a99bc52e6"); //aeon
|
||||
if (_configService.TraktAuthToken != null)
|
||||
{
|
||||
request.HttpRequest.Headers.Add("Authorization", "Bearer " + _configService.TraktAuthToken);
|
||||
}
|
||||
|
||||
yield return request;
|
||||
}
|
||||
|
|
|
@ -6,11 +6,45 @@ require('../../../Mixins/TagInput');
|
|||
require('bootstrap');
|
||||
require('bootstrap.tagsinput');
|
||||
|
||||
var view = Marionette.ItemView.extend({
|
||||
template : 'Settings/NetImport/Options/NetImportOptionsViewTemplate',
|
||||
//if ('searchParams' in HTMLAnchorElement.prototype) {
|
||||
// var URLSearchParams = require('url-search-params-polyfill');
|
||||
//}
|
||||
|
||||
ui : {
|
||||
importExclusions : '.x-import-exclusions'
|
||||
var URLSearchParams = require('url-search-params');
|
||||
|
||||
var q = window.location;
|
||||
var callback_url = q.protocol+'//'+q.hostname+(q.port ? ':' + q.port : '')+'/settings/netimport';
|
||||
var view = Marionette.ItemView.extend({
|
||||
template : 'Settings/NetImport/Options/NetImportOptionsViewTemplate',
|
||||
events : {
|
||||
'click .x-reset-trakt-tokens' : '_resetTraktTokens',
|
||||
'click .x-revoke-trakt-tokens' : '_revokeTraktTokens'
|
||||
},
|
||||
|
||||
initialize : function() {
|
||||
|
||||
},
|
||||
|
||||
onShow : function() {
|
||||
var params = new URLSearchParams(window.location.search);
|
||||
var oauth = params.get('access');
|
||||
var refresh=params.get('refresh');
|
||||
if (oauth && refresh){
|
||||
history.pushState('object', 'title', callback_url);
|
||||
this.ui.authToken.val(oauth).trigger('change');
|
||||
this.ui.refreshToken.val(refresh).trigger('change');
|
||||
this.ui.tokenExpiry.val(Math.floor(Date.now() / 1000) + 4838400).trigger('change'); // this means the token will expire in 8 weeks (4838400 seconds)
|
||||
//this.model.isSaved = false;
|
||||
window.alert("Trakt Authentication Complete - Click Save to make the change take effect");
|
||||
}
|
||||
if (this.ui.authToken.val() && this.ui.refreshToken.val()){
|
||||
this.ui.resetTokensButton.hide();
|
||||
this.ui.revokeTokensButton.show();
|
||||
} else {
|
||||
this.ui.resetTokensButton.show();
|
||||
this.ui.revokeTokensButton.hide();
|
||||
}
|
||||
|
||||
},
|
||||
|
||||
onRender : function() {
|
||||
|
@ -20,53 +54,82 @@ var view = Marionette.ItemView.extend({
|
|||
/*itemText : function(item) {
|
||||
var uri;
|
||||
var text;
|
||||
if (item.startsWith('tt')) {
|
||||
uri = window.NzbDrone.ApiRoot + '/movies/lookup/imdb?imdbId='+item;
|
||||
}
|
||||
else {
|
||||
uri = window.NzbDrone.ApiRoot + '/movies/lookup/tmdb?tmdbId='+item;
|
||||
}
|
||||
var promise = $.ajax({
|
||||
url : uri,
|
||||
type : 'GET',
|
||||
async : false,
|
||||
});
|
||||
promise.success(function(response) {
|
||||
if (item.startsWith('tt')) {
|
||||
uri = window.NzbDrone.ApiRoot + '/movies/lookup/imdb?imdbId='+item;
|
||||
}
|
||||
else {
|
||||
uri = window.NzbDrone.ApiRoot + '/movies/lookup/tmdb?tmdbId='+item;
|
||||
}
|
||||
var promise = $.ajax({
|
||||
url : uri,
|
||||
type : 'GET',
|
||||
async : false,
|
||||
});
|
||||
promise.success(function(response) {
|
||||
text=response['title']+' ('+response['year']+')';
|
||||
});
|
||||
});
|
||||
|
||||
promise.error(function(request, status, error) {
|
||||
promise.error(function(request, status, error) {
|
||||
text=item;
|
||||
});
|
||||
});
|
||||
return text;
|
||||
}*/
|
||||
});
|
||||
this.ui.importExclusions.on('beforeItemAdd', function(event) {
|
||||
});
|
||||
this.ui.importExclusions.on('beforeItemAdd', function(event) {
|
||||
var uri;
|
||||
if (event.item.startsWith('tt')) {
|
||||
if (event.item.startsWith('tt')) {
|
||||
uri = window.NzbDrone.ApiRoot + '/movies/lookup/imdb?imdbId='+event.item;
|
||||
}
|
||||
else {
|
||||
uri = window.NzbDrone.ApiRoot + '/movies/lookup/tmdb?tmdbId='+event.item;
|
||||
uri = window.NzbDrone.ApiRoot + '/movies/lookup/tmdb?tmdbId='+event.item;
|
||||
}
|
||||
var promise = $.ajax({
|
||||
url : uri,
|
||||
type : 'GET',
|
||||
url : uri,
|
||||
type : 'GET',
|
||||
async : false,
|
||||
});
|
||||
});
|
||||
promise.success(function(response) {
|
||||
event.cancel=false;
|
||||
});
|
||||
});
|
||||
|
||||
promise.error(function(request, status, error) {
|
||||
promise.error(function(request, status, error) {
|
||||
event.cancel = true;
|
||||
window.alert(event.item+' is not a valid! Must be valid tt#### IMDB ID or #### TMDB ID');
|
||||
});
|
||||
});
|
||||
window.alert(event.item+' is not a valid! Must be valid tt#### IMDB ID or #### TMDB ID');
|
||||
});
|
||||
});
|
||||
},
|
||||
|
||||
ui : {
|
||||
resetTraktTokens : '.x-reset-trakt-tokens',
|
||||
authToken : '.x-trakt-auth-token',
|
||||
refreshToken : '.x-trakt-refresh-token',
|
||||
resetTokensButton : '.x-reset-trakt-tokens',
|
||||
revokeTokensButton : '.x-revoke-trakt-tokens',
|
||||
tokenExpiry : '.x-trakt-token-expiry',
|
||||
importExclusions : '.x-import-exclusions'
|
||||
},
|
||||
|
||||
_resetTraktTokens : function() {
|
||||
if (window.confirm("Proceed to trakt.tv for authentication?\nYou will then be redirected back here.")){
|
||||
window.location='http://radarr.aeonlucid.com/v1/trakt/redirect?target='+callback_url;
|
||||
//this.ui.resetTokensButton.hide();
|
||||
}
|
||||
},
|
||||
|
||||
_revokeTraktTokens : function() {
|
||||
if (window.confirm("Log out of trakt.tv?")){
|
||||
//TODO: need to implement this: http://docs.trakt.apiary.io/#reference/authentication-oauth/revoke-token/revoke-an-access_token
|
||||
this.ui.authToken.val('').trigger('change');
|
||||
this.ui.refreshToken.val('').trigger('change');
|
||||
this.ui.tokenExpiry.val(0).trigger('change');
|
||||
this.ui.resetTokensButton.show();
|
||||
this.ui.revokeTokensButton.hide();
|
||||
window.alert("Logged out of Trakt.tv - Click Save to make the change take effect");
|
||||
}
|
||||
}
|
||||
});
|
||||
|
||||
|
||||
AsModelBoundView.call(view);
|
||||
AsValidatedView.call(view);
|
||||
|
||||
|
|
|
@ -41,4 +41,22 @@
|
|||
<input type="text" name="importExclusions" class="form-control x-import-exclusions"/>
|
||||
</div>
|
||||
</div>
|
||||
</fieldset>
|
||||
<legend>Trakt Authentication</legend>
|
||||
<div class="form-group">
|
||||
<label class="col-sm-1 control-label">Auth Token</label>
|
||||
<div class="col-sm-4">
|
||||
<input type="text" readonly="readonly" name="traktAuthToken" class="form-control x-trakt-auth-token"/>
|
||||
<input type="hidden" readonly="readonly" name="traktTokenExpiry" class="form-control x-trakt-token-expiry"/>
|
||||
</div>
|
||||
</div>
|
||||
<div class="form-group">
|
||||
<label class="col-sm-1 control-label">Refresh Token</label>
|
||||
<div class="col-sm-4">
|
||||
<input type="text" readonly="readonly" name="traktRefreshToken" class="form-control x-trakt-refresh-token"/>
|
||||
</div>
|
||||
<div class="input-group-btn">
|
||||
<button class="btn btn-danger btn-icon-only x-reset-trakt-tokens" title="Reset Trakt Tokens"><i class="icon-sonarr-refresh"></i></button>
|
||||
<button class="btn btn-danger btn-icon-only x-revoke-trakt-tokens" title="Revoke Trakt Tokens"><i class="icon-sonarr-logout"></i></button>
|
||||
</div >
|
||||
</div>
|
||||
</fieldset>
|
||||
|
|
Loading…
Reference in a new issue