Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Fixes for project file removal support #1694

Open
wants to merge 24 commits into
base: master
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
24 commits
Select commit Hold shift + click to select a range
994aebd
Basic implementation for unwatch project files method
savpek Feb 2, 2020
d11b0b4
Implemented routines to analyzers to remove diagnostics if file/proje…
savpek Feb 2, 2020
1282837
Merge branch 'master' into feature/support-removal-of-files
savpek Feb 2, 2020
81d1c81
Fixed bug in unwatch method
savpek Feb 2, 2020
f319e0a
Merge remote-tracking branch 'upstream/master' into feature/support-r…
savpek Feb 15, 2020
7e311a5
Merge branch 'master' into feature/support-removal-of-files
savpek Mar 1, 2020
4bd9a14
Fixes for project manager
savpek Mar 1, 2020
f65179e
Merge branch 'master' into feature/support-removal-of-files
savpek Mar 3, 2020
af56be9
Merge branch 'master' into feature/support-removal-of-files
savpek Mar 11, 2020
d171efc
Merge branch 'master' into feature/support-removal-of-files
savpek Mar 29, 2020
9d30063
Added test that checks removed project in workspace
savpek Mar 29, 2020
d82113d
Added test for cleanup analysis
savpek Mar 29, 2020
af5b575
Rebuild
savpek Mar 29, 2020
30f6c42
Rebuild
savpek Mar 30, 2020
b7c2b57
Merge branch 'master' into feature/support-removal-of-files
savpek Apr 7, 2020
4666f05
Merge branch 'master' into feature/support-removal-of-files
savpek Apr 11, 2020
f20a016
Merge branch 'master' into feature/support-removal-of-files
savpek Apr 18, 2020
fec0244
Merge from upstream
savpek Jun 5, 2020
9c91d2c
Merge branch 'master' into feature/support-removal-of-files
savpek Jun 6, 2020
d25034d
Merge branch 'master' into feature/support-removal-of-files
savpek Jun 6, 2020
776bf26
Mergefix
savpek Jun 6, 2020
ea06c9b
Merge branch 'master' into feature/support-removal-of-files
savpek Jun 9, 2020
e08fb70
Merge branch 'master' into feature/support-removal-of-files
savpek Jun 22, 2020
0d0ac64
Merge branch 'master' into feature/support-removal-of-files
savpek Oct 31, 2020
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
2 changes: 2 additions & 0 deletions src/OmniSharp.Abstractions/FileWatching/IFileSystemWatcher.cs
Original file line number Diff line number Diff line change
Expand Up @@ -6,6 +6,8 @@ namespace OmniSharp.FileWatching

public interface IFileSystemWatcher
{
void Unwatch(string pathOrExtension);

/// <summary>
/// Call to watch a file or directory path for changes.
/// </summary>
Expand Down

This file was deleted.

40 changes: 39 additions & 1 deletion src/OmniSharp.Host/FileWatching/ManualFileSystemWatcher.cs
Original file line number Diff line number Diff line change
Expand Up @@ -5,7 +5,7 @@

namespace OmniSharp.FileWatching
{
internal partial class ManualFileSystemWatcher : IFileSystemWatcher, IFileSystemNotifier
internal class ManualFileSystemWatcher : IFileSystemWatcher, IFileSystemNotifier
{
private readonly object _gate = new object();
private readonly Dictionary<string, Callbacks> _callbacksMap;
Expand Down Expand Up @@ -73,5 +73,43 @@ public void Watch(string pathOrExtension, FileSystemNotificationCallback callbac
callbacks.Add(callback);
}
}

public void Unwatch(string pathOrExtension)
{
if (pathOrExtension == null)
{
throw new ArgumentNullException(nameof(pathOrExtension));
}

lock (_gate)
{
if (_callbacksMap.ContainsKey(pathOrExtension))
{
_callbacksMap.Remove(pathOrExtension);
}
}
}

private class Callbacks
{
private readonly List<FileSystemNotificationCallback> _callbacks = new List<FileSystemNotificationCallback>();
private readonly HashSet<FileSystemNotificationCallback> _callbackSet = new HashSet<FileSystemNotificationCallback>();

public void Add(FileSystemNotificationCallback callback)
{
if (_callbackSet.Add(callback))
{
_callbacks.Add(callback);
}
}

public void Invoke(string filePath, FileChangeType changeType)
{
foreach (var callback in _callbacks)
{
callback(filePath, changeType);
}
}
}
}
}
130 changes: 67 additions & 63 deletions src/OmniSharp.MSBuild/ProjectManager.cs
Original file line number Diff line number Diff line change
Expand Up @@ -10,21 +10,17 @@
using Microsoft.CodeAnalysis;
using Microsoft.CodeAnalysis.CSharp;
using Microsoft.Extensions.Logging;
using Newtonsoft.Json.Linq;
using OmniSharp.Eventing;
using OmniSharp.FileWatching;
using OmniSharp.Models.UpdateBuffer;
using OmniSharp.MSBuild.Logging;
using OmniSharp.MSBuild.Models.Events;
using OmniSharp.MSBuild.Notification;
using OmniSharp.MSBuild.ProjectFile;
using OmniSharp.Roslyn.CSharp.Services.Diagnostics;
using OmniSharp.Roslyn.CSharp.Services.Refactoring.V2;
using OmniSharp.Options;
using OmniSharp.Roslyn.Utilities;
using OmniSharp.Services;
using OmniSharp.Utilities;
using System.Reflection;
using Microsoft.CodeAnalysis.Diagnostics;
using OmniSharp.Roslyn.EditorConfig;

Expand All @@ -37,11 +33,13 @@ private class ProjectToUpdate
public ProjectIdInfo ProjectIdInfo;
public string FilePath { get; }
public bool AllowAutoRestore { get; set; }
public FileChangeType ChangeType { get; }
public ProjectLoadedEventArgs LoadedEventArgs { get; set; }

public ProjectToUpdate(string filePath, bool allowAutoRestore, ProjectIdInfo projectIdInfo)
public ProjectToUpdate(string filePath, bool allowAutoRestore, ProjectIdInfo projectIdInfo, FileChangeType changeType)
{
ProjectIdInfo = projectIdInfo ?? throw new ArgumentNullException(nameof(projectIdInfo));
ChangeType = changeType;
FilePath = filePath ?? throw new ArgumentNullException(nameof(filePath));
AllowAutoRestore = allowAutoRestore;
}
Expand Down Expand Up @@ -127,7 +125,7 @@ private async Task WaitForProjectModelReadyAsync(string documentPath)
if (_projectsRequestedOnDemand.TryAdd(csProjFile, 0 /*unused*/))
{
var projectIdInfo = new ProjectIdInfo(ProjectId.CreateNewId(csProjFile), false);
QueueProjectUpdate(csProjFile, allowAutoRestore: true, projectId: projectIdInfo);
QueueProjectUpdate(csProjFile, allowAutoRestore: true, projectId: projectIdInfo, FileChangeType.Unspecified);
}
}

Expand Down Expand Up @@ -158,10 +156,10 @@ protected override void DisposeCore(bool disposing)
public IEnumerable<ProjectFileInfo> GetAllProjects() => _projectFiles.GetItems();
public bool TryGetProject(string projectFilePath, out ProjectFileInfo projectFileInfo) => _projectFiles.TryGetValue(projectFilePath, out projectFileInfo);

public void QueueProjectUpdate(string projectFilePath, bool allowAutoRestore, ProjectIdInfo projectId)
public void QueueProjectUpdate(string projectFilePath, bool allowAutoRestore, ProjectIdInfo projectId, FileChangeType changeType = FileChangeType.Unspecified)
{
_logger.LogInformation($"Queue project update for '{projectFilePath}'");
_queue.Post(new ProjectToUpdate(projectFilePath, allowAutoRestore, projectId));
_queue.Post(new ProjectToUpdate(projectFilePath, allowAutoRestore, projectId, changeType));
}

public async Task WaitForQueueEmptyAsync()
Expand Down Expand Up @@ -193,8 +191,8 @@ private void ProcessQueue(CancellationToken cancellationToken)
_processingQueue = true;
try
{
Dictionary<string, ProjectToUpdate> projectByFilePathMap = null;
List<ProjectToUpdate> projectList = null;
var projectByFilePathMap = new Dictionary<string, ProjectToUpdate>(StringComparer.OrdinalIgnoreCase);
var projectList = new List<ProjectToUpdate>();

while (_queue.TryReceive(out var currentProject))
{
Expand All @@ -203,12 +201,6 @@ private void ProcessQueue(CancellationToken cancellationToken)
break;
}

if (projectByFilePathMap == null)
{
projectByFilePathMap = new Dictionary<string, ProjectToUpdate>(StringComparer.OrdinalIgnoreCase);
projectList = new List<ProjectToUpdate>();
}

// Ensure that we don't process the same project twice. However, if a project *does*
// appear more than once in the update queue, ensure that AllowAutoRestore is set to true
// if any of the updates requires it.
Expand All @@ -222,7 +214,11 @@ private void ProcessQueue(CancellationToken cancellationToken)
continue;
}

// TODO: Handle removing project
if(currentProject.ChangeType == FileChangeType.Delete)
{
RemoveProject(currentProject.FilePath);
continue;
}

projectByFilePathMap.Add(currentProject.FilePath, currentProject);
projectList.Add(currentProject);
Expand Down Expand Up @@ -258,35 +254,32 @@ private void ProcessQueue(CancellationToken cancellationToken)
}
}

if (projectByFilePathMap != null)
foreach (var project in projectList)
{
foreach (var project in projectList)
{
UpdateProject(project.FilePath);
UpdateProject(project.FilePath);

// Fire loaded events
if (project.LoadedEventArgs != null)
// Fire loaded events
if (project.LoadedEventArgs != null)
{
foreach (var eventSink in _eventSinks)
{
foreach (var eventSink in _eventSinks)
try
{
eventSink.ProjectLoaded(project.LoadedEventArgs);
}
catch (Exception ex)
{
try
{
eventSink.ProjectLoaded(project.LoadedEventArgs);
}
catch (Exception ex)
{
_logger.LogError(ex, "Exception thrown while calling event sinks");
}
_logger.LogError(ex, "Exception thrown while calling event sinks");
}
}
}
}

foreach (var project in projectList)
foreach (var project in projectList)
{
if (_projectFiles.TryGetValue(project.FilePath, out var projectFileInfo))
{
if (_projectFiles.TryGetValue(project.FilePath, out var projectFileInfo))
{
_packageDependencyChecker.CheckForUnresolvedDependences(projectFileInfo, project.AllowAutoRestore);
}
_packageDependencyChecker.CheckForUnresolvedDependences(projectFileInfo, project.AllowAutoRestore);
}
}
}
Expand Down Expand Up @@ -348,9 +341,10 @@ private bool RemoveProject(string projectFilePath)
if (!_workspace.TryApplyChanges(newSolution))
{
_logger.LogError($"Failed to remove project from workspace: '{projectFileInfo.FilePath}'");
return false;
}

// TODO: Stop watching project files
UnwatchProjectFiles(projectFileInfo);

return true;
}
Expand Down Expand Up @@ -378,11 +372,9 @@ private void AddProject(ProjectFileInfo projectFileInfo)

private void WatchProjectFiles(ProjectFileInfo projectFileInfo)
{
// TODO: This needs some improvement. Currently, it tracks both deletions and changes
// as "updates". We should properly remove projects that are deleted.
_fileSystemWatcher.Watch(projectFileInfo.FilePath, (file, changeType) =>
{
QueueProjectUpdate(projectFileInfo.FilePath, allowAutoRestore: true, projectFileInfo.ProjectIdInfo);
QueueProjectUpdate(projectFileInfo.FilePath, allowAutoRestore: true, projectFileInfo.ProjectIdInfo, changeType);
});

if (_workspace.EditorConfigEnabled)
Expand Down Expand Up @@ -413,38 +405,50 @@ private void WatchProjectFiles(ProjectFileInfo projectFileInfo)
{
_fileSystemWatcher.Watch(projectFileInfo.RuleSet.FilePath, (file, changeType) =>
{
QueueProjectUpdate(projectFileInfo.FilePath, allowAutoRestore: false, projectFileInfo.ProjectIdInfo);
QueueProjectUpdate(projectFileInfo.FilePath, allowAutoRestore: false, projectFileInfo.ProjectIdInfo, changeType);
});
}

if (!string.IsNullOrEmpty(projectFileInfo.ProjectAssetsFile))
foreach(var projectAssetFile in GetProjectAssetFilePaths(projectFileInfo))
{
_fileSystemWatcher.Watch(projectFileInfo.ProjectAssetsFile, (file, changeType) =>
_fileSystemWatcher.Watch(projectAssetFile, (file, changeType) =>
{
QueueProjectUpdate(projectFileInfo.FilePath, allowAutoRestore: false, projectFileInfo.ProjectIdInfo);
QueueProjectUpdate(projectFileInfo.FilePath, allowAutoRestore: false, projectFileInfo.ProjectIdInfo, changeType);
});
}
}

var restoreDirectory = Path.GetDirectoryName(projectFileInfo.ProjectAssetsFile);
var nugetFileBase = Path.Combine(restoreDirectory, Path.GetFileName(projectFileInfo.FilePath) + ".nuget");
var nugetCacheFile = nugetFileBase + ".cache";
var nugetPropsFile = nugetFileBase + ".g.props";
var nugetTargetsFile = nugetFileBase + ".g.targets";
private void UnwatchProjectFiles(ProjectFileInfo projectFileInfo)
{
_fileSystemWatcher.Unwatch(projectFileInfo.FilePath);

_fileSystemWatcher.Watch(nugetCacheFile, (file, changeType) =>
{
QueueProjectUpdate(projectFileInfo.FilePath, allowAutoRestore: false, projectFileInfo.ProjectIdInfo);
});
if (projectFileInfo.RuleSet?.FilePath != null)
{
_fileSystemWatcher.Unwatch(projectFileInfo.RuleSet.FilePath);
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Will this be problematic if multiple projects use the same ruleset file? project.AnalyzerConfigDocuments are another group of files that could be unwatched, but they are likely shared with other projects so it may require reference counting.

}

_fileSystemWatcher.Watch(nugetPropsFile, (file, changeType) =>
{
QueueProjectUpdate(projectFileInfo.FilePath, allowAutoRestore: false, projectFileInfo.ProjectIdInfo);
});
foreach(var projectAssetFile in GetProjectAssetFilePaths(projectFileInfo))
{
_fileSystemWatcher.Unwatch(projectAssetFile);
}
}

_fileSystemWatcher.Watch(nugetTargetsFile, (file, changeType) =>
{
QueueProjectUpdate(projectFileInfo.FilePath, allowAutoRestore: false, projectFileInfo.ProjectIdInfo);
});
private static IEnumerable<string> GetProjectAssetFilePaths(ProjectFileInfo projectFileInfo)
{
if (string.IsNullOrEmpty(projectFileInfo.ProjectAssetsFile))
{
return new string[] {};
}

var restoreDirectory = Path.GetDirectoryName(projectFileInfo.ProjectAssetsFile);
var nugetFileBase = Path.Combine(restoreDirectory, Path.GetFileName(projectFileInfo.FilePath) + ".nuget");
return new[]
{
projectFileInfo.ProjectAssetsFile,
nugetFileBase + ".cache",
nugetFileBase + ".g.props",
nugetFileBase + ".g.targets"
};
}

private void UpdateProject(string projectFilePath)
Expand Down Expand Up @@ -661,7 +665,7 @@ private void UpdateProjectReferences(Project project, ImmutableArray<string> pro
referencedProject = ProjectFileInfo.CreateNoBuild(projectReferencePath, _projectLoader, _dotNetInfo);
AddProject(referencedProject);

QueueProjectUpdate(projectReferencePath, allowAutoRestore: true, referencedProject.ProjectIdInfo);
QueueProjectUpdate(projectReferencePath, allowAutoRestore: true, referencedProject.ProjectIdInfo, FileChangeType.Unspecified);
}
}

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -168,9 +168,10 @@ private void OnWorkspaceChanged(object sender, WorkspaceChangeEventArgs changeEv
QueueForAnalysis(ImmutableArray.Create(changeEvent.DocumentId), AnalyzerWorkType.Foreground);
break;
case WorkspaceChangeKind.DocumentRemoved:
if (!_currentDiagnosticResultLookup.TryRemove(changeEvent.DocumentId, out _))
if(_currentDiagnosticResultLookup.TryRemove(changeEvent.DocumentId, out _))
{
_logger.LogDebug($"Tried to remove non existent document from analysis, document: {changeEvent.DocumentId}");
var existingDocumentsInProject = _workspace.CurrentSolution.GetProject(changeEvent.ProjectId).Documents.Select(x => x.Id).ToImmutableArray();
QueueForAnalysis(existingDocumentsInProject, AnalyzerWorkType.Background);
}
break;
case WorkspaceChangeKind.ProjectAdded:
Expand Down
4 changes: 4 additions & 0 deletions tests/OmniSharp.Cake.Tests/LineIndexHelperFacts.cs
Original file line number Diff line number Diff line change
Expand Up @@ -151,6 +151,10 @@ public async Task TranslateFromGenerated_Should_Translate_To_Negative_If_Outside

private class DummyFileSystemWatcher : IFileSystemWatcher
{
public void Unwatch(string pathOrExtension)
{
}

public void Watch(string pathOrExtension, FileSystemNotificationCallback callback)
{
}
Expand Down
4 changes: 2 additions & 2 deletions tests/OmniSharp.MSBuild.Tests/ProjectLoadTestEventEmitter.cs
Original file line number Diff line number Diff line change
Expand Up @@ -36,9 +36,9 @@ public ExportDescriptorProvider[] AsExportDescriptionProvider(ILoggerFactory log

public void Emit(string kind, object args)
{
if(args is ProjectConfigurationMessage)
if(args is ProjectConfigurationMessage message)
{
ReceivedMessages = ReceivedMessages.Add((ProjectConfigurationMessage)args);
ReceivedMessages = ReceivedMessages.Add(message);
_messageEvent.Set();
}
}
Expand Down
Loading