duplicati/Duplicati/WebserverCore/Abstractions/IJWTTokenProvider.cs
Kenneth Skovhede e3f1aefec2 Implemented a nonce for refresh tokens
This adds a nonce to the refresh token such that each request to obtain a refresh token must now also provide a matching nonce.

When using non-persisted logins, the request to the server is the same, but the "remember me" flag toggles a shorter duration for the refresh token.

The FE can then store the nonce in either local storage for persisted logins or in session storage for non-persisted logins.

The default is currently to always issue refresh tokens with a nonce, but this can be toggled with the JWT configuration.

The ngax client does not have the non-persisted login so it stores the nonce in local storage, using a name that is compatible with ngclient so the user can swap between them without needing to re-login.

The server util was updated to also store the nonce.

This fixes #6451
2025-08-07 23:10:54 +02:00

129 lines
6.6 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.
namespace Duplicati.WebserverCore.Abstractions;
/// <summary>
/// Provides methods to create and read JWT tokens.
/// </summary>
public interface IJWTTokenProvider
{
/// <summary>
/// Represents a JWT token that can be used for a single operation.
/// </summary>
/// <param name="ValidFrom">The time the token was created.</param>
/// <param name="Expiration">The time the token expires.</param>
/// <param name="UserId">The user ID the token is for.</param>
/// <param name="Operation">The operation the token is for.</param>
public record SingleOperationToken(DateTimeOffset ValidFrom, DateTimeOffset Expiration, string UserId, string Operation);
/// <summary>
/// Represents a JWT token that can be used to sign in, instead of using a password.
/// </summary>
/// <param name="ValidFrom">The time the token was created.</param>
/// <param name="Expiration">The time the token expires.</param>
/// <param name="UserId">The user ID the token is for.</param>
public record SigninToken(DateTimeOffset ValidFrom, DateTimeOffset Expiration, string UserId);
/// <summary>
/// Represents a JWT token that can be used to access resources.
/// </summary>
/// <param name="ValidFrom">The time the token was created.</param>
/// <param name="Expiration">The time the token expires.</param>
/// <param name="TokenFamilyId">The token family ID the token is for.</param>
/// <param name="UserId">The user ID the token is for.</param>
public record AccessToken(DateTimeOffset ValidFrom, DateTimeOffset Expiration, string TokenFamilyId, string UserId);
/// <summary>
/// Represents a JWT token that can be used to refresh an access token.
/// </summary>
/// <param name="ValidFrom">The time the token was created.</param>
/// <param name="Expiration">The time the token expires.</param>
/// <param name="TokenFamilyId">The token family ID the token is for.</param>
/// <param name="UserId">The user ID the token is for.</param>
/// <param name="Counter">The counter of the token family the token is for.</param>
public record RefreshToken(DateTimeOffset ValidFrom, DateTimeOffset Expiration, string TokenFamilyId, string UserId, int Counter);
/// <summary>
/// Creates a JWT token that only works for a single operation.
/// </summary>
/// <param name="userId">The user ID the token is for.</param>
/// <param name="operation">The operation the token is for.</param>
/// <returns>The JWT token.</returns>
string CreateSingleOperationToken(string userId, string operation);
/// <summary>
/// Creates a JWT token that can be used to sign in, instead of using a password.
/// </summary>
/// <param name="userId">The user ID the token is for.</param>
/// <returns>The JWT token.</returns>
string CreateSigninToken(string userId);
/// <summary>
/// Creates a JWT token that can be used to access resources.
/// </summary>
/// <param name="userId">The user ID the token is for.</param>
/// <param name="tokenFamilyId">The token family ID the token is for.</param>
/// <param name="expiration">The expiration time of the token, can only be shorter than the current.</param>
/// <returns>The JWT token.</returns>
string CreateAccessToken(string userId, string tokenFamilyId, TimeSpan? expiration = null);
/// <summary>
/// Creates a JWT token that can be used to access resources &quot;forever&quot;.
/// </summary>
/// <returns>The JWT token.</returns>
string CreateForeverToken();
/// <summary>
/// Creates a JWT token that can be used to refresh an access token.
/// </summary>
/// <param name="userId">The user ID the token is for.</param>
/// <param name="tokenFamilyId">The token family ID the token is for.</param>
/// <param name="counter">The counter of the token family the token is for.</param>
/// <param name="shortLived">Whether the token should be short-lived.</param>
/// <returns>The JWT token and the nonce.</returns>
(string RefreshToken, string? Nonce) CreateRefreshToken(string userId, string tokenFamilyId, int counter, bool shortLived);
/// <summary>
/// Reads a JWT token that only works for a single operation.
/// </summary>
/// <param name="token">The JWT token.</param>
/// <returns>The parsed and validated single operation token.</returns>
SingleOperationToken ReadSingleOperationToken(string token);
/// <summary>
/// Reads a JWT token that can be used to sign in, instead of using a password.
/// </summary>
/// <param name="token">The JWT token.</param>
/// <returns>The parsed and validated sign-in token.</returns>
SigninToken ReadSigninToken(string token);
/// <summary>
/// Reads a JWT token that can be used to access resources.
/// </summary>
/// <param name="token">The JWT token.</param>
/// <returns>The parsed and validated access token.</returns>
AccessToken ReadAccessToken(string token);
/// <summary>
/// Reads a JWT token that can be used to refresh an access token.
/// </summary>
/// <param name="token">The JWT token.</param>
/// <param name="nonce">The nonce used to validate the token.</param>
/// <returns>The parsed and validated refresh token.</returns>
RefreshToken ReadRefreshToken(string token, string? nonce);
/// <summary>
/// Gets the family ID from a JWT token with no family counter.
/// </summary>
string TemporaryFamilyId { get; }
}