// // System.IO.FileSystemWatcher.cs // // Authors: // Tim Coleman (tim@timcoleman.com) // Gonzalo Paniagua Javier (gonzalo@ximian.com) // // Copyright (C) Tim Coleman, 2002 // (c) 2003 Ximian, Inc. (http://www.ximian.com) // Copyright (C) 2004, 2006 Novell, Inc (http://www.novell.com) // // // Permission is hereby granted, free of charge, to any person obtaining // a copy of this software and associated documentation files (the // "Software"), to deal in the Software without restriction, including // without limitation the rights to use, copy, modify, merge, publish, // distribute, sublicense, and/or sell copies of the Software, and to // permit persons to whom the Software is furnished to do so, subject to // the following conditions: // // The above copyright notice and this permission notice shall be // included in all copies or substantial portions of the Software. // // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, // EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF // MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND // NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE // LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION // OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION // WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. // using System.ComponentModel; using System.Diagnostics; using System.Runtime.CompilerServices; using System.Runtime.InteropServices; using System.Security.Permissions; using System.Threading; using System.Threading.Tasks; namespace System.IO { [DefaultEvent("Changed")] [IODescription ("")] public class FileSystemWatcher : Component, ISupportInitialize { #region Fields bool inited; bool start_requested; bool enableRaisingEvents; string filter; bool includeSubdirectories; int internalBufferSize; NotifyFilters notifyFilter; string path; string fullpath; ISynchronizeInvoke synchronizingObject; WaitForChangedResult lastData; bool waiting; SearchPattern2 pattern; bool disposed; string mangledFilter; static IFileWatcher watcher; object watcher_handle; static object lockobj = new object (); #endregion // Fields #region Constructors public FileSystemWatcher () { this.notifyFilter = NotifyFilters.LastWrite | NotifyFilters.FileName | NotifyFilters.DirectoryName; this.enableRaisingEvents = false; this.filter = "*"; this.includeSubdirectories = false; this.internalBufferSize = 8192; this.path = ""; InitWatcher (); } public FileSystemWatcher (string path) : this (path, "*") { } public FileSystemWatcher (string path, string filter) { if (path == null) throw new ArgumentNullException ("path"); if (filter == null) throw new ArgumentNullException ("filter"); if (path == String.Empty) throw new ArgumentException ("Empty path", "path"); if (!Directory.Exists (path)) throw new ArgumentException ("Directory does not exist", "path"); this.inited = false; this.start_requested = false; this.enableRaisingEvents = false; this.filter = filter; if (this.filter == "*.*") this.filter = "*"; this.includeSubdirectories = false; this.internalBufferSize = 8192; this.notifyFilter = NotifyFilters.LastWrite | NotifyFilters.FileName | NotifyFilters.DirectoryName; this.path = path; this.synchronizingObject = null; InitWatcher (); } [EnvironmentPermission (SecurityAction.Assert, Read="MONO_MANAGED_WATCHER")] void InitWatcher () { lock (lockobj) { if (watcher_handle != null) return; string managed = Environment.GetEnvironmentVariable ("MONO_MANAGED_WATCHER"); int mode = 0; if (managed == null) mode = InternalSupportsFSW (); bool ok = false; switch (mode) { case 1: // windows ok = DefaultWatcher.GetInstance (out watcher); watcher_handle = this; break; case 2: // libfam ok = FAMWatcher.GetInstance (out watcher, false); watcher_handle = this; break; case 3: // kevent ok = KeventWatcher.GetInstance (out watcher); watcher_handle = this; break; case 4: // libgamin ok = FAMWatcher.GetInstance (out watcher, true); watcher_handle = this; break; case 5: // inotify ok = InotifyWatcher.GetInstance (out watcher, true); watcher_handle = this; break; case 6: // CoreFX ok = CoreFXFileSystemWatcherProxy.GetInstance (out watcher); watcher_handle = (watcher as CoreFXFileSystemWatcherProxy).NewWatcher (this); break; } if (mode == 0 || !ok) { if (String.Compare (managed, "disabled", true) == 0) NullFileWatcher.GetInstance (out watcher); else { DefaultWatcher.GetInstance (out watcher); watcher_handle = this; } } this.inited = true; ShowWatcherInfo (); } } [Conditional ("DEBUG"), Conditional ("TRACE")] void ShowWatcherInfo () { Console.WriteLine ("Watcher implementation: {0}", watcher != null ? watcher.GetType ().ToString () : ""); } #endregion // Constructors #region Properties /* If this is enabled, we Pulse this instance */ internal bool Waiting { get { return waiting; } set { waiting = value; } } internal string MangledFilter { get { if (filter != "*.*") return filter; if (mangledFilter != null) return mangledFilter; string filterLocal = "*.*"; return filterLocal; } } internal SearchPattern2 Pattern { get { if (pattern == null) { if (watcher?.GetType () == typeof (KeventWatcher)) pattern = new SearchPattern2 (MangledFilter, true); //assume we want to ignore case (OS X) else pattern = new SearchPattern2 (MangledFilter); } return pattern; } } internal string FullPath { get { if (fullpath == null) { if (path == null || path == "") fullpath = Environment.CurrentDirectory; else fullpath = System.IO.Path.GetFullPath (path); } return fullpath; } } [DefaultValue(false)] [IODescription("Flag to indicate if this instance is active")] public bool EnableRaisingEvents { get { return enableRaisingEvents; } set { if (disposed) throw new ObjectDisposedException (GetType().Name); start_requested = true; if (!inited) return; if (value == enableRaisingEvents) return; // Do nothing enableRaisingEvents = value; if (value) { Start (); } else { Stop (); start_requested = false; } } } [DefaultValue("*.*")] [IODescription("File name filter pattern")] [SettingsBindable(true)] [TypeConverter ("System.Diagnostics.Design.StringValueConverter, " + Consts.AssemblySystem_Design)] public string Filter { get { return filter; } set { if (value == null || value == "") value = "*"; if (!string.Equals(filter, value, PathInternal.StringComparison)) { filter = value == "*.*" ? "*" : value; pattern = null; mangledFilter = null; } } } [DefaultValue(false)] [IODescription("Flag to indicate we want to watch subdirectories")] public bool IncludeSubdirectories { get { return includeSubdirectories; } set { if (includeSubdirectories == value) return; includeSubdirectories = value; if (value && enableRaisingEvents) { Stop (); Start (); } } } [Browsable(false)] [DefaultValue(8192)] public int InternalBufferSize { get { return internalBufferSize; } set { if (internalBufferSize == value) return; if (value < 4096) value = 4096; internalBufferSize = value; if (enableRaisingEvents) { Stop (); Start (); } } } [DefaultValue(NotifyFilters.FileName | NotifyFilters.DirectoryName | NotifyFilters.LastWrite)] [IODescription("Flag to indicate which change event we want to monitor")] public NotifyFilters NotifyFilter { get { return notifyFilter; } set { if (notifyFilter == value) return; notifyFilter = value; if (enableRaisingEvents) { Stop (); Start (); } } } [DefaultValue("")] [IODescription("The directory to monitor")] [SettingsBindable(true)] [TypeConverter ("System.Diagnostics.Design.StringValueConverter, " + Consts.AssemblySystem_Design)] [Editor ("System.Diagnostics.Design.FSWPathEditor, " + Consts.AssemblySystem_Design, "System.Drawing.Design.UITypeEditor, " + Consts.AssemblySystem_Drawing)] public string Path { get { return path; } set { if (disposed) throw new ObjectDisposedException (GetType().Name); value = (value == null) ? string.Empty : value; if (string.Equals(path, value, PathInternal.StringComparison)) return; bool exists = false; Exception exc = null; try { exists = Directory.Exists (value); } catch (Exception e) { exc = e; } if (exc != null) throw new ArgumentException(SR.Format(SR.InvalidDirName, value), nameof(Path)); if (!exists) throw new ArgumentException(SR.Format(SR.InvalidDirName_NotExists, value), nameof(Path)); path = value; fullpath = null; if (enableRaisingEvents) { Stop (); Start (); } } } [Browsable(false)] public override ISite Site { get { return base.Site; } set { base.Site = value; if (Site != null && Site.DesignMode) this.EnableRaisingEvents = true; } } [DefaultValue(null)] [IODescription("The object used to marshal the event handler calls resulting from a directory change")] [Browsable (false)] public ISynchronizeInvoke SynchronizingObject { get { return synchronizingObject; } set { synchronizingObject = value; } } #endregion // Properties #region Methods public void BeginInit () { // Not necessary in Mono // but if called, EndInit() must be called inited = false; } protected override void Dispose (bool disposing) { if (disposed) return; try { watcher?.StopDispatching (watcher_handle); watcher?.Dispose (watcher_handle); } catch (Exception) { } watcher_handle = null; watcher = null; disposed = true; base.Dispose (disposing); GC.SuppressFinalize (this); } ~FileSystemWatcher () { if (disposed) return; Dispose (false); } public void EndInit () { inited = true; if (start_requested) this.EnableRaisingEvents = true; } enum EventType { FileSystemEvent, ErrorEvent, RenameEvent } private void RaiseEvent (Delegate ev, EventArgs arg, EventType evtype) { if (ev == null) return; if (synchronizingObject == null) { foreach (var target in ev.GetInvocationList()) { switch (evtype) { case EventType.RenameEvent: ((RenamedEventHandler)target).Invoke (this, (RenamedEventArgs)arg); break; case EventType.ErrorEvent: ((ErrorEventHandler)target).Invoke (this, (ErrorEventArgs)arg); break; case EventType.FileSystemEvent: ((FileSystemEventHandler)target).Invoke (this, (FileSystemEventArgs)arg); break; } } return; } synchronizingObject.BeginInvoke (ev, new object [] {this, arg}); } protected void OnChanged (FileSystemEventArgs e) { RaiseEvent (Changed, e, EventType.FileSystemEvent); } protected void OnCreated (FileSystemEventArgs e) { RaiseEvent (Created, e, EventType.FileSystemEvent); } protected void OnDeleted (FileSystemEventArgs e) { RaiseEvent (Deleted, e, EventType.FileSystemEvent); } protected void OnError (ErrorEventArgs e) { RaiseEvent (Error, e, EventType.ErrorEvent); } protected void OnRenamed (RenamedEventArgs e) { RaiseEvent (Renamed, e, EventType.RenameEvent); } public WaitForChangedResult WaitForChanged (WatcherChangeTypes changeType) { return WaitForChanged (changeType, Timeout.Infinite); } public WaitForChangedResult WaitForChanged (WatcherChangeTypes changeType, int timeout) { WaitForChangedResult result = new WaitForChangedResult (); bool prevEnabled = EnableRaisingEvents; if (!prevEnabled) EnableRaisingEvents = true; bool gotData; lock (this) { waiting = true; gotData = Monitor.Wait (this, timeout); if (gotData) result = this.lastData; } EnableRaisingEvents = prevEnabled; if (!gotData) result.TimedOut = true; return result; } internal void DispatchErrorEvents (ErrorEventArgs args) { OnError (args); } internal void DispatchEvents (FileAction act, string filename, ref RenamedEventArgs renamed) { if (waiting) { lastData = new WaitForChangedResult (); } switch (act) { case FileAction.Added: lastData.Name = filename; lastData.ChangeType = WatcherChangeTypes.Created; Task.Run (() => OnCreated (new FileSystemEventArgs (WatcherChangeTypes.Created, path, filename))); break; case FileAction.Removed: lastData.Name = filename; lastData.ChangeType = WatcherChangeTypes.Deleted; Task.Run (() => OnDeleted (new FileSystemEventArgs (WatcherChangeTypes.Deleted, path, filename))); break; case FileAction.Modified: lastData.Name = filename; lastData.ChangeType = WatcherChangeTypes.Changed; Task.Run (() => OnChanged (new FileSystemEventArgs (WatcherChangeTypes.Changed, path, filename))); break; case FileAction.RenamedOldName: if (renamed != null) { OnRenamed (renamed); } lastData.OldName = filename; lastData.ChangeType = WatcherChangeTypes.Renamed; renamed = new RenamedEventArgs (WatcherChangeTypes.Renamed, path, filename, ""); break; case FileAction.RenamedNewName: lastData.Name = filename; lastData.ChangeType = WatcherChangeTypes.Renamed; if (renamed == null) { renamed = new RenamedEventArgs (WatcherChangeTypes.Renamed, path, "", filename); } var renamed_ref = renamed; Task.Run (() => OnRenamed (renamed_ref)); renamed = null; break; default: break; } } void Start () { if (disposed) return; if (watcher_handle == null) return; watcher?.StartDispatching (watcher_handle); } void Stop () { if (disposed) return; if (watcher_handle == null) return; watcher?.StopDispatching (watcher_handle); } #endregion // Methods #region Events and Delegates [IODescription("Occurs when a file/directory change matches the filter")] public event FileSystemEventHandler Changed; [IODescription("Occurs when a file/directory creation matches the filter")] public event FileSystemEventHandler Created; [IODescription("Occurs when a file/directory deletion matches the filter")] public event FileSystemEventHandler Deleted; [Browsable(false)] public event ErrorEventHandler Error; [IODescription("Occurs when a file/directory rename matches the filter")] public event RenamedEventHandler Renamed; #endregion // Events and Delegates /* 0 -> not supported */ /* 1 -> windows */ /* 2 -> FAM */ /* 3 -> Kevent */ /* 4 -> gamin */ /* 5 -> inotify */ /* 6 -> CoreFX */ [MethodImplAttribute(MethodImplOptions.InternalCall)] static extern int InternalSupportsFSW (); } }