233 lines
9.0 KiB
C#
233 lines
9.0 KiB
C#
using System.Collections.Generic;
|
|
using Umbraco.Core;
|
|
using Umbraco.Core.Configuration;
|
|
using Umbraco.Web.Models.ContentEditing;
|
|
using Newtonsoft.Json.Linq;
|
|
using System.Threading.Tasks;
|
|
using System.Net.Http;
|
|
using System;
|
|
using System.Linq;
|
|
using System.Text;
|
|
using Microsoft.AspNetCore.Mvc;
|
|
using Microsoft.Extensions.Logging;
|
|
using Umbraco.Core.Cache;
|
|
using Umbraco.Core.Persistence;
|
|
using Umbraco.Core.Services;
|
|
using Umbraco.Core.Strings;
|
|
using Umbraco.Core.Dashboards;
|
|
using Umbraco.Core.Security;
|
|
using Umbraco.Web.Services;
|
|
using Umbraco.Web.BackOffice.Filters;
|
|
using Umbraco.Web.Common.Attributes;
|
|
using Umbraco.Web.Common.Controllers;
|
|
using Umbraco.Web.Common.Filters;
|
|
using Microsoft.AspNetCore.Authorization;
|
|
using Umbraco.Web.Common.Authorization;
|
|
|
|
namespace Umbraco.Web.BackOffice.Controllers
|
|
{
|
|
//we need to fire up the controller like this to enable loading of remote css directly from this controller
|
|
[PluginController(Constants.Web.Mvc.BackOfficeApiArea)]
|
|
[ValidationFilter]
|
|
[AngularJsonOnlyConfiguration] // TODO: This could be applied with our Application Model conventions
|
|
[IsBackOffice]
|
|
[Authorize(Policy = AuthorizationPolicies.BackOfficeAccess)]
|
|
public class DashboardController : UmbracoApiController
|
|
{
|
|
private readonly IBackOfficeSecurityAccessor _backOfficeSecurityAccessor;
|
|
private readonly AppCaches _appCaches;
|
|
private readonly ILogger<DashboardController> _logger;
|
|
private readonly IDashboardService _dashboardService;
|
|
private readonly IUmbracoVersion _umbracoVersion;
|
|
private readonly IShortStringHelper _shortStringHelper;
|
|
|
|
/// <summary>
|
|
/// Initializes a new instance of the <see cref="DashboardController"/> with all its dependencies.
|
|
/// </summary>
|
|
public DashboardController(
|
|
IBackOfficeSecurityAccessor backOfficeSecurityAccessor,
|
|
AppCaches appCaches,
|
|
ILogger<DashboardController> logger,
|
|
IDashboardService dashboardService,
|
|
IUmbracoVersion umbracoVersion,
|
|
IShortStringHelper shortStringHelper)
|
|
|
|
{
|
|
_backOfficeSecurityAccessor = backOfficeSecurityAccessor;
|
|
_appCaches = appCaches;
|
|
_logger = logger;
|
|
_dashboardService = dashboardService;
|
|
_umbracoVersion = umbracoVersion;
|
|
_shortStringHelper = shortStringHelper;
|
|
}
|
|
|
|
//we have just one instance of HttpClient shared for the entire application
|
|
private static readonly HttpClient HttpClient = new HttpClient();
|
|
|
|
//we have baseurl as a param to make previewing easier, so we can test with a dev domain from client side
|
|
[ValidateAngularAntiForgeryToken]
|
|
public async Task<JObject> GetRemoteDashboardContent(string section, string baseUrl = "https://dashboard.umbraco.org/")
|
|
{
|
|
var user = _backOfficeSecurityAccessor.BackOfficeSecurity.CurrentUser;
|
|
var allowedSections = string.Join(",", user.AllowedSections);
|
|
var language = user.Language;
|
|
var version = _umbracoVersion.SemanticVersion.ToSemanticString();
|
|
|
|
var url = string.Format(baseUrl + "{0}?section={0}&allowed={1}&lang={2}&version={3}", section, allowedSections, language, version);
|
|
var key = "umbraco-dynamic-dashboard-" + language + allowedSections.Replace(",", "-") + section;
|
|
|
|
var content = _appCaches.RuntimeCache.GetCacheItem<JObject>(key);
|
|
var result = new JObject();
|
|
if (content != null)
|
|
{
|
|
result = content;
|
|
}
|
|
else
|
|
{
|
|
//content is null, go get it
|
|
try
|
|
{
|
|
//fetch dashboard json and parse to JObject
|
|
var json = await HttpClient.GetStringAsync(url);
|
|
content = JObject.Parse(json);
|
|
result = content;
|
|
|
|
_appCaches.RuntimeCache.InsertCacheItem<JObject>(key, () => result, new TimeSpan(0, 30, 0));
|
|
}
|
|
catch (HttpRequestException ex)
|
|
{
|
|
_logger.LogError(ex.InnerException ?? ex, "Error getting dashboard content from {Url}", url);
|
|
|
|
//it's still new JObject() - we return it like this to avoid error codes which triggers UI warnings
|
|
_appCaches.RuntimeCache.InsertCacheItem<JObject>(key, () => result, new TimeSpan(0, 5, 0));
|
|
}
|
|
}
|
|
|
|
return result;
|
|
}
|
|
|
|
public async Task<IActionResult> GetRemoteDashboardCss(string section, string baseUrl = "https://dashboard.umbraco.org/")
|
|
{
|
|
var url = string.Format(baseUrl + "css/dashboard.css?section={0}", section);
|
|
var key = "umbraco-dynamic-dashboard-css-" + section;
|
|
|
|
var content = _appCaches.RuntimeCache.GetCacheItem<string>(key);
|
|
var result = string.Empty;
|
|
|
|
if (content != null)
|
|
{
|
|
result = content;
|
|
}
|
|
else
|
|
{
|
|
//content is null, go get it
|
|
try
|
|
{
|
|
//fetch remote css
|
|
content = await HttpClient.GetStringAsync(url);
|
|
|
|
//can't use content directly, modified closure problem
|
|
result = content;
|
|
|
|
//save server content for 30 mins
|
|
_appCaches.RuntimeCache.InsertCacheItem<string>(key, () => result, new TimeSpan(0, 30, 0));
|
|
}
|
|
catch (HttpRequestException ex)
|
|
{
|
|
_logger.LogError(ex.InnerException ?? ex, "Error getting dashboard CSS from {Url}", url);
|
|
|
|
//it's still string.Empty - we return it like this to avoid error codes which triggers UI warnings
|
|
_appCaches.RuntimeCache.InsertCacheItem<string>(key, () => result, new TimeSpan(0, 5, 0));
|
|
}
|
|
}
|
|
|
|
|
|
return Content(result,"text/css", Encoding.UTF8);
|
|
|
|
}
|
|
|
|
public async Task<IActionResult> GetRemoteXml(string site, string url)
|
|
{
|
|
// This is used in place of the old feedproxy.config
|
|
// Which was used to grab data from our.umbraco.com, umbraco.com or umbraco.tv
|
|
// for certain dashboards or the help drawer
|
|
var urlPrefix = string.Empty;
|
|
switch (site.ToUpper())
|
|
{
|
|
case "TV":
|
|
urlPrefix = "https://umbraco.tv/";
|
|
break;
|
|
|
|
case "OUR":
|
|
urlPrefix = "https://our.umbraco.com/";
|
|
break;
|
|
|
|
case "COM":
|
|
urlPrefix = "https://umbraco.com/";
|
|
break;
|
|
|
|
default:
|
|
return NotFound();
|
|
}
|
|
|
|
|
|
//Make remote call to fetch videos or remote dashboard feed data
|
|
var key = $"umbraco-XML-feed-{site}-{url.ToCleanString(_shortStringHelper, CleanStringType.UrlSegment)}";
|
|
|
|
var content = _appCaches.RuntimeCache.GetCacheItem<string>(key);
|
|
var result = string.Empty;
|
|
|
|
if (content != null)
|
|
{
|
|
result = content;
|
|
}
|
|
else
|
|
{
|
|
//content is null, go get it
|
|
try
|
|
{
|
|
//fetch remote css
|
|
content = await HttpClient.GetStringAsync($"{urlPrefix}{url}");
|
|
|
|
//can't use content directly, modified closure problem
|
|
result = content;
|
|
|
|
//save server content for 30 mins
|
|
_appCaches.RuntimeCache.InsertCacheItem<string>(key, () => result, new TimeSpan(0, 30, 0));
|
|
}
|
|
catch (HttpRequestException ex)
|
|
{
|
|
_logger.LogError(ex.InnerException ?? ex, "Error getting remote dashboard data from {UrlPrefix}{Url}", urlPrefix, url);
|
|
|
|
//it's still string.Empty - we return it like this to avoid error codes which triggers UI warnings
|
|
_appCaches.RuntimeCache.InsertCacheItem<string>(key, () => result, new TimeSpan(0, 5, 0));
|
|
}
|
|
}
|
|
|
|
return Content(result,"text/xml", Encoding.UTF8);
|
|
|
|
}
|
|
|
|
// return IDashboardSlim - we don't need sections nor access rules
|
|
[ValidateAngularAntiForgeryToken]
|
|
[OutgoingEditorModelEvent]
|
|
public IEnumerable<Tab<IDashboardSlim>> GetDashboard(string section)
|
|
{
|
|
var currentUser = _backOfficeSecurityAccessor.BackOfficeSecurity.CurrentUser;
|
|
return _dashboardService.GetDashboards(section, currentUser).Select(x => new Tab<IDashboardSlim>
|
|
{
|
|
Id = x.Id,
|
|
Alias = x.Alias,
|
|
Label = x.Label,
|
|
Expanded = x.Expanded,
|
|
IsActive = x.IsActive,
|
|
Properties = x.Properties.Select(y => new DashboardSlim
|
|
{
|
|
Alias = y.Alias,
|
|
View = y.View
|
|
})
|
|
}).ToList();
|
|
}
|
|
}
|
|
}
|