Skip to content

Commit

Permalink
Listen for directory changes of network shares in FTP
Browse files Browse the repository at this point in the history
  • Loading branch information
gave92 committed Apr 1, 2022
1 parent efac043 commit 9c6461f
Show file tree
Hide file tree
Showing 2 changed files with 145 additions and 6 deletions.
94 changes: 93 additions & 1 deletion src/Files.Launcher/MessageHandlers/Win32MessageHandler.cs
Original file line number Diff line number Diff line change
Expand Up @@ -8,6 +8,7 @@
using System.Diagnostics;
using System.IO;
using System.IO.Pipes;
using System.Linq;
using System.Runtime.Versioning;
using System.Threading.Tasks;
using Vanara.Windows.Shell;
Expand All @@ -20,11 +21,18 @@ namespace FilesFullTrust.MessageHandlers
[SupportedOSPlatform("Windows10.0.10240")]
public class Win32MessageHandler : Disposable, IMessageHandler
{
private IList<FileSystemWatcher> dirWatchers;
private PipeStream connection;

public void Initialize(PipeStream connection)
{
this.connection = connection;

DetectIsSetAsDefaultFileManager();
DetectIsSetAsOpenFileDialog();
ApplicationData.Current.LocalSettings.Values["TEMP"] = Environment.GetEnvironmentVariable("TEMP");

dirWatchers = new List<FileSystemWatcher>();
}

private static void DetectIsSetAsDefaultFileManager()
Expand Down Expand Up @@ -159,7 +167,7 @@ public async Task ParseArgumentsAsync(PipeStream connection, Dictionary<string,
return;
}
}

var dataPath = Environment.ExpandEnvironmentVariables("%LocalAppData%\\Files");
if (enable)
{
Expand Down Expand Up @@ -234,6 +242,90 @@ public async Task ParseArgumentsAsync(PipeStream connection, Dictionary<string,
await Win32API.SendMessageAsync(connection, new ValueSet() { { "FileAssociation", await Win32API.GetFileAssociationAsync(filePath, true) } }, message.Get("RequestID", (string)null));
}
break;

case "WatchDirectory":
var watchAction = (string)message["action"];
await ParseWatchDirectoryActionAsync(connection, message, watchAction);
break;
}
}

private async Task ParseWatchDirectoryActionAsync(PipeStream connection, Dictionary<string, object> message, string action)
{
switch (action)
{
case "start":
{
var res = new ValueSet();
var folderPath = (string)message["folderPath"];
if (Directory.Exists(folderPath))
{
var watcher = new FileSystemWatcher
{
Path = folderPath,
Filter = "*.*",
NotifyFilter = NotifyFilters.LastWrite | NotifyFilters.FileName | NotifyFilters.DirectoryName
};
watcher.Created += DirectoryWatcher_Changed;
watcher.Deleted += DirectoryWatcher_Changed;
watcher.Renamed += DirectoryWatcher_Changed;
watcher.EnableRaisingEvents = true;
res.Add("watcherID", watcher.GetHashCode());
dirWatchers.Add(watcher);
}
await Win32API.SendMessageAsync(connection, res, message.Get("RequestID", (string)null));
}
break;

case "cancel":
{
var watcherID = (long)message["watcherID"];
var watcher = dirWatchers.SingleOrDefault(x => x.GetHashCode() == watcherID);
if (watcher != null)
{
dirWatchers.Remove(watcher);
watcher.Dispose();
}
}
break;
}
}

private async void DirectoryWatcher_Changed(object sender, FileSystemEventArgs e)
{
System.Diagnostics.Debug.WriteLine($"Directory watcher event: {e.ChangeType}, {e.FullPath}");
if (connection?.IsConnected ?? false)
{
var response = new ValueSet()
{
{ "FileSystem", Path.GetDirectoryName(e.FullPath) },
{ "Name", e.Name },
{ "Path", e.FullPath },
{ "Type", e.ChangeType.ToString() }
};
if (e.ChangeType == WatcherChangeTypes.Created)
{
using var folderItem = new ShellItem(e.FullPath);
var shellFileItem = ShellFolderExtensions.GetShellFileItem(folderItem);
response["Item"] = JsonConvert.SerializeObject(shellFileItem);
}
else if (e.ChangeType == WatcherChangeTypes.Renamed)
{
response["OldPath"] = (e as RenamedEventArgs).OldFullPath;
}
// Send message to UWP app to refresh items
await Win32API.SendMessageAsync(connection, response);
}
}

protected override void Dispose(bool disposing)
{
if (disposing)
{
foreach (var watcher in dirWatchers)
{
watcher.Dispose();
}
}
}
}
Expand Down
57 changes: 52 additions & 5 deletions src/Files.Uwp/ViewModels/ItemViewModel.cs
Original file line number Diff line number Diff line change
Expand Up @@ -431,6 +431,20 @@ private async void Connection_RequestReceived(object sender, Dictionary<string,
}
break;

case "Renamed":
var matchingItem = filesAndFolders.FirstOrDefault(x => x.ItemPath.Equals((string)message["OldPath"], StringComparison.OrdinalIgnoreCase));
if (matchingItem != null)
{
await CoreApplication.MainView.DispatcherQueue.EnqueueAsync(() =>
{
matchingItem.ItemPath = itemPath;
matchingItem.ItemNameRaw = (string)message["Name"];
});
await OrderFilesAndFoldersAsync();
await ApplySingleFileChangeAsync(matchingItem);
}
break;

case "Deleted":
// get the item that immediately follows matching item to be removed
// if the matching item is the last item, try to get the previous item; otherwise, null
Expand Down Expand Up @@ -1287,7 +1301,6 @@ private async Task RapidAddItemsToCollection(string path, LibraryItem library =
currentStorageFolder ??= await FilesystemTasks.Wrap(() => StorageFileExtensions.DangerousGetFolderWithPathFromPathAsync(path));
var syncStatus = await CheckCloudDriveSyncStatusAsync(currentStorageFolder?.Item);
PageTypeUpdated?.Invoke(this, new PageTypeUpdatedEventArgs() { IsTypeCloudDrive = syncStatus != CloudDriveSyncStatus.NotSynced && syncStatus != CloudDriveSyncStatus.Unknown });

WatchForDirectoryChanges(path, syncStatus);
break;

Expand All @@ -1297,7 +1310,11 @@ private async Task RapidAddItemsToCollection(string path, LibraryItem library =
WatchForStorageFolderChanges(currentStorageFolder?.Folder);
break;

case 2: // Do no watch for changes in Box Drive folder to avoid constant refresh (#7428)
case 2: // Watch for changes using FTP in Box Drive folder (#7428) and network drives (#5869)
PageTypeUpdated?.Invoke(this, new PageTypeUpdatedEventArgs() { IsTypeCloudDrive = false });
WatchForWin32FolderChanges(path);
break;

case -1: // Enumeration failed
default:
break;
Expand Down Expand Up @@ -1490,8 +1507,9 @@ public async Task<int> EnumerateItemsFromStandardFolderAsync(string path, Type s
{
// Flag to use FindFirstFileExFromApp or StorageFolder enumeration
var isBoxFolder = App.CloudDrivesManager.Drives.FirstOrDefault(x => x.Text == "Box")?.Path?.TrimEnd('\\') is string boxFolder ?
path.StartsWith(boxFolder) : false;
bool enumFromStorageFolder = isBoxFolder; // Use storage folder for Box Drive (#4629)
path.StartsWith(boxFolder) : false; // Use storage folder for Box Drive (#4629)
var isNetworkFolder = System.Text.RegularExpressions.Regex.IsMatch(path, @"^\\(?!\?)"); // Use storage folder for network drives (*FromApp methods return access denied)
bool enumFromStorageFolder = isBoxFolder || isNetworkFolder;

BaseStorageFolder rootFolder = null;

Expand Down Expand Up @@ -1582,7 +1600,7 @@ await DialogDisplayHelper.ShowDialogAsync(
}
CurrentFolder = currentFolder;
await EnumFromStorageFolderAsync(path, currentFolder, rootFolder, currentStorageFolder, sourcePageType, cancellationToken);
return isBoxFolder ? 2 : 1; // Workaround for #7428
return isBoxFolder || isNetworkFolder ? 2 : 1; // Workaround for #7428
}
else
{
Expand Down Expand Up @@ -1758,6 +1776,35 @@ await Task.Run(() =>
});
}

private async void WatchForWin32FolderChanges(string folderPath)
{
if (Connection != null)
{
var (status, response) = await Connection.SendMessageForResponseAsync(new ValueSet()
{
{ "Arguments", "WatchDirectory" },
{ "action", "start" },
{ "folderPath", folderPath }
});
if (status == AppServiceResponseStatus.Success
&& response.ContainsKey("watcherID"))
{
watcherCTS.Token.Register(async () =>
{
if (Connection != null)
{
await Connection.SendMessageAsync(new ValueSet()
{
{ "Arguments", "WatchDirectory" },
{ "action", "cancel" },
{ "watcherID", (long)response["watcherID"] }
});
}
});
}
}
}

private async void ItemQueryResult_ContentsChanged(IStorageQueryResultBase sender, object args)
{
//query options have to be reapplied otherwise old results are returned
Expand Down

0 comments on commit 9c6461f

Please sign in to comment.