mirror of
https://github.com/bitwarden/server.git
synced 2025-04-05 21:18:13 -05:00
Merge branch 'main' into ac/pm-15621/refactor-delete-command
This commit is contained in:
commit
b57545811e
@ -3,7 +3,7 @@
|
|||||||
<PropertyGroup>
|
<PropertyGroup>
|
||||||
<TargetFramework>net8.0</TargetFramework>
|
<TargetFramework>net8.0</TargetFramework>
|
||||||
|
|
||||||
<Version>2025.2.2</Version>
|
<Version>2025.2.3</Version>
|
||||||
|
|
||||||
<RootNamespace>Bit.$(MSBuildProjectName)</RootNamespace>
|
<RootNamespace>Bit.$(MSBuildProjectName)</RootNamespace>
|
||||||
<ImplicitUsings>enable</ImplicitUsings>
|
<ImplicitUsings>enable</ImplicitUsings>
|
||||||
|
@ -475,20 +475,17 @@ public class ProviderBillingService(
|
|||||||
Provider provider,
|
Provider provider,
|
||||||
TaxInfo taxInfo)
|
TaxInfo taxInfo)
|
||||||
{
|
{
|
||||||
ArgumentNullException.ThrowIfNull(provider);
|
if (taxInfo is not
|
||||||
ArgumentNullException.ThrowIfNull(taxInfo);
|
{
|
||||||
|
BillingAddressCountry: not null and not "",
|
||||||
if (string.IsNullOrEmpty(taxInfo.BillingAddressCountry) ||
|
BillingAddressPostalCode: not null and not ""
|
||||||
string.IsNullOrEmpty(taxInfo.BillingAddressPostalCode))
|
})
|
||||||
{
|
{
|
||||||
logger.LogError("Cannot create customer for provider ({ProviderID}) without both a country and postal code", provider.Id);
|
logger.LogError("Cannot create customer for provider ({ProviderID}) without both a country and postal code", provider.Id);
|
||||||
|
|
||||||
throw new BillingException();
|
throw new BillingException();
|
||||||
}
|
}
|
||||||
|
|
||||||
var providerDisplayName = provider.DisplayName();
|
var options = new CustomerCreateOptions
|
||||||
|
|
||||||
var customerCreateOptions = new CustomerCreateOptions
|
|
||||||
{
|
{
|
||||||
Address = new AddressOptions
|
Address = new AddressOptions
|
||||||
{
|
{
|
||||||
@ -508,9 +505,9 @@ public class ProviderBillingService(
|
|||||||
new CustomerInvoiceSettingsCustomFieldOptions
|
new CustomerInvoiceSettingsCustomFieldOptions
|
||||||
{
|
{
|
||||||
Name = provider.SubscriberType(),
|
Name = provider.SubscriberType(),
|
||||||
Value = providerDisplayName?.Length <= 30
|
Value = provider.DisplayName()?.Length <= 30
|
||||||
? providerDisplayName
|
? provider.DisplayName()
|
||||||
: providerDisplayName?[..30]
|
: provider.DisplayName()?[..30]
|
||||||
}
|
}
|
||||||
]
|
]
|
||||||
},
|
},
|
||||||
@ -522,7 +519,8 @@ public class ProviderBillingService(
|
|||||||
|
|
||||||
if (!string.IsNullOrEmpty(taxInfo.TaxIdNumber))
|
if (!string.IsNullOrEmpty(taxInfo.TaxIdNumber))
|
||||||
{
|
{
|
||||||
var taxIdType = taxService.GetStripeTaxCode(taxInfo.BillingAddressCountry,
|
var taxIdType = taxService.GetStripeTaxCode(
|
||||||
|
taxInfo.BillingAddressCountry,
|
||||||
taxInfo.TaxIdNumber);
|
taxInfo.TaxIdNumber);
|
||||||
|
|
||||||
if (taxIdType == null)
|
if (taxIdType == null)
|
||||||
@ -533,15 +531,20 @@ public class ProviderBillingService(
|
|||||||
throw new BadRequestException("billingTaxIdTypeInferenceError");
|
throw new BadRequestException("billingTaxIdTypeInferenceError");
|
||||||
}
|
}
|
||||||
|
|
||||||
customerCreateOptions.TaxIdData =
|
options.TaxIdData =
|
||||||
[
|
[
|
||||||
new CustomerTaxIdDataOptions { Type = taxIdType, Value = taxInfo.TaxIdNumber }
|
new CustomerTaxIdDataOptions { Type = taxIdType, Value = taxInfo.TaxIdNumber }
|
||||||
];
|
];
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (!string.IsNullOrEmpty(provider.DiscountId))
|
||||||
|
{
|
||||||
|
options.Coupon = provider.DiscountId;
|
||||||
|
}
|
||||||
|
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
return await stripeAdapter.CustomerCreateAsync(customerCreateOptions);
|
return await stripeAdapter.CustomerCreateAsync(options);
|
||||||
}
|
}
|
||||||
catch (StripeException stripeException) when (stripeException.StripeError?.Code == StripeConstants.ErrorCodes.TaxIdInvalid)
|
catch (StripeException stripeException) when (stripeException.StripeError?.Code == StripeConstants.ErrorCodes.TaxIdInvalid)
|
||||||
{
|
{
|
||||||
|
@ -731,18 +731,6 @@ public class ProviderBillingServiceTests
|
|||||||
|
|
||||||
#region SetupCustomer
|
#region SetupCustomer
|
||||||
|
|
||||||
[Theory, BitAutoData]
|
|
||||||
public async Task SetupCustomer_NullProvider_ThrowsArgumentNullException(
|
|
||||||
SutProvider<ProviderBillingService> sutProvider,
|
|
||||||
TaxInfo taxInfo) =>
|
|
||||||
await Assert.ThrowsAsync<ArgumentNullException>(() => sutProvider.Sut.SetupCustomer(null, taxInfo));
|
|
||||||
|
|
||||||
[Theory, BitAutoData]
|
|
||||||
public async Task SetupCustomer_NullTaxInfo_ThrowsArgumentNullException(
|
|
||||||
SutProvider<ProviderBillingService> sutProvider,
|
|
||||||
Provider provider) =>
|
|
||||||
await Assert.ThrowsAsync<ArgumentNullException>(() => sutProvider.Sut.SetupCustomer(provider, null));
|
|
||||||
|
|
||||||
[Theory, BitAutoData]
|
[Theory, BitAutoData]
|
||||||
public async Task SetupCustomer_MissingCountry_ContactSupport(
|
public async Task SetupCustomer_MissingCountry_ContactSupport(
|
||||||
SutProvider<ProviderBillingService> sutProvider,
|
SutProvider<ProviderBillingService> sutProvider,
|
||||||
|
@ -10,6 +10,9 @@ public class CreateMspProviderModel : IValidatableObject
|
|||||||
[Display(Name = "Owner Email")]
|
[Display(Name = "Owner Email")]
|
||||||
public string OwnerEmail { get; set; }
|
public string OwnerEmail { get; set; }
|
||||||
|
|
||||||
|
[Display(Name = "Subscription Discount")]
|
||||||
|
public string DiscountId { get; set; }
|
||||||
|
|
||||||
[Display(Name = "Teams (Monthly) Seat Minimum")]
|
[Display(Name = "Teams (Monthly) Seat Minimum")]
|
||||||
public int TeamsMonthlySeatMinimum { get; set; }
|
public int TeamsMonthlySeatMinimum { get; set; }
|
||||||
|
|
||||||
@ -20,7 +23,8 @@ public class CreateMspProviderModel : IValidatableObject
|
|||||||
{
|
{
|
||||||
return new Provider
|
return new Provider
|
||||||
{
|
{
|
||||||
Type = ProviderType.Msp
|
Type = ProviderType.Msp,
|
||||||
|
DiscountId = DiscountId
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1,3 +1,4 @@
|
|||||||
|
@using Bit.Core.Billing.Constants
|
||||||
@model CreateMspProviderModel
|
@model CreateMspProviderModel
|
||||||
|
|
||||||
@{
|
@{
|
||||||
@ -12,6 +13,19 @@
|
|||||||
<label asp-for="OwnerEmail" class="form-label"></label>
|
<label asp-for="OwnerEmail" class="form-label"></label>
|
||||||
<input type="text" class="form-control" asp-for="OwnerEmail">
|
<input type="text" class="form-control" asp-for="OwnerEmail">
|
||||||
</div>
|
</div>
|
||||||
|
<div class="mb-3">
|
||||||
|
@{
|
||||||
|
var selectList = new List<SelectListItem>
|
||||||
|
{
|
||||||
|
new ("No discount", string.Empty, true),
|
||||||
|
new ("20% - Open", StripeConstants.CouponIDs.MSPDiscounts.Open),
|
||||||
|
new ("35% - Silver", StripeConstants.CouponIDs.MSPDiscounts.Silver),
|
||||||
|
new ("50% - Gold", StripeConstants.CouponIDs.MSPDiscounts.Gold)
|
||||||
|
};
|
||||||
|
}
|
||||||
|
<label asp-for="DiscountId" class="form-label"></label>
|
||||||
|
<select class="form-select" asp-for="DiscountId" asp-items="selectList"></select>
|
||||||
|
</div>
|
||||||
<div class="row">
|
<div class="row">
|
||||||
<div class="col-sm">
|
<div class="col-sm">
|
||||||
<div class="mb-3">
|
<div class="mb-3">
|
||||||
|
31
src/Api/Utilities/CommandResultExtensions.cs
Normal file
31
src/Api/Utilities/CommandResultExtensions.cs
Normal file
@ -0,0 +1,31 @@
|
|||||||
|
using Bit.Core.Models.Commands;
|
||||||
|
using Microsoft.AspNetCore.Mvc;
|
||||||
|
|
||||||
|
namespace Bit.Api.Utilities;
|
||||||
|
|
||||||
|
public static class CommandResultExtensions
|
||||||
|
{
|
||||||
|
public static IActionResult MapToActionResult<T>(this CommandResult<T> commandResult)
|
||||||
|
{
|
||||||
|
return commandResult switch
|
||||||
|
{
|
||||||
|
NoRecordFoundFailure<T> failure => new ObjectResult(failure.ErrorMessages) { StatusCode = StatusCodes.Status404NotFound },
|
||||||
|
BadRequestFailure<T> failure => new ObjectResult(failure.ErrorMessages) { StatusCode = StatusCodes.Status400BadRequest },
|
||||||
|
Failure<T> failure => new ObjectResult(failure.ErrorMessages) { StatusCode = StatusCodes.Status400BadRequest },
|
||||||
|
Success<T> success => new ObjectResult(success.Data) { StatusCode = StatusCodes.Status200OK },
|
||||||
|
_ => throw new InvalidOperationException($"Unhandled commandResult type: {commandResult.GetType().Name}")
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
public static IActionResult MapToActionResult(this CommandResult commandResult)
|
||||||
|
{
|
||||||
|
return commandResult switch
|
||||||
|
{
|
||||||
|
NoRecordFoundFailure failure => new ObjectResult(failure.ErrorMessages) { StatusCode = StatusCodes.Status404NotFound },
|
||||||
|
BadRequestFailure failure => new ObjectResult(failure.ErrorMessages) { StatusCode = StatusCodes.Status400BadRequest },
|
||||||
|
Failure failure => new ObjectResult(failure.ErrorMessages) { StatusCode = StatusCodes.Status400BadRequest },
|
||||||
|
Success => new ObjectResult(new { }) { StatusCode = StatusCodes.Status200OK },
|
||||||
|
_ => throw new InvalidOperationException($"Unhandled commandResult type: {commandResult.GetType().Name}")
|
||||||
|
};
|
||||||
|
}
|
||||||
|
}
|
@ -41,10 +41,15 @@ public class SubscriptionDeletedHandler : ISubscriptionDeletedHandler
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (organizationId.HasValue &&
|
if (organizationId.HasValue)
|
||||||
subscription.CancellationDetails.Comment != providerMigrationCancellationComment &&
|
|
||||||
!subscription.CancellationDetails.Comment.Contains(addedToProviderCancellationComment))
|
|
||||||
{
|
{
|
||||||
|
if (!string.IsNullOrEmpty(subscription.CancellationDetails?.Comment) &&
|
||||||
|
(subscription.CancellationDetails.Comment == providerMigrationCancellationComment ||
|
||||||
|
subscription.CancellationDetails.Comment.Contains(addedToProviderCancellationComment)))
|
||||||
|
{
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
await _organizationDisableCommand.DisableAsync(organizationId.Value, subscription.CurrentPeriodEnd);
|
await _organizationDisableCommand.DisableAsync(organizationId.Value, subscription.CurrentPeriodEnd);
|
||||||
}
|
}
|
||||||
else if (userId.HasValue)
|
else if (userId.HasValue)
|
||||||
|
@ -35,6 +35,7 @@ public class Provider : ITableObject<Guid>, ISubscriber
|
|||||||
public GatewayType? Gateway { get; set; }
|
public GatewayType? Gateway { get; set; }
|
||||||
public string? GatewayCustomerId { get; set; }
|
public string? GatewayCustomerId { get; set; }
|
||||||
public string? GatewaySubscriptionId { get; set; }
|
public string? GatewaySubscriptionId { get; set; }
|
||||||
|
public string? DiscountId { get; set; }
|
||||||
|
|
||||||
public string? BillingEmailAddress() => BillingEmail?.ToLowerInvariant().Trim();
|
public string? BillingEmailAddress() => BillingEmail?.ToLowerInvariant().Trim();
|
||||||
|
|
||||||
|
@ -18,8 +18,15 @@ public static class StripeConstants
|
|||||||
|
|
||||||
public static class CouponIDs
|
public static class CouponIDs
|
||||||
{
|
{
|
||||||
public const string MSPDiscount35 = "msp-discount-35";
|
public const string LegacyMSPDiscount = "msp-discount-35";
|
||||||
public const string SecretsManagerStandalone = "sm-standalone";
|
public const string SecretsManagerStandalone = "sm-standalone";
|
||||||
|
|
||||||
|
public static class MSPDiscounts
|
||||||
|
{
|
||||||
|
public const string Open = "msp-open-discount";
|
||||||
|
public const string Silver = "msp-silver-discount";
|
||||||
|
public const string Gold = "msp-gold-discount";
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public static class ErrorCodes
|
public static class ErrorCodes
|
||||||
|
@ -254,7 +254,7 @@ public class ProviderMigrator(
|
|||||||
|
|
||||||
await stripeAdapter.CustomerUpdateAsync(customer.Id, new CustomerUpdateOptions
|
await stripeAdapter.CustomerUpdateAsync(customer.Id, new CustomerUpdateOptions
|
||||||
{
|
{
|
||||||
Coupon = StripeConstants.CouponIDs.MSPDiscount35
|
Coupon = StripeConstants.CouponIDs.LegacyMSPDiscount
|
||||||
});
|
});
|
||||||
|
|
||||||
provider.GatewayCustomerId = customer.Id;
|
provider.GatewayCustomerId = customer.Id;
|
||||||
|
@ -46,7 +46,8 @@ public class OrganizationSale
|
|||||||
var customerSetup = new CustomerSetup
|
var customerSetup = new CustomerSetup
|
||||||
{
|
{
|
||||||
Coupon = signup.IsFromProvider
|
Coupon = signup.IsFromProvider
|
||||||
? StripeConstants.CouponIDs.MSPDiscount35
|
// TODO: Remove when last of the legacy providers has been migrated.
|
||||||
|
? StripeConstants.CouponIDs.LegacyMSPDiscount
|
||||||
: signup.IsFromSecretsManagerTrial
|
: signup.IsFromSecretsManagerTrial
|
||||||
? StripeConstants.CouponIDs.SecretsManagerStandalone
|
? StripeConstants.CouponIDs.SecretsManagerStandalone
|
||||||
: null
|
: null
|
||||||
|
@ -119,7 +119,6 @@ public static class FeatureFlagKeys
|
|||||||
public const string UseTreeWalkerApiForPageDetailsCollection = "use-tree-walker-api-for-page-details-collection";
|
public const string UseTreeWalkerApiForPageDetailsCollection = "use-tree-walker-api-for-page-details-collection";
|
||||||
public const string DuoRedirect = "duo-redirect";
|
public const string DuoRedirect = "duo-redirect";
|
||||||
public const string AC2101UpdateTrialInitiationEmail = "AC-2101-update-trial-initiation-email";
|
public const string AC2101UpdateTrialInitiationEmail = "AC-2101-update-trial-initiation-email";
|
||||||
public const string AC1795_UpdatedSubscriptionStatusSection = "AC-1795_updated-subscription-status-section";
|
|
||||||
public const string EmailVerification = "email-verification";
|
public const string EmailVerification = "email-verification";
|
||||||
public const string EmailVerificationDisableTimingDelays = "email-verification-disable-timing-delays";
|
public const string EmailVerificationDisableTimingDelays = "email-verification-disable-timing-delays";
|
||||||
public const string ExtensionRefresh = "extension-refresh";
|
public const string ExtensionRefresh = "extension-refresh";
|
||||||
|
23
src/Core/Models/Commands/BadRequestFailure.cs
Normal file
23
src/Core/Models/Commands/BadRequestFailure.cs
Normal file
@ -0,0 +1,23 @@
|
|||||||
|
namespace Bit.Core.Models.Commands;
|
||||||
|
|
||||||
|
public class BadRequestFailure<T> : Failure<T>
|
||||||
|
{
|
||||||
|
public BadRequestFailure(IEnumerable<string> errorMessage) : base(errorMessage)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
public BadRequestFailure(string errorMessage) : base(errorMessage)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public class BadRequestFailure : Failure
|
||||||
|
{
|
||||||
|
public BadRequestFailure(IEnumerable<string> errorMessage) : base(errorMessage)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
public BadRequestFailure(string errorMessage) : base(errorMessage)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
}
|
@ -1,4 +1,6 @@
|
|||||||
namespace Bit.Core.Models.Commands;
|
#nullable enable
|
||||||
|
|
||||||
|
namespace Bit.Core.Models.Commands;
|
||||||
|
|
||||||
public class CommandResult(IEnumerable<string> errors)
|
public class CommandResult(IEnumerable<string> errors)
|
||||||
{
|
{
|
||||||
@ -10,3 +12,39 @@ public class CommandResult(IEnumerable<string> errors)
|
|||||||
|
|
||||||
public CommandResult() : this(Array.Empty<string>()) { }
|
public CommandResult() : this(Array.Empty<string>()) { }
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public class Failure : CommandResult
|
||||||
|
{
|
||||||
|
protected Failure(IEnumerable<string> errorMessages) : base(errorMessages)
|
||||||
|
{
|
||||||
|
|
||||||
|
}
|
||||||
|
public Failure(string errorMessage) : base(errorMessage)
|
||||||
|
{
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public class Success : CommandResult
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
public abstract class CommandResult<T>
|
||||||
|
{
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
public class Success<T>(T data) : CommandResult<T>
|
||||||
|
{
|
||||||
|
public T? Data { get; init; } = data;
|
||||||
|
}
|
||||||
|
|
||||||
|
public class Failure<T>(IEnumerable<string> errorMessage) : CommandResult<T>
|
||||||
|
{
|
||||||
|
public IEnumerable<string> ErrorMessages { get; init; } = errorMessage;
|
||||||
|
|
||||||
|
public Failure(string errorMessage) : this(new[] { errorMessage })
|
||||||
|
{
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
24
src/Core/Models/Commands/NoRecordFoundFailure.cs
Normal file
24
src/Core/Models/Commands/NoRecordFoundFailure.cs
Normal file
@ -0,0 +1,24 @@
|
|||||||
|
namespace Bit.Core.Models.Commands;
|
||||||
|
|
||||||
|
public class NoRecordFoundFailure<T> : Failure<T>
|
||||||
|
{
|
||||||
|
public NoRecordFoundFailure(IEnumerable<string> errorMessage) : base(errorMessage)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
public NoRecordFoundFailure(string errorMessage) : base(errorMessage)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public class NoRecordFoundFailure : Failure
|
||||||
|
{
|
||||||
|
public NoRecordFoundFailure(IEnumerable<string> errorMessage) : base(errorMessage)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
public NoRecordFoundFailure(string errorMessage) : base(errorMessage)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
@ -1609,15 +1609,12 @@ public class StripePaymentService : IPaymentService
|
|||||||
{
|
{
|
||||||
subscriptionInfo.Subscription = new SubscriptionInfo.BillingSubscription(sub);
|
subscriptionInfo.Subscription = new SubscriptionInfo.BillingSubscription(sub);
|
||||||
|
|
||||||
if (_featureService.IsEnabled(FeatureFlagKeys.AC1795_UpdatedSubscriptionStatusSection))
|
var (suspensionDate, unpaidPeriodEndDate) = await GetSuspensionDateAsync(sub);
|
||||||
{
|
|
||||||
var (suspensionDate, unpaidPeriodEndDate) = await GetSuspensionDateAsync(sub);
|
|
||||||
|
|
||||||
if (suspensionDate.HasValue && unpaidPeriodEndDate.HasValue)
|
if (suspensionDate.HasValue && unpaidPeriodEndDate.HasValue)
|
||||||
{
|
{
|
||||||
subscriptionInfo.Subscription.SuspensionDate = suspensionDate;
|
subscriptionInfo.Subscription.SuspensionDate = suspensionDate;
|
||||||
subscriptionInfo.Subscription.UnpaidPeriodEndDate = unpaidPeriodEndDate;
|
subscriptionInfo.Subscription.UnpaidPeriodEndDate = unpaidPeriodEndDate;
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -17,7 +17,8 @@
|
|||||||
@RevisionDate DATETIME2(7),
|
@RevisionDate DATETIME2(7),
|
||||||
@Gateway TINYINT = 0,
|
@Gateway TINYINT = 0,
|
||||||
@GatewayCustomerId VARCHAR(50) = NULL,
|
@GatewayCustomerId VARCHAR(50) = NULL,
|
||||||
@GatewaySubscriptionId VARCHAR(50) = NULL
|
@GatewaySubscriptionId VARCHAR(50) = NULL,
|
||||||
|
@DiscountId VARCHAR(50) = NULL
|
||||||
AS
|
AS
|
||||||
BEGIN
|
BEGIN
|
||||||
SET NOCOUNT ON
|
SET NOCOUNT ON
|
||||||
@ -42,7 +43,8 @@ BEGIN
|
|||||||
[RevisionDate],
|
[RevisionDate],
|
||||||
[Gateway],
|
[Gateway],
|
||||||
[GatewayCustomerId],
|
[GatewayCustomerId],
|
||||||
[GatewaySubscriptionId]
|
[GatewaySubscriptionId],
|
||||||
|
[DiscountId]
|
||||||
)
|
)
|
||||||
VALUES
|
VALUES
|
||||||
(
|
(
|
||||||
@ -64,6 +66,7 @@ BEGIN
|
|||||||
@RevisionDate,
|
@RevisionDate,
|
||||||
@Gateway,
|
@Gateway,
|
||||||
@GatewayCustomerId,
|
@GatewayCustomerId,
|
||||||
@GatewaySubscriptionId
|
@GatewaySubscriptionId,
|
||||||
|
@DiscountId
|
||||||
)
|
)
|
||||||
END
|
END
|
||||||
|
@ -17,7 +17,8 @@
|
|||||||
@RevisionDate DATETIME2(7),
|
@RevisionDate DATETIME2(7),
|
||||||
@Gateway TINYINT = 0,
|
@Gateway TINYINT = 0,
|
||||||
@GatewayCustomerId VARCHAR(50) = NULL,
|
@GatewayCustomerId VARCHAR(50) = NULL,
|
||||||
@GatewaySubscriptionId VARCHAR(50) = NULL
|
@GatewaySubscriptionId VARCHAR(50) = NULL,
|
||||||
|
@DiscountId VARCHAR(50) = NULL
|
||||||
AS
|
AS
|
||||||
BEGIN
|
BEGIN
|
||||||
SET NOCOUNT ON
|
SET NOCOUNT ON
|
||||||
@ -42,7 +43,8 @@ BEGIN
|
|||||||
[RevisionDate] = @RevisionDate,
|
[RevisionDate] = @RevisionDate,
|
||||||
[Gateway] = @Gateway,
|
[Gateway] = @Gateway,
|
||||||
[GatewayCustomerId] = @GatewayCustomerId,
|
[GatewayCustomerId] = @GatewayCustomerId,
|
||||||
[GatewaySubscriptionId] = @GatewaySubscriptionId
|
[GatewaySubscriptionId] = @GatewaySubscriptionId,
|
||||||
|
[DiscountId] = @DiscountId
|
||||||
WHERE
|
WHERE
|
||||||
[Id] = @Id
|
[Id] = @Id
|
||||||
END
|
END
|
||||||
|
@ -18,5 +18,6 @@
|
|||||||
[Gateway] TINYINT NULL,
|
[Gateway] TINYINT NULL,
|
||||||
[GatewayCustomerId] VARCHAR (50) NULL,
|
[GatewayCustomerId] VARCHAR (50) NULL,
|
||||||
[GatewaySubscriptionId] VARCHAR (50) NULL,
|
[GatewaySubscriptionId] VARCHAR (50) NULL,
|
||||||
|
[DiscountId] VARCHAR (50) NULL,
|
||||||
CONSTRAINT [PK_Provider] PRIMARY KEY CLUSTERED ([Id] ASC)
|
CONSTRAINT [PK_Provider] PRIMARY KEY CLUSTERED ([Id] ASC)
|
||||||
);
|
);
|
||||||
|
107
test/Api.Test/Utilities/CommandResultExtensionTests.cs
Normal file
107
test/Api.Test/Utilities/CommandResultExtensionTests.cs
Normal file
@ -0,0 +1,107 @@
|
|||||||
|
using Bit.Api.Utilities;
|
||||||
|
using Bit.Core.Models.Commands;
|
||||||
|
using Bit.Core.Vault.Entities;
|
||||||
|
using Microsoft.AspNetCore.Http;
|
||||||
|
using Microsoft.AspNetCore.Mvc;
|
||||||
|
using Xunit;
|
||||||
|
|
||||||
|
namespace Bit.Api.Test.Utilities;
|
||||||
|
|
||||||
|
public class CommandResultExtensionTests
|
||||||
|
{
|
||||||
|
public static IEnumerable<object[]> WithGenericTypeTestCases()
|
||||||
|
{
|
||||||
|
yield return new object[]
|
||||||
|
{
|
||||||
|
new NoRecordFoundFailure<Cipher>(new[] { "Error 1", "Error 2" }),
|
||||||
|
new ObjectResult(new[] { "Error 1", "Error 2" }) { StatusCode = StatusCodes.Status404NotFound }
|
||||||
|
};
|
||||||
|
yield return new object[]
|
||||||
|
{
|
||||||
|
new BadRequestFailure<Cipher>("Error 3"),
|
||||||
|
new ObjectResult(new[] { "Error 3" }) { StatusCode = StatusCodes.Status400BadRequest }
|
||||||
|
};
|
||||||
|
yield return new object[]
|
||||||
|
{
|
||||||
|
new Failure<Cipher>("Error 4"),
|
||||||
|
new ObjectResult(new[] { "Error 4" }) { StatusCode = StatusCodes.Status400BadRequest }
|
||||||
|
};
|
||||||
|
var cipher = new Cipher() { Id = Guid.NewGuid() };
|
||||||
|
|
||||||
|
yield return new object[]
|
||||||
|
{
|
||||||
|
new Success<Cipher>(cipher),
|
||||||
|
new ObjectResult(cipher) { StatusCode = StatusCodes.Status200OK }
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
[Theory]
|
||||||
|
[MemberData(nameof(WithGenericTypeTestCases))]
|
||||||
|
public void MapToActionResult_WithGenericType_ShouldMapToHttpResponse(CommandResult<Cipher> input, ObjectResult expected)
|
||||||
|
{
|
||||||
|
var result = input.MapToActionResult();
|
||||||
|
|
||||||
|
Assert.Equivalent(expected, result);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
[Fact]
|
||||||
|
public void MapToActionResult_WithGenericType_ShouldThrowExceptionForUnhandledCommandResult()
|
||||||
|
{
|
||||||
|
var result = new NotImplementedCommandResult();
|
||||||
|
|
||||||
|
Assert.Throws<InvalidOperationException>(() => result.MapToActionResult());
|
||||||
|
}
|
||||||
|
|
||||||
|
public static IEnumerable<object[]> TestCases()
|
||||||
|
{
|
||||||
|
yield return new object[]
|
||||||
|
{
|
||||||
|
new NoRecordFoundFailure(new[] { "Error 1", "Error 2" }),
|
||||||
|
new ObjectResult(new[] { "Error 1", "Error 2" }) { StatusCode = StatusCodes.Status404NotFound }
|
||||||
|
};
|
||||||
|
yield return new object[]
|
||||||
|
{
|
||||||
|
new BadRequestFailure("Error 3"),
|
||||||
|
new ObjectResult(new[] { "Error 3" }) { StatusCode = StatusCodes.Status400BadRequest }
|
||||||
|
};
|
||||||
|
yield return new object[]
|
||||||
|
{
|
||||||
|
new Failure("Error 4"),
|
||||||
|
new ObjectResult(new[] { "Error 4" }) { StatusCode = StatusCodes.Status400BadRequest }
|
||||||
|
};
|
||||||
|
yield return new object[]
|
||||||
|
{
|
||||||
|
new Success(),
|
||||||
|
new ObjectResult(new { }) { StatusCode = StatusCodes.Status200OK }
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
[Theory]
|
||||||
|
[MemberData(nameof(TestCases))]
|
||||||
|
public void MapToActionResult_ShouldMapToHttpResponse(CommandResult input, ObjectResult expected)
|
||||||
|
{
|
||||||
|
var result = input.MapToActionResult();
|
||||||
|
|
||||||
|
Assert.Equivalent(expected, result);
|
||||||
|
}
|
||||||
|
|
||||||
|
[Fact]
|
||||||
|
public void MapToActionResult_ShouldThrowExceptionForUnhandledCommandResult()
|
||||||
|
{
|
||||||
|
var result = new NotImplementedCommandResult<Cipher>();
|
||||||
|
|
||||||
|
Assert.Throws<InvalidOperationException>(() => result.MapToActionResult());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public class NotImplementedCommandResult<T> : CommandResult<T>
|
||||||
|
{
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
public class NotImplementedCommandResult : CommandResult
|
||||||
|
{
|
||||||
|
|
||||||
|
}
|
@ -0,0 +1,171 @@
|
|||||||
|
-- Add 'DiscountId' column to 'Provider' table.
|
||||||
|
IF COL_LENGTH('[dbo].[Provider]', 'DiscountId') IS NULL
|
||||||
|
BEGIN
|
||||||
|
ALTER TABLE
|
||||||
|
[dbo].[Provider]
|
||||||
|
ADD
|
||||||
|
[DiscountId] VARCHAR(50) NULL;
|
||||||
|
END
|
||||||
|
GO
|
||||||
|
|
||||||
|
-- Recreate 'ProviderView' so that it includes the 'DiscountId' column.
|
||||||
|
CREATE OR ALTER VIEW [dbo].[ProviderView]
|
||||||
|
AS
|
||||||
|
SELECT
|
||||||
|
*
|
||||||
|
FROM
|
||||||
|
[dbo].[Provider]
|
||||||
|
GO
|
||||||
|
|
||||||
|
-- Alter 'Provider_Create' SPROC to add 'DiscountId' column.
|
||||||
|
CREATE OR ALTER PROCEDURE [dbo].[Provider_Create]
|
||||||
|
@Id UNIQUEIDENTIFIER OUTPUT,
|
||||||
|
@Name NVARCHAR(50),
|
||||||
|
@BusinessName NVARCHAR(50),
|
||||||
|
@BusinessAddress1 NVARCHAR(50),
|
||||||
|
@BusinessAddress2 NVARCHAR(50),
|
||||||
|
@BusinessAddress3 NVARCHAR(50),
|
||||||
|
@BusinessCountry VARCHAR(2),
|
||||||
|
@BusinessTaxNumber NVARCHAR(30),
|
||||||
|
@BillingEmail NVARCHAR(256),
|
||||||
|
@BillingPhone NVARCHAR(50) = NULL,
|
||||||
|
@Status TINYINT,
|
||||||
|
@Type TINYINT = 0,
|
||||||
|
@UseEvents BIT,
|
||||||
|
@Enabled BIT,
|
||||||
|
@CreationDate DATETIME2(7),
|
||||||
|
@RevisionDate DATETIME2(7),
|
||||||
|
@Gateway TINYINT = 0,
|
||||||
|
@GatewayCustomerId VARCHAR(50) = NULL,
|
||||||
|
@GatewaySubscriptionId VARCHAR(50) = NULL,
|
||||||
|
@DiscountId VARCHAR(50) = NULL
|
||||||
|
AS
|
||||||
|
BEGIN
|
||||||
|
SET NOCOUNT ON
|
||||||
|
|
||||||
|
INSERT INTO [dbo].[Provider]
|
||||||
|
(
|
||||||
|
[Id],
|
||||||
|
[Name],
|
||||||
|
[BusinessName],
|
||||||
|
[BusinessAddress1],
|
||||||
|
[BusinessAddress2],
|
||||||
|
[BusinessAddress3],
|
||||||
|
[BusinessCountry],
|
||||||
|
[BusinessTaxNumber],
|
||||||
|
[BillingEmail],
|
||||||
|
[BillingPhone],
|
||||||
|
[Status],
|
||||||
|
[Type],
|
||||||
|
[UseEvents],
|
||||||
|
[Enabled],
|
||||||
|
[CreationDate],
|
||||||
|
[RevisionDate],
|
||||||
|
[Gateway],
|
||||||
|
[GatewayCustomerId],
|
||||||
|
[GatewaySubscriptionId],
|
||||||
|
[DiscountId]
|
||||||
|
)
|
||||||
|
VALUES
|
||||||
|
(
|
||||||
|
@Id,
|
||||||
|
@Name,
|
||||||
|
@BusinessName,
|
||||||
|
@BusinessAddress1,
|
||||||
|
@BusinessAddress2,
|
||||||
|
@BusinessAddress3,
|
||||||
|
@BusinessCountry,
|
||||||
|
@BusinessTaxNumber,
|
||||||
|
@BillingEmail,
|
||||||
|
@BillingPhone,
|
||||||
|
@Status,
|
||||||
|
@Type,
|
||||||
|
@UseEvents,
|
||||||
|
@Enabled,
|
||||||
|
@CreationDate,
|
||||||
|
@RevisionDate,
|
||||||
|
@Gateway,
|
||||||
|
@GatewayCustomerId,
|
||||||
|
@GatewaySubscriptionId,
|
||||||
|
@DiscountId
|
||||||
|
)
|
||||||
|
END
|
||||||
|
GO
|
||||||
|
|
||||||
|
-- Alter 'Provider_Update' SPROC to add 'DiscountId' column.
|
||||||
|
CREATE OR ALTER PROCEDURE [dbo].[Provider_Update]
|
||||||
|
@Id UNIQUEIDENTIFIER,
|
||||||
|
@Name NVARCHAR(50),
|
||||||
|
@BusinessName NVARCHAR(50),
|
||||||
|
@BusinessAddress1 NVARCHAR(50),
|
||||||
|
@BusinessAddress2 NVARCHAR(50),
|
||||||
|
@BusinessAddress3 NVARCHAR(50),
|
||||||
|
@BusinessCountry VARCHAR(2),
|
||||||
|
@BusinessTaxNumber NVARCHAR(30),
|
||||||
|
@BillingEmail NVARCHAR(256),
|
||||||
|
@BillingPhone NVARCHAR(50) = NULL,
|
||||||
|
@Status TINYINT,
|
||||||
|
@Type TINYINT = 0,
|
||||||
|
@UseEvents BIT,
|
||||||
|
@Enabled BIT,
|
||||||
|
@CreationDate DATETIME2(7),
|
||||||
|
@RevisionDate DATETIME2(7),
|
||||||
|
@Gateway TINYINT = 0,
|
||||||
|
@GatewayCustomerId VARCHAR(50) = NULL,
|
||||||
|
@GatewaySubscriptionId VARCHAR(50) = NULL,
|
||||||
|
@DiscountId VARCHAR(50) = NULL
|
||||||
|
AS
|
||||||
|
BEGIN
|
||||||
|
SET NOCOUNT ON
|
||||||
|
|
||||||
|
UPDATE
|
||||||
|
[dbo].[Provider]
|
||||||
|
SET
|
||||||
|
[Name] = @Name,
|
||||||
|
[BusinessName] = @BusinessName,
|
||||||
|
[BusinessAddress1] = @BusinessAddress1,
|
||||||
|
[BusinessAddress2] = @BusinessAddress2,
|
||||||
|
[BusinessAddress3] = @BusinessAddress3,
|
||||||
|
[BusinessCountry] = @BusinessCountry,
|
||||||
|
[BusinessTaxNumber] = @BusinessTaxNumber,
|
||||||
|
[BillingEmail] = @BillingEmail,
|
||||||
|
[BillingPhone] = @BillingPhone,
|
||||||
|
[Status] = @Status,
|
||||||
|
[Type] = @Type,
|
||||||
|
[UseEvents] = @UseEvents,
|
||||||
|
[Enabled] = @Enabled,
|
||||||
|
[CreationDate] = @CreationDate,
|
||||||
|
[RevisionDate] = @RevisionDate,
|
||||||
|
[Gateway] = @Gateway,
|
||||||
|
[GatewayCustomerId] = @GatewayCustomerId,
|
||||||
|
[GatewaySubscriptionId] = @GatewaySubscriptionId,
|
||||||
|
[DiscountId] = @DiscountId
|
||||||
|
WHERE
|
||||||
|
[Id] = @Id
|
||||||
|
END
|
||||||
|
GO
|
||||||
|
|
||||||
|
-- Refresh modules for SPROCs reliant on 'Provider' table/view.
|
||||||
|
IF OBJECT_ID('[dbo].[Provider_ReadAbilities]') IS NOT NULL
|
||||||
|
BEGIN
|
||||||
|
EXECUTE sp_refreshsqlmodule N'[dbo].[Provider_ReadAbilities]';
|
||||||
|
END
|
||||||
|
GO
|
||||||
|
|
||||||
|
IF OBJECT_ID('[dbo].[Provider_ReadById]') IS NOT NULL
|
||||||
|
BEGIN
|
||||||
|
EXECUTE sp_refreshsqlmodule N'[dbo].[Provider_ReadById]';
|
||||||
|
END
|
||||||
|
GO
|
||||||
|
|
||||||
|
IF OBJECT_ID('[dbo].[Provider_ReadByOrganizationId]') IS NOT NULL
|
||||||
|
BEGIN
|
||||||
|
EXECUTE sp_refreshsqlmodule N'[dbo].[Provider_ReadByOrganizationId]';
|
||||||
|
END
|
||||||
|
GO
|
||||||
|
|
||||||
|
IF OBJECT_ID('[dbo].[Provider_Search]') IS NOT NULL
|
||||||
|
BEGIN
|
||||||
|
EXECUTE sp_refreshsqlmodule N'[dbo].[Provider_Search]';
|
||||||
|
END
|
||||||
|
GO
|
3013
util/MySqlMigrations/Migrations/20250213140357_AddColumn_ProviderDiscountId.Designer.cs
generated
Normal file
3013
util/MySqlMigrations/Migrations/20250213140357_AddColumn_ProviderDiscountId.Designer.cs
generated
Normal file
File diff suppressed because it is too large
Load Diff
@ -0,0 +1,28 @@
|
|||||||
|
using Microsoft.EntityFrameworkCore.Migrations;
|
||||||
|
|
||||||
|
#nullable disable
|
||||||
|
|
||||||
|
namespace Bit.MySqlMigrations.Migrations;
|
||||||
|
|
||||||
|
/// <inheritdoc />
|
||||||
|
public partial class AddColumn_ProviderDiscountId : Migration
|
||||||
|
{
|
||||||
|
/// <inheritdoc />
|
||||||
|
protected override void Up(MigrationBuilder migrationBuilder)
|
||||||
|
{
|
||||||
|
migrationBuilder.AddColumn<string>(
|
||||||
|
name: "DiscountId",
|
||||||
|
table: "Provider",
|
||||||
|
type: "longtext",
|
||||||
|
nullable: true)
|
||||||
|
.Annotation("MySql:CharSet", "utf8mb4");
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <inheritdoc />
|
||||||
|
protected override void Down(MigrationBuilder migrationBuilder)
|
||||||
|
{
|
||||||
|
migrationBuilder.DropColumn(
|
||||||
|
name: "DiscountId",
|
||||||
|
table: "Provider");
|
||||||
|
}
|
||||||
|
}
|
@ -284,6 +284,9 @@ namespace Bit.MySqlMigrations.Migrations
|
|||||||
b.Property<DateTime>("CreationDate")
|
b.Property<DateTime>("CreationDate")
|
||||||
.HasColumnType("datetime(6)");
|
.HasColumnType("datetime(6)");
|
||||||
|
|
||||||
|
b.Property<string>("DiscountId")
|
||||||
|
.HasColumnType("longtext");
|
||||||
|
|
||||||
b.Property<bool>("Enabled")
|
b.Property<bool>("Enabled")
|
||||||
.HasColumnType("tinyint(1)");
|
.HasColumnType("tinyint(1)");
|
||||||
|
|
||||||
|
3019
util/PostgresMigrations/Migrations/20250213140406_AddColumn_ProviderDiscountId.Designer.cs
generated
Normal file
3019
util/PostgresMigrations/Migrations/20250213140406_AddColumn_ProviderDiscountId.Designer.cs
generated
Normal file
File diff suppressed because it is too large
Load Diff
@ -0,0 +1,27 @@
|
|||||||
|
using Microsoft.EntityFrameworkCore.Migrations;
|
||||||
|
|
||||||
|
#nullable disable
|
||||||
|
|
||||||
|
namespace Bit.PostgresMigrations.Migrations;
|
||||||
|
|
||||||
|
/// <inheritdoc />
|
||||||
|
public partial class AddColumn_ProviderDiscountId : Migration
|
||||||
|
{
|
||||||
|
/// <inheritdoc />
|
||||||
|
protected override void Up(MigrationBuilder migrationBuilder)
|
||||||
|
{
|
||||||
|
migrationBuilder.AddColumn<string>(
|
||||||
|
name: "DiscountId",
|
||||||
|
table: "Provider",
|
||||||
|
type: "text",
|
||||||
|
nullable: true);
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <inheritdoc />
|
||||||
|
protected override void Down(MigrationBuilder migrationBuilder)
|
||||||
|
{
|
||||||
|
migrationBuilder.DropColumn(
|
||||||
|
name: "DiscountId",
|
||||||
|
table: "Provider");
|
||||||
|
}
|
||||||
|
}
|
@ -287,6 +287,9 @@ namespace Bit.PostgresMigrations.Migrations
|
|||||||
b.Property<DateTime>("CreationDate")
|
b.Property<DateTime>("CreationDate")
|
||||||
.HasColumnType("timestamp with time zone");
|
.HasColumnType("timestamp with time zone");
|
||||||
|
|
||||||
|
b.Property<string>("DiscountId")
|
||||||
|
.HasColumnType("text");
|
||||||
|
|
||||||
b.Property<bool>("Enabled")
|
b.Property<bool>("Enabled")
|
||||||
.HasColumnType("boolean");
|
.HasColumnType("boolean");
|
||||||
|
|
||||||
|
3002
util/SqliteMigrations/Migrations/20250213140401_AddColumn_ProviderDiscountId.Designer.cs
generated
Normal file
3002
util/SqliteMigrations/Migrations/20250213140401_AddColumn_ProviderDiscountId.Designer.cs
generated
Normal file
File diff suppressed because it is too large
Load Diff
@ -0,0 +1,27 @@
|
|||||||
|
using Microsoft.EntityFrameworkCore.Migrations;
|
||||||
|
|
||||||
|
#nullable disable
|
||||||
|
|
||||||
|
namespace Bit.SqliteMigrations.Migrations;
|
||||||
|
|
||||||
|
/// <inheritdoc />
|
||||||
|
public partial class AddColumn_ProviderDiscountId : Migration
|
||||||
|
{
|
||||||
|
/// <inheritdoc />
|
||||||
|
protected override void Up(MigrationBuilder migrationBuilder)
|
||||||
|
{
|
||||||
|
migrationBuilder.AddColumn<string>(
|
||||||
|
name: "DiscountId",
|
||||||
|
table: "Provider",
|
||||||
|
type: "TEXT",
|
||||||
|
nullable: true);
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <inheritdoc />
|
||||||
|
protected override void Down(MigrationBuilder migrationBuilder)
|
||||||
|
{
|
||||||
|
migrationBuilder.DropColumn(
|
||||||
|
name: "DiscountId",
|
||||||
|
table: "Provider");
|
||||||
|
}
|
||||||
|
}
|
@ -279,6 +279,9 @@ namespace Bit.SqliteMigrations.Migrations
|
|||||||
b.Property<DateTime>("CreationDate")
|
b.Property<DateTime>("CreationDate")
|
||||||
.HasColumnType("TEXT");
|
.HasColumnType("TEXT");
|
||||||
|
|
||||||
|
b.Property<string>("DiscountId")
|
||||||
|
.HasColumnType("TEXT");
|
||||||
|
|
||||||
b.Property<bool>("Enabled")
|
b.Property<bool>("Enabled")
|
||||||
.HasColumnType("INTEGER");
|
.HasColumnType("INTEGER");
|
||||||
|
|
||||||
|
Loading…
x
Reference in New Issue
Block a user