duplicati/Duplicati/Library/Backend/Jottacloud/JottacloudAuthHelper.cs

89 lines
No EOL
3.9 KiB
C#

// Copyright (C) 2025, The Duplicati Team
// https://duplicati.com, hello@duplicati.com
//
// 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 Duplicati.Library.Interface;
using Duplicati.Library.Utility;
using Newtonsoft.Json;
namespace Duplicati.Library.Backend;
public class JottacloudAuthHelper : OAuthHelperHttpClient, IDisposable
{
private const string USERINFO_URL = "https://id.jottacloud.com/auth/realms/jottacloud/protocol/openid-connect/userinfo";
private JottacloudAuthHelper(HttpClient httpClient, string accessToken, string oauthUrl)
: base(accessToken, "jottacloud", oauthUrl, httpClient) // This will purposefully create a new HttpClient instance per AuthHelper instance
{
AutoAuthHeader = true;
AutoV2 = false; // Jottacloud is not v2 compatible because it generates a new refresh token with every access token refresh and invalidates the old.
Username = null!; // Setting via initializer
}
public static async Task<JottacloudAuthHelper> CreateAsync(string accessToken, string oauthUrl, CancellationToken cancellationToken = default)
{
var httpClient = HttpClientHelper.CreateClient();
httpClient.Timeout = Timeout.InfiniteTimeSpan;
try
{
var inst = new JottacloudAuthHelper(httpClient, accessToken, oauthUrl);
var userinfo = await inst.GetJsonDataAsync<UserInfo>(USERINFO_URL, cancellationToken);
if (userinfo == null || string.IsNullOrEmpty(userinfo.Username))
throw new UserInformationException(Strings.Jottacloud.NoUsernameError, "JottaNoUsername");
inst.Username = userinfo.Username;
return inst;
}
catch (Exception ex)
{
httpClient.Dispose();
throw new UserInformationException($"Failed to retrieve user information: {ex.Message}", "JottaUserInfoError", ex);
}
}
public string Username { get; private set; }
private class UserInfo
{
[JsonProperty("sub")]
public string? Subject { get; set; }
[JsonProperty("email_verified")]
public bool EmailVerified { get; set; }
[JsonProperty("name")]
public string? Name { get; set; }
[JsonProperty("realm")]
public string? Realm { get; set; }
[JsonProperty("preferred_username")]
public string? PreferredUsername { get; set; } // The numeric internal username, same as Username
[JsonProperty("given_name")]
public string? GivenName { get; set; }
[JsonProperty("family_name")]
public string? FamilyName { get; set; }
[JsonProperty("email")]
public string? Email { get; set; }
[JsonProperty("username")]
public string? Username { get; set; } // The numeric internal username, same as PreferredUsername
}
public void Dispose()
{
_httpClient?.Dispose();
}
}