Fix #2439 - ensure resource urls are constructed consistently on client and server
This commit is contained in:
parent
72cc44641b
commit
b7a3713946
|
@ -75,7 +75,7 @@ namespace Oqtane.Modules
|
||||||
var scripts = new List<object>();
|
var scripts = new List<object>();
|
||||||
foreach (Resource resource in Resources.Where(item => item.ResourceType == ResourceType.Script))
|
foreach (Resource resource in Resources.Where(item => item.ResourceType == ResourceType.Script))
|
||||||
{
|
{
|
||||||
var url = (resource.Url.Contains("://")) ? resource.Url : PageState.Alias.BaseUrl + (!resource.Url.StartsWith("/") ? "/" : "") + resource.Url;
|
var url = (resource.Url.Contains("://")) ? resource.Url : PageState.Alias.BaseUrl + resource.Url;
|
||||||
scripts.Add(new { href = url, bundle = resource.Bundle ?? "", integrity = resource.Integrity ?? "", crossorigin = resource.CrossOrigin ?? "", es6module = resource.ES6Module });
|
scripts.Add(new { href = url, bundle = resource.Bundle ?? "", integrity = resource.Integrity ?? "", crossorigin = resource.CrossOrigin ?? "", es6module = resource.ES6Module });
|
||||||
}
|
}
|
||||||
if (scripts.Any())
|
if (scripts.Any())
|
||||||
|
|
|
@ -35,7 +35,8 @@ namespace Oqtane.Themes
|
||||||
var scripts = new List<object>();
|
var scripts = new List<object>();
|
||||||
foreach (Resource resource in Resources.Where(item => item.ResourceType == ResourceType.Script))
|
foreach (Resource resource in Resources.Where(item => item.ResourceType == ResourceType.Script))
|
||||||
{
|
{
|
||||||
scripts.Add(new { href = resource.Url, bundle = resource.Bundle ?? "", integrity = resource.Integrity ?? "", crossorigin = resource.CrossOrigin ?? "", es6module = resource.ES6Module });
|
var url = (resource.Url.Contains("://")) ? resource.Url : PageState.Alias.BaseUrl + resource.Url;
|
||||||
|
scripts.Add(new { href = url, bundle = resource.Bundle ?? "", integrity = resource.Integrity ?? "", crossorigin = resource.CrossOrigin ?? "", es6module = resource.ES6Module });
|
||||||
}
|
}
|
||||||
if (scripts.Any())
|
if (scripts.Any())
|
||||||
{
|
{
|
||||||
|
|
|
@ -36,7 +36,7 @@
|
||||||
foreach (Resource resource in PageState.Page.Resources.Where(item => item.ResourceType == ResourceType.Stylesheet))
|
foreach (Resource resource in PageState.Page.Resources.Where(item => item.ResourceType == ResourceType.Stylesheet))
|
||||||
{
|
{
|
||||||
var prefix = "app-stylesheet-" + resource.Level.ToString().ToLower();
|
var prefix = "app-stylesheet-" + resource.Level.ToString().ToLower();
|
||||||
var url = (resource.Url.Contains("://")) ? resource.Url : PageState.Alias.BaseUrl + (!resource.Url.StartsWith("/") ? "/" : "") + resource.Url;
|
var url = (resource.Url.Contains("://")) ? resource.Url : PageState.Alias.BaseUrl + resource.Url;
|
||||||
links.Add(new { id = prefix + "-" + batch + "-" + (links.Count + 1).ToString("00"), rel = "stylesheet", href = url, type = "text/css", integrity = resource.Integrity ?? "", crossorigin = resource.CrossOrigin ?? "", insertbefore = prefix });
|
links.Add(new { id = prefix + "-" + batch + "-" + (links.Count + 1).ToString("00"), rel = "stylesheet", href = url, type = "text/css", integrity = resource.Integrity ?? "", crossorigin = resource.CrossOrigin ?? "", insertbefore = prefix });
|
||||||
}
|
}
|
||||||
if (links.Any())
|
if (links.Any())
|
||||||
|
|
|
@ -20,6 +20,7 @@ using Oqtane.Enums;
|
||||||
using Oqtane.Security;
|
using Oqtane.Security;
|
||||||
using Oqtane.Extensions;
|
using Oqtane.Extensions;
|
||||||
using Oqtane.Themes;
|
using Oqtane.Themes;
|
||||||
|
using Oqtane.UI;
|
||||||
|
|
||||||
namespace Oqtane.Pages
|
namespace Oqtane.Pages
|
||||||
{
|
{
|
||||||
|
@ -179,7 +180,7 @@ namespace Oqtane.Pages
|
||||||
{
|
{
|
||||||
ThemeType = page.ThemeType;
|
ThemeType = page.ThemeType;
|
||||||
}
|
}
|
||||||
ProcessThemeResources(ThemeType);
|
ProcessThemeResources(ThemeType, alias);
|
||||||
}
|
}
|
||||||
else // page not found
|
else // page not found
|
||||||
{
|
{
|
||||||
|
@ -203,7 +204,7 @@ namespace Oqtane.Pages
|
||||||
var assemblies = AppDomain.CurrentDomain.GetOqtaneAssemblies();
|
var assemblies = AppDomain.CurrentDomain.GetOqtaneAssemblies();
|
||||||
foreach (Assembly assembly in assemblies)
|
foreach (Assembly assembly in assemblies)
|
||||||
{
|
{
|
||||||
ProcessHostResources(assembly);
|
ProcessHostResources(assembly, alias);
|
||||||
}
|
}
|
||||||
|
|
||||||
// set culture if not specified
|
// set culture if not specified
|
||||||
|
@ -436,7 +437,7 @@ namespace Oqtane.Pages
|
||||||
"</script>";
|
"</script>";
|
||||||
}
|
}
|
||||||
|
|
||||||
private void ProcessHostResources(Assembly assembly)
|
private void ProcessHostResources(Assembly assembly, Alias alias)
|
||||||
{
|
{
|
||||||
var types = assembly.GetTypes().Where(item => item.GetInterfaces().Contains(typeof(IHostResources)));
|
var types = assembly.GetTypes().Where(item => item.GetInterfaces().Contains(typeof(IHostResources)));
|
||||||
foreach (var type in types)
|
foreach (var type in types)
|
||||||
|
@ -445,12 +446,12 @@ namespace Oqtane.Pages
|
||||||
foreach (var resource in obj.Resources)
|
foreach (var resource in obj.Resources)
|
||||||
{
|
{
|
||||||
resource.Level = ResourceLevel.App;
|
resource.Level = ResourceLevel.App;
|
||||||
ProcessResource(resource, 0);
|
ProcessResource(resource, 0, alias);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private void ProcessThemeResources(string ThemeType)
|
private void ProcessThemeResources(string ThemeType, Alias alias)
|
||||||
{
|
{
|
||||||
var type = Type.GetType(ThemeType);
|
var type = Type.GetType(ThemeType);
|
||||||
if (type != null)
|
if (type != null)
|
||||||
|
@ -462,40 +463,41 @@ namespace Oqtane.Pages
|
||||||
foreach (var resource in obj.Resources.Where(item => item.ResourceType == ResourceType.Stylesheet))
|
foreach (var resource in obj.Resources.Where(item => item.ResourceType == ResourceType.Stylesheet))
|
||||||
{
|
{
|
||||||
resource.Level = ResourceLevel.Page;
|
resource.Level = ResourceLevel.Page;
|
||||||
ProcessResource(resource, count++);
|
ProcessResource(resource, count++, alias);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private void ProcessResource(Resource resource, int count)
|
private void ProcessResource(Resource resource, int count, Alias alias)
|
||||||
{
|
{
|
||||||
|
var url = (resource.Url.Contains("://")) ? resource.Url : alias.BaseUrl + resource.Url;
|
||||||
switch (resource.ResourceType)
|
switch (resource.ResourceType)
|
||||||
{
|
{
|
||||||
case ResourceType.Stylesheet:
|
case ResourceType.Stylesheet:
|
||||||
if (!HeadResources.Contains(resource.Url, StringComparison.OrdinalIgnoreCase))
|
if (!HeadResources.Contains(url, StringComparison.OrdinalIgnoreCase))
|
||||||
{
|
{
|
||||||
string id = "";
|
string id = "";
|
||||||
if (resource.Level == ResourceLevel.Page)
|
if (resource.Level == ResourceLevel.Page)
|
||||||
{
|
{
|
||||||
id = "id=\"app-stylesheet-" + resource.Level.ToString().ToLower() + "-" + DateTime.UtcNow.ToString("yyyyMMddHHmmssfff") + "-" + count.ToString("00") + "\" ";
|
id = "id=\"app-stylesheet-" + resource.Level.ToString().ToLower() + "-" + DateTime.UtcNow.ToString("yyyyMMddHHmmssfff") + "-" + count.ToString("00") + "\" ";
|
||||||
}
|
}
|
||||||
HeadResources += "<link " + id + "rel=\"stylesheet\" href=\"" + resource.Url + "\"" + CrossOrigin(resource.CrossOrigin) + Integrity(resource.Integrity) + " />" + Environment.NewLine;
|
HeadResources += "<link " + id + "rel=\"stylesheet\" href=\"" + url + "\"" + CrossOrigin(resource.CrossOrigin) + Integrity(resource.Integrity) + " type=\"text/css\"/>" + Environment.NewLine;
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
case ResourceType.Script:
|
case ResourceType.Script:
|
||||||
if (resource.Location == Shared.ResourceLocation.Body)
|
if (resource.Location == Shared.ResourceLocation.Body)
|
||||||
{
|
{
|
||||||
if (!BodyResources.Contains(resource.Url, StringComparison.OrdinalIgnoreCase))
|
if (!BodyResources.Contains(url, StringComparison.OrdinalIgnoreCase))
|
||||||
{
|
{
|
||||||
BodyResources += "<script src=\"" + resource.Url + "\"" + CrossOrigin(resource.CrossOrigin) + Integrity(resource.Integrity) + "></script>" + Environment.NewLine;
|
BodyResources += "<script src=\"" + url + "\"" + CrossOrigin(resource.CrossOrigin) + Integrity(resource.Integrity) + "></script>" + Environment.NewLine;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
if (!HeadResources.Contains(resource.Url, StringComparison.OrdinalIgnoreCase))
|
if (!HeadResources.Contains(resource.Url, StringComparison.OrdinalIgnoreCase))
|
||||||
{
|
{
|
||||||
HeadResources += "<script src=\"" + resource.Url + "\"" + CrossOrigin(resource.CrossOrigin) + Integrity(resource.Integrity) + "></script>" + Environment.NewLine;
|
HeadResources += "<script src=\"" + url + "\"" + CrossOrigin(resource.CrossOrigin) + Integrity(resource.Integrity) + "></script>" + Environment.NewLine;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
|
|
|
@ -8,6 +8,8 @@ namespace Oqtane.Models
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public class Resource
|
public class Resource
|
||||||
{
|
{
|
||||||
|
private string _url;
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// A <see cref="ResourceType"/> so the Interop can properly create `script` or `link` tags
|
/// A <see cref="ResourceType"/> so the Interop can properly create `script` or `link` tags
|
||||||
/// </summary>
|
/// </summary>
|
||||||
|
@ -16,7 +18,14 @@ namespace Oqtane.Models
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Path to the resources.
|
/// Path to the resources.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public string Url { get; set; }
|
public string Url
|
||||||
|
{
|
||||||
|
get => _url;
|
||||||
|
set
|
||||||
|
{
|
||||||
|
_url = (value.Contains("://")) ? value : (!value.StartsWith("/") ? "/" : "") + value;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Integrity checks to increase the security of resources accessed. Especially common in CDN resources.
|
/// Integrity checks to increase the security of resources accessed. Especially common in CDN resources.
|
||||||
|
|
Loading…
Reference in New Issue
Block a user