Query the project API (Visual Studio SDK)
Important
This documentation refers to Project Query API functionality for the Visual Studio SDK. For VisualStudio.Extensibility API documentation, see Query the project API for VisualStudio.Extensibility.
The Project Query API in the Visual Studio SDK enables querying information from the project system. Project systems are components of Visual Studio that assist users in managing projects, running builds to generate outputs, and testing those outputs.
With the Project Query API, you can:
- Work with Project Systems
- Retrieve data from projects
- Make changes to projects
Some examples of what the Project Query API can do include giving you a way to understand the files included in a project, the NuGet packages referenced by a project, add new files to a project, or adjust project properties.
Find further information on project systems in our Visual Studio Project System Extensibility Documentation reference.
This overview covers top scenarios for working with the project query API:
- Access the project query space
- Query the project system for a project
- Specify the project parameters to be included in the query result
- Filter the query result
- Use nested queries to specify desired properties
- Retrieve a child collection using the Get method
- Query additional information from a previously returned item
- Modify a project
- Query for project properties
- Query for solutions
- Query for solution folders
- Enumerating source files with additional information in a project
- Query for projects that own a specific source file
- Query for project configurations and their properties
- Query for project-to-project references
- Query for package references
- Query for project output groups
- Query for Startup Projects
- Query for Solution Configurations
- Action Query to Load/Unload a Project
- Action Query to Build Solutions/Projects
- Action Query to Save a Solution/Project
- Query to Subscribe to Query Results
- Query to Track Query Changes
- Action Query to Skip
You'll need to obtain an instance of the project query space object to query the project system. This object has several asynchronous methods that query or update the project system. The term project query space and the term workspace both mean the same thing and refer to the object that provides access to all the data for a project.
In the following code excerpt, package
represents an instance of AsyncPackage, a class utilized in the development of Visual Studio extensions. The method GetServiceAsync
is employed to asynchronously procure the query service from the Visual Studio's service container.
IProjectSystemQueryService queryService = await package.GetServiceAsync<IProjectSystemQueryService, IProjectSystemQueryService>();
ProjectQueryableSpace workSpace = queryService.QueryableSpace;
The API lets you query for an individual project: if you have the project GUID. There are usually two GUIDs associated with a project, one that represents the project type, and other that uniquely represents the project. You can find the project's unique GUID in the solution file, or from an extension, you can query for the Guid
property as demonstrated in the next section.
IAsyncEnumerable<IQueryResultItem<IProjectSnapshot>> projectList = workspace
.ProjectsByProjectGuid(knownGuid)
.QueryAsync(cancellationToken);
When querying the project system, utilize With
clauses to determine which parameters or metadata are included in the query results. There are several valid methods to specify which parameters should be included.
IAsyncEnumerable<IQueryResultItem<IProjectSnapshot>> allProjects = workSpace
.Projects
.With(p => p.Path)
.With(p => p.Guid)
.With(p => p.Kind) // DTE.Project.Kind
.With(p => p.Type) // VSHPROPID_ProjectType
.With(p => p.TypeGuid) // VSHPROPID_TypeGuid
.With(p => p.Capabilities)
.QueryAsync(cancellationToken);
await foreach (IQueryResultItem<IProjectSnapshot> project in allProjects)
{
var projectGuid = project.Value.Guid;
// Checking whether 'Capabilities' property has been retrieved.
// Otherwise, it can throw for projects which do not support it. (Like SQL projects)
bool capabilities = project.Value.PropertiesAvailableStatus.Capabilities;
}
You can also specify multiple desired parameters in a single With
clause.
IAsyncEnumerable<IQueryResultItem<IProjectSnapshot>> allProjects = workSpace
.Projects
.With(p => new { p.Path, p.Guid, p.Kind, p.Type, p.TypeGuid, p.Capabilities })
.QueryAsync(cancellationToken);
Using WithRequired
ensures that only projects containing specified properties are retrieved. For instance, in the following example, only projects that include files named information
are selected.
IAsyncEnumerable<IQueryResultItem<IProjectSnapshot>> projectWithFiles = workSpace
.Projects
.With(p => new { p.Path, p.Guid })
.WithRequired(p => p.Files.Where(f => f.FileName == "information"))
.QueryAsync(cancellationToken);
You can apply conditional filtering in two ways to limit query results:
Different project types support different sets of capabilities. With a Where
clause, you can filter for projects that support specific capabilities. Queries can fail if you don't filter to projects that support the relevant capabilities.
The following code returns the Path
and Guid
of all .NET Core web projects in the workspace:
IAsyncEnumerable<IQueryResultItem<IProjectSnapshot>> webProjects = workspace
.Projects
.Where(p => p.Capabilities.Contains("DotNetCoreWeb"))
.With(p => new { p.Path, p.Guid })
.QueryAsync(cancellationToken);
At the level of individual projects, each project possesses a RulesResults
attribute, which includes a RuleName
and Items
. The API call RuleResultsByRuleName
can be used to filter by rule name.
In the following query, rather than retrieving every rule present in ActiveConfigurations, we specifically target the CompilerCommandLineArgs rule. The query results will include both the rule name and the items associated with it.
var results = workSpace
.Projects
.With(p => p.Path)
.With(p => p.ActiveConfigurations
.With(c => c.RuleResultsByRuleName("CompilerCommandLineArgs")
.With(r => r.RuleName)
.With(r => r.Items
.With(i => i.Name))))
.QueryAsync();
You can also use query methods like ProjectsByCapabilities
that have filtering built into the query.
IAsyncEnumerable<IQueryResultItem<IProjectSnapshot>> webProjects = workspace
.ProjectsByCapabilities("DotNetCoreWeb | DotNetCoreRazor")
.With(p => new { p.Path, p.Guid })
.QueryAsync(cancellationToken);
While the With
clause returns the default set of properties, you can use PropertiesByName
to return results with the properties you're interested in.
IQueryResults<IPropertySnapshot> properties = myproject
.PropertiesByName("FileName", "LocalPath")
.QueryAsync();
Some parameters are collections themselves, and you can use nested queries to do similar specification and filtering on those child collections.
In the following example, a nested query lets you filter and specify the collection of files to be included with each project returned by the outer query.
The query detailed below yields an IProjectSnapshot for projects featuring the ApplicationIcon property. It specifically searches for .ico files within these projects, aiming to determine their paths and whether they are hidden.
IAsyncEnumerable<IQueryResultItem<IProjectSnapshot>> projects = workspace
.ProjectsByCapabilities("CPS")
.With(p => new { p.Path, p.IsProjectFileSearchable })
.With(p => p.PropertiesByName("ApplicationIcon")) // Retrieve a single property, if it exists
.With(p => p.Files // Without any condition, retrieve all files in the project, but filter them
.Where(f => f.Extension == ".ico")
.With(f => new { f.Path, f.IsHidden }))
.QueryAsync(cancellationToken);
await foreach (IQueryResultItem<IProjectSnapshot> project in projects)
{
IPropertySnapshot property = project.Value.Properties.FirstOrDefault();
string? applicationIcon = (string?)property?.Value;
foreach (var iconFile in project.Value.Files)
{
string filePath = iconFile.Path;
bool isHidden = iconFile.IsHidden;
}
}
The project model in Visual Studio includes collections for projects and their child collections, which can encompass files or project capabilities among others. To access a specific child collection, use the Get
clause. This clause, similar to other types of queries, allows for the incorporation of additional clauses like the With
clause, which helps refine or constrain the query results.
In the following query, the Get
method retrieves the child collection, Files
, from a project identified by its specific Guid.
IAsyncEnumerable<IQueryResultItem<IFileSnapshot>> files = querySpace
.Projects
.Where(p => p.Guid == knownGuid)
.Get(p => p.Files
.With(f => new { f.Path, f.IsHidden, f.IsSearchable }))
.QueryAsync();
await foreach (IQueryResultItem<IFileSnapshot> file in files)
{
...
}
You can use the results from a previous query as the base for additional queries.
IQueryResults<IProjectSnapshot> allProjects = querySpace
.Projects
.With(p => p.Path)
.With(p => p.Guid)
.ExecuteQueryAsync();
foreach (IProjectSnapshot project in allProjects)
{
// Gets child collections
IAsyncEnumerable<IQueryResultItem<IFileSnapshot>> files = project.Files
.With(f => new { f.Path, f.ItemType }).QueryAsync();
}
Query results are normally immutable. Use the query API's AsUpdatable
clause to access mutable versions of the query results, which will allow you to make changes to the projects and project items.
IQueryResult<IProjectSnapshot> updatedProjects = workSpace
.ProjectsByProjectGuid(knownGuid)
.AsUpdatable()
.CreateFile("AdditionalInformation.txt", textContent)
.ExecuteAsync(cancellationToken);
IQueryResult<IProjectSnapshot> updatedProjects = myproject
.AsUpdatable()
.CreateFile("AdditionalInformation2.txt", textContent)
.ExecuteAsync(cancellationToken);
IQueryResult<IProjectSnapshot> updatedProjects = myproject
.AsUpdatable()
.Rename("NewProjectName")
.ExecuteAsync(cancellationToken);
A Get
clause can query for project properties. The following query returns a collection of IPropertySnapshot
that contains entries for the two properties requested. IPropertySnapshot
contains the property name and value at a point in time.
This query asynchronously retrieves properties named RootNamespace
and AssemblyVersion
from a collection of projects. It operates on myProjects
, which is a previously obtained IProjectSnapshot
object. The query first transforms this collection into a queryable format using AsQueryable
, then identifies the specific properties to be retrieved using Get. Finally, it executes the query asynchronously with QueryAsync, incorporating a cancellation token to control the duration of the operation.
// We assume that we can find the "RootNamespace" property in the result.
// However it isn't true from query API point of view.
// The query tries to retrieve items based on the condition, and if there is no such item, it will run successfully, only without returning items.
IAsyncEnumerable<IQueryResultItem<IPropertySnapshot>> properties = myProjects
.AsQueryable()
.Get(p => p.PropertiesByName("RootNamespace", "AssemblyVersion"))
.QueryAsync(cancellationToken);
In addition to working with projects as shown previously, you can use similar techniques to work with solutions.
IAsyncEnumerable<IQueryResultItem<ISolutionSnapshot>> solutions = workSpace
.Solutions
.With(s => new { s.Path, s.Guid, s.ActiveConfiguration, s.ActivePlatform })
.QueryAsync(cancellationToken);
Likewise, you can use a Get
clause to query for solution folders. The IsNested
property lets you include or exclude nested folders from your results. Solution Explorer can have nested folders, such as for configuration setting or resources.
IAsyncEnumerable<IQueryResultItem<ISolutionFolderSnapshot>> solutionFolders = workSpace
.Solutions
.Get(s => s.SolutionFolders)
.With(folder => folder.Name)
.With(folder => folder.IsNested)
.With(folder => folder.VisualPath) // it's a relative (virtual) path to represent how the folder is nested.
.QueryAsync(cancellationToken);
This example gets all nested solution folders, projects, files inside a solution folder (not recursively nested):
IAsyncEnumerable<IQueryResultItem<ISolutionSnapshot>> solutionFoldersWithExtraInformation = mySolutionFolder
.AsQueryable()
.With(folder => folder.Files
.With(f => f.Path))
.With(folder => folder.Projects
.With(p => new { p.Name, p.Guid }))
.With(folder => folder.SolutionFolders
.With(nested => nested.Name))
.QueryAsync(cancellationToken);
This example gets all recursively nested solution folders. The VisualPath
is the path as it shows up in Solution Explorer.
string visualPath = mySolutionFolder.VisualPath;
IAsyncEnumerable<IQueryResultItem<ISolutionFolderSnapshot>> recursivelyNestedFolders = await workSpace
.Solutions
.Get(s => s.SolutionFolders)
.Where(f => f.VisualPath.StartsWith(visualPath) && f.VisualPath != visualPath)
.With(f => f.Name)
.QueryAsync(cancellationToken);
Here's an example enumerating all .xaml files in a project and its code generator:
IAsyncEnumerable<IQueryResultItem<IFileSnapshot>> files =
workSpace.ProjectsByProjectGuid(knownGuid)
.Get(p => p.Files)
.Where(file => file.Extension == ".xaml")
.With(file => file.Path)
.With(file => file.PropertiesByName("Generator"))
.QueryAsync(cancellationToken);
This example starts with a project returned from the previous query:
IAsyncEnumerable<IQueryResultItem<IFileSnapshot>> files = myProject
.FilesEndingWith(".xaml") // use built-in filter instead of 'Where' condition
.With(file => file.Path)
.With(file => file.PropertiesByName("Generator"))
.QueryAsync(cancellationToken);
Or to get all content files, which are non-compiled files that are required at runtime, such as HTML and CSS files:
IAsyncEnumerable<IQueryResultItem<IFileSnapshot>> files =
myProject.FilesWithItemTypes("Content")
.With(file => file.Path)
.QueryAsync(cancellationToken);
Or to enumerate all files with a certain extension, such as XML Schema files (.xsd
files) in all projects:
IAsyncEnumerable<IQueryResultItem<IFileSnapshot>> schemaFiles =
workSpace.Projects
.Get(proj => proj.FilesEndingWith(".xsd"))
.With(file => file.Path)
.QueryAsync(cancellationToken);
await foreach (IQueryResultItem<IFileSnapshot> fileResult in schemaFiles)
{
DoSomething(fileResult.Value.Path);
}
As projects and folders have information about which files they own or contain, you can use a WithRequired
clause to query for projects that include certain files.
string myFilePath = "c://my/file//path";
IAsyncEnumerable<IQueryResultItem<IProjectSnapshot>> projects = workspace
.Projects
.WithRequired(proj => proj.FilesByPath(myFilePath))
.With(proj => proj.Guid)
.QueryAsync(cancellationToken);
IAsyncEnumerable<IQueryResultItem<ISolutionFolderSnapshot>> solutionFolders = workspace
.Solutions
.Get(s => s.SolutionFolders)
.WithRequired(folder => folder.FilesByPath(myFilePath))
.With(folder => folder.Name)
.With(folder => folder.Guid)
.QueryAsync(cancellationToken);
Projects have a ConfigurationDimension
property, which you can use to find project configuration information. Project configuration information relates to project build configurations (for example, Debug
and Release
).
IAsyncEnumerable<IQueryResultItem<IProjectSnapshot>> projects = workspace
.Projects
.With(p => new { p.Guid, p.Name })
.With(p => p.Configurations
.With(c => c.Name)
.With(c => c.PropertiesByName("OutputPath"))
.With(c => c.ConfigurationDimensions)) // ConfigurationDimension is essentially Name, Value pairs, both are default properties.
.QueryAsync(cancellationToken);
await foreach (IQueryResultItem<IProjectSnapshot> project in projects)
{
foreach (var configuration in project.Value.Configuration)
{
// ...
}
}
You can also query to find projects that reference a given project.
IAsyncEnumerable<IQueryResultItem<IProjectReferenceSnapshot>> projectReferences = myProject
.ProjectReferences
.With(r => r.ProjectGuid)
.With(r => r.ReferencedProjectId)
.QueryAsync(cancellationToken);
IAsyncEnumerable<IQueryResultItem<IProjectSnapshot>> projects = workSpace
.Projects
.With(p => p.Guid)
.WithRequired(p => p.ProjectReferences
.Where(r => r.ProjectGuid == knownGuid))
.QueryAsync(cancellationToken);
Likewise, you can query for NuGet package references.
IAsyncEnumerable<IQueryResultItem<IProjectConfigurationSnapshot>> configurationsWithPackageReferences = myProject
.ActiveConfigurations
.With(c => c.Name)
.With(c => c.PackageReferences
.With(p => new { p.Name, p.Version }))
.QueryAsync(cancellationToken);
string packageName = "Newtonsoft.Json";
IAsyncEnumerable<IQueryResultItem<IProjectSnapshot>> projects = workSpace
.Projects
.With(p => p.Guid)
.WithRequired(p => p.ActiveConfigurations
.WithRequired(c => c.PackageReferences
.Where(package => package.Name == packageName)))
.QueryAsync(cancellationToken);
Project configurations have information about project output groups.
// From our list of active configurations, we need to get the first one in the list
IAsyncEnumerable<IQueryResultItem<IProjectConfigurationSnapshot>> configurations = myProject
.ActiveConfigurations
.QueryAsync(cancellationToken);
IProjectConfigurationSnapshot myConfiguration = null;
await foreach (IQueryResultItem<IProjectConfigurationSnapshot> config in configurations)
{
myConfiguration = config.Value;
break;
}
// A multi-target project may have multiple active configurations
IAsyncEnumerable<IQueryResultItem<IOutputGroupSnapshot>> outputGroups = myConfiguration
.OutputGroupsByName("Built", "Symbols")
.With(g => g.Name)
.With(g => g.Outputs)
.QueryAsync(cancellationToken);
The solution has a set of startup projects that can be executed as an executable.
// A query to get the list of startup project's name and path
var result = await workSpace.Solutions
.With(solution => solution.StartupProjects
.With(startupproject => startupproject.Name)
.With(startupproject => startupproject.Path))
.QueryAsync(CancellationToken.None);
Using the Project Query API, you also can select which projects get executed. The following sample shows how two project paths can be set as the startup projects.
// A query to set the startup project
var result = await workSpace.Solutions
.With(solution => solution.StartupProjects)
.AsUpdatable()
.SetStartupProjects("full\\path\\to\\project1.csproj",
"full\\path\\to\\project2.csproj")
.ExecuteAsync(CancellationToken.None);
A solution configuration is a collection of projects that are included in the build when the configuration is active. The example that follows shows how to query for the names of the solution configurations.
var result = workSpace.Solutions
.With(solution => solution.SolutionConfigurations
.With(solutionconfiguration => solutionconfiguration.Name))
.QueryAsync();
The AddSolutionConfiguration
method takes three parameters:
- The first parameter is the new name for the new solution configuration. In this example, the new solution configuration will be called
Foo
. - The next parameter is the configuration on which the new configuration should be based. Below, the new solution configuration is based on the existing solution configuration
Debug
. - Lastly, the boolean represents if the solution configuration should be propagated.
var result = await workSpace.Solutions
.AsUpdatable()
.AddSolutionConfiguration("Foo", "Debug", false)
.ExecuteAsync();
DeleteSolutionConfiguration
is an API call that removes the solution configuration. In the example below, The solution configuration called Foo
is removed.
var result = await workSpace.Solutions
.AsUpdatable()
.DeleteSolutionConfiguration("Foo")
.ExecuteAsync();
If loading/unloading a project is necessary, you need to specify the solution and the path to the desired project in question.
// Unload Project
var result = await workSpace.Solutions
.AsUpdatable()
.UnloadProject("full\\path\\to\\project.csproj")
.ExecuteAsync();
// Reload Project
var result = await querySpace.Solutions
.AsUpdatable()
.ReloadProject("full\\path\\to\\project.csproj")
.ExecuteAsync();
You can invoke build actions on the project or solution level. These build actions include:
BuildAsync
RebuildAsync
CleanAsync
DebugLaunchAsync
LaunchAsync
Building on the solution level will build all projects that are loaded into the solution. Below is an example of building a solution.
var result = await querySpace.Solutions
.BuildAsync(cancellationToken);
While building on the project level, determine the selected project you want to build. In the example below, myProject
is an IProjectSnapshot
that will be built.
var result = await myProject.BuildAsync(cancellationToken);
SaveAsync
can be used on the project or solution level.
var result = await querySpace.Solutions
.SaveAsync(cancellationToken);
myProject
is an IProjectSnapshot
of the target project to be saved.
var result = await myProject.SaveAsync(cancellationToken);
SubscribeAsync
can be used to track the most recent IQueryResults
in the project or solution.
In the example below, SubscribeAsync
will keep up to date with the solution's path and guid. The SubscribeObserver instance is passed in to receive notifications of changes.
var solutions = await workSpace.Solutions
.With(s => s.Path)
.With(s => s.Guid)
.SubscribeAsync(new SubscribeObserver(), CancellationToken.None);
The SubscribeObserver
is a component that implements IObserver interface and receives change notifications. For the example above, it would implement IObserver<IQueryResults<ISolutionSnapshot>>
.
private class SubscribeObserver : IObserver<IQueryResults<ISolutionSnapshot>>
{
public void OnCompleted()
{
...
}
public void OnError(Exception error)
{
...
}
public void OnNext(IQueryResults<ISolutionSnapshot> value)
{
...
}
public override int GetHashCode()
{
...
}
}
TrackUpdatesAsync
can be used to track all changes made in the project or solution.
In the example below, TrackUpdatesAsync
will track changes to files in the project. The TrackerObserver instance is passed in to receive notifications of changes.
var projects = await workSpace.Projects.ExecuteQueryAsync(cancellationToken: CancellationToken.None);
var singleProject = projects.FirstOrDefault();
var unsubscriber = await singleProject
.Files
.With(f => f.FileName)
.TrackUpdatesAsync(new TrackerObserver(), CancellationToken.None);
The TrackerObserver
is a component that implements IObserver interface and receives change notifications. For the example above, it would implement IObserver<IQueryTrackUpdates<IFileSnapshot>>
.
private class TrackerObserver : IObserver<IQueryTrackUpdates<IFileSnapshot>>
{
public void OnCompleted()
{
...
}
public void OnError(Exception error)
{
...
}
public void OnNext(IQueryTrackUpdates<IFileSnapshot> value)
{
...
}
public override int GetHashCode()
{
...
}
}
Skip
can be used to skip N results from a query.
In this code sample, the first result from the query is skipped. If there were three projects in the solution, for example, the first result is skipped, and the query will return the two remaining projects. The order is not guaranteed.
var projects = workSpace.Projects
.With(proj => proj.Name)
.Skip(1)
.QueryAsync();
To review keywords and concepts regarding the Project Query API, see Project Query Concepts.
Review the code for an extension that uses the Project Query API at VSProjectQueryAPISample.