* Run code cleanup * Start manual cleanup after dotnet format * Finish up manual pass * Fix up missed warnings * Fix after merge * Update src/Umbraco.Core/Cache/ContentTypeCacheRefresher.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Core/Cache/DataTypeCacheRefresher.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Core/Cache/DeepCloneAppCache.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Core/Cache/DomainCacheRefresher.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Core/Cache/FastDictionaryAppCacheBase.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Core/Cache/FastDictionaryAppCacheBase.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Core/Cache/FastDictionaryAppCacheBase.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Core/Cache/FastDictionaryAppCacheBase.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Core/Cache/FastDictionaryAppCacheBase.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Core/Cache/MacroCacheRefresher.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Core/Cache/MediaCacheRefresher.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Core/Cache/MemberCacheRefresher.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Core/Cache/MemberGroupCacheRefresher.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Core/Composing/OrderedCollectionBuilderBase.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Core/Configuration/Models/RequestHandlerSettings.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Core/Composing/SetCollectionBuilderBase.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Core/Composing/WeightedCollectionBuilderBase.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Core/ContentApps/ContentAppFactoryCollectionBuilder.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Core/Dashboards/DashboardCollectionBuilder.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Core/Extensions/StringExtensions.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Core/Extensions/StringExtensions.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Core/Extensions/StringExtensions.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Core/Extensions/StringExtensions.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Core/Extensions/StringExtensions.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Core/Extensions/StringExtensions.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Core/Extensions/StringExtensions.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Core/Notifications/ContentTypeRefreshedNotification.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Core/Notifications/ContentTypeSavedNotification.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Core/Notifications/ContentTypeSavingNotification.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Core/Notifications/DictionaryCacheRefresherNotification.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Core/Notifications/DataTypeCacheRefresherNotification.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Core/Notifications/ContentUnpublishingNotification.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Core/Notifications/DictionaryItemDeletingNotification.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Core/Notifications/DictionaryItemSavedNotification.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Core/Notifications/DictionaryItemSavingNotification.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Core/Notifications/DomainCacheRefresherNotification.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Core/Notifications/LanguageCacheRefresherNotification.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Core/Notifications/MacroCacheRefresherNotification.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Core/Notifications/MediaCacheRefresherNotification.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Core/Notifications/MediaEmptiedRecycleBinNotification.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Core/Notifications/MediaEmptyingRecycleBinNotification.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Core/Notifications/MediaMovedNotification.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Core/Notifications/MediaMovedToRecycleBinNotification.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Core/Extensions/StringExtensions.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Core/Extensions/StringExtensions.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Core/Extensions/StringExtensions.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Core/Extensions/StringExtensions.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Core/Extensions/StringExtensions.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Core/Extensions/StringExtensions.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Core/Notifications/MediaMovedToRecycleBinNotification.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Core/Notifications/MediaMovingNotification.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Core/Notifications/MediaMovingToRecycleBinNotification.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Core/Notifications/MediaTreeChangeNotification.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Core/Notifications/MediaTreeChangeNotification.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Core/Notifications/MediaTypeChangedNotification.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Core/Notifications/MediaTypeDeletingNotification.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Core/Notifications/MediaTypeMovedNotification.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Core/Notifications/MediaTypeMovingNotification.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Core/Notifications/MediaTypeMovingNotification.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Core/Notifications/MediaTypeRefreshedNotification.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Core/Notifications/MemberCacheRefresherNotification.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Core/Notifications/MemberGroupCacheRefresherNotification.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Core/Notifications/MemberGroupDeletingNotification.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Core/Notifications/MemberGroupSavedNotification.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Core/Notifications/MemberGroupSavingNotification.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Core/Notifications/MemberTypeChangedNotification.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Core/Notifications/MemberTypeDeletedNotification.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Core/Notifications/MemberTypeDeletingNotification.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Core/Notifications/MemberTypeMovedNotification.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Core/Notifications/MemberTypeMovedNotification.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Core/Notifications/MemberTypeMovingNotification.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Core/Notifications/MemberTypeMovingNotification.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Core/Notifications/MemberTypeRefreshedNotification.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Core/Notifications/MemberTypeSavingNotification.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Core/Extensions/StringExtensions.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Core/Extensions/UriExtensions.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Core/Notifications/MovedToRecycleBinNotification.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Core/Notifications/MovedToRecycleBinNotification.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Core/Notifications/MovingToRecycleBinNotification.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Core/Notifications/MovingToRecycleBinNotification.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Core/Notifications/PartialViewDeletingNotification.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Core/Extensions/UriExtensions.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Core/IO/PhysicalFileSystem.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Core/IO/PhysicalFileSystem.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Core/IO/ShadowWrapper.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Core/Logging/DisposableTimer.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Core/Install/InstallSteps/UpgradeStep.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Core/Mapping/MapDefinitionCollectionBuilder.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Core/Media/EmbedProviders/EmbedProvidersCollectionBuilder.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Core/Notifications/PartialViewSavedNotification.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Core/Models/ContentEditing/ContentVariationDisplay.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Core/Notifications/PartialViewSavingNotification.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Core/Notifications/PublicAccessCacheRefresherNotification.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Core/Notifications/PublicAccessEntryDeletedNotification.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Core/Notifications/PublicAccessEntryDeletingNotification.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Core/Notifications/PublicAccessEntryDeletingNotification.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Core/Notifications/PublicAccessEntrySavedNotification.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Core/Notifications/PublicAccessEntrySavingNotification.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Core/Notifications/PublicAccessEntrySavingNotification.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Core/Notifications/RelationTypeCacheRefresherNotification.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Core/Notifications/RelationTypeDeletingNotification.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Core/Notifications/RelationTypeSavedNotification.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Core/Notifications/RelationTypeSavingNotification.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Core/Notifications/StatefulNotification.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Core/Notifications/StylesheetDeletingNotification.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Core/Notifications/StylesheetSavingNotification.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Core/Models/DeepCloneHelper.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Core/Models/File.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Core/Models/PropertyGroupExtensions.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Update src/Umbraco.Core/Models/PropertyGroupExtensions.cs Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> * Fix based on review * Fix after merge Signed-off-by: Zeegaan <nge@umbraco.dk> Co-authored-by: Nikolaj Geisle <niko737@edu.ucl.dk> Co-authored-by: Mole <nikolajlauridsen@protonmail.ch> Co-authored-by: Zeegaan <nge@umbraco.dk>
410 lines
14 KiB
C#
410 lines
14 KiB
C#
// Copyright (c) Umbraco.
|
|
// See LICENSE for more details.
|
|
|
|
using System.Text;
|
|
using System.Xml;
|
|
using System.Xml.Linq;
|
|
using System.Xml.XPath;
|
|
using Umbraco.Cms.Core;
|
|
using Umbraco.Cms.Core.Xml;
|
|
|
|
namespace Umbraco.Extensions;
|
|
|
|
/// <summary>
|
|
/// Extension methods for xml objects
|
|
/// </summary>
|
|
public static class XmlExtensions
|
|
{
|
|
public static bool HasAttribute(this XmlAttributeCollection attributes, string attributeName) =>
|
|
attributes.Cast<XmlAttribute>().Any(x => x.Name == attributeName);
|
|
|
|
/// <summary>
|
|
/// Selects a list of XmlNode matching an XPath expression.
|
|
/// </summary>
|
|
/// <param name="source">A source XmlNode.</param>
|
|
/// <param name="expression">An XPath expression.</param>
|
|
/// <param name="variables">A set of XPathVariables.</param>
|
|
/// <returns>The list of XmlNode matching the XPath expression.</returns>
|
|
/// <remarks>
|
|
/// <para>
|
|
/// If
|
|
/// <param name="variables" />
|
|
/// is <c>null</c>, or is empty, or contains only one single
|
|
/// value which itself is <c>null</c>, then variables are ignored.
|
|
/// </para>
|
|
/// <para>The XPath expression should reference variables as <c>$var</c>.</para>
|
|
/// </remarks>
|
|
public static XmlNodeList? SelectNodes(this XmlNode source, string expression, IEnumerable<XPathVariable>? variables)
|
|
{
|
|
XPathVariable[]? av = variables?.ToArray();
|
|
return SelectNodes(source, expression, av);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Selects a list of XmlNode matching an XPath expression.
|
|
/// </summary>
|
|
/// <param name="source">A source XmlNode.</param>
|
|
/// <param name="expression">An XPath expression.</param>
|
|
/// <param name="variables">A set of XPathVariables.</param>
|
|
/// <returns>The list of XmlNode matching the XPath expression.</returns>
|
|
/// <remarks>
|
|
/// <para>
|
|
/// If
|
|
/// <param name="variables" />
|
|
/// is <c>null</c>, or is empty, or contains only one single
|
|
/// value which itself is <c>null</c>, then variables are ignored.
|
|
/// </para>
|
|
/// <para>The XPath expression should reference variables as <c>$var</c>.</para>
|
|
/// </remarks>
|
|
public static XmlNodeList? SelectNodes(this XmlNode source, XPathExpression expression, IEnumerable<XPathVariable>? variables)
|
|
{
|
|
XPathVariable[]? av = variables?.ToArray();
|
|
return SelectNodes(source, expression, av);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Selects a list of XmlNode matching an XPath expression.
|
|
/// </summary>
|
|
/// <param name="source">A source XmlNode.</param>
|
|
/// <param name="expression">An XPath expression.</param>
|
|
/// <param name="variables">A set of XPathVariables.</param>
|
|
/// <returns>The list of XmlNode matching the XPath expression.</returns>
|
|
/// <remarks>
|
|
/// <para>
|
|
/// If
|
|
/// <param name="variables" />
|
|
/// is <c>null</c>, or is empty, or contains only one single
|
|
/// value which itself is <c>null</c>, then variables are ignored.
|
|
/// </para>
|
|
/// <para>The XPath expression should reference variables as <c>$var</c>.</para>
|
|
/// </remarks>
|
|
public static XmlNodeList? SelectNodes(this XmlNode source, string? expression, params XPathVariable[]? variables)
|
|
{
|
|
if (variables == null || variables.Length == 0 || variables[0] == null)
|
|
{
|
|
return source.SelectNodes(expression ?? string.Empty);
|
|
}
|
|
|
|
XPathNodeIterator? iterator = source.CreateNavigator()?.Select(expression ?? string.Empty, variables);
|
|
return XmlNodeListFactory.CreateNodeList(iterator);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Selects a list of XmlNode matching an XPath expression.
|
|
/// </summary>
|
|
/// <param name="source">A source XmlNode.</param>
|
|
/// <param name="expression">An XPath expression.</param>
|
|
/// <param name="variables">A set of XPathVariables.</param>
|
|
/// <returns>The list of XmlNode matching the XPath expression.</returns>
|
|
/// <remarks>
|
|
/// <para>
|
|
/// If
|
|
/// <param name="variables" />
|
|
/// is <c>null</c>, or is empty, or contains only one single
|
|
/// value which itself is <c>null</c>, then variables are ignored.
|
|
/// </para>
|
|
/// <para>The XPath expression should reference variables as <c>$var</c>.</para>
|
|
/// </remarks>
|
|
public static XmlNodeList SelectNodes(this XmlNode source, XPathExpression expression, params XPathVariable[]? variables)
|
|
{
|
|
if (variables == null || variables.Length == 0 || variables[0] == null)
|
|
{
|
|
return source.SelectNodes(expression);
|
|
}
|
|
|
|
XPathNodeIterator? iterator = source.CreateNavigator()?.Select(expression, variables);
|
|
return XmlNodeListFactory.CreateNodeList(iterator);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Selects the first XmlNode that matches an XPath expression.
|
|
/// </summary>
|
|
/// <param name="source">A source XmlNode.</param>
|
|
/// <param name="expression">An XPath expression.</param>
|
|
/// <param name="variables">A set of XPathVariables.</param>
|
|
/// <returns>The first XmlNode that matches the XPath expression.</returns>
|
|
/// <remarks>
|
|
/// <para>
|
|
/// If
|
|
/// <param name="variables" />
|
|
/// is <c>null</c>, or is empty, or contains only one single
|
|
/// value which itself is <c>null</c>, then variables are ignored.
|
|
/// </para>
|
|
/// <para>The XPath expression should reference variables as <c>$var</c>.</para>
|
|
/// </remarks>
|
|
public static XmlNode? SelectSingleNode(this XmlNode source, string expression, IEnumerable<XPathVariable>? variables)
|
|
{
|
|
XPathVariable[]? av = variables?.ToArray();
|
|
return SelectSingleNode(source, expression, av);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Selects the first XmlNode that matches an XPath expression.
|
|
/// </summary>
|
|
/// <param name="source">A source XmlNode.</param>
|
|
/// <param name="expression">An XPath expression.</param>
|
|
/// <param name="variables">A set of XPathVariables.</param>
|
|
/// <returns>The first XmlNode that matches the XPath expression.</returns>
|
|
/// <remarks>
|
|
/// <para>
|
|
/// If
|
|
/// <param name="variables" />
|
|
/// is <c>null</c>, or is empty, or contains only one single
|
|
/// value which itself is <c>null</c>, then variables are ignored.
|
|
/// </para>
|
|
/// <para>The XPath expression should reference variables as <c>$var</c>.</para>
|
|
/// </remarks>
|
|
public static XmlNode? SelectSingleNode(this XmlNode source, XPathExpression expression, IEnumerable<XPathVariable>? variables)
|
|
{
|
|
XPathVariable[]? av = variables?.ToArray();
|
|
return SelectSingleNode(source, expression, av);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Selects the first XmlNode that matches an XPath expression.
|
|
/// </summary>
|
|
/// <param name="source">A source XmlNode.</param>
|
|
/// <param name="expression">An XPath expression.</param>
|
|
/// <param name="variables">A set of XPathVariables.</param>
|
|
/// <returns>The first XmlNode that matches the XPath expression.</returns>
|
|
/// <remarks>
|
|
/// <para>
|
|
/// If
|
|
/// <param name="variables" />
|
|
/// is <c>null</c>, or is empty, or contains only one single
|
|
/// value which itself is <c>null</c>, then variables are ignored.
|
|
/// </para>
|
|
/// <para>The XPath expression should reference variables as <c>$var</c>.</para>
|
|
/// </remarks>
|
|
public static XmlNode? SelectSingleNode(this XmlNode source, string expression, params XPathVariable[]? variables)
|
|
{
|
|
if (variables == null || variables.Length == 0 || variables[0] == null)
|
|
{
|
|
return source.SelectSingleNode(expression);
|
|
}
|
|
|
|
return SelectNodes(source, expression, variables)?.Cast<XmlNode>().FirstOrDefault();
|
|
}
|
|
|
|
/// <summary>
|
|
/// Selects the first XmlNode that matches an XPath expression.
|
|
/// </summary>
|
|
/// <param name="source">A source XmlNode.</param>
|
|
/// <param name="expression">An XPath expression.</param>
|
|
/// <param name="variables">A set of XPathVariables.</param>
|
|
/// <returns>The first XmlNode that matches the XPath expression.</returns>
|
|
/// <remarks>
|
|
/// <para>
|
|
/// If
|
|
/// <param name="variables" />
|
|
/// is <c>null</c>, or is empty, or contains only one single
|
|
/// value which itself is <c>null</c>, then variables are ignored.
|
|
/// </para>
|
|
/// <para>The XPath expression should reference variables as <c>$var</c>.</para>
|
|
/// </remarks>
|
|
public static XmlNode? SelectSingleNode(this XmlNode source, XPathExpression expression, params XPathVariable[]? variables)
|
|
{
|
|
if (variables == null || variables.Length == 0 || variables[0] == null)
|
|
{
|
|
return source.SelectSingleNode(expression);
|
|
}
|
|
|
|
return SelectNodes(source, expression, variables).Cast<XmlNode>().FirstOrDefault();
|
|
}
|
|
|
|
/// <summary>
|
|
/// Converts from an XDocument to an XmlDocument
|
|
/// </summary>
|
|
/// <param name="xDocument"></param>
|
|
/// <returns></returns>
|
|
public static XmlDocument ToXmlDocument(this XDocument xDocument)
|
|
{
|
|
var xmlDocument = new XmlDocument();
|
|
using (XmlReader xmlReader = xDocument.CreateReader())
|
|
{
|
|
xmlDocument.Load(xmlReader);
|
|
}
|
|
|
|
return xmlDocument;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Converts from an XmlDocument to an XDocument
|
|
/// </summary>
|
|
/// <param name="xmlDocument"></param>
|
|
/// <returns></returns>
|
|
public static XDocument ToXDocument(this XmlDocument xmlDocument)
|
|
{
|
|
using (var nodeReader = new XmlNodeReader(xmlDocument))
|
|
{
|
|
nodeReader.MoveToContent();
|
|
return XDocument.Load(nodeReader);
|
|
}
|
|
}
|
|
|
|
///// <summary>
|
|
///// Converts from an XElement to an XmlElement
|
|
///// </summary>
|
|
///// <param name="xElement"></param>
|
|
///// <returns></returns>
|
|
public static XmlNode? ToXmlElement(this XContainer xElement)
|
|
{
|
|
var xmlDocument = new XmlDocument();
|
|
using (XmlReader xmlReader = xElement.CreateReader())
|
|
{
|
|
xmlDocument.Load(xmlReader);
|
|
}
|
|
|
|
return xmlDocument.DocumentElement;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Converts from an XmlElement to an XElement
|
|
/// </summary>
|
|
/// <param name="xmlElement"></param>
|
|
/// <returns></returns>
|
|
public static XElement ToXElement(this XmlNode xmlElement)
|
|
{
|
|
using (var nodeReader = new XmlNodeReader(xmlElement))
|
|
{
|
|
nodeReader.MoveToContent();
|
|
return XElement.Load(nodeReader);
|
|
}
|
|
}
|
|
|
|
public static T? RequiredAttributeValue<T>(this XElement xml, string attributeName)
|
|
{
|
|
if (xml == null)
|
|
{
|
|
throw new ArgumentNullException(nameof(xml));
|
|
}
|
|
|
|
if (xml.HasAttributes == false)
|
|
{
|
|
throw new InvalidOperationException($"{attributeName} not found in xml");
|
|
}
|
|
|
|
XAttribute? attribute = xml.Attribute(attributeName);
|
|
if (attribute is null)
|
|
{
|
|
throw new InvalidOperationException($"{attributeName} not found in xml");
|
|
}
|
|
|
|
Attempt<T> result = attribute.Value.TryConvertTo<T>();
|
|
if (result.Success)
|
|
{
|
|
return result.Result;
|
|
}
|
|
|
|
throw new InvalidOperationException($"{attribute.Value} attribute value cannot be converted to {typeof(T)}");
|
|
}
|
|
|
|
public static T? AttributeValue<T>(this XElement xml, string attributeName)
|
|
{
|
|
if (xml == null)
|
|
{
|
|
throw new ArgumentNullException("xml");
|
|
}
|
|
|
|
if (xml.HasAttributes == false)
|
|
{
|
|
return default;
|
|
}
|
|
|
|
if (xml.Attribute(attributeName) == null)
|
|
{
|
|
return default;
|
|
}
|
|
|
|
var val = xml.Attribute(attributeName)?.Value;
|
|
Attempt<T> result = val.TryConvertTo<T>();
|
|
if (result.Success)
|
|
{
|
|
return result.Result;
|
|
}
|
|
|
|
return default;
|
|
}
|
|
|
|
public static T? AttributeValue<T>(this XmlNode xml, string attributeName)
|
|
{
|
|
if (xml == null)
|
|
{
|
|
throw new ArgumentNullException("xml");
|
|
}
|
|
|
|
if (xml.Attributes == null)
|
|
{
|
|
return default;
|
|
}
|
|
|
|
if (xml.Attributes[attributeName] == null)
|
|
{
|
|
return default;
|
|
}
|
|
|
|
var val = xml.Attributes[attributeName]?.Value;
|
|
Attempt<T> result = val.TryConvertTo<T>();
|
|
if (result.Success)
|
|
{
|
|
return result.Result;
|
|
}
|
|
|
|
return default;
|
|
}
|
|
|
|
public static XElement? GetXElement(this XmlNode node)
|
|
{
|
|
var xDoc = new XDocument();
|
|
using (XmlWriter xmlWriter = xDoc.CreateWriter())
|
|
{
|
|
node.WriteTo(xmlWriter);
|
|
}
|
|
|
|
return xDoc.Root;
|
|
}
|
|
|
|
public static XmlNode? GetXmlNode(this XContainer element)
|
|
{
|
|
using (XmlReader xmlReader = element.CreateReader())
|
|
{
|
|
var xmlDoc = new XmlDocument();
|
|
xmlDoc.Load(xmlReader);
|
|
return xmlDoc.DocumentElement;
|
|
}
|
|
}
|
|
|
|
public static XmlNode? GetXmlNode(this XContainer element, XmlDocument xmlDoc)
|
|
{
|
|
XmlNode? node = element.GetXmlNode();
|
|
if (node is not null)
|
|
{
|
|
return xmlDoc.ImportNode(node, true);
|
|
}
|
|
|
|
return null;
|
|
}
|
|
|
|
// this exists because
|
|
// new XElement("root", "a\nb").Value is "a\nb" but
|
|
// .ToString(SaveOptions.*) is "a\r\nb" and cannot figure out how to get rid of "\r"
|
|
// and when saving data we want nothing to change
|
|
// this method will produce a string that respects the \r and \n in the data value
|
|
public static string ToDataString(this XElement xml)
|
|
{
|
|
var settings = new XmlWriterSettings
|
|
{
|
|
OmitXmlDeclaration = true,
|
|
NewLineHandling = NewLineHandling.None,
|
|
Indent = false,
|
|
};
|
|
var output = new StringBuilder();
|
|
using (var writer = XmlWriter.Create(output, settings))
|
|
{
|
|
xml.WriteTo(writer);
|
|
}
|
|
|
|
return output.ToString();
|
|
}
|
|
}
|