separate metadata refresh from validation

pull/702/head
Luke Pulverenti 11 years ago
parent 8b29e67e22
commit 14084fdd87

@ -1,4 +1,5 @@
using MediaBrowser.Model.Configuration; using MediaBrowser.Controller.Providers;
using MediaBrowser.Model.Configuration;
using MediaBrowser.Model.Dto; using MediaBrowser.Model.Dto;
using MediaBrowser.Model.Entities; using MediaBrowser.Model.Entities;
using System; using System;
@ -49,7 +50,7 @@ namespace MediaBrowser.Controller.Entities.Audio
} }
private readonly Task _cachedTask = Task.FromResult(true); private readonly Task _cachedTask = Task.FromResult(true);
protected override Task ValidateChildrenInternal(IProgress<double> progress, CancellationToken cancellationToken, bool? recursive = null, bool forceRefreshMetadata = false) protected override Task ValidateChildrenInternal(IProgress<double> progress, CancellationToken cancellationToken, bool recursive, bool refreshChildMetadata, MetadataRefreshOptions refreshOptions)
{ {
if (IsAccessedByName) if (IsAccessedByName)
{ {
@ -57,7 +58,7 @@ namespace MediaBrowser.Controller.Entities.Audio
return _cachedTask; return _cachedTask;
} }
return base.ValidateChildrenInternal(progress, cancellationToken, recursive, forceRefreshMetadata); return base.ValidateChildrenInternal(progress, cancellationToken, recursive, refreshChildMetadata, refreshOptions);
} }
public override string GetClientTypeName() public override string GetClientTypeName()

@ -472,75 +472,30 @@ namespace MediaBrowser.Controller.Entities
/// <returns>List{Video}.</returns> /// <returns>List{Video}.</returns>
private IEnumerable<Trailer> LoadLocalTrailers(List<FileSystemInfo> fileSystemChildren) private IEnumerable<Trailer> LoadLocalTrailers(List<FileSystemInfo> fileSystemChildren)
{ {
return new List<Trailer>(); var files = fileSystemChildren.OfType<DirectoryInfo>()
//ItemResolveArgs resolveArgs; .Where(i => string.Equals(i.Name, TrailerFolderName, StringComparison.OrdinalIgnoreCase))
.SelectMany(i => i.EnumerateFiles("*", SearchOption.TopDirectoryOnly))
//try .ToList();
//{
// resolveArgs = ResolveArgs; // Support plex/xbmc convention
files.AddRange(fileSystemChildren.OfType<FileInfo>()
// if (!resolveArgs.IsDirectory) .Where(i => System.IO.Path.GetFileNameWithoutExtension(i.Name).EndsWith(XbmcTrailerFileSuffix, StringComparison.OrdinalIgnoreCase) && !string.Equals(Path, i.FullName, StringComparison.OrdinalIgnoreCase))
// { );
// return new List<Trailer>();
// } return LibraryManager.ResolvePaths<Trailer>(files, null).Select(video =>
//} {
//catch (IOException ex) // Try to retrieve it from the db. If we don't find it, use the resolved version
//{ var dbItem = LibraryManager.GetItemById(video.Id) as Trailer;
// Logger.ErrorException("Error getting ResolveArgs for {0}", ex, Path);
// return new List<Trailer>(); if (dbItem != null)
//} {
video = dbItem;
//var files = new List<FileSystemInfo>(); }
//var folder = resolveArgs.GetFileSystemEntryByName(TrailerFolderName); return video;
//// Path doesn't exist. No biggie // Sort them so that the list can be easily compared for changes
//if (folder != null) }).OrderBy(i => i.Path).ToList();
//{
// try
// {
// files.AddRange(new DirectoryInfo(folder.FullName).EnumerateFiles());
// }
// catch (IOException ex)
// {
// Logger.ErrorException("Error loading trailers for {0}", ex, Name);
// }
//}
//// Support xbmc trailers (-trailer suffix on video file names)
//files.AddRange(resolveArgs.FileSystemChildren.Where(i =>
//{
// try
// {
// if ((i.Attributes & FileAttributes.Directory) != FileAttributes.Directory)
// {
// if (System.IO.Path.GetFileNameWithoutExtension(i.Name).EndsWith(XbmcTrailerFileSuffix, StringComparison.OrdinalIgnoreCase) && !string.Equals(Path, i.FullName, StringComparison.OrdinalIgnoreCase))
// {
// return true;
// }
// }
// }
// catch (IOException ex)
// {
// Logger.ErrorException("Error accessing path {0}", ex, i.FullName);
// }
// return false;
//}));
//return LibraryManager.ResolvePaths<Trailer>(files, null).Select(video =>
//{
// // Try to retrieve it from the db. If we don't find it, use the resolved version
// var dbItem = LibraryManager.GetItemById(video.Id) as Trailer;
// if (dbItem != null)
// {
// video = dbItem;
// }
// return video;
//}).ToList();
} }
/// <summary> /// <summary>
@ -656,19 +611,8 @@ namespace MediaBrowser.Controller.Entities
} }
} }
if (themeSongsChanged) if (themeSongsChanged || themeVideosChanged || localTrailersChanged)
{
Logger.Debug("Theme songs have changed for {0}", Path);
options.ForceSave = true;
}
if (themeVideosChanged)
{
Logger.Debug("Theme videos have changed for {0}", Path);
options.ForceSave = true;
}
if (localTrailersChanged)
{ {
Logger.Debug("Local trailers have changed for {0}", Path);
options.ForceSave = true; options.ForceSave = true;
} }
} }

@ -7,6 +7,7 @@ using System.Linq;
using System.Runtime.Serialization; using System.Runtime.Serialization;
using System.Threading; using System.Threading;
using System.Threading.Tasks; using System.Threading.Tasks;
using MediaBrowser.Controller.Providers;
namespace MediaBrowser.Controller.Entities namespace MediaBrowser.Controller.Entities
{ {
@ -116,9 +117,10 @@ namespace MediaBrowser.Controller.Entities
/// <param name="progress">The progress.</param> /// <param name="progress">The progress.</param>
/// <param name="cancellationToken">The cancellation token.</param> /// <param name="cancellationToken">The cancellation token.</param>
/// <param name="recursive">if set to <c>true</c> [recursive].</param> /// <param name="recursive">if set to <c>true</c> [recursive].</param>
/// <param name="forceRefreshMetadata">if set to <c>true</c> [force refresh metadata].</param> /// <param name="refreshChildMetadata">if set to <c>true</c> [refresh child metadata].</param>
/// <param name="refreshOptions">The refresh options.</param>
/// <returns>Task.</returns> /// <returns>Task.</returns>
protected override Task ValidateChildrenInternal(IProgress<double> progress, CancellationToken cancellationToken, bool? recursive = null, bool forceRefreshMetadata = false) protected override Task ValidateChildrenInternal(IProgress<double> progress, CancellationToken cancellationToken, bool recursive, bool refreshChildMetadata, MetadataRefreshOptions refreshOptions)
{ {
CreateResolveArgs(); CreateResolveArgs();
ResetDynamicChildren(); ResetDynamicChildren();

@ -307,7 +307,17 @@ namespace MediaBrowser.Controller.Entities
/// <param name="recursive">if set to <c>true</c> [recursive].</param> /// <param name="recursive">if set to <c>true</c> [recursive].</param>
/// <param name="forceRefreshMetadata">if set to <c>true</c> [force refresh metadata].</param> /// <param name="forceRefreshMetadata">if set to <c>true</c> [force refresh metadata].</param>
/// <returns>Task.</returns> /// <returns>Task.</returns>
public async Task ValidateChildren(IProgress<double> progress, CancellationToken cancellationToken, bool? recursive = null, bool forceRefreshMetadata = false) public Task ValidateChildren(IProgress<double> progress, CancellationToken cancellationToken, bool? recursive = null, bool forceRefreshMetadata = false)
{
return ValidateChildrenWithCancellationSupport(progress, cancellationToken, recursive ?? true, true,
new MetadataRefreshOptions
{
ReplaceAllMetadata = forceRefreshMetadata
});
}
private async Task ValidateChildrenWithCancellationSupport(IProgress<double> progress, CancellationToken cancellationToken, bool recursive, bool refreshChildMetadata, MetadataRefreshOptions refreshOptions)
{ {
cancellationToken.ThrowIfCancellationRequested(); cancellationToken.ThrowIfCancellationRequested();
@ -327,7 +337,7 @@ namespace MediaBrowser.Controller.Entities
var linkedCancellationTokenSource = CancellationTokenSource.CreateLinkedTokenSource(innerCancellationTokenSource.Token, cancellationToken); var linkedCancellationTokenSource = CancellationTokenSource.CreateLinkedTokenSource(innerCancellationTokenSource.Token, cancellationToken);
await ValidateChildrenInternal(progress, linkedCancellationTokenSource.Token, recursive, forceRefreshMetadata).ConfigureAwait(false); await ValidateChildrenInternal(progress, linkedCancellationTokenSource.Token, recursive, refreshChildMetadata, refreshOptions).ConfigureAwait(false);
} }
catch (OperationCanceledException ex) catch (OperationCanceledException ex)
{ {
@ -352,15 +362,15 @@ namespace MediaBrowser.Controller.Entities
} }
/// <summary> /// <summary>
/// Compare our current children (presumably just read from the repo) with the current state of the file system and adjust for any changes /// Validates the children internal.
/// ***Currently does not contain logic to maintain items that are unavailable in the file system***
/// </summary> /// </summary>
/// <param name="progress">The progress.</param> /// <param name="progress">The progress.</param>
/// <param name="cancellationToken">The cancellation token.</param> /// <param name="cancellationToken">The cancellation token.</param>
/// <param name="recursive">if set to <c>true</c> [recursive].</param> /// <param name="recursive">if set to <c>true</c> [recursive].</param>
/// <param name="forceRefreshMetadata">if set to <c>true</c> [force refresh metadata].</param> /// <param name="refreshChildMetadata">if set to <c>true</c> [refresh child metadata].</param>
/// <param name="refreshOptions">The refresh options.</param>
/// <returns>Task.</returns> /// <returns>Task.</returns>
protected async virtual Task ValidateChildrenInternal(IProgress<double> progress, CancellationToken cancellationToken, bool? recursive = null, bool forceRefreshMetadata = false) protected async virtual Task ValidateChildrenInternal(IProgress<double> progress, CancellationToken cancellationToken, bool recursive, bool refreshChildMetadata, MetadataRefreshOptions refreshOptions)
{ {
var locationType = LocationType; var locationType = LocationType;
@ -400,9 +410,6 @@ namespace MediaBrowser.Controller.Entities
BaseItem currentChild; BaseItem currentChild;
if (currentChildren.TryGetValue(child.Id, out currentChild)) if (currentChildren.TryGetValue(child.Id, out currentChild))
{
//existing item - check if it has changed
if (currentChild.HasChanged(child))
{ {
var currentChildLocationType = currentChild.LocationType; var currentChildLocationType = currentChild.LocationType;
if (currentChildLocationType != LocationType.Remote && if (currentChildLocationType != LocationType.Remote &&
@ -412,22 +419,15 @@ namespace MediaBrowser.Controller.Entities
} }
currentChild.IsInMixedFolder = child.IsInMixedFolder; currentChild.IsInMixedFolder = child.IsInMixedFolder;
validChildren.Add(currentChild);
}
else
{
validChildren.Add(currentChild);
}
currentChild.IsOffline = false; currentChild.IsOffline = false;
} }
else else
{ {
//brand new item - needs to be added //brand new item - needs to be added
newItems.Add(child); newItems.Add(child);
validChildren.Add(child);
} }
validChildren.Add(currentChild);
} }
// If any items were added or removed.... // If any items were added or removed....
@ -435,7 +435,6 @@ namespace MediaBrowser.Controller.Entities
{ {
// That's all the new and changed ones - now see if there are any that are missing // That's all the new and changed ones - now see if there are any that are missing
var itemsRemoved = currentChildren.Values.Except(validChildren).ToList(); var itemsRemoved = currentChildren.Values.Except(validChildren).ToList();
var actualRemovals = new List<BaseItem>(); var actualRemovals = new List<BaseItem>();
foreach (var item in itemsRemoved) foreach (var item in itemsRemoved)
@ -450,7 +449,6 @@ namespace MediaBrowser.Controller.Entities
else if (!string.IsNullOrEmpty(item.Path) && IsPathOffline(item.Path)) else if (!string.IsNullOrEmpty(item.Path) && IsPathOffline(item.Path))
{ {
item.IsOffline = true; item.IsOffline = true;
validChildren.Add(item); validChildren.Add(item);
} }
else else
@ -486,78 +484,135 @@ namespace MediaBrowser.Controller.Entities
cancellationToken.ThrowIfCancellationRequested(); cancellationToken.ThrowIfCancellationRequested();
await RefreshChildren(validChildren, progress, cancellationToken, recursive, forceRefreshMetadata).ConfigureAwait(false); if (recursive)
{
await ValidateSubFolders(validChildren.OfType<Folder>().ToList(), progress, cancellationToken).ConfigureAwait(false);
}
progress.Report(20);
if (refreshChildMetadata)
{
var container = this as IMetadataContainer;
var innerProgress = new ActionableProgress<double>();
innerProgress.RegisterAction(p => progress.Report((.80 * p) + 20));
if (container != null)
{
await container.RefreshAllMetadata(refreshOptions, innerProgress, cancellationToken).ConfigureAwait(false);
}
else
{
await RefreshMetadataRecursive(refreshOptions, recursive, innerProgress, cancellationToken);
}
}
progress.Report(100); progress.Report(100);
} }
/// <summary> private async Task RefreshMetadataRecursive(MetadataRefreshOptions refreshOptions, bool recursive, IProgress<double> progress, CancellationToken cancellationToken)
/// Refreshes the children.
/// </summary>
/// <param name="children">The children.</param>
/// <param name="progress">The progress.</param>
/// <param name="cancellationToken">The cancellation token.</param>
/// <param name="recursive">if set to <c>true</c> [recursive].</param>
/// <param name="forceRefreshMetadata">if set to <c>true</c> [force refresh metadata].</param>
/// <returns>Task.</returns>
private async Task RefreshChildren(IList<BaseItem> children, IProgress<double> progress, CancellationToken cancellationToken, bool? recursive, bool forceRefreshMetadata = false)
{ {
var list = children; var children = ActualChildren.ToList();
var percentages = new Dictionary<Guid, double>(list.Count); var percentages = new Dictionary<Guid, double>(children.Count);
var tasks = new List<Task>(); var tasks = new List<Task>();
foreach (var tuple in list) foreach (var child in children)
{ {
if (tasks.Count > 10) if (tasks.Count > 3)
{ {
await Task.WhenAll(tasks).ConfigureAwait(false); await Task.WhenAll(tasks).ConfigureAwait(false);
tasks.Clear();
} }
tasks.Add(RefreshChild(tuple, progress, percentages, list.Count, cancellationToken, recursive, forceRefreshMetadata)); cancellationToken.ThrowIfCancellationRequested();
var innerProgress = new ActionableProgress<double>();
// Avoid implicitly captured closure
var currentChild = child;
innerProgress.RegisterAction(p =>
{
lock (percentages)
{
percentages[currentChild.Id] = p / 100;
var percent = percentages.Values.Sum();
percent /= children.Count;
percent *= 100;
progress.Report(percent);
} }
});
cancellationToken.ThrowIfCancellationRequested(); if (child.IsFolder)
{
await RefreshChildMetadata(child, refreshOptions, recursive, innerProgress, cancellationToken)
.ConfigureAwait(false);
}
else
{
tasks.Add(RefreshChildMetadata(child, refreshOptions, recursive, innerProgress, cancellationToken));
}
}
await Task.WhenAll(tasks).ConfigureAwait(false); await Task.WhenAll(tasks).ConfigureAwait(false);
progress.Report(100);
} }
private async Task RefreshChild(BaseItem item, IProgress<double> progress, Dictionary<Guid, double> percentages, int childCount, CancellationToken cancellationToken, bool? recursive, bool forceRefreshMetadata = false) private async Task RefreshChildMetadata(BaseItem child, MetadataRefreshOptions refreshOptions, bool recursive, IProgress<double> progress, CancellationToken cancellationToken)
{ {
cancellationToken.ThrowIfCancellationRequested(); var container = child as IMetadataContainer;
var child = item; if (container != null)
try {
await container.RefreshAllMetadata(refreshOptions, progress, cancellationToken).ConfigureAwait(false);
}
else
{ {
//refresh it await child.RefreshMetadata(refreshOptions, cancellationToken).ConfigureAwait(false);
await child.RefreshMetadata(new MetadataRefreshOptions
if (recursive)
{ {
ReplaceAllMetadata = forceRefreshMetadata var folder = child as Folder;
}, cancellationToken).ConfigureAwait(false); if (folder != null)
}
catch (IOException ex)
{ {
Logger.ErrorException("Error refreshing {0}", ex, child.Path ?? child.Name); await folder.RefreshMetadataRecursive(refreshOptions, true, progress, cancellationToken);
}
}
} }
progress.Report(100);
}
/// <summary>
/// Refreshes the children.
/// </summary>
/// <param name="children">The children.</param>
/// <param name="progress">The progress.</param>
/// <param name="cancellationToken">The cancellation token.</param>
/// <returns>Task.</returns>
private async Task ValidateSubFolders(IList<Folder> children, IProgress<double> progress, CancellationToken cancellationToken)
{
var list = children;
var childCount = list.Count;
var percentages = new Dictionary<Guid, double>(list.Count);
// Refresh children if a folder and the item changed or recursive is set to true var tasks = new List<Task>();
var refreshChildren = child.IsFolder;
if (refreshChildren) foreach (var item in list)
{ {
// Don't refresh children if explicitly set to false if (tasks.Count > 10)
if (recursive.HasValue && recursive.Value == false)
{ {
refreshChildren = false; await Task.WhenAll(tasks).ConfigureAwait(false);
}
} }
if (refreshChildren)
{
cancellationToken.ThrowIfCancellationRequested(); cancellationToken.ThrowIfCancellationRequested();
var child = item;
var innerProgress = new ActionableProgress<double>(); var innerProgress = new ActionableProgress<double>();
innerProgress.RegisterAction(p => innerProgress.RegisterAction(p =>
@ -569,24 +624,16 @@ namespace MediaBrowser.Controller.Entities
var percent = percentages.Values.Sum(); var percent = percentages.Values.Sum();
percent /= childCount; percent /= childCount;
progress.Report((90 * percent) + 10); progress.Report((10 * percent) + 10);
} }
}); });
await ((Folder)child).ValidateChildren(innerProgress, cancellationToken, recursive, forceRefreshMetadata).ConfigureAwait(false); tasks.Add(child.ValidateChildrenWithCancellationSupport(innerProgress, cancellationToken, true, false, null));
} }
else
{
lock (percentages)
{
percentages[child.Id] = 1;
var percent = percentages.Values.Sum(); cancellationToken.ThrowIfCancellationRequested();
percent /= childCount;
progress.Report((90 * percent) + 10); await Task.WhenAll(tasks).ConfigureAwait(false);
}
}
} }
/// <summary> /// <summary>
@ -962,11 +1009,11 @@ namespace MediaBrowser.Controller.Entities
/// <returns>Task.</returns> /// <returns>Task.</returns>
public override async Task ChangedExternally() public override async Task ChangedExternally()
{ {
await base.ChangedExternally().ConfigureAwait(false);
var progress = new Progress<double>(); var progress = new Progress<double>();
await ValidateChildren(progress, CancellationToken.None).ConfigureAwait(false); await ValidateChildren(progress, CancellationToken.None).ConfigureAwait(false);
await base.ChangedExternally().ConfigureAwait(false);
} }
/// <summary> /// <summary>

@ -0,0 +1,19 @@
using MediaBrowser.Controller.Providers;
using System;
using System.Threading;
using System.Threading.Tasks;
namespace MediaBrowser.Controller.Entities
{
public interface IMetadataContainer
{
/// <summary>
/// Refreshes all metadata.
/// </summary>
/// <param name="refreshOptions">The refresh options.</param>
/// <param name="progress">The progress.</param>
/// <param name="cancellationToken">The cancellation token.</param>
/// <returns>Task.</returns>
Task RefreshAllMetadata(MetadataRefreshOptions refreshOptions, IProgress<double> progress, CancellationToken cancellationToken);
}
}

@ -215,8 +215,9 @@ namespace MediaBrowser.Controller.Entities
return RefreshMetadata(new MetadataRefreshOptions return RefreshMetadata(new MetadataRefreshOptions
{ {
ForceSave = true, ReplaceAllMetadata = true,
ReplaceAllMetadata = true ImageRefreshMode = ImageRefreshMode.FullRefresh,
MetadataRefreshMode = MetadataRefreshMode.FullRefresh
}, CancellationToken.None); }, CancellationToken.None);
} }

@ -103,6 +103,7 @@
<Compile Include="Entities\IItemByName.cs" /> <Compile Include="Entities\IItemByName.cs" />
<Compile Include="Entities\ILibraryItem.cs" /> <Compile Include="Entities\ILibraryItem.cs" />
<Compile Include="Entities\ImageSourceInfo.cs" /> <Compile Include="Entities\ImageSourceInfo.cs" />
<Compile Include="Entities\IMetadataContainer.cs" />
<Compile Include="Entities\LinkedChild.cs" /> <Compile Include="Entities\LinkedChild.cs" />
<Compile Include="Entities\MusicVideo.cs" /> <Compile Include="Entities\MusicVideo.cs" />
<Compile Include="Entities\IHasAwards.cs" /> <Compile Include="Entities\IHasAwards.cs" />

@ -197,6 +197,11 @@ namespace MediaBrowser.Providers.Manager
.Where(i => i.HasChanged(currentItem, currentItem.DateLastSaved)) .Where(i => i.HasChanged(currentItem, currentItem.DateLastSaved))
.ToList(); .ToList();
if (providersWithChanges.Count > 0)
{
var b = true;
}
// If local providers are the only ones with changes, then just run those // If local providers are the only ones with changes, then just run those
if (providersWithChanges.All(i => i is ILocalMetadataProvider)) if (providersWithChanges.All(i => i is ILocalMetadataProvider))
{ {

@ -51,12 +51,15 @@ namespace MediaBrowser.Providers.TV
var dateLastEpisodeAdded = item.DateLastEpisodeAdded; var dateLastEpisodeAdded = item.DateLastEpisodeAdded;
item.DateLastEpisodeAdded = episodes.Select(i => i.DateCreated) item.DateLastEpisodeAdded = episodes
.Where(i => i.LocationType != LocationType.Virtual)
.Select(i => i.DateCreated)
.OrderByDescending(i => i) .OrderByDescending(i => i)
.FirstOrDefault(); .FirstOrDefault();
if (dateLastEpisodeAdded != item.DateLastEpisodeAdded) if (dateLastEpisodeAdded != item.DateLastEpisodeAdded)
{ {
Logger.Debug("DateLastEpisodeAdded changed for {0}", item.Path);
updateType = updateType | ItemUpdateType.MetadataImport; updateType = updateType | ItemUpdateType.MetadataImport;
} }

@ -1088,32 +1088,17 @@ namespace MediaBrowser.Providers.TV
{ {
var seriesDataPath = GetSeriesDataPath(_config.ApplicationPaths, seriesId); var seriesDataPath = GetSeriesDataPath(_config.ApplicationPaths, seriesId);
try
{
var files = new DirectoryInfo(seriesDataPath).EnumerateFiles("*.xml", SearchOption.TopDirectoryOnly)
.ToList();
var seriesXmlFilename = item.GetPreferredMetadataLanguage() + ".xml"; var seriesXmlFilename = item.GetPreferredMetadataLanguage() + ".xml";
var seriesFile = files.FirstOrDefault(i => string.Equals(seriesXmlFilename, i.Name, StringComparison.OrdinalIgnoreCase)); var filePath = Path.Combine(seriesDataPath, seriesXmlFilename);
if (seriesFile != null && seriesFile.Exists && _fileSystem.GetLastWriteTimeUtc(seriesFile) > date) var seriesFile = new FileInfo(filePath);
{
return true;
}
var actorsXml = files.FirstOrDefault(i => string.Equals("actors.xml", i.Name, StringComparison.OrdinalIgnoreCase));
if (actorsXml != null && actorsXml.Exists && _fileSystem.GetLastWriteTimeUtc(actorsXml) > date) if (seriesFile.Exists && _fileSystem.GetLastWriteTimeUtc(seriesFile) > date)
{ {
return true; return true;
} }
} }
catch (DirectoryNotFoundException)
{
// Don't blow up
}
}
return false; return false;
} }

@ -1110,7 +1110,6 @@ namespace MediaBrowser.Server.Implementations.Library
cancellationToken.ThrowIfCancellationRequested(); cancellationToken.ThrowIfCancellationRequested();
await userRootFolder.ValidateChildren(new Progress<double>(), cancellationToken, recursive: false).ConfigureAwait(false); await userRootFolder.ValidateChildren(new Progress<double>(), cancellationToken, recursive: false).ConfigureAwait(false);
var b = true;
} }
/// <summary> /// <summary>

Loading…
Cancel
Save