1
0
mirror of https://github.com/bitwarden/server.git synced 2025-04-05 05:00:19 -05:00

[fix] Address QA found defects for the Stripe Subscriptions admin tool (#2150)

* [fix] Clear the page on Stripe Subscription search change
[SG-404]

* [fix] Ensure page is null when selecting all Stripe Subscriptions for an action
[SG-404]

* [feat] Allow Stripe Subscriptions to be filtered by a test clock
[SG-404]
This commit is contained in:
Addison Beck 2022-07-26 13:59:41 -04:00 committed by GitHub
parent d1a2e58ce1
commit d1db4d31cb
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
6 changed files with 38 additions and 2 deletions

View File

@ -453,6 +453,7 @@ namespace Bit.Admin.Controllers
{ {
Items = subscriptions.Select(s => new StripeSubscriptionRowModel(s)).ToList(), Items = subscriptions.Select(s => new StripeSubscriptionRowModel(s)).ToList(),
Prices = (await _stripeAdapter.PriceListAsync(new Stripe.PriceListOptions() { Limit = 100 })).Data, Prices = (await _stripeAdapter.PriceListAsync(new Stripe.PriceListOptions() { Limit = 100 })).Data,
TestClocks = await _stripeAdapter.TestClockListAsync(),
Filter = options Filter = options
}; };
return View(model); return View(model);
@ -464,6 +465,7 @@ namespace Bit.Admin.Controllers
if (!ModelState.IsValid) if (!ModelState.IsValid)
{ {
model.Prices = (await _stripeAdapter.PriceListAsync(new Stripe.PriceListOptions() { Limit = 100 })).Data; model.Prices = (await _stripeAdapter.PriceListAsync(new Stripe.PriceListOptions() { Limit = 100 })).Data;
model.TestClocks = await _stripeAdapter.TestClockListAsync();
return View(model); return View(model);
} }
@ -485,11 +487,11 @@ namespace Bit.Admin.Controllers
} }
else else
{ {
if (model.Action == StripeSubscriptionsAction.PreviousPage) if (model.Action == StripeSubscriptionsAction.PreviousPage || model.Action == StripeSubscriptionsAction.Search)
{ {
model.Filter.StartingAfter = null; model.Filter.StartingAfter = null;
} }
if (model.Action == StripeSubscriptionsAction.NextPage) if (model.Action == StripeSubscriptionsAction.NextPage || model.Action == StripeSubscriptionsAction.Search)
{ {
model.Filter.EndingBefore = null; model.Filter.EndingBefore = null;
} }

View File

@ -30,6 +30,7 @@ namespace Bit.Admin.Models
public StripeSubscriptionsAction Action { get; set; } = StripeSubscriptionsAction.Search; public StripeSubscriptionsAction Action { get; set; } = StripeSubscriptionsAction.Search;
public string Message { get; set; } public string Message { get; set; }
public List<Stripe.Price> Prices { get; set; } public List<Stripe.Price> Prices { get; set; }
public List<Stripe.TestHelpers.TestClock> TestClocks { get; set; }
public StripeSubscriptionListOptions Filter { get; set; } = new StripeSubscriptionListOptions(); public StripeSubscriptionListOptions Filter { get; set; } = new StripeSubscriptionListOptions();
public IEnumerable<ValidationResult> Validate(ValidationContext validationContext) public IEnumerable<ValidationResult> Validate(ValidationContext validationContext)
{ {

View File

@ -112,6 +112,14 @@
{<option asp-selected='@Model.Filter.Price == @price.Id' value="@price.Id">@price.Id</option>} {<option asp-selected='@Model.Filter.Price == @price.Id' value="@price.Id">@price.Id</option>}
</select> </select>
</div> </div>
<div class="col-6">
<label asp-for="Filter.TestClock">Test Clock</label>
<select asp-for="Filter.TestClock" name="filter.TestClock" class="form-control mr-2">
<option asp-selected="Model.Filter.TestClock == null" value="@null">All</option>
@foreach (var clock in Model.TestClocks)
{<option asp-selected='@Model.Filter.TestClock == @clock.Id' value="@clock.Id">@clock.Name</option>}
</select>
</div>
</div> </div>
<div class="row col-12 d-flex justify-content-end my-3"> <div class="row col-12 d-flex justify-content-end my-3">
<button type="submit" class="btn btn-primary" title="Search" name="action" asp-for="Action" value="@StripeSubscriptionsAction.Search"><i class="fa fa-search"></i> Search</button> <button type="submit" class="btn btn-primary" title="Search" name="action" asp-for="Action" value="@StripeSubscriptionsAction.Search"><i class="fa fa-search"></i> Search</button>

View File

@ -27,6 +27,13 @@
public Stripe.SubscriptionListOptions ToStripeApiOptions() public Stripe.SubscriptionListOptions ToStripeApiOptions()
{ {
var stripeApiOptions = (Stripe.SubscriptionListOptions)this; var stripeApiOptions = (Stripe.SubscriptionListOptions)this;
if (SelectAll)
{
stripeApiOptions.EndingBefore = null;
stripeApiOptions.StartingAfter = null;
}
if (CurrentPeriodEndDate.HasValue) if (CurrentPeriodEndDate.HasValue)
{ {
stripeApiOptions.CurrentPeriodEnd = new Stripe.DateRangeOptions() stripeApiOptions.CurrentPeriodEnd = new Stripe.DateRangeOptions()
@ -35,6 +42,7 @@
GreaterThan = CurrentPeriodEndRange == "gt" ? CurrentPeriodEndDate : null GreaterThan = CurrentPeriodEndRange == "gt" ? CurrentPeriodEndDate : null
}; };
} }
return stripeApiOptions; return stripeApiOptions;
} }
} }

View File

@ -35,5 +35,6 @@ namespace Bit.Core.Services
Task<Stripe.BankAccount> BankAccountCreateAsync(string customerId, Stripe.BankAccountCreateOptions options = null); Task<Stripe.BankAccount> BankAccountCreateAsync(string customerId, Stripe.BankAccountCreateOptions options = null);
Task<Stripe.BankAccount> BankAccountDeleteAsync(string customerId, string bankAccount, Stripe.BankAccountDeleteOptions options = null); Task<Stripe.BankAccount> BankAccountDeleteAsync(string customerId, string bankAccount, Stripe.BankAccountDeleteOptions options = null);
Task<Stripe.StripeList<Stripe.Price>> PriceListAsync(Stripe.PriceListOptions options = null); Task<Stripe.StripeList<Stripe.Price>> PriceListAsync(Stripe.PriceListOptions options = null);
Task<List<Stripe.TestHelpers.TestClock>> TestClockListAsync();
} }
} }

View File

@ -15,6 +15,7 @@ namespace Bit.Core.Services
private readonly Stripe.CardService _cardService; private readonly Stripe.CardService _cardService;
private readonly Stripe.BankAccountService _bankAccountService; private readonly Stripe.BankAccountService _bankAccountService;
private readonly Stripe.PriceService _priceService; private readonly Stripe.PriceService _priceService;
private readonly Stripe.TestHelpers.TestClockService _testClockService;
public StripeAdapter() public StripeAdapter()
{ {
@ -29,6 +30,7 @@ namespace Bit.Core.Services
_cardService = new Stripe.CardService(); _cardService = new Stripe.CardService();
_bankAccountService = new Stripe.BankAccountService(); _bankAccountService = new Stripe.BankAccountService();
_priceService = new Stripe.PriceService(); _priceService = new Stripe.PriceService();
_testClockService = new Stripe.TestHelpers.TestClockService();
} }
public Task<Stripe.Customer> CustomerCreateAsync(Stripe.CustomerCreateOptions options) public Task<Stripe.Customer> CustomerCreateAsync(Stripe.CustomerCreateOptions options)
@ -198,5 +200,19 @@ namespace Bit.Core.Services
{ {
return await _priceService.ListAsync(options); return await _priceService.ListAsync(options);
} }
public async Task<List<Stripe.TestHelpers.TestClock>> TestClockListAsync()
{
var items = new List<Stripe.TestHelpers.TestClock>();
var options = new Stripe.TestHelpers.TestClockListOptions()
{
Limit = 100
};
await foreach (var i in _testClockService.ListAutoPagingAsync(options))
{
items.Add(i);
}
return items;
}
} }
} }