-
Notifications
You must be signed in to change notification settings - Fork 1
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking โSign up for GitHubโ, you agree to our terms of service and privacy statement. Weโll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Adds LicensingService
#40
base: main
Are you sure you want to change the base?
Changes from 1 commit
14e878b
42d5b7b
5448139
aede3ce
b08a2ab
e49220d
7dd8a28
43eb08c
2cc8f7a
2abf460
bdf7a03
5597d20
8154ce5
04c405f
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
- Loading branch information
There are no files selected for viewing
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,93 @@ | ||
using System.IdentityModel.Tokens.Jwt; | ||
using System.Security.Claims; | ||
using Microsoft.Extensions.Options; | ||
using Microsoft.IdentityModel.Tokens; | ||
|
||
namespace Bitwarden.Extensions.Hosting.Licensing; | ||
|
||
internal sealed class DefaultLicensingService : ILicensingService | ||
{ | ||
private readonly LicensingOptions _licensingOptions; | ||
private readonly TimeProvider _timeProvider; | ||
|
||
public DefaultLicensingService(IOptions<LicensingOptions> licensingOptions, TimeProvider timeProvider) | ||
{ | ||
ArgumentNullException.ThrowIfNull(licensingOptions); | ||
|
||
// TODO: Do we need to support runtime changes to these settings at all, I don't think we do... | ||
_licensingOptions = licensingOptions.Value; | ||
_timeProvider = timeProvider; | ||
|
||
// We are cloud if the signing certificate has a private key that can sign licenses and local development | ||
// hasn't forced self host. | ||
IsCloud = _licensingOptions.SigningCertificate.HasPrivateKey && !_licensingOptions.ForceSelfHost; | ||
} | ||
|
||
public bool IsCloud { get; } | ||
|
||
public string CreateLicense(IEnumerable<Claim> claims, TimeSpan validFor) | ||
{ | ||
ArgumentNullException.ThrowIfNull(claims); | ||
ArgumentOutOfRangeException.ThrowIfLessThanOrEqual(validFor, TimeSpan.Zero); | ||
|
||
if (!IsCloud) | ||
{ | ||
throw new InvalidOperationException("Self-hosted services can not create a license, please check 'IsCloud' before calling this method."); | ||
} | ||
|
||
var now = _timeProvider.GetUtcNow().UtcDateTime; | ||
|
||
|
||
var tokenDescriptor = new SecurityTokenDescriptor | ||
{ | ||
Subject = new ClaimsIdentity(claims), | ||
// Issuer = "bitwarden.com", // TODO: Is this what we want? | ||
// Audience = "configurable_product_name??", // TODO: Is this what we want? | ||
SigningCredentials = new SigningCredentials( | ||
new X509SecurityKey(_licensingOptions.SigningCertificate), SecurityAlgorithms.RsaSha256), | ||
IssuedAt = now, | ||
NotBefore = now, | ||
Expires = now.Add(validFor), | ||
}; | ||
|
||
var tokenHandler = new JwtSecurityTokenHandler(); | ||
|
||
var token = tokenHandler.CreateToken(tokenDescriptor); | ||
|
||
return tokenHandler.WriteToken(token); | ||
} | ||
|
||
public async Task<IEnumerable<Claim>> VerifyLicenseAsync(string license) | ||
{ | ||
ArgumentNullException.ThrowIfNull(license); | ||
// TODO: Should we validate that this is self host? | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. โน๏ธ I'd say no. I can make up some ideas around how we'd offer license verification capabilities somewhere new so we don't need to constrain this. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Sweet, I concur. |
||
// It's not technically wrong to be able to do that but we don't do it currently | ||
// so we could disallow it. | ||
|
||
var tokenHandler = new JwtSecurityTokenHandler(); | ||
|
||
if (!tokenHandler.CanReadToken(license)) | ||
{ | ||
throw new InvalidLicenseException(InvalidLicenseReason.InvalidFormat); | ||
} | ||
|
||
var tokenValidateParameters = new TokenValidationParameters | ||
{ | ||
IssuerSigningKey = new X509SecurityKey(_licensingOptions.SigningCertificate), | ||
ValidateIssuerSigningKey = true, | ||
ValidateLifetime = true, | ||
ValidateIssuer = false, // TODO: Do we want to have no issuer? | ||
ValidateAudience = false, // TODO: Do we want to have no audience? | ||
}; | ||
|
||
var tokenValidationResult = await tokenHandler.ValidateTokenAsync(license, tokenValidateParameters); | ||
|
||
if (!tokenValidationResult.IsValid) | ||
{ | ||
throw tokenValidationResult.Exception; | ||
} | ||
|
||
// Should I even take a ClaimsIdentity and return it here instead of a list of claims? | ||
return tokenValidationResult.ClaimsIdentity.Claims; | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. โน๏ธ I took a look and don't see anything else we'd want to return really. |
||
} | ||
} |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,35 @@ | ||
using System.Security.Claims; | ||
|
||
namespace Bitwarden.Extensions.Hosting.Licensing; | ||
|
||
/// <summary> | ||
/// A service with the ability to consume and create licenses. | ||
/// </summary> | ||
public interface ILicensingService | ||
{ | ||
/// <summary> | ||
/// Returns whether or not the current service is running as a cloud instance. | ||
/// </summary> | ||
bool IsCloud { get; } | ||
|
||
// TODO: Any other options than valid for? | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. ๐ญ Limit the product names enabled perhaps? Made up constraint by me. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. I personally don't want to limit the product names in anyway, if we did it would require that we update this library whenever we create a new product, which feels like an artificial slowdown. It would also do nothing to stop one product from using another products name. But I do think that maybe taking the product name in this method could be a good idea. Right now, this has a limitation of creating licenses for only a single products licenses per project. Maybe that is a limitation we don't want. Just adding the product name here though would still have all products share the same signing certificate which maybe we'd want that configurable per product too. But forcing the product name to be passed in here does force them to configure it, vs now I allow it to be configured but grab the Also, I think I am going back on accepting a There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. I'm not opposed to updating this when a product is added -- that's a major event that requires preparation. That said, it has pitfalls and I am just making this up so we don't need to add it; if we want to license differently in the future we can figure it out then. Renaming a I like the idea of allowing a set time to be the input instead. Let the caller figure out when they want it to expire. |
||
/// <summary> | ||
/// Creates a signed license that can be consumed on self-hosted instances. | ||
/// </summary> | ||
/// <remarks> | ||
/// This method can only be called when <see cref="IsCloud"/> returns <see langword="true" />. | ||
/// </remarks> | ||
/// <param name="claims">The claims to include in the license file.</param> | ||
/// <param name="validFor">How long the license should be valid for.</param> | ||
/// <returns> | ||
/// A string representation of the license that can be given to people to store with their self hosted instance. | ||
/// </returns> | ||
string CreateLicense(IEnumerable<Claim> claims, TimeSpan validFor); | ||
|
||
/// <summary> | ||
/// Verifies that the given license is valid and can have it's contents be trusted. | ||
/// </summary> | ||
/// <param name="license">The license to check.</param> | ||
/// <returns>An enumerable of claims included in the license.</returns> | ||
Task<IEnumerable<Claim>> VerifyLicenseAsync(string license); | ||
} |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,66 @@ | ||
namespace Bitwarden.Extensions.Hosting.Licensing; | ||
|
||
/// <summary> | ||
/// A set of reasons explaining why a license was invalid. | ||
/// </summary> | ||
public enum InvalidLicenseReason | ||
{ | ||
/// <summary> | ||
/// The given license was in an invalid format and could not be read further. | ||
/// </summary> | ||
InvalidFormat, | ||
|
||
/// <summary> | ||
/// The given license may have been valid previously but has expired and should no longer be used. | ||
/// </summary> | ||
Expired, | ||
|
||
/// <summary> | ||
/// The license is invalid for an unknown reason, checks logs for additional details. | ||
/// </summary> | ||
Unknown, | ||
} | ||
|
||
/// <summary> | ||
/// The exception that is thrown when a license is invalid and cannot be verified. | ||
/// </summary> | ||
public class InvalidLicenseException : Exception | ||
{ | ||
private const string DefaultMessage = "The license is invalid and cannot be trusted."; | ||
|
||
/// <summary> | ||
/// Initializes a new instance of <see cref="InvalidLicenseException"/>. | ||
/// </summary> | ||
/// <param name="reason"></param> | ||
public InvalidLicenseException(InvalidLicenseReason reason) | ||
: base(DefaultMessage) | ||
{ | ||
Reason = reason; | ||
} | ||
|
||
/// <summary> | ||
/// Initializes a new instance of <see cref="InvalidLicenseException"/>. | ||
/// </summary> | ||
/// <param name="reason"></param> | ||
/// <param name="message"></param> | ||
public InvalidLicenseException(InvalidLicenseReason reason, string? message) | ||
: this(reason, message, null) | ||
{ } | ||
|
||
/// <summary> | ||
/// Initializes a new instance of <see cref="InvalidLicenseException"/>. | ||
/// </summary> | ||
/// <param name="reason"></param> | ||
/// <param name="message"></param> | ||
/// <param name="innerException"></param> | ||
public InvalidLicenseException(InvalidLicenseReason reason, string? message, Exception? innerException) | ||
: base(message ?? DefaultMessage, innerException) | ||
{ | ||
Reason = reason; | ||
} | ||
|
||
/// <summary> | ||
/// The reason the license was found to be invalid. | ||
/// </summary> | ||
public InvalidLicenseReason Reason { get; } | ||
} |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,63 @@ | ||
using System.Security.Cryptography.X509Certificates; | ||
|
||
namespace Bitwarden.Extensions.Hosting.Licensing; | ||
|
||
/// <summary> | ||
/// A set of options for customizing how licensing behaves. | ||
/// </summary> | ||
public sealed class LicensingOptions | ||
{ | ||
/// <summary> | ||
/// Options for configuring license retrieval from azure blob storage. | ||
/// </summary> | ||
public AzureBlobLicensingOptions AzureBlob { get; set; } = new AzureBlobLicensingOptions(); | ||
|
||
/// <summary> | ||
/// The certificate that will be used to either sign or validate licenses. | ||
/// </summary> | ||
public X509Certificate2 SigningCertificate { get; set; } = null!; | ||
|
||
/// <summary> | ||
/// Development option to force the usage of self hosted organization even though a certificate | ||
/// that can sign licenses it available. | ||
/// </summary> | ||
public bool ForceSelfHost { get; set; } | ||
} | ||
|
||
/// <summary> | ||
/// A set of options for customizing how to find a certificate in Azure blob storage. | ||
/// </summary> | ||
public sealed class AzureBlobLicensingOptions | ||
{ | ||
/// <summary> | ||
/// The connection string to the azure blob storage account. | ||
/// </summary> | ||
public string? ConnectionString { get; set; } | ||
|
||
/// <summary> | ||
/// The password for the certificate stored in azure blob storage. | ||
/// </summary> | ||
public string? CertificatePassword { get; set; } | ||
|
||
/// <summary> | ||
/// The name of the blob the certificate is stored in, defaults to <c>certificates</c>. | ||
/// </summary> | ||
public string BlobName { get; set; } = "certificates"; | ||
|
||
/// <summary> | ||
/// The name of the license stored in azure blob storage, defaults to <c>licensing.pfx</c>. | ||
/// </summary> | ||
public string LicenseName { get; set; } = "licensing.pfx"; | ||
} | ||
|
||
/// <summary> | ||
/// It's important that these can't be set through configuration, only through code. | ||
/// </summary> | ||
/// <remarks> | ||
/// We would need to make this public for services to customize this. | ||
/// </remarks> | ||
internal sealed class InternalLicensingOptions | ||
{ | ||
public string DevelopmentThumbprint { get; set; } = "207E64A231E8AA32AAF68A61037C075EBEBD553F"; | ||
public string NonDevelopmentThumbprint { get; set; } = "B34876439FCDA2846505B2EFBBA6C4A951313EBE"; | ||
} |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
โน๏ธ No.