btcpayserver/BTCPayServer/Controllers/StoresController.Integrations.cs

178 lines
6.9 KiB
C#
Raw Normal View History

2020-09-18 17:20:31 +02:00
using System;
using System.Globalization;
using System.IO;
using System.Linq;
2020-09-18 17:20:31 +02:00
using System.Net.Http;
using System.Text;
using System.Threading.Tasks;
using BTCPayServer.Client.Models;
2020-09-18 17:20:31 +02:00
using BTCPayServer.Data;
using BTCPayServer.Models.StoreViewModels;
using BTCPayServer.Services.Invoices;
2020-09-18 17:20:31 +02:00
using BTCPayServer.Services.Shopify;
using BTCPayServer.Services.Shopify.Models;
using Microsoft.AspNetCore.Authorization;
using Microsoft.AspNetCore.Mvc;
using Nethereum.Util;
2020-09-18 17:20:31 +02:00
namespace BTCPayServer.Controllers
{
public partial class StoresController
{
private static string _cachedShopifyJavascript;
2020-09-19 12:13:55 +02:00
private async Task<string> GetJavascript()
{
if (!string.IsNullOrEmpty(_cachedShopifyJavascript))
2020-09-18 17:20:31 +02:00
{
return _cachedShopifyJavascript;
2020-09-18 17:20:31 +02:00
}
2020-09-19 12:13:55 +02:00
string[] fileList = _BtcpayServerOptions.BundleJsCss
? new[] { "bundles/shopify-bundle.min.js"}
: new[] {"modal/btcpay.js", "shopify/btcpay-browser-client.js", "shopify/btcpay-shopify-checkout.js"};
2020-09-18 17:20:31 +02:00
foreach (var file in fileList)
{
await using var stream = _webHostEnvironment.WebRootFileProvider
.GetFileInfo(file).CreateReadStream();
using var reader = new StreamReader(stream);
_cachedShopifyJavascript += Environment.NewLine + await reader.ReadToEndAsync();
2020-09-18 17:20:31 +02:00
}
return _cachedShopifyJavascript;
2020-09-19 12:13:55 +02:00
}
[AllowAnonymous]
[HttpGet("{storeId}/integrations/shopify/shopify.js")]
public async Task<IActionResult> ShopifyJavascript(string storeId)
{
var jsFile = $"var BTCPAYSERVER_URL = \"{Request.GetAbsoluteRoot()}\"; var STORE_ID = \"{storeId}\"; { await GetJavascript()}";
2020-09-18 17:20:31 +02:00
return Content(jsFile, "text/javascript");
}
[HttpGet]
[Route("{storeId}/integrations")]
[Route("{storeId}/integrations/shopify")]
public IActionResult Integrations()
{
var blob = CurrentStore.GetStoreBlob();
var vm = new IntegrationsViewModel {Shopify = blob.Shopify};
return View("Integrations", vm);
}
[HttpPost]
[Route("{storeId}/integrations/shopify")]
public async Task<IActionResult> Integrations([FromServices] IHttpClientFactory clientFactory,
IntegrationsViewModel vm, string command = "", string exampleUrl = "")
{
if (!string.IsNullOrEmpty(exampleUrl))
{
try
{
//https://{apikey}:{password}@{hostname}/admin/api/{version}/{resource}.json
2020-09-18 17:20:31 +02:00
var parsedUrl = new Uri(exampleUrl);
var userInfo = parsedUrl.UserInfo.Split(":");
vm.Shopify = new ShopifySettings()
{
ApiKey = userInfo[0],
Password = userInfo[1],
ShopName = parsedUrl.Host.Replace(".myshopify.com", "", StringComparison.InvariantCultureIgnoreCase)
};
command = "ShopifySaveCredentials";
}
catch (Exception)
{
TempData[WellKnownTempData.ErrorMessage] = "The provided example url was invalid.";
return View("Integrations", vm);
}
}
switch (command)
{
case "ShopifySaveCredentials":
{
var shopify = vm.Shopify;
var validCreds = shopify != null && shopify?.CredentialsPopulated() == true;
if (!validCreds)
{
TempData[WellKnownTempData.ErrorMessage] = "Please provide valid Shopify credentials";
return View("Integrations", vm);
}
var apiClient = new ShopifyApiClient(clientFactory, shopify.CreateShopifyApiCredentials());
try
{
await apiClient.OrdersCount();
}
catch
{
TempData[WellKnownTempData.ErrorMessage] =
"Shopify rejected provided credentials, please correct values and try again";
2020-09-18 17:20:31 +02:00
return View("Integrations", vm);
}
var scopesGranted = await apiClient.CheckScopes();
if (!scopesGranted.Contains("read_orders") || !scopesGranted.Contains("write_script_tags"))
2020-09-18 17:20:31 +02:00
{
TempData[WellKnownTempData.ErrorMessage] =
"Please grant the private app permissions for read_orders, write_script_tags";
return View("Integrations", vm);
2020-09-18 17:20:31 +02:00
}
var result = await apiClient.CreateScript(Url.Action("ShopifyJavascript", "Stores",
new {storeId = CurrentStore.Id}, Request.Scheme));
shopify.ScriptId = result.ScriptTag?.Id.ToString(CultureInfo.InvariantCulture);
shopify.IntegratedAt = DateTimeOffset.Now;
2020-09-18 17:20:31 +02:00
var blob = CurrentStore.GetStoreBlob();
blob.Shopify = shopify;
2020-09-18 17:20:31 +02:00
if (CurrentStore.SetStoreBlob(blob))
{
await _Repo.UpdateStore(CurrentStore);
}
TempData[WellKnownTempData.SuccessMessage] = "Shopify integration successfully updated";
2020-09-18 17:20:31 +02:00
break;
}
case "ShopifyClearCredentials":
{
var blob = CurrentStore.GetStoreBlob();
if (blob.Shopify.IntegratedAt.HasValue)
{
if (!string.IsNullOrEmpty(blob.Shopify.ScriptId))
{
try
{
var apiClient = new ShopifyApiClient(clientFactory,
blob.Shopify.CreateShopifyApiCredentials());
await apiClient.RemoveScript(blob.Shopify.ScriptId);
}
catch (Exception e)
{
//couldnt remove the script but that's ok
}
}
}
blob.Shopify = null;
if (CurrentStore.SetStoreBlob(blob))
{
await _Repo.UpdateStore(CurrentStore);
}
TempData[WellKnownTempData.SuccessMessage] = "Shopify integration credentials cleared";
break;
}
}
return RedirectToAction(nameof(Integrations), new {storeId = CurrentStore.Id});
}
}
}