2022-04-11 10:48:12 +02:00
#nullable enable
2020-06-29 04:44:35 +02:00
using System ;
2018-08-22 13:57:54 +02:00
using System.Collections.Generic ;
using System.Globalization ;
using System.Linq ;
2019-03-05 09:09:17 +01:00
using System.Threading ;
2018-08-22 13:57:54 +02:00
using System.Threading.Tasks ;
2020-11-17 13:46:23 +01:00
using BTCPayServer.Abstractions.Constants ;
2020-03-19 11:11:15 +01:00
using BTCPayServer.Client ;
2018-02-26 10:58:02 +01:00
using BTCPayServer.Configuration ;
2017-10-17 06:52:30 +02:00
using BTCPayServer.Data ;
2019-08-20 10:38:15 +02:00
using BTCPayServer.HostedServices ;
2017-09-13 16:50:36 +02:00
using BTCPayServer.Models ;
using BTCPayServer.Models.StoreViewModels ;
2019-01-07 09:52:27 +01:00
using BTCPayServer.Payments ;
using BTCPayServer.Payments.Lightning ;
2018-05-03 18:46:52 +02:00
using BTCPayServer.Rating ;
2019-10-18 17:54:20 +02:00
using BTCPayServer.Security.Bitpay ;
2018-02-12 19:27:36 +01:00
using BTCPayServer.Services ;
2020-03-15 10:51:57 +01:00
using BTCPayServer.Services.Apps ;
2019-05-29 16:33:31 +02:00
using BTCPayServer.Services.Invoices ;
2018-04-18 09:38:17 +02:00
using BTCPayServer.Services.Rates ;
2017-09-15 09:06:57 +02:00
using BTCPayServer.Services.Stores ;
using BTCPayServer.Services.Wallets ;
2017-09-13 16:50:36 +02:00
using Microsoft.AspNetCore.Authorization ;
2021-06-17 11:27:17 +02:00
using Microsoft.AspNetCore.DataProtection ;
2017-09-13 16:50:36 +02:00
using Microsoft.AspNetCore.Identity ;
using Microsoft.AspNetCore.Mvc ;
2017-10-17 06:52:30 +02:00
using Microsoft.AspNetCore.Mvc.Rendering ;
2022-01-19 03:52:05 +01:00
using Microsoft.Extensions.Options ;
2017-09-13 16:50:36 +02:00
using NBitcoin ;
2017-12-06 10:08:21 +01:00
using NBitcoin.DataEncoders ;
2020-05-23 21:13:18 +02:00
using StoreData = BTCPayServer . Data . StoreData ;
2017-09-13 16:50:36 +02:00
namespace BTCPayServer.Controllers
{
2017-10-27 10:53:04 +02:00
[Route("stores")]
2019-10-12 13:35:30 +02:00
[Authorize(AuthenticationSchemes = AuthenticationSchemes.Cookie)]
2020-03-20 05:41:47 +01:00
[Authorize(Policy = Policies.CanModifyStoreSettings, AuthenticationSchemes = AuthenticationSchemes.Cookie)]
2017-10-27 10:53:04 +02:00
[AutoValidateAntiforgeryToken]
2022-01-07 04:32:00 +01:00
public partial class UIStoresController : Controller
2017-10-27 10:53:04 +02:00
{
2022-01-07 04:32:00 +01:00
public UIStoresController (
2018-02-25 16:48:12 +01:00
IServiceProvider serviceProvider ,
2018-02-26 10:58:02 +01:00
BTCPayServerOptions btcpayServerOptions ,
BTCPayServerEnvironment btcpayEnv ,
2017-10-27 10:53:04 +02:00
StoreRepository repo ,
TokenRepository tokenRepo ,
UserManager < ApplicationUser > userManager ,
2022-01-07 04:08:28 +01:00
BitpayAccessTokenController tokenController ,
2018-01-11 06:36:12 +01:00
BTCPayWalletProvider walletProvider ,
2017-12-21 07:52:04 +01:00
BTCPayNetworkProvider networkProvider ,
2018-08-22 09:53:40 +02:00
RateFetcher rateFactory ,
2018-01-08 14:45:09 +01:00
ExplorerClientProvider explorerProvider ,
2018-03-23 09:27:48 +01:00
LanguageService langService ,
2019-08-20 10:38:15 +02:00
PaymentMethodHandlerDictionary paymentMethodHandlerDictionary ,
2022-05-24 06:18:16 +02:00
PoliciesSettings policiesSettings ,
2019-12-18 14:28:03 +01:00
IAuthorizationService authorizationService ,
2020-01-18 06:12:27 +01:00
EventAggregator eventAggregator ,
2020-09-18 17:20:31 +02:00
AppService appService ,
2022-01-11 05:15:48 +01:00
WebhookSender webhookNotificationManager ,
2021-09-03 08:37:12 +02:00
IDataProtectionProvider dataProtector ,
2022-01-19 03:52:05 +01:00
IOptions < ExternalServicesOptions > externalServiceOptions )
2017-10-27 10:53:04 +02:00
{
2018-05-03 18:46:52 +02:00
_RateFactory = rateFactory ;
2017-10-27 10:53:04 +02:00
_Repo = repo ;
_TokenRepository = tokenRepo ;
_UserManager = userManager ;
2018-03-23 09:27:48 +01:00
_LangService = langService ;
2017-10-27 10:53:04 +02:00
_TokenController = tokenController ;
2018-01-11 06:36:12 +01:00
_WalletProvider = walletProvider ;
2019-05-29 16:33:31 +02:00
_paymentMethodHandlerDictionary = paymentMethodHandlerDictionary ;
2022-05-24 06:18:16 +02:00
_policiesSettings = policiesSettings ;
2019-12-18 14:28:03 +01:00
_authorizationService = authorizationService ;
2020-03-15 10:51:57 +01:00
_appService = appService ;
2021-06-17 11:27:17 +02:00
DataProtector = dataProtector . CreateProtector ( "ConfigProtector" ) ;
2020-11-06 12:42:26 +01:00
WebhookNotificationManager = webhookNotificationManager ;
2020-01-18 06:12:27 +01:00
_EventAggregator = eventAggregator ;
2018-01-06 10:57:56 +01:00
_NetworkProvider = networkProvider ;
2018-01-08 14:45:09 +01:00
_ExplorerProvider = explorerProvider ;
2018-02-25 16:48:12 +01:00
_ServiceProvider = serviceProvider ;
2018-02-26 10:58:02 +01:00
_BtcpayServerOptions = btcpayServerOptions ;
_BTCPayEnv = btcpayEnv ;
2022-01-19 03:52:05 +01:00
_externalServiceOptions = externalServiceOptions ;
2017-10-27 10:53:04 +02:00
}
2020-06-29 05:07:48 +02:00
readonly BTCPayServerOptions _BtcpayServerOptions ;
readonly BTCPayServerEnvironment _BTCPayEnv ;
readonly IServiceProvider _ServiceProvider ;
readonly BTCPayNetworkProvider _NetworkProvider ;
readonly BTCPayWalletProvider _WalletProvider ;
2022-01-07 04:08:28 +01:00
readonly BitpayAccessTokenController _TokenController ;
2020-06-29 05:07:48 +02:00
readonly StoreRepository _Repo ;
readonly TokenRepository _TokenRepository ;
readonly UserManager < ApplicationUser > _UserManager ;
2022-01-19 03:52:05 +01:00
readonly RateFetcher _RateFactory ;
private readonly ExplorerClientProvider _ExplorerProvider ;
2020-06-29 05:07:48 +02:00
private readonly LanguageService _LangService ;
2019-05-29 16:33:31 +02:00
private readonly PaymentMethodHandlerDictionary _paymentMethodHandlerDictionary ;
2022-05-24 06:18:16 +02:00
private readonly PoliciesSettings _policiesSettings ;
2019-12-18 14:28:03 +01:00
private readonly IAuthorizationService _authorizationService ;
2020-03-15 10:51:57 +01:00
private readonly AppService _appService ;
2020-01-18 06:12:27 +01:00
private readonly EventAggregator _EventAggregator ;
2022-01-19 03:52:05 +01:00
private readonly IOptions < ExternalServicesOptions > _externalServiceOptions ;
2017-10-27 10:53:04 +02:00
2019-01-18 11:15:31 +01:00
[TempData]
public bool StoreNotConfigured
{
get ; set ;
}
2017-10-27 10:53:04 +02:00
[HttpGet]
2018-03-23 08:24:57 +01:00
[Route("{storeId}/users")]
2018-04-30 15:00:43 +02:00
public async Task < IActionResult > StoreUsers ( )
2018-03-23 08:24:57 +01:00
{
StoreUsersViewModel vm = new StoreUsersViewModel ( ) ;
2018-04-30 15:00:43 +02:00
await FillUsers ( vm ) ;
2018-03-23 08:24:57 +01:00
return View ( vm ) ;
}
2018-04-30 15:00:43 +02:00
private async Task FillUsers ( StoreUsersViewModel vm )
2017-10-27 10:53:04 +02:00
{
2019-10-12 13:35:30 +02:00
var users = await _Repo . GetStoreUsers ( CurrentStore . Id ) ;
vm . StoreId = CurrentStore . Id ;
2018-03-23 08:24:57 +01:00
vm . Users = users . Select ( u = > new StoreUsersViewModel . StoreUserViewModel ( )
2017-10-27 10:53:04 +02:00
{
2018-03-23 08:24:57 +01:00
Email = u . Email ,
Id = u . Id ,
Role = u . Role
} ) . ToList ( ) ;
2017-10-27 10:53:04 +02:00
}
2022-04-12 09:55:10 +02:00
public StoreData CurrentStore = > HttpContext . GetStoreData ( ) ;
2018-03-23 08:24:57 +01:00
[HttpPost]
[Route("{storeId}/users")]
2018-04-30 15:00:43 +02:00
public async Task < IActionResult > StoreUsers ( StoreUsersViewModel vm )
2018-02-15 05:33:29 +01:00
{
2018-04-30 15:00:43 +02:00
await FillUsers ( vm ) ;
2018-03-24 12:40:26 +01:00
if ( ! ModelState . IsValid )
2018-02-15 05:33:29 +01:00
{
2018-03-23 08:24:57 +01:00
return View ( vm ) ;
}
var user = await _UserManager . FindByEmailAsync ( vm . Email ) ;
2018-03-24 12:40:26 +01:00
if ( user = = null )
2018-03-23 08:24:57 +01:00
{
ModelState . AddModelError ( nameof ( vm . Email ) , "User not found" ) ;
return View ( vm ) ;
}
2018-03-24 12:40:26 +01:00
if ( ! StoreRoles . AllRoles . Contains ( vm . Role ) )
2018-03-23 08:24:57 +01:00
{
ModelState . AddModelError ( nameof ( vm . Role ) , "Invalid role" ) ;
return View ( vm ) ;
}
2019-10-12 13:35:30 +02:00
if ( ! await _Repo . AddStoreUser ( CurrentStore . Id , user . Id , vm . Role ) )
2018-03-23 08:24:57 +01:00
{
ModelState . AddModelError ( nameof ( vm . Email ) , "The user already has access to this store" ) ;
return View ( vm ) ;
2018-02-15 05:33:29 +01:00
}
2021-09-07 04:55:53 +02:00
TempData [ WellKnownTempData . SuccessMessage ] = "User added successfully." ;
2018-03-23 08:24:57 +01:00
return RedirectToAction ( nameof ( StoreUsers ) ) ;
2018-02-15 05:33:29 +01:00
}
2021-09-07 04:55:53 +02:00
[HttpGet("{storeId}/users/{userId}/delete")]
2018-04-30 15:00:43 +02:00
public async Task < IActionResult > DeleteStoreUser ( string userId )
2017-10-27 10:53:04 +02:00
{
2018-03-23 08:24:57 +01:00
var user = await _UserManager . FindByIdAsync ( userId ) ;
if ( user = = null )
return NotFound ( ) ;
2021-09-07 04:55:53 +02:00
return View ( "Confirm" , new ConfirmModel ( "Remove store user" , $"This action will prevent <strong>{user.Email}</strong> from accessing this store and its settings. Are you sure?" , "Remove" ) ) ;
2017-10-27 10:53:04 +02:00
}
2021-09-07 04:55:53 +02:00
[HttpPost("{storeId}/users/{userId}/delete")]
2018-03-23 08:24:57 +01:00
public async Task < IActionResult > DeleteStoreUserPost ( string storeId , string userId )
2017-10-27 10:53:04 +02:00
{
2022-02-10 06:51:10 +01:00
if ( await _Repo . RemoveStoreUser ( storeId , userId ) )
TempData [ WellKnownTempData . SuccessMessage ] = "User removed successfully." ;
else
{
TempData [ WellKnownTempData . ErrorMessage ] = "Removing this user would result in the store having no owner." ;
}
2021-09-07 04:55:53 +02:00
return RedirectToAction ( nameof ( StoreUsers ) , new { storeId , userId } ) ;
2017-10-27 10:53:04 +02:00
}
2021-09-07 04:55:53 +02:00
[HttpGet("{storeId}/rates")]
2020-01-13 14:20:45 +01:00
public IActionResult Rates ( )
2018-05-03 18:46:52 +02:00
{
2020-01-13 14:20:45 +01:00
var exchanges = GetSupportedExchanges ( ) ;
2019-10-12 13:35:30 +02:00
var storeBlob = CurrentStore . GetStoreBlob ( ) ;
2018-05-03 18:46:52 +02:00
var vm = new RatesViewModel ( ) ;
2020-01-10 14:50:39 +01:00
vm . SetExchangeRates ( exchanges , storeBlob . PreferredExchange ? ? CoinGeckoRateProvider . CoinGeckoName ) ;
2018-08-01 11:38:46 +02:00
vm . Spread = ( double ) ( storeBlob . Spread * 100 m ) ;
2019-10-12 13:35:30 +02:00
vm . StoreId = CurrentStore . Id ;
2018-05-03 18:46:52 +02:00
vm . Script = storeBlob . GetRateRules ( _NetworkProvider ) . ToString ( ) ;
vm . DefaultScript = storeBlob . GetDefaultRateRules ( _NetworkProvider ) . ToString ( ) ;
2020-01-10 14:50:39 +01:00
vm . AvailableExchanges = exchanges ;
2019-03-11 10:39:21 +01:00
vm . DefaultCurrencyPairs = storeBlob . GetDefaultCurrencyPairString ( ) ;
2018-05-03 18:46:52 +02:00
vm . ShowScripting = storeBlob . RateScripting ;
return View ( vm ) ;
}
2021-09-07 04:55:53 +02:00
[HttpPost("{storeId}/rates")]
2022-04-11 10:48:12 +02:00
public async Task < IActionResult > Rates ( RatesViewModel model , string? command = null , string? storeId = null , CancellationToken cancellationToken = default )
2018-05-03 18:46:52 +02:00
{
2020-01-10 14:50:39 +01:00
if ( command = = "scripting-on" )
{
2020-06-28 10:55:27 +02:00
return RedirectToAction ( nameof ( ShowRateRules ) , new { scripting = true , storeId = model . StoreId } ) ;
}
else if ( command = = "scripting-off" )
2020-01-10 14:50:39 +01:00
{
2020-06-28 10:55:27 +02:00
return RedirectToAction ( nameof ( ShowRateRules ) , new { scripting = false , storeId = model . StoreId } ) ;
2020-01-10 14:50:39 +01:00
}
2020-01-13 14:20:45 +01:00
var exchanges = GetSupportedExchanges ( ) ;
2020-01-10 14:50:39 +01:00
model . SetExchangeRates ( exchanges , model . PreferredExchange ) ;
2019-03-11 10:39:21 +01:00
model . StoreId = storeId ? ? model . StoreId ;
2022-04-11 10:48:12 +02:00
CurrencyPair [ ] ? currencyPairs = null ;
2019-03-11 10:39:21 +01:00
try
{
currencyPairs = model . DefaultCurrencyPairs ?
. Split ( new [ ] { ',' } , StringSplitOptions . RemoveEmptyEntries )
. Select ( p = > CurrencyPair . Parse ( p ) )
. ToArray ( ) ;
}
catch
{
ModelState . AddModelError ( nameof ( model . DefaultCurrencyPairs ) , "Invalid currency pairs (should be for example: BTC_USD,BTC_CAD,BTC_JPY)" ) ;
}
2018-05-03 18:46:52 +02:00
if ( ! ModelState . IsValid )
{
return View ( model ) ;
}
if ( model . PreferredExchange ! = null )
model . PreferredExchange = model . PreferredExchange . Trim ( ) . ToLowerInvariant ( ) ;
2019-10-12 13:35:30 +02:00
var blob = CurrentStore . GetStoreBlob ( ) ;
2018-05-03 18:46:52 +02:00
model . DefaultScript = blob . GetDefaultRateRules ( _NetworkProvider ) . ToString ( ) ;
2020-01-10 14:50:39 +01:00
model . AvailableExchanges = exchanges ;
2018-05-03 18:46:52 +02:00
blob . PreferredExchange = model . PreferredExchange ;
2018-08-01 11:38:46 +02:00
blob . Spread = ( decimal ) model . Spread / 100.0 m ;
2019-03-11 10:39:21 +01:00
blob . DefaultCurrencyPairs = currencyPairs ;
2018-05-03 18:46:52 +02:00
if ( ! model . ShowScripting )
{
2020-01-10 14:50:39 +01:00
if ( ! exchanges . Any ( provider = > provider . Id . Equals ( model . PreferredExchange , StringComparison . InvariantCultureIgnoreCase ) ) )
2018-05-03 18:46:52 +02:00
{
ModelState . AddModelError ( nameof ( model . PreferredExchange ) , $"Unsupported exchange ({model.RateSource})" ) ;
return View ( model ) ;
}
}
2022-04-11 10:48:12 +02:00
RateRules ? rules = null ;
2018-05-03 18:46:52 +02:00
if ( model . ShowScripting )
{
if ( ! RateRules . TryParse ( model . Script , out rules , out var errors ) )
{
errors = errors ? ? new List < RateRulesErrors > ( ) ;
var errorString = String . Join ( ", " , errors . ToArray ( ) ) ;
ModelState . AddModelError ( nameof ( model . Script ) , $"Parsing error ({errorString})" ) ;
return View ( model ) ;
}
else
{
blob . RateScript = rules . ToString ( ) ;
2018-05-04 04:48:03 +02:00
ModelState . Remove ( nameof ( model . Script ) ) ;
model . Script = blob . RateScript ;
2018-05-03 18:46:52 +02:00
}
}
rules = blob . GetRateRules ( _NetworkProvider ) ;
if ( command = = "Test" )
{
if ( string . IsNullOrWhiteSpace ( model . ScriptTest ) )
{
ModelState . AddModelError ( nameof ( model . ScriptTest ) , "Fill out currency pair to test for (like BTC_USD,BTC_CAD)" ) ;
return View ( model ) ;
}
var splitted = model . ScriptTest . Split ( ',' , StringSplitOptions . RemoveEmptyEntries ) ;
var pairs = new List < CurrencyPair > ( ) ;
foreach ( var pair in splitted )
{
if ( ! CurrencyPair . TryParse ( pair , out var currencyPair ) )
{
ModelState . AddModelError ( nameof ( model . ScriptTest ) , $"Invalid currency pair '{pair}' (it should be formatted like BTC_USD,BTC_CAD)" ) ;
return View ( model ) ;
}
pairs . Add ( currencyPair ) ;
}
2019-03-05 09:09:17 +01:00
var fetchs = _RateFactory . FetchRates ( pairs . ToHashSet ( ) , rules , cancellationToken ) ;
2018-05-03 18:46:52 +02:00
var testResults = new List < RatesViewModel . TestResultViewModel > ( ) ;
foreach ( var fetch in fetchs )
{
var testResult = await ( fetch . Value ) ;
testResults . Add ( new RatesViewModel . TestResultViewModel ( )
{
CurrencyPair = fetch . Key . ToString ( ) ,
Error = testResult . Errors . Count ! = 0 ,
2018-07-27 11:04:41 +02:00
Rule = testResult . Errors . Count = = 0 ? testResult . Rule + " = " + testResult . BidAsk . Bid . ToString ( CultureInfo . InvariantCulture )
2018-05-03 18:46:52 +02:00
: testResult . EvaluatedRule
} ) ;
}
model . TestRateRules = testResults ;
return View ( model ) ;
}
else // command == Save
{
2019-10-12 13:35:30 +02:00
if ( CurrentStore . SetStoreBlob ( blob ) )
2018-05-03 18:46:52 +02:00
{
2019-10-12 13:35:30 +02:00
await _Repo . UpdateStore ( CurrentStore ) ;
2019-10-31 04:29:59 +01:00
TempData [ WellKnownTempData . SuccessMessage ] = "Rate settings updated" ;
2018-05-03 18:46:52 +02:00
}
return RedirectToAction ( nameof ( Rates ) , new
{
2019-10-12 13:35:30 +02:00
storeId = CurrentStore . Id
2018-05-03 18:46:52 +02:00
} ) ;
}
}
2021-09-07 04:55:53 +02:00
[HttpGet("{storeId}/rates/confirm")]
2018-05-03 18:46:52 +02:00
public IActionResult ShowRateRules ( bool scripting )
{
2021-09-07 04:55:53 +02:00
return View ( "Confirm" , new ConfirmModel
2018-05-03 18:46:52 +02:00
{
2018-05-04 09:09:43 +02:00
Action = "Continue" ,
2018-05-03 18:46:52 +02:00
Title = "Rate rule scripting" ,
Description = scripting ?
2018-06-25 04:58:07 +02:00
"This action will modify your current rate sources. Are you sure to turn on rate rules scripting? (Advanced users)"
2018-05-03 18:46:52 +02:00
: "This action will delete your rate script. Are you sure to turn off rate rules scripting?" ,
2020-01-13 14:20:45 +01:00
ButtonClass = scripting ? "btn-primary" : "btn-danger"
2018-05-03 18:46:52 +02:00
} ) ;
}
2021-09-07 04:55:53 +02:00
[HttpPost("{storeId}/rates/confirm")]
2018-05-03 18:46:52 +02:00
public async Task < IActionResult > ShowRateRulesPost ( bool scripting )
{
2019-10-12 13:35:30 +02:00
var blob = CurrentStore . GetStoreBlob ( ) ;
2018-05-03 18:46:52 +02:00
blob . RateScripting = scripting ;
blob . RateScript = blob . GetDefaultRateRules ( _NetworkProvider ) . ToString ( ) ;
2019-10-12 13:35:30 +02:00
CurrentStore . SetStoreBlob ( blob ) ;
await _Repo . UpdateStore ( CurrentStore ) ;
2021-09-07 04:55:53 +02:00
TempData [ WellKnownTempData . SuccessMessage ] = "Rate rules scripting " + ( scripting ? "activated" : "deactivated" ) ;
2019-10-12 13:35:30 +02:00
return RedirectToAction ( nameof ( Rates ) , new { storeId = CurrentStore . Id } ) ;
2018-05-03 18:46:52 +02:00
}
2021-09-29 17:23:01 +02:00
[HttpGet("{storeId}/checkout")]
2021-10-29 08:25:43 +02:00
public IActionResult CheckoutAppearance ( )
2018-03-27 07:48:32 +02:00
{
2019-10-12 13:35:30 +02:00
var storeBlob = CurrentStore . GetStoreBlob ( ) ;
2021-10-29 08:25:43 +02:00
var vm = new CheckoutAppearanceViewModel ( ) ;
2019-10-12 13:35:30 +02:00
SetCryptoCurrencies ( vm , CurrentStore ) ;
2021-10-25 08:18:02 +02:00
vm . PaymentMethodCriteria = CurrentStore . GetSupportedPaymentMethods ( _NetworkProvider )
. Where ( s = > ! storeBlob . GetExcludedPaymentMethods ( ) . Match ( s . PaymentId ) )
. Where ( s = > _NetworkProvider . GetNetwork ( s . PaymentId . CryptoCode ) ! = null )
. Where ( s = > s . PaymentId . PaymentType ! = PaymentTypes . LNURLPay )
. Select ( method = >
2020-12-29 09:58:35 +01:00
{
var existing =
storeBlob . PaymentMethodCriteria . SingleOrDefault ( criteria = >
criteria . PaymentMethod = = method . PaymentId ) ;
if ( existing is null )
{
return new PaymentMethodCriteriaViewModel ( )
2020-09-15 11:09:09 +02:00
{
2020-12-29 09:58:35 +01:00
PaymentMethod = method . PaymentId . ToString ( ) ,
Value = ""
} ;
}
else
{
return new PaymentMethodCriteriaViewModel ( )
{
PaymentMethod = existing . PaymentMethod . ToString ( ) ,
Type = existing . Above
? PaymentMethodCriteriaViewModel . CriteriaType . GreaterThan
: PaymentMethodCriteriaViewModel . CriteriaType . LessThan ,
Value = existing . Value ? . ToString ( ) ? ? ""
} ;
}
} ) . ToList ( ) ;
2021-12-31 08:59:02 +01:00
2019-03-16 04:43:57 +01:00
vm . RequiresRefundEmail = storeBlob . RequiresRefundEmail ;
2021-04-07 06:08:42 +02:00
vm . LazyPaymentMethods = storeBlob . LazyPaymentMethods ;
2019-04-11 11:53:31 +02:00
vm . RedirectAutomatically = storeBlob . RedirectAutomatically ;
2020-11-09 08:11:03 +01:00
vm . CustomCSS = storeBlob . CustomCSS ;
vm . CustomLogo = storeBlob . CustomLogo ;
vm . HtmlTitle = storeBlob . HtmlTitle ;
2021-07-27 08:17:56 +02:00
vm . AutoDetectLanguage = storeBlob . AutoDetectLanguage ;
2020-11-09 08:11:03 +01:00
vm . SetLanguages ( _LangService , storeBlob . DefaultLang ) ;
2021-12-31 08:59:02 +01:00
2018-03-27 07:48:32 +02:00
return View ( vm ) ;
}
2020-09-15 11:09:09 +02:00
2021-10-29 08:25:43 +02:00
void SetCryptoCurrencies ( CheckoutAppearanceViewModel vm , Data . StoreData storeData )
2022-04-11 10:48:12 +02:00
{
var choices = GetEnabledPaymentMethodChoices ( storeData ) ;
var chosen = GetDefaultPaymentMethodChoice ( storeData ) ;
vm . PaymentMethods = new SelectList ( choices , nameof ( chosen . Value ) , nameof ( chosen . Name ) , chosen ? . Value ) ;
vm . DefaultPaymentMethod = chosen ? . Value ;
}
2022-06-15 04:32:46 +02:00
public PaymentMethodOptionViewModel . Format [ ] GetEnabledPaymentMethodChoices ( StoreData storeData )
2022-04-11 10:48:12 +02:00
{
var enabled = storeData . GetEnabledPaymentIds ( _NetworkProvider ) ;
return enabled
. Select ( o = >
new PaymentMethodOptionViewModel . Format ( )
{
Name = o . ToPrettyString ( ) ,
Value = o . ToString ( ) ,
PaymentId = o
} ) . ToArray ( ) ;
}
2022-06-15 04:32:46 +02:00
PaymentMethodOptionViewModel . Format ? GetDefaultPaymentMethodChoice ( StoreData storeData )
2019-01-31 14:03:28 +01:00
{
2021-10-18 09:56:47 +02:00
var enabled = storeData . GetEnabledPaymentIds ( _NetworkProvider ) ;
var defaultPaymentId = storeData . GetDefaultPaymentId ( ) ;
2022-01-14 09:48:15 +01:00
var defaultChoice = defaultPaymentId is not null ? defaultPaymentId . FindNearest ( enabled ) : null ;
2021-11-04 10:21:38 +01:00
if ( defaultChoice is null )
{
2022-04-20 03:20:39 +02:00
defaultChoice = enabled . FirstOrDefault ( e = > e . CryptoCode = = _NetworkProvider . DefaultNetwork . CryptoCode & & e . PaymentType = = PaymentTypes . BTCLike ) ? ?
enabled . FirstOrDefault ( e = > e . CryptoCode = = _NetworkProvider . DefaultNetwork . CryptoCode & & e . PaymentType = = PaymentTypes . LightningLike ) ? ?
2021-11-04 10:21:38 +01:00
enabled . FirstOrDefault ( ) ;
}
2022-04-11 10:48:12 +02:00
var choices = GetEnabledPaymentMethodChoices ( storeData ) ;
return defaultChoice is null ? null : choices . FirstOrDefault ( c = > defaultChoice . ToString ( ) . Equals ( c . Value , StringComparison . OrdinalIgnoreCase ) ) ;
2019-01-31 14:03:28 +01:00
}
2021-12-31 08:59:02 +01:00
2018-03-27 07:48:32 +02:00
[HttpPost]
[Route("{storeId}/checkout")]
2021-10-29 08:25:43 +02:00
public async Task < IActionResult > CheckoutAppearance ( CheckoutAppearanceViewModel model )
2018-03-27 07:48:32 +02:00
{
bool needUpdate = false ;
2019-10-12 13:35:30 +02:00
var blob = CurrentStore . GetStoreBlob ( ) ;
2019-01-31 11:07:38 +01:00
var defaultPaymentMethodId = model . DefaultPaymentMethod = = null ? null : PaymentMethodId . Parse ( model . DefaultPaymentMethod ) ;
2021-10-18 09:56:47 +02:00
if ( CurrentStore . GetDefaultPaymentId ( ) ! = defaultPaymentMethodId )
2018-03-27 07:48:32 +02:00
{
needUpdate = true ;
2019-10-12 13:35:30 +02:00
CurrentStore . SetDefaultPaymentId ( defaultPaymentMethodId ) ;
2018-03-27 07:48:32 +02:00
}
2019-10-12 13:35:30 +02:00
SetCryptoCurrencies ( model , CurrentStore ) ;
2018-03-27 07:48:32 +02:00
model . SetLanguages ( _LangService , model . DefaultLang ) ;
2021-09-03 08:37:12 +02:00
model . PaymentMethodCriteria ? ? = new List < PaymentMethodCriteriaViewModel > ( ) ;
2020-09-15 11:09:09 +02:00
for ( var index = 0 ; index < model . PaymentMethodCriteria . Count ; index + + )
{
var methodCriterion = model . PaymentMethodCriteria [ index ] ;
if ( ! string . IsNullOrWhiteSpace ( methodCriterion . Value ) )
{
if ( ! CurrencyValue . TryParse ( methodCriterion . Value , out var value ) )
{
2020-10-31 11:34:40 +01:00
model . AddModelError ( viewModel = > viewModel . PaymentMethodCriteria [ index ] . Value ,
2021-10-20 07:02:20 +02:00
$"{methodCriterion.PaymentMethod}: Invalid format. Make sure to enter a valid amount and currency code. Examples: '5 USD', '0.001 BTC'" , this ) ;
2020-09-15 11:09:09 +02:00
}
}
}
2021-09-03 08:37:12 +02:00
2018-04-18 09:38:17 +02:00
if ( ! ModelState . IsValid )
2018-04-03 10:54:50 +02:00
{
return View ( model ) ;
}
2020-09-15 11:09:09 +02:00
2021-10-25 08:18:02 +02:00
// Payment criteria for Off-Chain should also affect LNUrl
foreach ( var newCriteria in model . PaymentMethodCriteria . ToList ( ) )
{
var paymentMethodId = PaymentMethodId . Parse ( newCriteria . PaymentMethod ) ;
if ( paymentMethodId . PaymentType = = PaymentTypes . LightningLike )
model . PaymentMethodCriteria . Add ( new PaymentMethodCriteriaViewModel ( )
{
PaymentMethod = new PaymentMethodId ( paymentMethodId . CryptoCode , PaymentTypes . LNURLPay ) . ToString ( ) ,
Type = newCriteria . Type ,
Value = newCriteria . Value
} ) ;
// Should not be able to set LNUrlPay criteria directly in UI
if ( paymentMethodId . PaymentType = = PaymentTypes . LNURLPay )
model . PaymentMethodCriteria . Remove ( newCriteria ) ;
}
blob . PaymentMethodCriteria ? ? = new List < PaymentMethodCriteria > ( ) ;
foreach ( var newCriteria in model . PaymentMethodCriteria )
{
var paymentMethodId = PaymentMethodId . Parse ( newCriteria . PaymentMethod ) ;
var existingCriteria = blob . PaymentMethodCriteria . FirstOrDefault ( c = > c . PaymentMethod = = paymentMethodId ) ;
if ( existingCriteria ! = null )
blob . PaymentMethodCriteria . Remove ( existingCriteria ) ;
CurrencyValue . TryParse ( newCriteria . Value , out var cv ) ;
blob . PaymentMethodCriteria . Add ( new PaymentMethodCriteria ( )
2020-09-15 11:09:09 +02:00
{
2021-10-25 08:18:02 +02:00
Above = newCriteria . Type = = PaymentMethodCriteriaViewModel . CriteriaType . GreaterThan ,
Value = cv ,
PaymentMethod = paymentMethodId
} ) ;
}
2019-03-16 04:43:57 +01:00
blob . RequiresRefundEmail = model . RequiresRefundEmail ;
2021-04-07 06:08:42 +02:00
blob . LazyPaymentMethods = model . LazyPaymentMethods ;
2019-04-11 11:53:31 +02:00
blob . RedirectAutomatically = model . RedirectAutomatically ;
2020-11-09 08:11:03 +01:00
blob . CustomLogo = model . CustomLogo ;
blob . CustomCSS = model . CustomCSS ;
blob . HtmlTitle = string . IsNullOrWhiteSpace ( model . HtmlTitle ) ? null : model . HtmlTitle ;
2021-07-27 08:17:56 +02:00
blob . AutoDetectLanguage = model . AutoDetectLanguage ;
2020-11-09 08:11:03 +01:00
blob . DefaultLang = model . DefaultLang ;
2019-10-12 13:35:30 +02:00
if ( CurrentStore . SetStoreBlob ( blob ) )
2018-03-27 07:48:32 +02:00
{
needUpdate = true ;
}
if ( needUpdate )
{
2019-10-12 13:35:30 +02:00
await _Repo . UpdateStore ( CurrentStore ) ;
2019-10-31 04:29:59 +01:00
TempData [ WellKnownTempData . SuccessMessage ] = "Store successfully updated" ;
2018-03-27 07:48:32 +02:00
}
2021-10-29 08:25:43 +02:00
return RedirectToAction ( nameof ( CheckoutAppearance ) , new
2018-03-27 07:48:32 +02:00
{
2019-10-12 13:35:30 +02:00
storeId = CurrentStore . Id
2018-03-27 07:48:32 +02:00
} ) ;
}
2021-12-31 08:59:02 +01:00
internal void AddPaymentMethods ( StoreData store , StoreBlob storeBlob ,
2021-10-25 08:18:02 +02:00
out List < StoreDerivationScheme > derivationSchemes , out List < StoreLightningNode > lightningNodes )
2017-10-27 10:53:04 +02:00
{
2018-07-27 13:37:16 +02:00
var excludeFilters = storeBlob . GetExcludedPaymentMethods ( ) ;
2018-03-24 12:40:26 +01:00
var derivationByCryptoCode =
2018-03-20 18:48:11 +01:00
store
. GetSupportedPaymentMethods ( _NetworkProvider )
2019-05-08 16:39:11 +02:00
. OfType < DerivationSchemeSettings > ( )
2019-12-24 08:20:44 +01:00
. ToDictionary ( c = > c . Network . CryptoCode . ToUpperInvariant ( ) ) ;
2018-02-07 13:59:16 +01:00
2018-03-20 18:48:11 +01:00
var lightningByCryptoCode = store
2019-05-29 16:33:31 +02:00
. GetSupportedPaymentMethods ( _NetworkProvider )
. OfType < LightningSupportedPaymentMethod > ( )
2021-10-25 08:18:02 +02:00
. Where ( method = > method . PaymentId . PaymentType = = LightningPaymentType . Instance )
2019-12-24 08:20:44 +01:00
. ToDictionary ( c = > c . CryptoCode . ToUpperInvariant ( ) ) ;
2018-03-20 18:48:11 +01:00
2021-10-25 08:18:02 +02:00
derivationSchemes = new List < StoreDerivationScheme > ( ) ;
lightningNodes = new List < StoreLightningNode > ( ) ;
2019-05-29 16:33:31 +02:00
foreach ( var paymentMethodId in _paymentMethodHandlerDictionary . Distinct ( ) . SelectMany ( handler = > handler . GetSupportedPaymentMethods ( ) ) )
2018-01-08 14:45:09 +01:00
{
2020-06-28 10:55:27 +02:00
switch ( paymentMethodId . PaymentType )
2017-10-27 10:53:04 +02:00
{
2019-06-04 01:59:01 +02:00
case BitcoinPaymentType _ :
2019-05-29 16:33:31 +02:00
var strategy = derivationByCryptoCode . TryGet ( paymentMethodId . CryptoCode ) ;
2019-12-24 08:20:44 +01:00
var network = _NetworkProvider . GetNetwork < BTCPayNetwork > ( paymentMethodId . CryptoCode ) ;
2020-01-26 11:45:24 +01:00
var value = strategy ? . ToPrettyString ( ) ? ? string . Empty ;
2020-06-28 10:55:27 +02:00
2021-10-25 08:18:02 +02:00
derivationSchemes . Add ( new StoreDerivationScheme
2019-05-29 16:33:31 +02:00
{
Crypto = paymentMethodId . CryptoCode ,
2019-12-24 08:20:44 +01:00
WalletSupported = network . WalletSupported ,
2020-01-26 11:45:24 +01:00
Value = value ,
2019-05-29 16:33:31 +02:00
WalletId = new WalletId ( store . Id , paymentMethodId . CryptoCode ) ,
2020-01-26 11:45:24 +01:00
Enabled = ! excludeFilters . Match ( paymentMethodId ) & & strategy ! = null ,
2020-07-29 11:55:28 +02:00
#if ALTCOINS
2020-01-26 11:45:24 +01:00
Collapsed = network is ElementsBTCPayNetwork elementsBTCPayNetwork & & elementsBTCPayNetwork . NetworkCryptoCode ! = elementsBTCPayNetwork . CryptoCode & & string . IsNullOrEmpty ( value )
2020-07-28 22:48:51 +02:00
#endif
2019-05-29 16:33:31 +02:00
} ) ;
break ;
2021-12-31 08:59:02 +01:00
2021-10-25 08:18:02 +02:00
case LNURLPayPaymentType lnurlPayPaymentType :
break ;
2021-12-31 08:59:02 +01:00
2019-06-04 01:59:01 +02:00
case LightningPaymentType _ :
2019-05-29 16:33:31 +02:00
var lightning = lightningByCryptoCode . TryGet ( paymentMethodId . CryptoCode ) ;
2021-04-16 15:31:09 +02:00
var isEnabled = ! excludeFilters . Match ( paymentMethodId ) & & lightning ! = null ;
2021-10-25 08:18:02 +02:00
lightningNodes . Add ( new StoreLightningNode
2019-05-29 16:33:31 +02:00
{
CryptoCode = paymentMethodId . CryptoCode ,
2021-03-09 09:54:38 +01:00
Address = lightning ? . GetDisplayableConnectionString ( ) ,
2021-04-16 15:31:09 +02:00
Enabled = isEnabled
2019-05-29 16:33:31 +02:00
} ) ;
break ;
2020-06-28 10:55:27 +02:00
}
2018-01-08 14:45:09 +01:00
}
}
2021-09-03 08:37:12 +02:00
2022-01-20 12:52:31 +01:00
[HttpGet("{storeId}/settings")]
public IActionResult GeneralSettings ( )
2020-10-16 06:30:46 +02:00
{
var store = HttpContext . GetStoreData ( ) ;
if ( store = = null )
return NotFound ( ) ;
var storeBlob = store . GetStoreBlob ( ) ;
2022-01-20 12:52:31 +01:00
var vm = new GeneralSettingsViewModel
2021-10-25 08:18:02 +02:00
{
Id = store . Id ,
2022-01-20 12:52:31 +01:00
StoreName = store . StoreName ,
StoreWebsite = store . StoreWebsite ,
2021-10-29 08:25:43 +02:00
NetworkFeeMode = storeBlob . NetworkFeeMode ,
AnyoneCanCreateInvoice = storeBlob . AnyoneCanInvoice ,
PaymentTolerance = storeBlob . PaymentTolerance ,
InvoiceExpiration = ( int ) storeBlob . InvoiceExpiration . TotalMinutes ,
2022-01-24 12:17:09 +01:00
DefaultCurrency = storeBlob . DefaultCurrency ,
2022-01-20 12:52:31 +01:00
BOLT11Expiration = ( long ) storeBlob . RefundBOLT11Expiration . TotalDays ,
2021-10-29 08:25:43 +02:00
CanDelete = _Repo . CanDeleteStores ( )
2021-10-25 08:18:02 +02:00
} ;
2021-12-31 08:59:02 +01:00
2020-10-16 06:30:46 +02:00
return View ( vm ) ;
}
2021-09-03 08:37:12 +02:00
2021-10-29 08:25:43 +02:00
[HttpPost("{storeId}/settings")]
2022-04-11 10:48:12 +02:00
public async Task < IActionResult > GeneralSettings ( GeneralSettingsViewModel model , string? command = null )
2018-01-08 14:45:09 +01:00
{
bool needUpdate = false ;
2021-10-29 08:25:43 +02:00
if ( CurrentStore . StoreName ! = model . StoreName )
2018-01-08 14:45:09 +01:00
{
needUpdate = true ;
2021-10-29 08:25:43 +02:00
CurrentStore . StoreName = model . StoreName ;
2018-01-08 14:45:09 +01:00
}
2021-12-31 08:59:02 +01:00
2021-10-29 08:25:43 +02:00
if ( CurrentStore . StoreWebsite ! = model . StoreWebsite )
2018-01-08 14:45:09 +01:00
{
needUpdate = true ;
2021-10-29 08:25:43 +02:00
CurrentStore . StoreWebsite = model . StoreWebsite ;
2018-01-08 14:45:09 +01:00
}
2022-01-20 12:52:31 +01:00
var blob = CurrentStore . GetStoreBlob ( ) ;
blob . AnyoneCanInvoice = model . AnyoneCanCreateInvoice ;
blob . NetworkFeeMode = model . NetworkFeeMode ;
blob . PaymentTolerance = model . PaymentTolerance ;
blob . DefaultCurrency = model . DefaultCurrency ;
blob . InvoiceExpiration = TimeSpan . FromMinutes ( model . InvoiceExpiration ) ;
blob . RefundBOLT11Expiration = TimeSpan . FromDays ( model . BOLT11Expiration ) ;
if ( CurrentStore . SetStoreBlob ( blob ) )
{
needUpdate = true ;
}
2018-01-08 14:45:09 +01:00
if ( needUpdate )
{
2019-10-12 13:35:30 +02:00
await _Repo . UpdateStore ( CurrentStore ) ;
2020-06-28 10:55:27 +02:00
2021-10-29 08:25:43 +02:00
TempData [ WellKnownTempData . SuccessMessage ] = "Store successfully updated" ;
2018-01-08 14:45:09 +01:00
}
2021-10-29 08:25:43 +02:00
return RedirectToAction ( nameof ( GeneralSettings ) , new
2018-01-08 14:45:09 +01:00
{
2019-10-12 13:35:30 +02:00
storeId = CurrentStore . Id
2018-01-08 14:45:09 +01:00
} ) ;
2018-07-19 15:23:14 +02:00
}
2021-12-31 08:59:02 +01:00
2021-09-07 04:55:53 +02:00
[HttpGet("{storeId}/delete")]
2018-07-19 15:23:14 +02:00
public IActionResult DeleteStore ( string storeId )
{
2021-09-07 04:55:53 +02:00
return View ( "Confirm" , new ConfirmModel ( "Delete store" , "The store will be permanently deleted. This action will also delete all invoices, apps and data associated with the store. Are you sure?" , "Delete" ) ) ;
2018-07-19 15:23:14 +02:00
}
2021-09-07 04:55:53 +02:00
[HttpPost("{storeId}/delete")]
2018-07-19 15:23:14 +02:00
public async Task < IActionResult > DeleteStorePost ( string storeId )
{
2019-10-12 13:35:30 +02:00
await _Repo . DeleteStore ( CurrentStore . Id ) ;
2021-09-07 04:55:53 +02:00
TempData [ WellKnownTempData . SuccessMessage ] = "Store successfully deleted." ;
2022-01-07 04:32:00 +01:00
return RedirectToAction ( nameof ( UIHomeController . Index ) , "UIHome" ) ;
2017-10-27 10:53:04 +02:00
}
2020-01-13 14:20:45 +01:00
private IEnumerable < AvailableRateProvider > GetSupportedExchanges ( )
2018-04-18 09:38:17 +02:00
{
2020-01-13 14:20:45 +01:00
var exchanges = _RateFactory . RateProviderFactory . GetSupportedExchanges ( ) ;
2020-01-10 14:50:39 +01:00
return exchanges
. Where ( r = > ! string . IsNullOrWhiteSpace ( r . Name ) )
. OrderBy ( s = > s . Id , StringComparer . OrdinalIgnoreCase ) ;
2018-04-18 09:38:17 +02:00
}
2021-06-17 07:11:01 +02:00
private DerivationSchemeSettings ParseDerivationStrategy ( string derivationScheme , BTCPayNetwork network )
2017-10-27 10:53:04 +02:00
{
2019-05-09 09:05:18 +02:00
var parser = new DerivationSchemeParser ( network ) ;
2021-01-11 03:22:42 +01:00
try
{
var derivationSchemeSettings = new DerivationSchemeSettings ( ) ;
derivationSchemeSettings . Network = network ;
var result = parser . ParseOutputDescriptor ( derivationScheme ) ;
derivationSchemeSettings . AccountOriginal = derivationScheme . Trim ( ) ;
derivationSchemeSettings . AccountDerivation = result . Item1 ;
derivationSchemeSettings . AccountKeySettings = result . Item2 ? . Select ( ( path , i ) = > new AccountKeySettings ( )
{
RootFingerprint = path ? . MasterFingerprint ,
AccountKeyPath = path ? . KeyPath ,
AccountKey = result . Item1 . GetExtPubKeys ( ) . ElementAt ( i ) . GetWif ( parser . Network )
} ) . ToArray ( ) ? ? new AccountKeySettings [ result . Item1 . GetExtPubKeys ( ) . Count ( ) ] ;
return derivationSchemeSettings ;
}
catch ( Exception )
{
// ignored
}
2021-09-03 08:37:12 +02:00
2019-05-08 16:39:11 +02:00
return new DerivationSchemeSettings ( parser . Parse ( derivationScheme ) , network ) ;
2017-10-27 10:53:04 +02:00
}
[HttpGet]
[Route("{storeId}/Tokens")]
2018-04-30 15:00:43 +02:00
public async Task < IActionResult > ListTokens ( )
2017-10-27 10:53:04 +02:00
{
var model = new TokensViewModel ( ) ;
2019-10-12 13:35:30 +02:00
var tokens = await _TokenRepository . GetTokensByStoreIdAsync ( CurrentStore . Id ) ;
2019-01-18 11:15:31 +01:00
model . StoreNotConfigured = StoreNotConfigured ;
2017-10-27 10:53:04 +02:00
model . Tokens = tokens . Select ( t = > new TokenViewModel ( )
{
Label = t . Label ,
SIN = t . SIN ,
Id = t . Value
} ) . ToArray ( ) ;
2018-04-29 11:28:04 +02:00
2019-10-12 13:35:30 +02:00
model . ApiKey = ( await _TokenRepository . GetLegacyAPIKeys ( CurrentStore . Id ) ) . FirstOrDefault ( ) ;
2018-04-29 11:28:04 +02:00
if ( model . ApiKey = = null )
model . EncodedApiKey = "*API Key*" ;
else
model . EncodedApiKey = Encoders . Base64 . EncodeData ( Encoders . ASCII . DecodeData ( model . ApiKey ) ) ;
2017-10-27 10:53:04 +02:00
return View ( model ) ;
}
2021-12-31 08:59:02 +01:00
2021-09-07 04:55:53 +02:00
[HttpGet("{storeId}/tokens/{tokenId}/revoke")]
2018-10-31 09:59:09 +01:00
public async Task < IActionResult > RevokeToken ( string tokenId )
{
var token = await _TokenRepository . GetToken ( tokenId ) ;
2019-10-12 13:35:30 +02:00
if ( token = = null | | token . StoreId ! = CurrentStore . Id )
2018-10-31 09:59:09 +01:00
return NotFound ( ) ;
2021-09-07 04:55:53 +02:00
return View ( "Confirm" , new ConfirmModel ( "Revoke the token" , $"The access token with the label <strong>{token.Label}</strong> will be revoked. Do you wish to continue?" , "Revoke" ) ) ;
2018-10-31 09:59:09 +01:00
}
2021-12-31 08:59:02 +01:00
2021-09-07 04:55:53 +02:00
[HttpPost("{storeId}/tokens/{tokenId}/revoke")]
2018-10-31 09:59:09 +01:00
public async Task < IActionResult > RevokeTokenConfirm ( string tokenId )
{
var token = await _TokenRepository . GetToken ( tokenId ) ;
if ( token = = null | |
2019-10-12 13:35:30 +02:00
token . StoreId ! = CurrentStore . Id | |
2018-10-31 09:59:09 +01:00
! await _TokenRepository . DeleteToken ( tokenId ) )
2021-09-07 04:55:53 +02:00
TempData [ WellKnownTempData . ErrorMessage ] = "Failure to revoke this token." ;
2018-10-31 09:59:09 +01:00
else
2019-10-31 04:29:59 +01:00
TempData [ WellKnownTempData . SuccessMessage ] = "Token revoked" ;
2022-04-11 10:48:12 +02:00
return RedirectToAction ( nameof ( ListTokens ) , new { storeId = token ? . StoreId } ) ;
2018-10-31 09:59:09 +01:00
}
[HttpGet]
[Route("{storeId}/tokens/{tokenId}")]
public async Task < IActionResult > ShowToken ( string tokenId )
{
var token = await _TokenRepository . GetToken ( tokenId ) ;
2019-10-12 13:35:30 +02:00
if ( token = = null | | token . StoreId ! = CurrentStore . Id )
2018-10-31 09:59:09 +01:00
return NotFound ( ) ;
return View ( token ) ;
}
2017-10-27 10:53:04 +02:00
[HttpPost]
[Route("{storeId}/Tokens/Create")]
2019-11-03 08:16:52 +01:00
public async Task < IActionResult > CreateToken ( string storeId , CreateTokenViewModel model )
2017-10-27 10:53:04 +02:00
{
if ( ! ModelState . IsValid )
{
2019-11-03 08:16:52 +01:00
return View ( nameof ( CreateToken ) , model ) ;
2017-10-27 10:53:04 +02:00
}
model . Label = model . Label ? ? String . Empty ;
2018-03-23 08:24:57 +01:00
var userId = GetUserId ( ) ;
if ( userId = = null )
2019-10-12 13:35:30 +02:00
return Challenge ( AuthenticationSchemes . Cookie ) ;
2019-11-03 08:16:52 +01:00
storeId = model . StoreId ;
var store = CurrentStore ? ? await _Repo . FindStore ( storeId , userId ) ;
if ( store = = null )
return Challenge ( AuthenticationSchemes . Cookie ) ;
2017-10-27 10:53:04 +02:00
var tokenRequest = new TokenRequest ( )
{
Label = model . Label ,
2022-02-08 03:24:58 +01:00
Id = model . PublicKey = = null ? null : NBitpayClient . Extensions . BitIdExtensions . GetBitIDSIN ( new PubKey ( model . PublicKey ) . Compress ( ) )
2017-10-27 10:53:04 +02:00
} ;
2022-04-11 10:48:12 +02:00
string? pairingCode = null ;
2017-10-27 10:53:04 +02:00
if ( model . PublicKey = = null )
{
tokenRequest . PairingCode = await _TokenRepository . CreatePairingCodeAsync ( ) ;
await _TokenRepository . UpdatePairingCode ( new PairingCodeEntity ( )
{
Id = tokenRequest . PairingCode ,
Label = model . Label ,
} ) ;
await _TokenRepository . PairWithStoreAsync ( tokenRequest . PairingCode , storeId ) ;
pairingCode = tokenRequest . PairingCode ;
}
else
{
2020-06-29 05:07:48 +02:00
pairingCode = ( await _TokenController . Tokens ( tokenRequest ) ) . Data [ 0 ] . PairingCode ;
2017-10-27 10:53:04 +02:00
}
2018-01-09 18:07:42 +01:00
GeneratedPairingCode = pairingCode ;
2017-10-27 10:53:04 +02:00
return RedirectToAction ( nameof ( RequestPairing ) , new
{
pairingCode = pairingCode ,
selectedStore = storeId
} ) ;
}
2022-04-11 10:48:12 +02:00
public string? GeneratedPairingCode { get ; set ; }
2022-01-11 05:15:48 +01:00
public WebhookSender WebhookNotificationManager { get ; }
2021-06-17 11:27:17 +02:00
public IDataProtector DataProtector { get ; }
2018-01-09 18:07:42 +01:00
2017-10-27 10:53:04 +02:00
[HttpGet]
[Route("{storeId}/Tokens/Create")]
2019-11-03 08:16:52 +01:00
public IActionResult CreateToken ( string storeId )
{
var model = new CreateTokenViewModel ( ) ;
ViewBag . HidePublicKey = storeId = = null ;
ViewBag . ShowStores = storeId = = null ;
ViewBag . ShowMenu = storeId ! = null ;
model . StoreId = storeId ;
return View ( model ) ;
}
[HttpGet]
[Route("/api-tokens")]
[AllowAnonymous]
2018-04-30 15:00:43 +02:00
public async Task < IActionResult > CreateToken ( )
2017-10-27 10:53:04 +02:00
{
var userId = GetUserId ( ) ;
if ( string . IsNullOrWhiteSpace ( userId ) )
2019-10-12 13:35:30 +02:00
return Challenge ( AuthenticationSchemes . Cookie ) ;
2017-10-27 10:53:04 +02:00
var model = new CreateTokenViewModel ( ) ;
2019-11-03 08:16:52 +01:00
ViewBag . HidePublicKey = true ;
ViewBag . ShowStores = true ;
ViewBag . ShowMenu = false ;
var stores = await _Repo . GetStoresByUserId ( userId ) ;
model . Stores = new SelectList ( stores . Where ( s = > s . Role = = StoreRoles . Owner ) , nameof ( CurrentStore . Id ) , nameof ( CurrentStore . StoreName ) ) ;
2020-01-12 07:32:26 +01:00
if ( ! model . Stores . Any ( ) )
2019-11-03 08:16:52 +01:00
{
TempData [ WellKnownTempData . ErrorMessage ] = "You need to be owner of at least one store before pairing" ;
2022-01-07 04:32:00 +01:00
return RedirectToAction ( nameof ( UIHomeController . Index ) , "UIHome" ) ;
2017-10-27 10:53:04 +02:00
}
return View ( model ) ;
}
2019-11-03 08:16:52 +01:00
[HttpPost]
[Route("/api-tokens")]
[AllowAnonymous]
public Task < IActionResult > CreateToken2 ( CreateTokenViewModel model )
{
return CreateToken ( model . StoreId , model ) ;
}
2018-04-29 11:28:04 +02:00
[HttpPost]
[Route("{storeId}/tokens/apikey")]
2020-06-28 10:55:27 +02:00
public async Task < IActionResult > GenerateAPIKey ( string storeId , string command = "" )
2018-04-29 11:28:04 +02:00
{
2018-04-29 19:33:42 +02:00
var store = HttpContext . GetStoreData ( ) ;
2018-04-29 11:28:04 +02:00
if ( store = = null )
return NotFound ( ) ;
2020-02-21 05:40:00 +01:00
if ( command = = "revoke" )
{
await _TokenRepository . RevokeLegacyAPIKeys ( CurrentStore . Id ) ;
TempData [ WellKnownTempData . SuccessMessage ] = "API Key revoked" ;
}
else
{
await _TokenRepository . GenerateLegacyAPIKey ( CurrentStore . Id ) ;
TempData [ WellKnownTempData . SuccessMessage ] = "API Key re-generated" ;
}
2020-06-28 10:55:27 +02:00
2020-01-14 14:06:46 +01:00
return RedirectToAction ( nameof ( ListTokens ) , new
{
storeId
} ) ;
2018-04-29 11:28:04 +02:00
}
2017-10-27 10:53:04 +02:00
[HttpGet]
[Route("/api-access-request")]
2018-04-30 15:00:43 +02:00
[AllowAnonymous]
2022-04-11 10:48:12 +02:00
public async Task < IActionResult > RequestPairing ( string pairingCode , string? selectedStore = null )
2017-10-27 10:53:04 +02:00
{
2018-04-30 15:00:43 +02:00
var userId = GetUserId ( ) ;
if ( userId = = null )
2019-10-12 13:35:30 +02:00
return Challenge ( AuthenticationSchemes . Cookie ) ;
2018-03-23 08:24:57 +01:00
if ( pairingCode = = null )
return NotFound ( ) ;
2021-04-08 15:32:42 +02:00
if ( selectedStore ! = null )
{
var store = await _Repo . FindStore ( selectedStore , userId ) ;
if ( store = = null )
return NotFound ( ) ;
HttpContext . SetStoreData ( store ) ;
ViewBag . ShowStores = false ;
}
2017-10-27 10:53:04 +02:00
var pairing = await _TokenRepository . GetPairingAsync ( pairingCode ) ;
if ( pairing = = null )
{
2019-10-31 04:29:59 +01:00
TempData [ WellKnownTempData . ErrorMessage ] = "Unknown pairing code" ;
2022-01-07 04:32:00 +01:00
return RedirectToAction ( nameof ( UIHomeController . Index ) , "UIHome" ) ;
2017-10-27 10:53:04 +02:00
}
else
{
2018-04-30 15:00:43 +02:00
var stores = await _Repo . GetStoresByUserId ( userId ) ;
2021-04-08 15:32:42 +02:00
return View ( new PairingModel
2017-10-27 10:53:04 +02:00
{
Id = pairing . Id ,
Label = pairing . Label ,
SIN = pairing . SIN ? ? "Server-Initiated Pairing" ,
2019-10-12 13:35:30 +02:00
StoreId = selectedStore ? ? stores . FirstOrDefault ( ) ? . Id ,
Stores = stores . Where ( u = > u . Role = = StoreRoles . Owner ) . Select ( s = > new PairingModel . StoreViewModel ( )
2017-10-27 10:53:04 +02:00
{
Id = s . Id ,
Name = string . IsNullOrEmpty ( s . StoreName ) ? s . Id : s . StoreName
} ) . ToArray ( )
} ) ;
}
}
[HttpPost]
2018-03-23 08:24:57 +01:00
[Route("/api-access-request")]
2019-10-12 13:35:30 +02:00
public async Task < IActionResult > Pair ( string pairingCode , string storeId )
2017-10-27 10:53:04 +02:00
{
if ( pairingCode = = null )
return NotFound ( ) ;
2019-10-12 13:35:30 +02:00
var store = CurrentStore ;
2017-10-27 10:53:04 +02:00
var pairing = await _TokenRepository . GetPairingAsync ( pairingCode ) ;
if ( store = = null | | pairing = = null )
return NotFound ( ) ;
var pairingResult = await _TokenRepository . PairWithStoreAsync ( pairingCode , store . Id ) ;
if ( pairingResult = = PairingResult . Complete | | pairingResult = = PairingResult . Partial )
{
2019-01-18 11:15:31 +01:00
var excludeFilter = store . GetStoreBlob ( ) . GetExcludedPaymentMethods ( ) ;
2020-01-12 07:32:26 +01:00
StoreNotConfigured = ! store . GetSupportedPaymentMethods ( _NetworkProvider )
2019-01-18 11:15:31 +01:00
. Where ( p = > ! excludeFilter . Match ( p . PaymentId ) )
2020-01-12 07:32:26 +01:00
. Any ( ) ;
2019-10-31 04:29:59 +01:00
TempData [ WellKnownTempData . SuccessMessage ] = "Pairing is successful" ;
2017-10-27 10:53:04 +02:00
if ( pairingResult = = PairingResult . Partial )
2019-10-31 04:29:59 +01:00
TempData [ WellKnownTempData . SuccessMessage ] = "Server initiated pairing code: " + pairingCode ;
2017-10-27 10:53:04 +02:00
return RedirectToAction ( nameof ( ListTokens ) , new
{
2018-11-22 07:13:35 +01:00
storeId = store . Id ,
pairingCode = pairingCode
2017-10-27 10:53:04 +02:00
} ) ;
}
else
{
2019-10-31 04:29:59 +01:00
TempData [ WellKnownTempData . ErrorMessage ] = $"Pairing failed ({pairingResult})" ;
2017-10-27 10:53:04 +02:00
return RedirectToAction ( nameof ( ListTokens ) , new
{
storeId = store . Id
} ) ;
}
}
2022-04-11 10:48:12 +02:00
private string? GetUserId ( )
2017-10-27 10:53:04 +02:00
{
2022-04-11 10:48:12 +02:00
if ( User . Identity ? . AuthenticationType ! = AuthenticationSchemes . Cookie )
2018-04-30 15:00:43 +02:00
return null ;
2017-10-27 10:53:04 +02:00
return _UserManager . GetUserId ( User ) ;
}
}
2017-09-13 16:50:36 +02:00
}