/// This command creates a .NET project or other artifacts based on a template. The command calls the template engine to create the artifacts on disk based on the specified template and options.
///
-/// Args,
IEnumerable<(string name, string value)> Vars,
- string PackageName = "",
+ string Package = "",
string PackageVersion = "",
bool? ForceEnglishOutput = default,
string ApiKey = "",
@@ -1605,7 +1789,7 @@ public IStartInfo GetStartInfo(IHost host)
.WithVars(Vars.ToArray())
.AddArgs("nuget")
.AddArgs("delete")
- .AddNotEmptyArgs(PackageName.ToArg())
+ .AddNotEmptyArgs(Package.ToArg())
.AddNotEmptyArgs(PackageVersion.ToArg())
.AddArgs("--non-interactive")
.AddArgs(ApiKey.ToArgs("--api-key", ""))
@@ -1619,7 +1803,7 @@ public IStartInfo GetStartInfo(IHost host)
}
///
- public override string ToString() => (ExecutablePath == string.Empty ? "dotnet" : Path.GetFileNameWithoutExtension(ExecutablePath)).GetShortName("Deletes or unlists a package from the server.", ShortName, "nuget", "delete", PackageName.ToArg(), PackageVersion.ToArg());
+ public override string ToString() => (ExecutablePath == string.Empty ? "dotnet" : Path.GetFileNameWithoutExtension(ExecutablePath)).GetShortName("Deletes or unlists a package from the server.", ShortName, "nuget", "delete", Package.ToArg(), PackageVersion.ToArg());
}
///
@@ -3289,13 +3473,28 @@ public IStartInfo GetStartInfo(IHost host)
///
/// By default, dotnet pack builds the project first. If you wish to avoid this behavior, pass the --no-build option. This option is often useful in Continuous Integration (CI) build scenarios where you know the code was previously built.
///
+///
.NET CLI command
///
-///
-/// new DotNetPack()
+///
+/// // Adds the namespace "HostApi" to use .NET build API
+/// using HostApi;
+///
+/// // Creates a new library project, running a command like: "dotnet new classlib -n MyLib --force"
+/// var result = new DotNetNew()
+/// .WithTemplateName("classlib")
+/// .WithName("MyLib")
+/// .WithForce(true)
+/// .Build().EnsureSuccess();
+///
+/// result.ExitCode.ShouldBe(0);
+///
+/// // Creates a NuGet package of version 1.2.3 for the project, running a command like: "dotnet pack /p:version=1.2.3" from the directory "MyLib"
+/// result = new DotNetPack()
+/// .WithWorkingDirectory("MyLib")
+/// .AddProps(("version", "1.2.3"))
/// .Build().EnsureSuccess();
-///
+///
///
-///
.NET CLI command
///
/// Specifies the set of command line arguments to use when starting the tool.
/// Specifies the set of environment variables that apply to this process and its child processes.
@@ -3507,13 +3706,29 @@ public IStartInfo GetStartInfo(IHost host)
///
/// This command compiles the application, reads through its dependencies specified in the project file, and publishes the resulting set of files to a directory.
///
+///
.NET CLI command
///
-///
-/// new DotNetPublish().AddProps(("PublishDir", ".publish"))
+///
+/// // Adds the namespace "HostApi" to use .NET build API
+/// using HostApi;
+///
+/// // Creates a new library project, running a command like: "dotnet new classlib -n MyLib --force"
+/// var result = new DotNetNew()
+/// .WithTemplateName("classlib")
+/// .AddArgs("-f", "net8.0")
+/// .WithName("MyLib")
+/// .WithForce(true)
/// .Build().EnsureSuccess();
-///
+///
+/// result.ExitCode.ShouldBe(0);
+///
+/// // Publish the project, running a command like: "dotnet publish --framework net6.0" from the directory "MyLib"
+/// result = new DotNetPublish()
+/// .WithWorkingDirectory("MyLib")
+/// .WithFramework("net8.0")
+/// .Build().EnsureSuccess();
+///
///
-///
.NET CLI command
///
/// Specifies the set of command line arguments to use when starting the tool.
/// Specifies the set of environment variables that apply to this process and its child processes.
@@ -3644,13 +3859,27 @@ public IStartInfo GetStartInfo(IHost host)
///
/// Sometimes, it might be inconvenient to run the implicit NuGet restore with these commands. For example, some automated systems, such as build systems, need to call dotnet restore explicitly to control when the restore occurs so that they can control network usage. To prevent the implicit NuGet restore, you can use the --no-restore flag with any of these commands.
///
+///
.NET CLI command
///
-///
-/// new DotNetRestore()
+///
+/// // Adds the namespace "HostApi" to use .NET build API
+/// using HostApi;
+///
+/// // Creates a new library project, running a command like: "dotnet new classlib -n MyLib --force"
+/// var result = new DotNetNew()
+/// .WithTemplateName("classlib")
+/// .WithName("MyLib")
+/// .WithForce(true)
+/// .Build().EnsureSuccess();
+///
+/// result.ExitCode.ShouldBe(0);
+///
+/// // Restore the project, running a command like: "dotnet restore" from the directory "MyLib"
+/// result = new DotNetRestore()
+/// .WithWorkingDirectory("MyLib")
/// .Build().EnsureSuccess();
-///
+///
///
-///
.NET CLI command
///
/// Specifies the set of command line arguments to use when starting the tool.
/// Specifies the set of environment variables that apply to this process and its child processes.
@@ -3772,20 +4001,34 @@ public IStartInfo GetStartInfo(IHost host)
///
/// To run the application, the dotnet run command resolves the dependencies of the application that are outside of the shared runtime from the NuGet cache. Because it uses cached dependencies, it's not recommended to use dotnet run to run applications in production. Instead, create a deployment using the dotnet publish command and deploy the published output.
///
+///
.NET CLI command
///
-///
-/// new DotNetNew()
+///
+/// // Adds the namespace "HostApi" to use .NET build API
+/// using HostApi;
+///
+/// // Creates a new console project, running a command like: "dotnet new console -n MyApp --force"
+/// var result = new DotNetNew()
/// .WithTemplateName("console")
/// .WithName("MyApp")
/// .WithForce(true)
-/// .Run().EnsureSuccess();
+/// .Build().EnsureSuccess();
///
+/// result.ExitCode.ShouldBe(0);
///
-/// new DotNetRun().WithWorkingDirectory("MyApp")
-/// .Build().EnsureSuccess();
-///
+/// // Runs the console project using a command like: "dotnet run" from the directory "MyApp"
+/// var stdOut = new List<string>();
+/// result = new DotNetRun()
+/// .WithWorkingDirectory("MyApp")
+/// .Build(message => stdOut.Add(message.Text))
+/// .EnsureSuccess();
+///
+/// result.ExitCode.ShouldBe(0);
+///
+/// // Checks StdOut
+/// stdOut.ShouldBe(new[] {"Hello, World!"});
+///
///
-///
.NET CLI command
///
/// Specifies the set of command line arguments to use when starting the tool.
/// Specifies the set of environment variables that apply to this process and its child processes.
@@ -4224,20 +4467,33 @@ public IStartInfo GetStartInfo(IHost host)
///
/// For multi-targeted projects, tests are run for each targeted framework. The test host and the unit test framework are packaged as NuGet packages and are restored as ordinary dependencies for the project. Starting with the .NET 9 SDK, these tests are run in parallel by default. To disable parallel execution, set the TestTfmsInParallel MSBuild property to false.
///
+///
.NET CLI command
///
-///
-/// new DotNetNew()
+///
+/// // Adds the namespace "HostApi" to use .NET build API
+/// using HostApi;
+///
+/// // Creates a new test project, running a command like: "dotnet new mstest -n MyTests --force"
+/// var result = new DotNetNew()
/// .WithTemplateName("mstest")
/// .WithName("MyTests")
/// .WithForce(true)
-/// .Run().EnsureSuccess();
+/// .Build().EnsureSuccess();
///
+/// result.ExitCode.ShouldBe(0);
///
-/// new DotNetTest().WithWorkingDirectory("MyTests")
-/// .Build().EnsureSuccess();
-///
+/// // Runs tests via a command like: "dotnet test" from the directory "MyTests"
+/// result = new DotNetTest()
+/// .WithWorkingDirectory("MyTests")
+/// .Build()
+/// .EnsureSuccess();
+///
+/// // The "result" variable provides details about a build
+/// result.ExitCode.ShouldBe(0, result.ToString());
+/// result.Summary.Tests.ShouldBe(1, result.ToString());
+/// result.Tests.Count(test => test.State == TestState.Finished).ShouldBe(1, result.ToString());
+///
///
-///
.NET CLI command
///
/// Specifies the set of command line arguments to use when starting the tool.
/// Specifies the set of environment variables that apply to this process and its child processes.
@@ -4569,6 +4825,28 @@ public IStartInfo GetStartInfo(IHost host)
/// This command finds the tool manifest file that is in scope for the current directory and installs the tools that are listed in it. For information about manifest files, see Install a local tool and Invoke a local tool.
///
///
.NET CLI command
+///
+///
+/// // Adds the namespace "HostApi" to use .NET build API
+/// using HostApi;
+///
+/// var projectDir = Path.Combine(Path.GetTempPath(), Guid.NewGuid().ToString()[..4]);
+/// Directory.CreateDirectory(projectDir);
+///
+/// // Creates a local tool manifest
+/// new DotNetNew()
+/// .WithTemplateName("tool-manifest")
+/// .WithWorkingDirectory(projectDir)
+/// .Run()
+/// .EnsureSuccess();
+///
+/// // Restore local tools
+/// new DotNetToolRestore()
+/// .WithWorkingDirectory(projectDir)
+/// .Run()
+/// .EnsureSuccess();
+///
+///
///
/// Specifies the set of command line arguments to use when starting the tool.
/// Specifies the set of environment variables that apply to this process and its child processes.
@@ -5636,6 +5914,27 @@ public IStartInfo GetStartInfo(IHost host)
///
/// Runs a C# script.
///
C# interactive
+///
+///
+/// // Adds the namespace "HostApi" to use .NET build API
+/// using HostApi;
+///
+/// var script = Path.GetTempFileName();
+/// File.WriteAllText(script, "Console.WriteLine($\"Hello, {Args[0]}!\");");
+///
+/// var stdOut = new List<string>();
+/// var result = new DotNetCsi()
+/// .WithScript(script)
+/// .AddArgs("World")
+/// .Build(message => stdOut.Add(message.Text))
+/// .EnsureSuccess();
+///
+/// result.ExitCode.ShouldBe(0);
+///
+/// // Checks StdOut
+/// stdOut.Contains("Hello, World!").ShouldBeTrue(result.ToString());
+///
+///
///
/// Specifies the set of command line arguments to use when starting the tool.
/// Specifies the set of environment variables that apply to this process and its child processes.
diff --git a/CSharpInteractive.HostApi/DotNetCommands.tt b/CSharpInteractive.HostApi/DotNetCommands.tt
index b527e0a9..161d5fb3 100644
--- a/CSharpInteractive.HostApi/DotNetCommands.tt
+++ b/CSharpInteractive.HostApi/DotNetCommands.tt
@@ -1,7 +1,7 @@
<#@ import namespace="System.Linq" #>
-<#@ import namespace="System.Linq" #>
-<#@ import namespace="System.Linq" #>
<#@ import namespace="System.Collections.Generic" #>
+<#@ import namespace="System.IO" #>
+<#@ import namespace="System.Xml" #>
// ReSharper disable UnusedMember.Global
// ReSharper disable InconsistentNaming
namespace HostApi;
@@ -17,7 +17,7 @@ using Internal;
string CreateCliRef(string command) =>
$"
.NET CLI command
";
-
+
var projectArg = new Arg("Project", "", "string", "The project or solution file to operate on. If not specified, the command searches the current directory for one. If more than one solution or project is found, an error is thrown.") { IsProject = true };
var solutionArg = new Arg("Solution", "", "string", "The solution file to use. If this argument is omitted, the command searches the current directory for one. If it finds no solution file or multiple solution files, the command fails.") { IsProject = true };
var propsArg = new Arg("Props", "--property", "IEnumerable<(string name, string value)>", "MSBuild options for setting properties.") { IsCollection = true };
@@ -116,11 +116,11 @@ using Internal;
paraFinish,
CreateCliRef("dotnet-add-package")
],
- ["add", "$Project", "package", "$PackageName"],
+ ["add", "$Project", "package", "$Package"],
[
sourcesArg,
projectArg,
- new Arg("PackageName", "", "string", "The package reference to add."),
+ new Arg("Package", "", "string", "The package reference to add."),
frameworkArg with { Comments = "Adds a package reference only when targeting a specific framework." },
noRestoreArg,
new Arg("PackageDirectory", "--package-directory", "string", @"The directory where to restore the packages. The default package restore location is %userprofile%\.nuget\packages on Windows and ~/.nuget/packages on macOS and Linux."),
@@ -163,10 +163,10 @@ using Internal;
paraFinish,
CreateCliRef("dotnet-remove-package")
],
- ["remove", "$Project", "package", "$PackageName"],
+ ["remove", "$Project", "package", "$Package"],
[
projectArg,
- new Arg("PackageName", "", "string", "The package reference to add."),
+ new Arg("Package", "", "string", "The package reference to add."),
diagnosticsArg
]
),
@@ -229,15 +229,6 @@ using Internal;
paraStart,
"For executable projects targeting .NET Core 3.0 and later, library dependencies are copied to the output folder. This means that if there isn't any other publish-specific logic (such as Web projects have), the build output should be deployable.",
paraFinish,
- exampleStart,
- codeStart,
- "var configuration = Props.Get(\"configuration\", \"Release\");",
- "",
- "",
- "new DotNetBuild().WithConfiguration(configuration)",
- " .Build().EnsureSuccess();",
- codeFinish,
- exampleFinish,
CreateCliRef("dotnet-build")
],
["build", "$Project"],
@@ -344,15 +335,6 @@ using Internal;
paraStart,
"This command creates a .NET project or other artifacts based on a template. The command calls the template engine to create the artifacts on disk based on the specified template and options.",
paraFinish,
- exampleStart,
- codeStart,
- "new DotNetNew()",
- " .WithTemplateName(\"console\")",
- " .WithName(\"MyApp\")",
- " .WithForce(true)",
- " .Run().EnsureSuccess();",
- codeFinish,
- exampleFinish,
CreateCliRef("dotnet-new")
],
["new", "$TemplateName"],
@@ -498,9 +480,9 @@ using Internal;
paraFinish,
CreateCliRef("dotnet-nuget-delete")
],
- ["nuget", "delete", "$PackageName", "$PackageVersion"],
+ ["nuget", "delete", "$Package", "$PackageVersion"],
[
- new Arg("PackageName", "", "string", "Name/ID of the package to delete.") { IsProject = true },
+ new Arg("Package", "", "string", "Name/ID of the package to delete.") { IsProject = true },
new Arg("PackageVersion", "", "string", "Version of the package to delete.") { IsProject = true },
new Arg("ForceEnglishOutput", "--force-english-output", "bool?", "Forces the application to run using an invariant, English-based culture."),
new Arg("ApiKey", "--api-key", "string", "The API key for the server."),
@@ -931,12 +913,6 @@ using Internal;
paraStart,
"By default, dotnet pack builds the project first. If you wish to avoid this behavior, pass the --no-build option. This option is often useful in Continuous Integration (CI) build scenarios where you know the code was previously built.",
paraFinish,
- exampleStart,
- codeStart,
- "new DotNetPack()",
- " .Build().EnsureSuccess();",
- codeFinish,
- exampleFinish,
CreateCliRef("dotnet-pack")
],
["pack", "$Project"],
@@ -997,12 +973,6 @@ using Internal;
paraStart,
"This command compiles the application, reads through its dependencies specified in the project file, and publishes the resulting set of files to a directory.",
paraFinish,
- exampleStart,
- codeStart,
- "new DotNetPublish().AddProps((\"PublishDir\", \".publish\"))",
- " .Build().EnsureSuccess();",
- codeFinish,
- exampleFinish,
CreateCliRef("dotnet-publish")
],
["publish", "$Project"],
@@ -1044,12 +1014,6 @@ using Internal;
paraStart,
"Sometimes, it might be inconvenient to run the implicit NuGet restore with these commands. For example, some automated systems, such as build systems, need to call dotnet restore explicitly to control when the restore occurs so that they can control network usage. To prevent the implicit NuGet restore, you can use the --no-restore flag with any of these commands.",
paraFinish,
- exampleStart,
- codeStart,
- "new DotNetRestore()",
- " .Build().EnsureSuccess();",
- codeFinish,
- exampleFinish,
CreateCliRef("dotnet-restore")
],
["restore", "$Project"],
@@ -1088,19 +1052,6 @@ using Internal;
paraStart,
"To run the application, the dotnet run command resolves the dependencies of the application that are outside of the shared runtime from the NuGet cache. Because it uses cached dependencies, it's not recommended to use dotnet run to run applications in production. Instead, create a deployment using the dotnet publish command and deploy the published output.",
paraFinish,
- exampleStart,
- codeStart,
- "new DotNetNew()",
- " .WithTemplateName(\"console\")",
- " .WithName(\"MyApp\")",
- " .WithForce(true)",
- " .Run().EnsureSuccess();",
- "",
- "",
- "new DotNetRun().WithWorkingDirectory(\"MyApp\")",
- " .Build().EnsureSuccess();",
- codeFinish,
- exampleFinish,
CreateCliRef("dotnet-run")
],
["run"],
@@ -1210,19 +1161,6 @@ using Internal;
paraStart,
"For multi-targeted projects, tests are run for each targeted framework. The test host and the unit test framework are packaged as NuGet packages and are restored as ordinary dependencies for the project. Starting with the .NET 9 SDK, these tests are run in parallel by default. To disable parallel execution, set the TestTfmsInParallel MSBuild property to false.",
paraFinish,
- exampleStart,
- codeStart,
- "new DotNetNew()",
- " .WithTemplateName(\"mstest\")",
- " .WithName(\"MyTests\")",
- " .WithForce(true)",
- " .Run().EnsureSuccess();",
- "",
- "",
- "new DotNetTest().WithWorkingDirectory(\"MyTests\")",
- " .Build().EnsureSuccess();",
- codeFinish,
- exampleFinish,
CreateCliRef("dotnet-test")
],
["test", "$Project"],
@@ -1636,6 +1574,33 @@ using Internal;
/// <#= comment #>
<#
}
+
+ var exampleFile = Path.Combine("..", "CSharpInteractive.Tests", "UsageScenarios", "Comments", $"{command.Name}Scenario.txt");
+ if (File.Exists(exampleFile))
+ {
+ var exampleLines = File.ReadAllLines(exampleFile);
+ if (exampleLines.Length > 0)
+ {
+#>
+/// <#= exampleStart #>
+///<#= codeStart #>
+<#
+ var doc = new XmlDocument();
+ foreach (var exampleLine in exampleLines.Select(i => i.TrimEnd()))
+ {
+ var node = doc.CreateElement("root");
+ node.InnerText = exampleLine;
+ var line = node.InnerXml;
+#>
+/// <#= line #>
+<#
+ }
+#>
+///<#= codeFinish #>
+/// <#= exampleFinish #>
+<#
+ }
+ }
#>
///
/// Specifies the set of command line arguments to use when starting the tool.
diff --git a/CSharpInteractive.Tests/README_TEMPLATE.md b/CSharpInteractive.Tests/README_TEMPLATE.md
index 96051856..5e8325c9 100644
--- a/CSharpInteractive.Tests/README_TEMPLATE.md
+++ b/CSharpInteractive.Tests/README_TEMPLATE.md
@@ -31,19 +31,25 @@
- [Build a project in a docker container](#build-a-project-in-a-docker-container)
- [Running in docker](#running-in-docker)
- .NET CLI
+ - [Adding a NuGet package](#adding-a-nuget-package)
- [Build a project](#build-a-project)
- [Build a project using MSBuild](#build-a-project-using-msbuild)
- [Clean a project](#clean-a-project)
+ - [Execute a dotnet application](#execute-a-dotnet-application)
+ - [NuGet package listing](#nuget-package-listing)
- [Pack a project](#pack-a-project)
- [Publish a project](#publish-a-project)
+ - [Removing a NuGet package](#removing-a-nuget-package)
- [Restore a project](#restore-a-project)
- [Restore local tools](#restore-local-tools)
- [Run a custom .NET command](#run-a-custom-.net-command)
+ - [Run a dotnet application](#run-a-dotnet-application)
- [Run a project](#run-a-project)
- [Run tests under dotCover](#run-tests-under-dotcover)
- [Test a project](#test-a-project)
- [Test a project using the MSBuild VSTest target](#test-a-project-using-the-msbuild-vstest-target)
- [Test an assembly](#test-an-assembly)
+ - [Run C# script](#run-c#-script)
- [Shuts down build servers](#shuts-down-build-servers)
- TeamCity API
- [TeamCity integration via service messages](#teamcity-integration-via-service-messages)
@@ -411,8 +417,6 @@ int? exitCode = new CommandLine("cmd", "/c", "TIMEOUT", "/T", "120")
.Run(default, TimeSpan.FromMilliseconds(1))
.EnsureSuccess()
.ExitCode;
-
-exitCode.HasValue.ShouldBeFalse();
```
@@ -447,10 +451,6 @@ var result = dockerRun
.Build()
.EnsureSuccess();
-// The "result" variable provides details about a build
-result.Errors.Any(message => message.State == BuildMessageState.StdError).ShouldBeFalse();
-result.ExitCode.ShouldBe(0);
-
string ToAbsoluteLinuxPath(string path) =>
"/" + path.Replace(":", "").Replace('\\', '/');
```
@@ -477,6 +477,38 @@ var result = new DockerRun(cmd, "mcr.microsoft.com/dotnet/sdk")
+### Adding a NuGet package
+
+
+
+``` CSharp
+ using HostApi;
+
+ new DotNetNew()
+ .WithTemplateName("classlib")
+ .WithName("MyLib")
+ .WithForce(true)
+ .Build().EnsureSuccess();
+
+ var result = new DotNetAddPackage()
+ .WithWorkingDirectory("MyLib")
+ .WithPackage("Pure.DI")
+ .Run().EnsureSuccess();
+
+ var lines = new List();
+ new DotNetListPackage()
+ .WithWorkingDirectory("MyLib")
+ .WithVerbosity(DotNetVerbosity.Minimal)
+ .Run(output => lines.Add(output.Line));
+
+ result.ExitCode.ShouldBe(0, result.ToString());
+ lines.Any(i => i.Contains("Pure.DI")).ShouldBeTrue(result.ToString());
+ }
+}
+```
+
+
+
### Build a project
@@ -485,31 +517,28 @@ var result = new DockerRun(cmd, "mcr.microsoft.com/dotnet/sdk")
// Adds the namespace "HostApi" to use .NET build API
using HostApi;
-// Creates a new library project, running a command like: "dotnet new classlib -n MyLib --force"
+// Creates a new test project, running a command like: "dotnet new xunit -n MyTests --force"
new DotNetNew()
.WithTemplateName("xunit")
- .WithName("MyLib")
+ .WithName("MyTests")
.WithForce(true)
.Build().EnsureSuccess();
-// Builds the library project, running a command like: "dotnet build" from the directory "MyLib"
+// Builds this project, running a command like: "dotnet build" from the directory "MyLib"
+var messages = new List();
var result = new DotNetBuild()
- .WithWorkingDirectory("MyLib")
- .Build().EnsureSuccess();
+ .WithWorkingDirectory("MyTests")
+ .Build(message => messages.Add(message)).EnsureSuccess();
// The "result" variable provides details about a build
-result.Errors.Any(message => message.State == BuildMessageState.StdError).ShouldBeFalse();
-result.ExitCode.ShouldBe(0);
+messages.Count.ShouldBeGreaterThan(0, result.ToString());
+result.Errors.Any(message => message.State == BuildMessageState.StdError).ShouldBeFalse(result.ToString());
+result.ExitCode.ShouldBe(0, result.ToString());
// Runs tests in docker
result = new DotNetTest()
- .WithWorkingDirectory("MyLib")
- .Build()
- .EnsureSuccess();
-
-result.ExitCode.ShouldBe(0);
-result.Summary.Tests.ShouldBe(1);
-result.Tests.Count(test => test.State == TestState.Finished).ShouldBe(1);
+ .WithWorkingDirectory("MyTests")
+ .Build().EnsureSuccess();
```
@@ -568,6 +597,66 @@ version.ShouldNotBeNull();
+### Execute a dotnet application
+
+
+
+``` CSharp
+// Adds the namespace "HostApi" to use .NET build API
+using HostApi;
+
+// Creates a new console project, running a command like: "dotnet new console -n MyApp --force"
+new DotNetNew()
+ .WithTemplateName("console")
+ .WithName("MyApp")
+ .WithForce(true)
+ .Build().EnsureSuccess();
+
+var tempDirectory = Path.Combine(Path.GetTempPath(), Path.GetRandomFileName());
+new DotNetPublish()
+ .WithWorkingDirectory("MyApp")
+ .WithOutput(tempDirectory)
+ .Build().EnsureSuccess();
+
+new DotNetExec()
+ .WithPathToApplication(Path.Combine(tempDirectory, "MyApp.dll"))
+ .Run().EnsureSuccess();
+```
+
+
+
+### NuGet package listing
+
+
+
+``` CSharp
+ using HostApi;
+
+ new DotNetNew()
+ .WithTemplateName("classlib")
+ .WithName("MyLib")
+ .WithForce(true)
+ .Build().EnsureSuccess();
+
+ new DotNetAddPackage()
+ .WithWorkingDirectory("MyLib")
+ .WithPackage("Pure.DI")
+ .Run().EnsureSuccess();
+
+ var lines = new List();
+ var result = new DotNetListPackage()
+ .WithWorkingDirectory("MyLib")
+ .WithVerbosity(DotNetVerbosity.Minimal)
+ .Run(output => lines.Add(output.Line));
+
+ result.ExitCode.ShouldBe(0, result.ToString());
+ lines.Any(i => i.Contains("Pure.DI")).ShouldBeTrue(result.ToString());
+ }
+}
+```
+
+
+
### Test a project using the MSBuild VSTest target
@@ -621,8 +710,6 @@ result = new DotNetPack()
.WithWorkingDirectory("MyLib")
.AddProps(("version", "1.2.3"))
.Build().EnsureSuccess();
-
-result.ExitCode.ShouldBe(0);
```
@@ -650,8 +737,43 @@ result = new DotNetPublish()
.WithWorkingDirectory("MyLib")
.WithFramework("net8.0")
.Build().EnsureSuccess();
+```
-result.ExitCode.ShouldBe(0);
+
+
+### Removing a NuGet package
+
+
+
+``` CSharp
+ using HostApi;
+
+ new DotNetNew()
+ .WithTemplateName("classlib")
+ .WithName("MyLib")
+ .WithForce(true)
+ .Build().EnsureSuccess();
+
+ new DotNetAddPackage()
+ .WithWorkingDirectory("MyLib")
+ .WithPackage("Pure.DI")
+ .Run().EnsureSuccess();
+
+ var result = new DotNetRemovePackage()
+ .WithWorkingDirectory("MyLib")
+ .WithPackage("Pure.DI")
+ .Run().EnsureSuccess();
+
+ var lines = new List();
+ new DotNetListPackage()
+ .WithWorkingDirectory("MyLib")
+ .WithVerbosity(DotNetVerbosity.Minimal)
+ .Run(output => lines.Add(output.Line));
+
+ result.ExitCode.ShouldBe(0, result.ToString());
+ lines.Any(i => i.Contains("Pure.DI")).ShouldBeFalse(result.ToString());
+ }
+}
```
@@ -677,8 +799,6 @@ result.ExitCode.ShouldBe(0);
result = new DotNetRestore()
.WithWorkingDirectory("MyLib")
.Build().EnsureSuccess();
-
-result.ExitCode.ShouldBe(0);
```
@@ -702,7 +822,8 @@ result.ExitCode.ShouldBe(0);
// Runs the console project using a command like: "dotnet run" from the directory "MyApp"
var stdOut = new List();
-result = new DotNetRun().WithWorkingDirectory("MyApp")
+result = new DotNetRun()
+ .WithWorkingDirectory("MyApp")
.Build(message => stdOut.Add(message.Text))
.EnsureSuccess();
@@ -714,6 +835,34 @@ stdOut.ShouldBe(new[] {"Hello, World!"});
+### Run a dotnet application
+
+
+
+``` CSharp
+// Adds the namespace "HostApi" to use .NET build API
+using HostApi;
+
+// Creates a new console project, running a command like: "dotnet new console -n MyApp --force"
+new DotNetNew()
+ .WithTemplateName("console")
+ .WithName("MyApp")
+ .WithForce(true)
+ .Build().EnsureSuccess();
+
+var tempDirectory = Path.Combine(Path.GetTempPath(), Path.GetRandomFileName());
+new DotNetPublish()
+ .WithWorkingDirectory("MyApp")
+ .WithOutput(tempDirectory)
+ .Build().EnsureSuccess();
+
+new DotNet()
+ .WithPathToApplication(Path.Combine(tempDirectory, "MyApp.dll"))
+ .Run().EnsureSuccess();
+```
+
+
+
### Test a project
@@ -738,9 +887,9 @@ result = new DotNetTest()
.EnsureSuccess();
// The "result" variable provides details about a build
-result.ExitCode.ShouldBe(0);
-result.Summary.Tests.ShouldBe(1);
-result.Tests.Count(test => test.State == TestState.Finished).ShouldBe(1);
+result.ExitCode.ShouldBe(0, result.ToString());
+result.Summary.Tests.ShouldBe(1, result.ToString());
+result.Tests.Count(test => test.State == TestState.Finished).ShouldBe(1, result.ToString());
```
@@ -792,8 +941,8 @@ var result = testUnderDotCover
.Build().EnsureSuccess();
// The "result" variable provides details about a build
-result.ExitCode.ShouldBe(0);
-result.Tests.Count(i => i.State == TestState.Finished).ShouldBe(1);
+result.ExitCode.ShouldBe(0, result.ToString());
+result.Tests.Count(i => i.State == TestState.Finished).ShouldBe(1, result.ToString());
// Generates a HTML code coverage report.
new DotNetCustom("dotCover", "report", $"--source={dotCoverSnapshot}", $"--output={dotCoverReport}", "--reportType=HTML")
@@ -847,7 +996,7 @@ var result = new DotNetNew()
.WithForce(true)
.Build().EnsureSuccess();
-result.ExitCode.ShouldBe(0);
+result.ExitCode.ShouldBe(0, result.ToString());
// Builds the test project, running a command like: "dotnet build -c Release" from the directory "MyTests"
result = new DotNetBuild()
@@ -856,7 +1005,7 @@ result = new DotNetBuild()
.WithOutput("MyOutput")
.Build().EnsureSuccess();
-result.ExitCode.ShouldBe(0);
+result.ExitCode.ShouldBe(0, result.ToString());
// Runs tests via a command like: "dotnet vstest" from the directory "MyTests"
result = new VSTest()
@@ -865,9 +1014,9 @@ result = new VSTest()
.Build().EnsureSuccess();
// The "result" variable provides details about a build
-result.ExitCode.ShouldBe(0);
-result.Summary.Tests.ShouldBe(1);
-result.Tests.Count(test => test.State == TestState.Finished).ShouldBe(1);
+result.ExitCode.ShouldBe(0, result.ToString());
+result.Summary.Tests.ShouldBe(1, result.ToString());
+result.Tests.Count(test => test.State == TestState.Finished).ShouldBe(1, result.ToString());
```
@@ -887,7 +1036,7 @@ var result = new DotNetNew()
.WithForce(true)
.Build().EnsureSuccess();
-result.ExitCode.ShouldBe(0);
+result.ExitCode.ShouldBe(0, result.ToString());
// Builds the library project, running a command like: "dotnet msbuild /t:Build -restore /p:configuration=Release -verbosity=detailed" from the directory "MyLib"
result = new MSBuild()
@@ -899,8 +1048,8 @@ result = new MSBuild()
.Build().EnsureSuccess();
// The "result" variable provides details about a build
-result.Errors.Any(message => message.State == BuildMessageState.StdError).ShouldBeFalse();
-result.ExitCode.ShouldBe(0);
+result.Errors.Any(message => message.State == BuildMessageState.StdError).ShouldBeFalse(result.ToString());
+result.ExitCode.ShouldBe(0, result.ToString());
```
@@ -921,6 +1070,32 @@ new DotNetBuildServerShutdown()
+### Run C# script
+
+
+
+``` CSharp
+// Adds the namespace "HostApi" to use .NET build API
+using HostApi;
+
+var script = Path.GetTempFileName();
+File.WriteAllText(script, "Console.WriteLine($\"Hello, {Args[0]}!\");");
+
+var stdOut = new List();
+var result = new DotNetCsi()
+ .WithScript(script)
+ .AddArgs("World")
+ .Build(message => stdOut.Add(message.Text))
+ .EnsureSuccess();
+
+result.ExitCode.ShouldBe(0);
+
+// Checks StdOut
+stdOut.Contains("Hello, World!").ShouldBeTrue(result.ToString());
+```
+
+
+
### TeamCity integration via service messages
For more details how to use TeamCity service message API please see [this](https://github.com/JetBrains/TeamCity.ServiceMessages) page. Instead of creating a root message writer like in the following example:
diff --git a/CSharpInteractive.Tests/README_TEMPLATE.tt b/CSharpInteractive.Tests/README_TEMPLATE.tt
index 4ad63b1b..f0f995a4 100644
--- a/CSharpInteractive.Tests/README_TEMPLATE.tt
+++ b/CSharpInteractive.Tests/README_TEMPLATE.tt
@@ -12,6 +12,19 @@
var solutionPath = Path.Combine(currentPath, "..");
var infos = new List>();
+ var commentsDir = Path.Combine(scenariosPath, "Comments");
+ if (Directory.Exists(commentsDir))
+ {
+ foreach (var file in Directory.GetFiles(commentsDir, "*.txt"))
+ {
+ File.Delete(file);
+ }
+ }
+ else
+ {
+ Directory.CreateDirectory(commentsDir);
+ }
+
foreach (var file in Directory.GetFiles(scenariosPath, "*.cs"))
{
var isBody = false;
@@ -127,12 +140,15 @@
<#
foreach (var info in infos.OrderBy(i => i["tag"] + i["priority"]).Where(info => info["visible"] == "true"))
{
+ var file = info["file"];
+ var body = info["body"];
+ File.WriteAllText(Path.Combine(commentsDir, Path.GetFileNameWithoutExtension(file) + ".txt"), body);
#>### <#= info["description"] #>
<#= info["header"] #>
``` CSharp
-<#= info["body"] #>```
+<#= body #>```
<#= info["footer"] #>
diff --git a/CSharpInteractive.Tests/UsageScenarios/Comments/ArgsScenario.txt b/CSharpInteractive.Tests/UsageScenarios/Comments/ArgsScenario.txt
new file mode 100644
index 00000000..8f0d4ed1
--- /dev/null
+++ b/CSharpInteractive.Tests/UsageScenarios/Comments/ArgsScenario.txt
@@ -0,0 +1,9 @@
+if (Args.Count > 0)
+{
+ WriteLine(Args[0]);
+}
+
+if (Args.Count > 1)
+{
+ WriteLine(Args[1]);
+}
diff --git a/CSharpInteractive.Tests/UsageScenarios/Comments/CommandLineAsyncCancellation.txt b/CSharpInteractive.Tests/UsageScenarios/Comments/CommandLineAsyncCancellation.txt
new file mode 100644
index 00000000..20f98db1
--- /dev/null
+++ b/CSharpInteractive.Tests/UsageScenarios/Comments/CommandLineAsyncCancellation.txt
@@ -0,0 +1,9 @@
+// Adds the namespace "HostApi" to use Command Line API
+using HostApi;
+
+var cancellationTokenSource = new CancellationTokenSource();
+var task = new CommandLine("cmd", "/c", "TIMEOUT", "/T", "120")
+ .RunAsync(default, cancellationTokenSource.Token);
+
+cancellationTokenSource.CancelAfter(TimeSpan.FromMilliseconds(100));
+task.IsCompleted.ShouldBeFalse();
diff --git a/CSharpInteractive.Tests/UsageScenarios/Comments/CommandLineAsyncScenario.txt b/CSharpInteractive.Tests/UsageScenarios/Comments/CommandLineAsyncScenario.txt
new file mode 100644
index 00000000..1362e5d9
--- /dev/null
+++ b/CSharpInteractive.Tests/UsageScenarios/Comments/CommandLineAsyncScenario.txt
@@ -0,0 +1,11 @@
+// Adds the namespace "HostApi" to use Command Line API
+using HostApi;
+
+await GetService()
+ .RunAsync(new CommandLine("cmd", "/C", "DIR"))
+ .EnsureSuccess();
+
+// or the same thing using the extension method
+var result = await new CommandLine("cmd", "/c", "DIR")
+ .RunAsync()
+ .EnsureSuccess();
diff --git a/CSharpInteractive.Tests/UsageScenarios/Comments/CommandLineInParallelScenario.txt b/CSharpInteractive.Tests/UsageScenarios/Comments/CommandLineInParallelScenario.txt
new file mode 100644
index 00000000..f64e01e8
--- /dev/null
+++ b/CSharpInteractive.Tests/UsageScenarios/Comments/CommandLineInParallelScenario.txt
@@ -0,0 +1,12 @@
+// Adds the namespace "HostApi" to use Command Line API
+using HostApi;
+
+var task = new CommandLine("cmd", "/c", "DIR")
+ .RunAsync()
+ .EnsureSuccess();
+
+var result = new CommandLine("cmd", "/c", "SET")
+ .Run()
+ .EnsureSuccess();
+
+await task;
diff --git a/CSharpInteractive.Tests/UsageScenarios/Comments/CommandLineOutputScenario.txt b/CSharpInteractive.Tests/UsageScenarios/Comments/CommandLineOutputScenario.txt
new file mode 100644
index 00000000..b3ec27ed
--- /dev/null
+++ b/CSharpInteractive.Tests/UsageScenarios/Comments/CommandLineOutputScenario.txt
@@ -0,0 +1,10 @@
+// Adds the namespace "HostApi" to use Command Line API
+using HostApi;
+
+var lines = new List();
+var result = new CommandLine("cmd", "/c", "SET")
+ .AddVars(("MyEnv", "MyVal"))
+ .Run(output => lines.Add(output.Line))
+ .EnsureSuccess();
+
+lines.ShouldContain("MyEnv=MyVal");
diff --git a/CSharpInteractive.Tests/UsageScenarios/Comments/CommandLineScenario.txt b/CSharpInteractive.Tests/UsageScenarios/Comments/CommandLineScenario.txt
new file mode 100644
index 00000000..766798cd
--- /dev/null
+++ b/CSharpInteractive.Tests/UsageScenarios/Comments/CommandLineScenario.txt
@@ -0,0 +1,19 @@
+// Adds the namespace "HostApi" to use Command Line API
+using HostApi;
+
+GetService()
+ .Run(new CommandLine("cmd", "/c", "DIR"))
+ .EnsureSuccess();
+
+// or the same thing using the extension method
+new CommandLine("cmd", "/c", "DIR")
+ .Run()
+ .EnsureSuccess();
+
+// using operator '+'
+var cmd = new CommandLine("cmd") + "/c" + "DIR";
+cmd.Run().EnsureSuccess();
+
+// with environment variables
+cmd = new CommandLine("cmd") + "/c" + "DIR" + ("MyEnvVar", "Some Value");
+cmd.Run().EnsureSuccess();
diff --git a/CSharpInteractive.Tests/UsageScenarios/Comments/CommandLineWithTimeoutScenario.txt b/CSharpInteractive.Tests/UsageScenarios/Comments/CommandLineWithTimeoutScenario.txt
new file mode 100644
index 00000000..a145f360
--- /dev/null
+++ b/CSharpInteractive.Tests/UsageScenarios/Comments/CommandLineWithTimeoutScenario.txt
@@ -0,0 +1,7 @@
+// Adds the namespace "HostApi" to use Command Line API
+using HostApi;
+
+int? exitCode = new CommandLine("cmd", "/c", "TIMEOUT", "/T", "120")
+ .Run(default, TimeSpan.FromMilliseconds(1))
+ .EnsureSuccess()
+ .ExitCode;
diff --git a/CSharpInteractive.Tests/UsageScenarios/Comments/CommandLinesScenario.txt b/CSharpInteractive.Tests/UsageScenarios/Comments/CommandLinesScenario.txt
new file mode 100644
index 00000000..af7598a8
--- /dev/null
+++ b/CSharpInteractive.Tests/UsageScenarios/Comments/CommandLinesScenario.txt
@@ -0,0 +1,44 @@
+// Adds the namespace "Script.Cmd" to use Command Line API
+using HostApi;
+
+// Creates and run a simple command line
+"whoami".AsCommandLine().Run().EnsureSuccess();
+
+// Creates and run a simple command line
+new CommandLine("whoami").Run().EnsureSuccess();
+
+// Creates and run a command line with arguments
+new CommandLine("cmd", "/c", "echo", "Hello").Run();
+
+// Same as previous statement
+new CommandLine("cmd", "/c")
+ .AddArgs("echo", "Hello")
+ .Run()
+ .EnsureSuccess();
+
+(new CommandLine("cmd") + "/c" + "echo" + "Hello")
+ .Run()
+ .EnsureSuccess();
+
+"cmd".AsCommandLine("/c", "echo", "Hello")
+ .Run()
+ .EnsureSuccess();
+
+("cmd".AsCommandLine() + "/c" + "echo" + "Hello")
+ .Run()
+ .EnsureSuccess();
+
+// Just builds a command line with multiple environment variables
+var cmd = new CommandLine("cmd", "/c", "echo", "Hello")
+ .AddVars(("Var1", "val1"), ("var2", "Val2"));
+
+// Same as previous statement
+cmd = new CommandLine("cmd") + "/c" + "echo" + "Hello" + ("Var1", "val1") + ("var2", "Val2");
+
+// Builds a command line to run from a specific working directory
+cmd = new CommandLine("cmd", "/c", "echo", "Hello")
+ .WithWorkingDirectory("MyDyrectory");
+
+// Builds a command line and replaces all command line arguments
+cmd = new CommandLine("cmd", "/c", "echo", "Hello")
+ .WithArgs("/c", "echo", "Hello !!!");
diff --git a/CSharpInteractive.Tests/UsageScenarios/Comments/DockerDotNetBuildScenario.txt b/CSharpInteractive.Tests/UsageScenarios/Comments/DockerDotNetBuildScenario.txt
new file mode 100644
index 00000000..d4a6b1d0
--- /dev/null
+++ b/CSharpInteractive.Tests/UsageScenarios/Comments/DockerDotNetBuildScenario.txt
@@ -0,0 +1,27 @@
+// Adds the namespace "HostApi" to use .NET build API and Docker API
+using HostApi;
+
+// Creates a base docker command line
+var dockerRun = new DockerRun()
+ .WithAutoRemove(true)
+ .WithInteractive(true)
+ .WithImage("mcr.microsoft.com/dotnet/sdk")
+ .WithPlatform("linux")
+ .WithContainerWorkingDirectory("/MyProjects")
+ .AddVolumes((ToAbsoluteLinuxPath(Environment.CurrentDirectory), "/MyProjects"));
+
+
+// Creates a new library project in a docker container
+dockerRun
+ .WithCommandLine(new DotNetCustom("new", "classlib", "-n", "MyLib", "--force"))
+ .Run()
+ .EnsureSuccess();
+
+// Builds the library project in a docker container
+var result = dockerRun
+ .WithCommandLine(new DotNetBuild().WithProject("MyLib/MyLib.csproj"))
+ .Build()
+ .EnsureSuccess();
+
+string ToAbsoluteLinuxPath(string path) =>
+ "/" + path.Replace(":", "").Replace('\\', '/');
diff --git a/CSharpInteractive.Tests/UsageScenarios/Comments/DockerRunScenario.txt b/CSharpInteractive.Tests/UsageScenarios/Comments/DockerRunScenario.txt
new file mode 100644
index 00000000..3c218edf
--- /dev/null
+++ b/CSharpInteractive.Tests/UsageScenarios/Comments/DockerRunScenario.txt
@@ -0,0 +1,11 @@
+// Adds the namespace "HostApi" to use Command Line API and Docker API
+using HostApi;
+
+// Creates some command line to run in a docker container
+var cmd = new CommandLine("whoami");
+
+// Runs the command line in a docker container
+var result = new DockerRun(cmd, "mcr.microsoft.com/dotnet/sdk")
+ .WithAutoRemove(true)
+ .Run()
+ .EnsureSuccess();
diff --git a/CSharpInteractive.Tests/UsageScenarios/Comments/DotNetAddPackageScenario.txt b/CSharpInteractive.Tests/UsageScenarios/Comments/DotNetAddPackageScenario.txt
new file mode 100644
index 00000000..49b1ab94
--- /dev/null
+++ b/CSharpInteractive.Tests/UsageScenarios/Comments/DotNetAddPackageScenario.txt
@@ -0,0 +1,23 @@
+ using HostApi;
+
+ new DotNetNew()
+ .WithTemplateName("classlib")
+ .WithName("MyLib")
+ .WithForce(true)
+ .Build().EnsureSuccess();
+
+ var result = new DotNetAddPackage()
+ .WithWorkingDirectory("MyLib")
+ .WithPackage("Pure.DI")
+ .Run().EnsureSuccess();
+
+ var lines = new List();
+ new DotNetListPackage()
+ .WithWorkingDirectory("MyLib")
+ .WithVerbosity(DotNetVerbosity.Minimal)
+ .Run(output => lines.Add(output.Line));
+
+ result.ExitCode.ShouldBe(0, result.ToString());
+ lines.Any(i => i.Contains("Pure.DI")).ShouldBeTrue(result.ToString());
+ }
+}
diff --git a/CSharpInteractive.Tests/UsageScenarios/Comments/DotNetBuildScenario.txt b/CSharpInteractive.Tests/UsageScenarios/Comments/DotNetBuildScenario.txt
new file mode 100644
index 00000000..b158ce49
--- /dev/null
+++ b/CSharpInteractive.Tests/UsageScenarios/Comments/DotNetBuildScenario.txt
@@ -0,0 +1,25 @@
+// Adds the namespace "HostApi" to use .NET build API
+using HostApi;
+
+// Creates a new test project, running a command like: "dotnet new xunit -n MyTests --force"
+new DotNetNew()
+ .WithTemplateName("xunit")
+ .WithName("MyTests")
+ .WithForce(true)
+ .Build().EnsureSuccess();
+
+// Builds this project, running a command like: "dotnet build" from the directory "MyLib"
+var messages = new List();
+var result = new DotNetBuild()
+ .WithWorkingDirectory("MyTests")
+ .Build(message => messages.Add(message)).EnsureSuccess();
+
+// The "result" variable provides details about a build
+messages.Count.ShouldBeGreaterThan(0, result.ToString());
+result.Errors.Any(message => message.State == BuildMessageState.StdError).ShouldBeFalse(result.ToString());
+result.ExitCode.ShouldBe(0, result.ToString());
+
+// Runs tests in docker
+result = new DotNetTest()
+ .WithWorkingDirectory("MyTests")
+ .Build().EnsureSuccess();
diff --git a/CSharpInteractive.Tests/UsageScenarios/Comments/DotNetBuildServerShutdownScenario.txt b/CSharpInteractive.Tests/UsageScenarios/Comments/DotNetBuildServerShutdownScenario.txt
new file mode 100644
index 00000000..be7faf2d
--- /dev/null
+++ b/CSharpInteractive.Tests/UsageScenarios/Comments/DotNetBuildServerShutdownScenario.txt
@@ -0,0 +1,7 @@
+// Adds the namespace "HostApi" to use .NET build API
+using HostApi;
+
+// Shuts down all build servers that are started from dotnet.
+new DotNetBuildServerShutdown()
+ .Run()
+ .EnsureSuccess();
diff --git a/CSharpInteractive.Tests/UsageScenarios/Comments/DotNetCleanScenario.txt b/CSharpInteractive.Tests/UsageScenarios/Comments/DotNetCleanScenario.txt
new file mode 100644
index 00000000..7c1c96fd
--- /dev/null
+++ b/CSharpInteractive.Tests/UsageScenarios/Comments/DotNetCleanScenario.txt
@@ -0,0 +1,26 @@
+// Adds the namespace "HostApi" to use .NET build API
+using HostApi;
+
+// Creates a new library project, running a command like: "dotnet new classlib -n MyLib --force"
+var result = new DotNetNew()
+ .WithTemplateName("classlib")
+ .WithName("MyLib")
+ .WithForce(true)
+ .Build().EnsureSuccess();
+
+result.ExitCode.ShouldBe(0);
+
+// Builds the library project, running a command like: "dotnet build" from the directory "MyLib"
+result = new DotNetBuild()
+ .WithWorkingDirectory("MyLib")
+ .Build().EnsureSuccess();
+
+result.ExitCode.ShouldBe(0);
+
+// Clean the project, running a command like: "dotnet clean" from the directory "MyLib"
+result = new DotNetClean()
+ .WithWorkingDirectory("MyLib")
+ .Build().EnsureSuccess();
+
+// The "result" variable provides details about a build
+result.ExitCode.ShouldBe(0);
diff --git a/CSharpInteractive.Tests/UsageScenarios/Comments/DotNetCsiScenario.txt b/CSharpInteractive.Tests/UsageScenarios/Comments/DotNetCsiScenario.txt
new file mode 100644
index 00000000..07eaea82
--- /dev/null
+++ b/CSharpInteractive.Tests/UsageScenarios/Comments/DotNetCsiScenario.txt
@@ -0,0 +1,17 @@
+// Adds the namespace "HostApi" to use .NET build API
+using HostApi;
+
+var script = Path.GetTempFileName();
+File.WriteAllText(script, "Console.WriteLine($\"Hello, {Args[0]}!\");");
+
+var stdOut = new List();
+var result = new DotNetCsi()
+ .WithScript(script)
+ .AddArgs("World")
+ .Build(message => stdOut.Add(message.Text))
+ .EnsureSuccess();
+
+result.ExitCode.ShouldBe(0);
+
+// Checks StdOut
+stdOut.Contains("Hello, World!").ShouldBeTrue(result.ToString());
diff --git a/CSharpInteractive.Tests/UsageScenarios/Comments/DotNetCustomScenario.txt b/CSharpInteractive.Tests/UsageScenarios/Comments/DotNetCustomScenario.txt
new file mode 100644
index 00000000..2f6de909
--- /dev/null
+++ b/CSharpInteractive.Tests/UsageScenarios/Comments/DotNetCustomScenario.txt
@@ -0,0 +1,10 @@
+// Adds the namespace "HostApi" to use .NET build API
+using HostApi;
+
+// Gets the dotnet version, running a command like: "dotnet --version"
+NuGetVersion? version = default;
+new DotNetCustom("--version")
+ .Run(message => NuGetVersion.TryParse(message.Line, out version))
+ .EnsureSuccess();
+
+version.ShouldNotBeNull();
diff --git a/CSharpInteractive.Tests/UsageScenarios/Comments/DotNetExecScenario.txt b/CSharpInteractive.Tests/UsageScenarios/Comments/DotNetExecScenario.txt
new file mode 100644
index 00000000..16dd0e2b
--- /dev/null
+++ b/CSharpInteractive.Tests/UsageScenarios/Comments/DotNetExecScenario.txt
@@ -0,0 +1,19 @@
+// Adds the namespace "HostApi" to use .NET build API
+using HostApi;
+
+// Creates a new console project, running a command like: "dotnet new console -n MyApp --force"
+new DotNetNew()
+ .WithTemplateName("console")
+ .WithName("MyApp")
+ .WithForce(true)
+ .Build().EnsureSuccess();
+
+var tempDirectory = Path.Combine(Path.GetTempPath(), Path.GetRandomFileName());
+new DotNetPublish()
+ .WithWorkingDirectory("MyApp")
+ .WithOutput(tempDirectory)
+ .Build().EnsureSuccess();
+
+new DotNetExec()
+ .WithPathToApplication(Path.Combine(tempDirectory, "MyApp.dll"))
+ .Run().EnsureSuccess();
diff --git a/CSharpInteractive.Tests/UsageScenarios/Comments/DotNetListPackageScenario.txt b/CSharpInteractive.Tests/UsageScenarios/Comments/DotNetListPackageScenario.txt
new file mode 100644
index 00000000..d8bb31ec
--- /dev/null
+++ b/CSharpInteractive.Tests/UsageScenarios/Comments/DotNetListPackageScenario.txt
@@ -0,0 +1,23 @@
+ using HostApi;
+
+ new DotNetNew()
+ .WithTemplateName("classlib")
+ .WithName("MyLib")
+ .WithForce(true)
+ .Build().EnsureSuccess();
+
+ new DotNetAddPackage()
+ .WithWorkingDirectory("MyLib")
+ .WithPackage("Pure.DI")
+ .Run().EnsureSuccess();
+
+ var lines = new List();
+ var result = new DotNetListPackage()
+ .WithWorkingDirectory("MyLib")
+ .WithVerbosity(DotNetVerbosity.Minimal)
+ .Run(output => lines.Add(output.Line));
+
+ result.ExitCode.ShouldBe(0, result.ToString());
+ lines.Any(i => i.Contains("Pure.DI")).ShouldBeTrue(result.ToString());
+ }
+}
diff --git a/CSharpInteractive.Tests/UsageScenarios/Comments/DotNetMSBuildVSTestScenario.txt b/CSharpInteractive.Tests/UsageScenarios/Comments/DotNetMSBuildVSTestScenario.txt
new file mode 100644
index 00000000..c957f8c5
--- /dev/null
+++ b/CSharpInteractive.Tests/UsageScenarios/Comments/DotNetMSBuildVSTestScenario.txt
@@ -0,0 +1,22 @@
+// Adds the namespace "HostApi" to use .NET build API
+using HostApi;
+
+// Creates a new test project, running a command like: "dotnet new mstest -n MyTests --force"
+var result = new DotNetNew()
+ .WithTemplateName("mstest")
+ .WithName("MyTests")
+ .WithForce(true)
+ .Build().EnsureSuccess();
+
+result.ExitCode.ShouldBe(0, result.ToString());
+
+// Runs tests via a command like: "dotnet msbuild /t:VSTest" from the directory "MyTests"
+result = new MSBuild()
+ .WithTarget("VSTest")
+ .WithWorkingDirectory("MyTests")
+ .Build().EnsureSuccess();
+
+// The "result" variable provides details about a build
+result.ExitCode.ShouldBe(0, result.ToString());
+result.Summary.Tests.ShouldBe(1, result.ToString());
+result.Tests.Count(test => test.State == TestState.Finished).ShouldBe(1, result.ToString());
diff --git a/CSharpInteractive.Tests/UsageScenarios/Comments/DotNetPackScenario.txt b/CSharpInteractive.Tests/UsageScenarios/Comments/DotNetPackScenario.txt
new file mode 100644
index 00000000..d456d731
--- /dev/null
+++ b/CSharpInteractive.Tests/UsageScenarios/Comments/DotNetPackScenario.txt
@@ -0,0 +1,17 @@
+// Adds the namespace "HostApi" to use .NET build API
+using HostApi;
+
+// Creates a new library project, running a command like: "dotnet new classlib -n MyLib --force"
+var result = new DotNetNew()
+ .WithTemplateName("classlib")
+ .WithName("MyLib")
+ .WithForce(true)
+ .Build().EnsureSuccess();
+
+result.ExitCode.ShouldBe(0);
+
+// Creates a NuGet package of version 1.2.3 for the project, running a command like: "dotnet pack /p:version=1.2.3" from the directory "MyLib"
+result = new DotNetPack()
+ .WithWorkingDirectory("MyLib")
+ .AddProps(("version", "1.2.3"))
+ .Build().EnsureSuccess();
diff --git a/CSharpInteractive.Tests/UsageScenarios/Comments/DotNetPublishScenario.txt b/CSharpInteractive.Tests/UsageScenarios/Comments/DotNetPublishScenario.txt
new file mode 100644
index 00000000..d04abc99
--- /dev/null
+++ b/CSharpInteractive.Tests/UsageScenarios/Comments/DotNetPublishScenario.txt
@@ -0,0 +1,18 @@
+// Adds the namespace "HostApi" to use .NET build API
+using HostApi;
+
+// Creates a new library project, running a command like: "dotnet new classlib -n MyLib --force"
+var result = new DotNetNew()
+ .WithTemplateName("classlib")
+ .AddArgs("-f", "net8.0")
+ .WithName("MyLib")
+ .WithForce(true)
+ .Build().EnsureSuccess();
+
+result.ExitCode.ShouldBe(0);
+
+// Publish the project, running a command like: "dotnet publish --framework net6.0" from the directory "MyLib"
+result = new DotNetPublish()
+ .WithWorkingDirectory("MyLib")
+ .WithFramework("net8.0")
+ .Build().EnsureSuccess();
diff --git a/CSharpInteractive.Tests/UsageScenarios/Comments/DotNetRemovePackageScenario.txt b/CSharpInteractive.Tests/UsageScenarios/Comments/DotNetRemovePackageScenario.txt
new file mode 100644
index 00000000..3d11279f
--- /dev/null
+++ b/CSharpInteractive.Tests/UsageScenarios/Comments/DotNetRemovePackageScenario.txt
@@ -0,0 +1,28 @@
+ using HostApi;
+
+ new DotNetNew()
+ .WithTemplateName("classlib")
+ .WithName("MyLib")
+ .WithForce(true)
+ .Build().EnsureSuccess();
+
+ new DotNetAddPackage()
+ .WithWorkingDirectory("MyLib")
+ .WithPackage("Pure.DI")
+ .Run().EnsureSuccess();
+
+ var result = new DotNetRemovePackage()
+ .WithWorkingDirectory("MyLib")
+ .WithPackage("Pure.DI")
+ .Run().EnsureSuccess();
+
+ var lines = new List();
+ new DotNetListPackage()
+ .WithWorkingDirectory("MyLib")
+ .WithVerbosity(DotNetVerbosity.Minimal)
+ .Run(output => lines.Add(output.Line));
+
+ result.ExitCode.ShouldBe(0, result.ToString());
+ lines.Any(i => i.Contains("Pure.DI")).ShouldBeFalse(result.ToString());
+ }
+}
diff --git a/CSharpInteractive.Tests/UsageScenarios/Comments/DotNetRestoreScenario.txt b/CSharpInteractive.Tests/UsageScenarios/Comments/DotNetRestoreScenario.txt
new file mode 100644
index 00000000..64580122
--- /dev/null
+++ b/CSharpInteractive.Tests/UsageScenarios/Comments/DotNetRestoreScenario.txt
@@ -0,0 +1,16 @@
+// Adds the namespace "HostApi" to use .NET build API
+using HostApi;
+
+// Creates a new library project, running a command like: "dotnet new classlib -n MyLib --force"
+var result = new DotNetNew()
+ .WithTemplateName("classlib")
+ .WithName("MyLib")
+ .WithForce(true)
+ .Build().EnsureSuccess();
+
+result.ExitCode.ShouldBe(0);
+
+// Restore the project, running a command like: "dotnet restore" from the directory "MyLib"
+result = new DotNetRestore()
+ .WithWorkingDirectory("MyLib")
+ .Build().EnsureSuccess();
diff --git a/CSharpInteractive.Tests/UsageScenarios/Comments/DotNetRunScenario.txt b/CSharpInteractive.Tests/UsageScenarios/Comments/DotNetRunScenario.txt
new file mode 100644
index 00000000..f5b8b8ab
--- /dev/null
+++ b/CSharpInteractive.Tests/UsageScenarios/Comments/DotNetRunScenario.txt
@@ -0,0 +1,23 @@
+// Adds the namespace "HostApi" to use .NET build API
+using HostApi;
+
+// Creates a new console project, running a command like: "dotnet new console -n MyApp --force"
+var result = new DotNetNew()
+ .WithTemplateName("console")
+ .WithName("MyApp")
+ .WithForce(true)
+ .Build().EnsureSuccess();
+
+result.ExitCode.ShouldBe(0);
+
+// Runs the console project using a command like: "dotnet run" from the directory "MyApp"
+var stdOut = new List();
+result = new DotNetRun()
+ .WithWorkingDirectory("MyApp")
+ .Build(message => stdOut.Add(message.Text))
+ .EnsureSuccess();
+
+result.ExitCode.ShouldBe(0);
+
+// Checks StdOut
+stdOut.ShouldBe(new[] {"Hello, World!"});
diff --git a/CSharpInteractive.Tests/UsageScenarios/Comments/DotNetScenario.txt b/CSharpInteractive.Tests/UsageScenarios/Comments/DotNetScenario.txt
new file mode 100644
index 00000000..93e90176
--- /dev/null
+++ b/CSharpInteractive.Tests/UsageScenarios/Comments/DotNetScenario.txt
@@ -0,0 +1,19 @@
+// Adds the namespace "HostApi" to use .NET build API
+using HostApi;
+
+// Creates a new console project, running a command like: "dotnet new console -n MyApp --force"
+new DotNetNew()
+ .WithTemplateName("console")
+ .WithName("MyApp")
+ .WithForce(true)
+ .Build().EnsureSuccess();
+
+var tempDirectory = Path.Combine(Path.GetTempPath(), Path.GetRandomFileName());
+new DotNetPublish()
+ .WithWorkingDirectory("MyApp")
+ .WithOutput(tempDirectory)
+ .Build().EnsureSuccess();
+
+new DotNet()
+ .WithPathToApplication(Path.Combine(tempDirectory, "MyApp.dll"))
+ .Run().EnsureSuccess();
diff --git a/CSharpInteractive.Tests/UsageScenarios/Comments/DotNetTestScenario.txt b/CSharpInteractive.Tests/UsageScenarios/Comments/DotNetTestScenario.txt
new file mode 100644
index 00000000..dbf9f2f0
--- /dev/null
+++ b/CSharpInteractive.Tests/UsageScenarios/Comments/DotNetTestScenario.txt
@@ -0,0 +1,22 @@
+// Adds the namespace "HostApi" to use .NET build API
+using HostApi;
+
+// Creates a new test project, running a command like: "dotnet new mstest -n MyTests --force"
+var result = new DotNetNew()
+ .WithTemplateName("mstest")
+ .WithName("MyTests")
+ .WithForce(true)
+ .Build().EnsureSuccess();
+
+result.ExitCode.ShouldBe(0);
+
+// Runs tests via a command like: "dotnet test" from the directory "MyTests"
+result = new DotNetTest()
+ .WithWorkingDirectory("MyTests")
+ .Build()
+ .EnsureSuccess();
+
+// The "result" variable provides details about a build
+result.ExitCode.ShouldBe(0, result.ToString());
+result.Summary.Tests.ShouldBe(1, result.ToString());
+result.Tests.Count(test => test.State == TestState.Finished).ShouldBe(1, result.ToString());
diff --git a/CSharpInteractive.Tests/UsageScenarios/Comments/DotNetTestWithDotCoverScenario.txt b/CSharpInteractive.Tests/UsageScenarios/Comments/DotNetTestWithDotCoverScenario.txt
new file mode 100644
index 00000000..7d80fc1f
--- /dev/null
+++ b/CSharpInteractive.Tests/UsageScenarios/Comments/DotNetTestWithDotCoverScenario.txt
@@ -0,0 +1,51 @@
+// Adds the namespace "HostApi" to use .NET build API
+using HostApi;
+
+// Creates a new test project, running a command like: "dotnet new mstest -n MyTests --force"
+new DotNetNew()
+ .WithTemplateName("mstest")
+ .WithName("MyTests")
+ .WithForce(true)
+ .Run().EnsureSuccess();
+
+// Creates the tool manifest and installs the dotCover tool locally
+// It is better to run the following 2 commands manually
+// and commit these changes to a source control
+new DotNetNew()
+ .WithTemplateName("tool-manifest")
+ .Run().EnsureSuccess();
+
+new DotNetCustom("tool", "install", "--local", "JetBrains.dotCover.GlobalTool")
+ .Run().EnsureSuccess();
+
+// Creates a test command
+var test = new DotNetTest()
+ .WithProject("MyTests");
+
+var dotCoverSnapshot = Path.Combine("MyTests", "dotCover.dcvr");
+var dotCoverReport = Path.Combine("MyTests", "dotCover.html");
+// Modifies the test command by putting "dotCover" in front of all arguments
+// to have something like "dotnet dotcover test ..."
+// and adding few specific arguments to the end
+var testUnderDotCover = test.Customize(cmd =>
+ cmd.ClearArgs()
+ + "dotcover"
+ + cmd.Args
+ + $"--dcOutput={dotCoverSnapshot}"
+ + "--dcFilters=+:module=TeamCity.CSharpInteractive.HostApi;+:module=dotnet-csi"
+ + "--dcAttributeFilters=System.Diagnostics.CodeAnalysis.ExcludeFromCodeCoverage");
+
+// Runs tests under dotCover via a command like: "dotnet dotcover test ..."
+var result = testUnderDotCover
+ .Build().EnsureSuccess();
+
+// The "result" variable provides details about a build
+result.ExitCode.ShouldBe(0, result.ToString());
+result.Tests.Count(i => i.State == TestState.Finished).ShouldBe(1, result.ToString());
+
+// Generates a HTML code coverage report.
+new DotNetCustom("dotCover", "report", $"--source={dotCoverSnapshot}", $"--output={dotCoverReport}", "--reportType=HTML")
+ .Run().EnsureSuccess();
+
+// Check for a dotCover report
+File.Exists(dotCoverReport).ShouldBeTrue();
diff --git a/CSharpInteractive.Tests/UsageScenarios/Comments/DotNetToolRestoreScenario.txt b/CSharpInteractive.Tests/UsageScenarios/Comments/DotNetToolRestoreScenario.txt
new file mode 100644
index 00000000..5ea03c5f
--- /dev/null
+++ b/CSharpInteractive.Tests/UsageScenarios/Comments/DotNetToolRestoreScenario.txt
@@ -0,0 +1,18 @@
+// Adds the namespace "HostApi" to use .NET build API
+using HostApi;
+
+var projectDir = Path.Combine(Path.GetTempPath(), Guid.NewGuid().ToString()[..4]);
+Directory.CreateDirectory(projectDir);
+
+// Creates a local tool manifest
+new DotNetNew()
+ .WithTemplateName("tool-manifest")
+ .WithWorkingDirectory(projectDir)
+ .Run()
+ .EnsureSuccess();
+
+// Restore local tools
+new DotNetToolRestore()
+ .WithWorkingDirectory(projectDir)
+ .Run()
+ .EnsureSuccess();
diff --git a/CSharpInteractive.Tests/UsageScenarios/Comments/DotNetVSTestScenario.txt b/CSharpInteractive.Tests/UsageScenarios/Comments/DotNetVSTestScenario.txt
new file mode 100644
index 00000000..17eeb2c2
--- /dev/null
+++ b/CSharpInteractive.Tests/UsageScenarios/Comments/DotNetVSTestScenario.txt
@@ -0,0 +1,31 @@
+// Adds the namespace "HostApi" to use .NET build API
+using HostApi;
+
+// Creates a new test project, running a command like: "dotnet new mstest -n MyTests --force"
+var result = new DotNetNew()
+ .WithTemplateName("mstest")
+ .WithName("MyTests")
+ .WithForce(true)
+ .Build().EnsureSuccess();
+
+result.ExitCode.ShouldBe(0, result.ToString());
+
+// Builds the test project, running a command like: "dotnet build -c Release" from the directory "MyTests"
+result = new DotNetBuild()
+ .WithWorkingDirectory("MyTests")
+ .WithConfiguration("Release")
+ .WithOutput("MyOutput")
+ .Build().EnsureSuccess();
+
+result.ExitCode.ShouldBe(0, result.ToString());
+
+// Runs tests via a command like: "dotnet vstest" from the directory "MyTests"
+result = new VSTest()
+ .AddTestFileNames(Path.Combine("MyOutput", "MyTests.dll"))
+ .WithWorkingDirectory("MyTests")
+ .Build().EnsureSuccess();
+
+// The "result" variable provides details about a build
+result.ExitCode.ShouldBe(0, result.ToString());
+result.Summary.Tests.ShouldBe(1, result.ToString());
+result.Tests.Count(test => test.State == TestState.Finished).ShouldBe(1, result.ToString());
diff --git a/CSharpInteractive.Tests/UsageScenarios/Comments/GetServiceScenario.txt b/CSharpInteractive.Tests/UsageScenarios/Comments/GetServiceScenario.txt
new file mode 100644
index 00000000..d64ed0fb
--- /dev/null
+++ b/CSharpInteractive.Tests/UsageScenarios/Comments/GetServiceScenario.txt
@@ -0,0 +1,4 @@
+GetService();
+
+var serviceProvider = GetService();
+serviceProvider.GetService(typeof(INuGet));
diff --git a/CSharpInteractive.Tests/UsageScenarios/Comments/HostScenario.txt b/CSharpInteractive.Tests/UsageScenarios/Comments/HostScenario.txt
new file mode 100644
index 00000000..d7e1ca68
--- /dev/null
+++ b/CSharpInteractive.Tests/UsageScenarios/Comments/HostScenario.txt
@@ -0,0 +1,2 @@
+var packages = Host.GetService();
+Host.WriteLine("Hello");
diff --git a/CSharpInteractive.Tests/UsageScenarios/Comments/LogErrorScenario.txt b/CSharpInteractive.Tests/UsageScenarios/Comments/LogErrorScenario.txt
new file mode 100644
index 00000000..eede6e08
--- /dev/null
+++ b/CSharpInteractive.Tests/UsageScenarios/Comments/LogErrorScenario.txt
@@ -0,0 +1 @@
+Error("Error info", "Error identifier");
diff --git a/CSharpInteractive.Tests/UsageScenarios/Comments/LogInfoScenario.txt b/CSharpInteractive.Tests/UsageScenarios/Comments/LogInfoScenario.txt
new file mode 100644
index 00000000..d2d25ecb
--- /dev/null
+++ b/CSharpInteractive.Tests/UsageScenarios/Comments/LogInfoScenario.txt
@@ -0,0 +1 @@
+Info("Some info");
diff --git a/CSharpInteractive.Tests/UsageScenarios/Comments/LogTraceScenario.txt b/CSharpInteractive.Tests/UsageScenarios/Comments/LogTraceScenario.txt
new file mode 100644
index 00000000..3761cd7f
--- /dev/null
+++ b/CSharpInteractive.Tests/UsageScenarios/Comments/LogTraceScenario.txt
@@ -0,0 +1 @@
+Trace("Some trace info");
diff --git a/CSharpInteractive.Tests/UsageScenarios/Comments/LogWarningScenario.txt b/CSharpInteractive.Tests/UsageScenarios/Comments/LogWarningScenario.txt
new file mode 100644
index 00000000..063bbaeb
--- /dev/null
+++ b/CSharpInteractive.Tests/UsageScenarios/Comments/LogWarningScenario.txt
@@ -0,0 +1 @@
+Warning("Warning info");
diff --git a/CSharpInteractive.Tests/UsageScenarios/Comments/MSBuildScenario.txt b/CSharpInteractive.Tests/UsageScenarios/Comments/MSBuildScenario.txt
new file mode 100644
index 00000000..5f11c8ea
--- /dev/null
+++ b/CSharpInteractive.Tests/UsageScenarios/Comments/MSBuildScenario.txt
@@ -0,0 +1,24 @@
+// Adds the namespace "HostApi" to use .NET build API
+using HostApi;
+
+// Creates a new library project, running a command like: "dotnet new classlib -n MyLib --force"
+var result = new DotNetNew()
+ .WithTemplateName("classlib")
+ .WithName("MyLib")
+ .WithForce(true)
+ .Build().EnsureSuccess();
+
+result.ExitCode.ShouldBe(0, result.ToString());
+
+// Builds the library project, running a command like: "dotnet msbuild /t:Build -restore /p:configuration=Release -verbosity=detailed" from the directory "MyLib"
+result = new MSBuild()
+ .WithWorkingDirectory("MyLib")
+ .WithTarget("Build")
+ .WithRestore(true)
+ .AddProps(("configuration", "Release"))
+ .WithVerbosity(DotNetVerbosity.Detailed)
+ .Build().EnsureSuccess();
+
+// The "result" variable provides details about a build
+result.Errors.Any(message => message.State == BuildMessageState.StdError).ShouldBeFalse(result.ToString());
+result.ExitCode.ShouldBe(0, result.ToString());
diff --git a/CSharpInteractive.Tests/UsageScenarios/Comments/NuGetRestoreAdvanced.txt b/CSharpInteractive.Tests/UsageScenarios/Comments/NuGetRestoreAdvanced.txt
new file mode 100644
index 00000000..9aae8ff0
--- /dev/null
+++ b/CSharpInteractive.Tests/UsageScenarios/Comments/NuGetRestoreAdvanced.txt
@@ -0,0 +1,13 @@
+// Adds the namespace "HostApi" to use INuGet
+using HostApi;
+
+var packagesPath = Path.Combine(
+ Path.GetTempPath(),
+ Guid.NewGuid().ToString()[..4]);
+
+var settings = new NuGetRestoreSettings("IoC.Container")
+ .WithVersionRange(VersionRange.Parse("[1.3, 1.3.8)"))
+ .WithTargetFrameworkMoniker("net5.0")
+ .WithPackagesPath(packagesPath);
+
+IEnumerable packages = GetService().Restore(settings);
diff --git a/CSharpInteractive.Tests/UsageScenarios/Comments/NuGetRestoreScenario.txt b/CSharpInteractive.Tests/UsageScenarios/Comments/NuGetRestoreScenario.txt
new file mode 100644
index 00000000..f03990e8
--- /dev/null
+++ b/CSharpInteractive.Tests/UsageScenarios/Comments/NuGetRestoreScenario.txt
@@ -0,0 +1,5 @@
+// Adds the namespace "HostApi" to use INuGet
+using HostApi;
+
+IEnumerable packages = GetService()
+ .Restore(new NuGetRestoreSettings("IoC.Container").WithVersionRange(VersionRange.All));
diff --git a/CSharpInteractive.Tests/UsageScenarios/Comments/PropsScenario.txt b/CSharpInteractive.Tests/UsageScenarios/Comments/PropsScenario.txt
new file mode 100644
index 00000000..d01ce035
--- /dev/null
+++ b/CSharpInteractive.Tests/UsageScenarios/Comments/PropsScenario.txt
@@ -0,0 +1,6 @@
+WriteLine(Props["version"]);
+WriteLine(Props.Get("configuration", "Release"));
+
+// Some CI/CDs have integration of these properties.
+// For example in TeamCity this property with all changes will be available in the next TeamCity steps.
+Props["version"] = "1.1.6";
diff --git a/CSharpInteractive.Tests/UsageScenarios/Comments/ServiceCollectionScenario.txt b/CSharpInteractive.Tests/UsageScenarios/Comments/ServiceCollectionScenario.txt
new file mode 100644
index 00000000..e1b1b913
--- /dev/null
+++ b/CSharpInteractive.Tests/UsageScenarios/Comments/ServiceCollectionScenario.txt
@@ -0,0 +1,20 @@
+public void Run()
+{
+ var serviceProvider =
+ GetService()
+ .AddTransient()
+ .BuildServiceProvider();
+
+ var myTask = serviceProvider.GetRequiredService();
+ var exitCode = myTask.Run();
+ exitCode.ShouldBe(0);
+}
+
+private class MyTask(ICommandLineRunner runner)
+{
+ public int? Run() => runner
+ .Run(new CommandLine("whoami"))
+ .EnsureSuccess()
+ .ExitCode;
+}
+
diff --git a/CSharpInteractive.Tests/UsageScenarios/Comments/TeamCityServiceMessagesScenario.txt b/CSharpInteractive.Tests/UsageScenarios/Comments/TeamCityServiceMessagesScenario.txt
new file mode 100644
index 00000000..69611885
--- /dev/null
+++ b/CSharpInteractive.Tests/UsageScenarios/Comments/TeamCityServiceMessagesScenario.txt
@@ -0,0 +1,18 @@
+// Adds a namespace to use ITeamCityWriter
+using JetBrains.TeamCity.ServiceMessages.Write.Special;
+
+using var writer = GetService();
+using (var tests = writer.OpenBlock("My Tests"))
+{
+ using (var test = tests.OpenTest("Test1"))
+ {
+ test.WriteStdOutput("Hello");
+ test.WriteImage("TestsResults/Test1Screenshot.jpg", "Screenshot");
+ test.WriteDuration(TimeSpan.FromMilliseconds(10));
+ }
+
+ using (var test = tests.OpenTest("Test2"))
+ {
+ test.WriteIgnored("Some reason");
+ }
+}
diff --git a/CSharpInteractive.Tests/UsageScenarios/Comments/WriteEmptyLineScenario.txt b/CSharpInteractive.Tests/UsageScenarios/Comments/WriteEmptyLineScenario.txt
new file mode 100644
index 00000000..448b0400
--- /dev/null
+++ b/CSharpInteractive.Tests/UsageScenarios/Comments/WriteEmptyLineScenario.txt
@@ -0,0 +1 @@
+WriteLine();
diff --git a/CSharpInteractive.Tests/UsageScenarios/Comments/WriteLineScenario.txt b/CSharpInteractive.Tests/UsageScenarios/Comments/WriteLineScenario.txt
new file mode 100644
index 00000000..7f4afc14
--- /dev/null
+++ b/CSharpInteractive.Tests/UsageScenarios/Comments/WriteLineScenario.txt
@@ -0,0 +1 @@
+WriteLine("Hello");
diff --git a/CSharpInteractive.Tests/UsageScenarios/Comments/WriteLineWithColourScenario.txt b/CSharpInteractive.Tests/UsageScenarios/Comments/WriteLineWithColourScenario.txt
new file mode 100644
index 00000000..15d058b1
--- /dev/null
+++ b/CSharpInteractive.Tests/UsageScenarios/Comments/WriteLineWithColourScenario.txt
@@ -0,0 +1 @@
+WriteLine("Hello", Header);
diff --git a/CSharpInteractive.Tests/UsageScenarios/DotNetAddPackageScenario.cs b/CSharpInteractive.Tests/UsageScenarios/DotNetAddPackageScenario.cs
new file mode 100644
index 00000000..31f5e210
--- /dev/null
+++ b/CSharpInteractive.Tests/UsageScenarios/DotNetAddPackageScenario.cs
@@ -0,0 +1,44 @@
+// ReSharper disable StringLiteralTypo
+// ReSharper disable ObjectCreationAsStatement
+// ReSharper disable ReturnValueOfPureMethodIsNotUsed
+// ReSharper disable CommentTypo
+
+namespace CSharpInteractive.Tests.UsageScenarios;
+
+using HostApi;
+
+[CollectionDefinition("Integration", DisableParallelization = true)]
+[Trait("Integration", "True")]
+public class DotNetAddPackageScenario : BaseScenario
+{
+ [Fact]
+ public void Run()
+ {
+ // $visible=true
+ // $tag=07 .NET CLI
+ // $priority=01
+ // $description=Adding a NuGet package
+ // {
+ // ## using HostApi;
+
+ new DotNetNew()
+ .WithTemplateName("classlib")
+ .WithName("MyLib")
+ .WithForce(true)
+ .Build().EnsureSuccess();
+
+ var result = new DotNetAddPackage()
+ .WithWorkingDirectory("MyLib")
+ .WithPackage("Pure.DI")
+ .Run().EnsureSuccess();
+
+ var lines = new List();
+ new DotNetListPackage()
+ .WithWorkingDirectory("MyLib")
+ .WithVerbosity(DotNetVerbosity.Minimal)
+ .Run(output => lines.Add(output.Line));
+
+ result.ExitCode.ShouldBe(0, result.ToString());
+ lines.Any(i => i.Contains("Pure.DI")).ShouldBeTrue(result.ToString());
+ }
+}
\ No newline at end of file
diff --git a/CSharpInteractive.Tests/UsageScenarios/DotNetBuildScenario.cs b/CSharpInteractive.Tests/UsageScenarios/DotNetBuildScenario.cs
index afea731f..4af68eb5 100644
--- a/CSharpInteractive.Tests/UsageScenarios/DotNetBuildScenario.cs
+++ b/CSharpInteractive.Tests/UsageScenarios/DotNetBuildScenario.cs
@@ -22,33 +22,32 @@ public void Run()
// Adds the namespace "HostApi" to use .NET build API
// ## using HostApi;
- // Creates a new library project, running a command like: "dotnet new classlib -n MyLib --force"
+ // Creates a new test project, running a command like: "dotnet new xunit -n MyTests --force"
new DotNetNew()
.WithTemplateName("xunit")
- .WithName("MyLib")
+ .WithName("MyTests")
.WithForce(true)
.Build().EnsureSuccess();
- // Builds the library project, running a command like: "dotnet build" from the directory "MyLib"
+ // Builds this project, running a command like: "dotnet build" from the directory "MyLib"
var messages = new List();
var result = new DotNetBuild()
- .WithWorkingDirectory("MyLib")
+ .WithWorkingDirectory("MyTests")
.Build(message => messages.Add(message)).EnsureSuccess();
// The "result" variable provides details about a build
messages.Count.ShouldBeGreaterThan(0, result.ToString());
- result.Errors.Any(message => message.State == BuildMessageState.StdError).ShouldBeFalse();
- result.ExitCode.ShouldBe(0);
+ result.Errors.Any(message => message.State == BuildMessageState.StdError).ShouldBeFalse(result.ToString());
+ result.ExitCode.ShouldBe(0, result.ToString());
// Runs tests in docker
result = new DotNetTest()
- .WithWorkingDirectory("MyLib")
- .Build()
- .EnsureSuccess();
+ .WithWorkingDirectory("MyTests")
+ .Build().EnsureSuccess();
// }
- result.ExitCode.ShouldBe(0);
- result.Summary.Tests.ShouldBe(1);
- result.Tests.Count(test => test.State == TestState.Finished).ShouldBe(1);
+ result.ExitCode.ShouldBe(0, result.ToString());
+ result.Summary.Tests.ShouldBe(1, result.ToString());
+ result.Tests.Count(test => test.State == TestState.Finished).ShouldBe(1, result.ToString());
}
}
\ No newline at end of file
diff --git a/CSharpInteractive.Tests/UsageScenarios/DotNetExecScenario.cs b/CSharpInteractive.Tests/UsageScenarios/DotNetExecScenario.cs
new file mode 100644
index 00000000..77d04617
--- /dev/null
+++ b/CSharpInteractive.Tests/UsageScenarios/DotNetExecScenario.cs
@@ -0,0 +1,44 @@
+// ReSharper disable StringLiteralTypo
+// ReSharper disable ObjectCreationAsStatement
+// ReSharper disable ReturnValueOfPureMethodIsNotUsed
+
+namespace CSharpInteractive.Tests.UsageScenarios;
+
+using System.Diagnostics.CodeAnalysis;
+using HostApi;
+
+[CollectionDefinition("Integration", DisableParallelization = true)]
+[Trait("Integration", "True")]
+[SuppressMessage("Performance", "CA1861:Avoid constant arrays as arguments")]
+public class DotNetExecScenario : BaseScenario
+{
+ [Fact]
+ public void Run()
+ {
+ // $visible=true
+ // $tag=07 .NET CLI
+ // $priority=01
+ // $description=Execute a dotnet application
+ // {
+ // Adds the namespace "HostApi" to use .NET build API
+ // ## using HostApi;
+
+ // Creates a new console project, running a command like: "dotnet new console -n MyApp --force"
+ new DotNetNew()
+ .WithTemplateName("console")
+ .WithName("MyApp")
+ .WithForce(true)
+ .Build().EnsureSuccess();
+
+ var tempDirectory = Path.Combine(Path.GetTempPath(), Path.GetRandomFileName());
+ new DotNetPublish()
+ .WithWorkingDirectory("MyApp")
+ .WithOutput(tempDirectory)
+ .Build().EnsureSuccess();
+
+ new DotNetExec()
+ .WithPathToApplication(Path.Combine(tempDirectory, "MyApp.dll"))
+ .Run().EnsureSuccess();
+ // }
+ }
+}
\ No newline at end of file
diff --git a/CSharpInteractive.Tests/UsageScenarios/DotNetListPackageScenario.cs b/CSharpInteractive.Tests/UsageScenarios/DotNetListPackageScenario.cs
new file mode 100644
index 00000000..ccebbdd2
--- /dev/null
+++ b/CSharpInteractive.Tests/UsageScenarios/DotNetListPackageScenario.cs
@@ -0,0 +1,44 @@
+// ReSharper disable StringLiteralTypo
+// ReSharper disable ObjectCreationAsStatement
+// ReSharper disable ReturnValueOfPureMethodIsNotUsed
+// ReSharper disable CommentTypo
+
+namespace CSharpInteractive.Tests.UsageScenarios;
+
+using HostApi;
+
+[CollectionDefinition("Integration", DisableParallelization = true)]
+[Trait("Integration", "True")]
+public class DotNetListPackageScenario : BaseScenario
+{
+ [Fact]
+ public void Run()
+ {
+ // $visible=true
+ // $tag=07 .NET CLI
+ // $priority=01
+ // $description=NuGet package listing
+ // {
+ // ## using HostApi;
+
+ new DotNetNew()
+ .WithTemplateName("classlib")
+ .WithName("MyLib")
+ .WithForce(true)
+ .Build().EnsureSuccess();
+
+ new DotNetAddPackage()
+ .WithWorkingDirectory("MyLib")
+ .WithPackage("Pure.DI")
+ .Run().EnsureSuccess();
+
+ var lines = new List();
+ var result = new DotNetListPackage()
+ .WithWorkingDirectory("MyLib")
+ .WithVerbosity(DotNetVerbosity.Minimal)
+ .Run(output => lines.Add(output.Line));
+
+ result.ExitCode.ShouldBe(0, result.ToString());
+ lines.Any(i => i.Contains("Pure.DI")).ShouldBeTrue(result.ToString());
+ }
+}
\ No newline at end of file
diff --git a/CSharpInteractive.Tests/UsageScenarios/DotNetRemovePackageScenario.cs b/CSharpInteractive.Tests/UsageScenarios/DotNetRemovePackageScenario.cs
new file mode 100644
index 00000000..fe8503aa
--- /dev/null
+++ b/CSharpInteractive.Tests/UsageScenarios/DotNetRemovePackageScenario.cs
@@ -0,0 +1,49 @@
+// ReSharper disable StringLiteralTypo
+// ReSharper disable ObjectCreationAsStatement
+// ReSharper disable ReturnValueOfPureMethodIsNotUsed
+// ReSharper disable CommentTypo
+
+namespace CSharpInteractive.Tests.UsageScenarios;
+
+using HostApi;
+
+[CollectionDefinition("Integration", DisableParallelization = true)]
+[Trait("Integration", "True")]
+public class DotNetRemovePackageScenario : BaseScenario
+{
+ [Fact]
+ public void Run()
+ {
+ // $visible=true
+ // $tag=07 .NET CLI
+ // $priority=01
+ // $description=Removing a NuGet package
+ // {
+ // ## using HostApi;
+
+ new DotNetNew()
+ .WithTemplateName("classlib")
+ .WithName("MyLib")
+ .WithForce(true)
+ .Build().EnsureSuccess();
+
+ new DotNetAddPackage()
+ .WithWorkingDirectory("MyLib")
+ .WithPackage("Pure.DI")
+ .Run().EnsureSuccess();
+
+ var result = new DotNetRemovePackage()
+ .WithWorkingDirectory("MyLib")
+ .WithPackage("Pure.DI")
+ .Run().EnsureSuccess();
+
+ var lines = new List();
+ new DotNetListPackage()
+ .WithWorkingDirectory("MyLib")
+ .WithVerbosity(DotNetVerbosity.Minimal)
+ .Run(output => lines.Add(output.Line));
+
+ result.ExitCode.ShouldBe(0, result.ToString());
+ lines.Any(i => i.Contains("Pure.DI")).ShouldBeFalse(result.ToString());
+ }
+}
\ No newline at end of file
diff --git a/CSharpInteractive.Tests/UsageScenarios/DotNetScenario.cs b/CSharpInteractive.Tests/UsageScenarios/DotNetScenario.cs
new file mode 100644
index 00000000..2c0eb002
--- /dev/null
+++ b/CSharpInteractive.Tests/UsageScenarios/DotNetScenario.cs
@@ -0,0 +1,44 @@
+// ReSharper disable StringLiteralTypo
+// ReSharper disable ObjectCreationAsStatement
+// ReSharper disable ReturnValueOfPureMethodIsNotUsed
+
+namespace CSharpInteractive.Tests.UsageScenarios;
+
+using System.Diagnostics.CodeAnalysis;
+using HostApi;
+
+[CollectionDefinition("Integration", DisableParallelization = true)]
+[Trait("Integration", "True")]
+[SuppressMessage("Performance", "CA1861:Avoid constant arrays as arguments")]
+public class DotNetScenario : BaseScenario
+{
+ [Fact]
+ public void Run()
+ {
+ // $visible=true
+ // $tag=07 .NET CLI
+ // $priority=01
+ // $description=Run a dotnet application
+ // {
+ // Adds the namespace "HostApi" to use .NET build API
+ // ## using HostApi;
+
+ // Creates a new console project, running a command like: "dotnet new console -n MyApp --force"
+ new DotNetNew()
+ .WithTemplateName("console")
+ .WithName("MyApp")
+ .WithForce(true)
+ .Build().EnsureSuccess();
+
+ var tempDirectory = Path.Combine(Path.GetTempPath(), Path.GetRandomFileName());
+ new DotNetPublish()
+ .WithWorkingDirectory("MyApp")
+ .WithOutput(tempDirectory)
+ .Build().EnsureSuccess();
+
+ new DotNet()
+ .WithPathToApplication(Path.Combine(tempDirectory, "MyApp.dll"))
+ .Run().EnsureSuccess();
+ // }
+ }
+}
\ No newline at end of file
diff --git a/README.md b/README.md
index 0d9fec10..6da36a25 100644
--- a/README.md
+++ b/README.md
@@ -153,16 +153,19 @@ The created project contains 2 entry points:
- [Build a project](#build-a-project)
- [Build a project using MSBuild](#build-a-project-using-msbuild)
- [Clean a project](#clean-a-project)
+ - [Execute a dotnet application](#execute-a-dotnet-application)
- [Pack a project](#pack-a-project)
- [Publish a project](#publish-a-project)
- [Restore a project](#restore-a-project)
- [Restore local tools](#restore-local-tools)
- [Run a custom .NET command](#run-a-custom-.net-command)
+ - [Run a dotnet application](#run-a-dotnet-application)
- [Run a project](#run-a-project)
- [Run tests under dotCover](#run-tests-under-dotcover)
- [Test a project](#test-a-project)
- [Test a project using the MSBuild VSTest target](#test-a-project-using-the-msbuild-vstest-target)
- [Test an assembly](#test-an-assembly)
+ - [Run C# script](#run-c#-script)
- [Shuts down build servers](#shuts-down-build-servers)
- TeamCity API
- [TeamCity integration via service messages](#teamcity-integration-via-service-messages)
@@ -530,8 +533,6 @@ int? exitCode = new CommandLine("cmd", "/c", "TIMEOUT", "/T", "120")
.Run(default, TimeSpan.FromMilliseconds(1))
.EnsureSuccess()
.ExitCode;
-
-exitCode.HasValue.ShouldBeFalse();
```
@@ -566,10 +567,6 @@ var result = dockerRun
.Build()
.EnsureSuccess();
-// The "result" variable provides details about a build
-result.Errors.Any(message => message.State == BuildMessageState.StdError).ShouldBeFalse();
-result.ExitCode.ShouldBe(0);
-
string ToAbsoluteLinuxPath(string path) =>
"/" + path.Replace(":", "").Replace('\\', '/');
```
@@ -605,17 +602,20 @@ var result = new DockerRun(cmd, "mcr.microsoft.com/dotnet/sdk")
using HostApi;
// Creates a new library project, running a command like: "dotnet new classlib -n MyLib --force"
-new DotNetNew("xunit", "-n", "MyLib", "--force")
- .Build()
- .EnsureSuccess();
+new DotNetNew()
+ .WithTemplateName("xunit")
+ .WithName("MyLib")
+ .WithForce(true)
+ .Build().EnsureSuccess();
// Builds the library project, running a command like: "dotnet build" from the directory "MyLib"
+var messages = new List();
var result = new DotNetBuild()
.WithWorkingDirectory("MyLib")
- .Build()
- .EnsureSuccess();
+ .Build(message => messages.Add(message)).EnsureSuccess();
// The "result" variable provides details about a build
+messages.Count.ShouldBeGreaterThan(0, result.ToString());
result.Errors.Any(message => message.State == BuildMessageState.StdError).ShouldBeFalse();
result.ExitCode.ShouldBe(0);
@@ -624,10 +624,6 @@ result = new DotNetTest()
.WithWorkingDirectory("MyLib")
.Build()
.EnsureSuccess();
-
-result.ExitCode.ShouldBe(0);
-result.Summary.Tests.ShouldBe(1);
-result.Tests.Count(test => test.State == TestState.Finished).ShouldBe(1);
```
@@ -641,25 +637,25 @@ result.Tests.Count(test => test.State == TestState.Finished).ShouldBe(1);
using HostApi;
// Creates a new library project, running a command like: "dotnet new classlib -n MyLib --force"
-var result = new DotNetNew("classlib", "-n", "MyLib", "--force")
- .Build()
- .EnsureSuccess();
+var result = new DotNetNew()
+ .WithTemplateName("classlib")
+ .WithName("MyLib")
+ .WithForce(true)
+ .Build().EnsureSuccess();
result.ExitCode.ShouldBe(0);
// Builds the library project, running a command like: "dotnet build" from the directory "MyLib"
result = new DotNetBuild()
.WithWorkingDirectory("MyLib")
- .Build()
- .EnsureSuccess();
+ .Build().EnsureSuccess();
result.ExitCode.ShouldBe(0);
// Clean the project, running a command like: "dotnet clean" from the directory "MyLib"
result = new DotNetClean()
.WithWorkingDirectory("MyLib")
- .Build()
- .EnsureSuccess();
+ .Build().EnsureSuccess();
// The "result" variable provides details about a build
result.ExitCode.ShouldBe(0);
@@ -686,6 +682,34 @@ version.ShouldNotBeNull();
+### Execute a dotnet application
+
+
+
+``` CSharp
+// Adds the namespace "HostApi" to use .NET build API
+using HostApi;
+
+// Creates a new console project, running a command like: "dotnet new console -n MyApp --force"
+new DotNetNew()
+ .WithTemplateName("console")
+ .WithName("MyApp")
+ .WithForce(true)
+ .Build().EnsureSuccess();
+
+var tempDirectory = Path.Combine(Path.GetTempPath(), Path.GetRandomFileName());
+new DotNetPublish()
+ .WithWorkingDirectory("MyApp")
+ .WithOutput(tempDirectory)
+ .Build().EnsureSuccess();
+
+new DotNetExec()
+ .WithPathToApplication(Path.Combine(tempDirectory, "MyApp.dll"))
+ .Run().EnsureSuccess();
+```
+
+
+
### Test a project using the MSBuild VSTest target
@@ -695,23 +719,24 @@ version.ShouldNotBeNull();
using HostApi;
// Creates a new test project, running a command like: "dotnet new mstest -n MyTests --force"
-var result = new DotNetNew("mstest", "-n", "MyTests", "--force")
- .Build()
- .EnsureSuccess();
+var result = new DotNetNew()
+ .WithTemplateName("mstest")
+ .WithName("MyTests")
+ .WithForce(true)
+ .Build().EnsureSuccess();
-result.ExitCode.ShouldBe(0);
+result.ExitCode.ShouldBe(0, result.ToString());
// Runs tests via a command like: "dotnet msbuild /t:VSTest" from the directory "MyTests"
result = new MSBuild()
.WithTarget("VSTest")
.WithWorkingDirectory("MyTests")
- .Build()
- .EnsureSuccess();
+ .Build().EnsureSuccess();
// The "result" variable provides details about a build
-result.ExitCode.ShouldBe(0);
-result.Summary.Tests.ShouldBe(1);
-result.Tests.Count(test => test.State == TestState.Finished).ShouldBe(1);
+result.ExitCode.ShouldBe(0, result.ToString());
+result.Summary.Tests.ShouldBe(1, result.ToString());
+result.Tests.Count(test => test.State == TestState.Finished).ShouldBe(1, result.ToString());
```
@@ -725,9 +750,11 @@ result.Tests.Count(test => test.State == TestState.Finished).ShouldBe(1);
using HostApi;
// Creates a new library project, running a command like: "dotnet new classlib -n MyLib --force"
-var result = new DotNetNew("classlib", "-n", "MyLib", "--force")
- .Build()
- .EnsureSuccess();
+var result = new DotNetNew()
+ .WithTemplateName("classlib")
+ .WithName("MyLib")
+ .WithForce(true)
+ .Build().EnsureSuccess();
result.ExitCode.ShouldBe(0);
@@ -735,10 +762,7 @@ result.ExitCode.ShouldBe(0);
result = new DotNetPack()
.WithWorkingDirectory("MyLib")
.AddProps(("version", "1.2.3"))
- .Build()
- .EnsureSuccess();
-
-result.ExitCode.ShouldBe(0);
+ .Build().EnsureSuccess();
```
@@ -752,9 +776,12 @@ result.ExitCode.ShouldBe(0);
using HostApi;
// Creates a new library project, running a command like: "dotnet new classlib -n MyLib --force"
-var result = new DotNetNew("classlib", "-n", "MyLib", "--force", "-f", "net8.0")
- .Build()
- .EnsureSuccess();
+var result = new DotNetNew()
+ .WithTemplateName("classlib")
+ .AddArgs("-f", "net8.0")
+ .WithName("MyLib")
+ .WithForce(true)
+ .Build().EnsureSuccess();
result.ExitCode.ShouldBe(0);
@@ -762,10 +789,7 @@ result.ExitCode.ShouldBe(0);
result = new DotNetPublish()
.WithWorkingDirectory("MyLib")
.WithFramework("net8.0")
- .Build()
- .EnsureSuccess();
-
-result.ExitCode.ShouldBe(0);
+ .Build().EnsureSuccess();
```
@@ -779,19 +803,18 @@ result.ExitCode.ShouldBe(0);
using HostApi;
// Creates a new library project, running a command like: "dotnet new classlib -n MyLib --force"
-var result = new DotNetNew("classlib", "-n", "MyLib", "--force")
- .Build()
- .EnsureSuccess();
+var result = new DotNetNew()
+ .WithTemplateName("classlib")
+ .WithName("MyLib")
+ .WithForce(true)
+ .Build().EnsureSuccess();
result.ExitCode.ShouldBe(0);
// Restore the project, running a command like: "dotnet restore" from the directory "MyLib"
result = new DotNetRestore()
.WithWorkingDirectory("MyLib")
- .Build()
- .EnsureSuccess();
-
-result.ExitCode.ShouldBe(0);
+ .Build().EnsureSuccess();
```
@@ -805,15 +828,18 @@ result.ExitCode.ShouldBe(0);
using HostApi;
// Creates a new console project, running a command like: "dotnet new console -n MyApp --force"
-var result = new DotNetNew("console", "-n", "MyApp", "--force")
- .Build()
- .EnsureSuccess();
+var result = new DotNetNew()
+ .WithTemplateName("console")
+ .WithName("MyApp")
+ .WithForce(true)
+ .Build().EnsureSuccess();
result.ExitCode.ShouldBe(0);
// Runs the console project using a command like: "dotnet run" from the directory "MyApp"
var stdOut = new List();
-result = new DotNetRun().WithWorkingDirectory("MyApp")
+result = new DotNetRun()
+ .WithWorkingDirectory("MyApp")
.Build(message => stdOut.Add(message.Text))
.EnsureSuccess();
@@ -825,6 +851,34 @@ stdOut.ShouldBe(new[] {"Hello, World!"});
+### Run a dotnet application
+
+
+
+``` CSharp
+// Adds the namespace "HostApi" to use .NET build API
+using HostApi;
+
+// Creates a new console project, running a command like: "dotnet new console -n MyApp --force"
+new DotNetNew()
+ .WithTemplateName("console")
+ .WithName("MyApp")
+ .WithForce(true)
+ .Build().EnsureSuccess();
+
+var tempDirectory = Path.Combine(Path.GetTempPath(), Path.GetRandomFileName());
+new DotNetPublish()
+ .WithWorkingDirectory("MyApp")
+ .WithOutput(tempDirectory)
+ .Build().EnsureSuccess();
+
+new DotNet()
+ .WithPathToApplication(Path.Combine(tempDirectory, "MyApp.dll"))
+ .Run().EnsureSuccess();
+```
+
+
+
### Test a project
@@ -834,9 +888,11 @@ stdOut.ShouldBe(new[] {"Hello, World!"});
using HostApi;
// Creates a new test project, running a command like: "dotnet new mstest -n MyTests --force"
-var result = new DotNetNew("mstest", "-n", "MyTests", "--force")
- .Build()
- .EnsureSuccess();
+var result = new DotNetNew()
+ .WithTemplateName("mstest")
+ .WithName("MyTests")
+ .WithForce(true)
+ .Build().EnsureSuccess();
result.ExitCode.ShouldBe(0);
@@ -847,9 +903,9 @@ result = new DotNetTest()
.EnsureSuccess();
// The "result" variable provides details about a build
-result.ExitCode.ShouldBe(0);
-result.Summary.Tests.ShouldBe(1);
-result.Tests.Count(test => test.State == TestState.Finished).ShouldBe(1);
+result.ExitCode.ShouldBe(0, result.ToString());
+result.Summary.Tests.ShouldBe(1, result.ToString());
+result.Tests.Count(test => test.State == TestState.Finished).ShouldBe(1, result.ToString());
```
@@ -863,20 +919,21 @@ result.Tests.Count(test => test.State == TestState.Finished).ShouldBe(1);
using HostApi;
// Creates a new test project, running a command like: "dotnet new mstest -n MyTests --force"
-new DotNetNew("mstest", "-n", "MyTests", "--force")
- .Run()
- .EnsureSuccess();
+new DotNetNew()
+ .WithTemplateName("mstest")
+ .WithName("MyTests")
+ .WithForce(true)
+ .Run().EnsureSuccess();
// Creates the tool manifest and installs the dotCover tool locally
// It is better to run the following 2 commands manually
// and commit these changes to a source control
-new DotNetNew("tool-manifest")
- .Run()
- .EnsureSuccess();
+new DotNetNew()
+ .WithTemplateName("tool-manifest")
+ .Run().EnsureSuccess();
new DotNetCustom("tool", "install", "--local", "JetBrains.dotCover.GlobalTool")
- .Run()
- .EnsureSuccess();
+ .Run().EnsureSuccess();
// Creates a test command
var test = new DotNetTest()
@@ -897,12 +954,11 @@ var testUnderDotCover = test.Customize(cmd =>
// Runs tests under dotCover via a command like: "dotnet dotcover test ..."
var result = testUnderDotCover
- .Build()
- .EnsureSuccess();
+ .Build().EnsureSuccess();
// The "result" variable provides details about a build
-result.ExitCode.ShouldBe(0);
-result.Tests.Count(i => i.State == TestState.Finished).ShouldBe(1);
+result.ExitCode.ShouldBe(0, result.ToString());
+result.Tests.Count(i => i.State == TestState.Finished).ShouldBe(1, result.ToString());
// Generates a HTML code coverage report.
new DotNetCustom("dotCover", "report", $"--source={dotCoverSnapshot}", $"--output={dotCoverReport}", "--reportType=HTML")
@@ -926,7 +982,8 @@ var projectDir = Path.Combine(Path.GetTempPath(), Guid.NewGuid().ToString()[..4]
Directory.CreateDirectory(projectDir);
// Creates a local tool manifest
-new DotNetNew("tool-manifest")
+new DotNetNew()
+ .WithTemplateName("tool-manifest")
.WithWorkingDirectory(projectDir)
.Run()
.EnsureSuccess();
@@ -949,33 +1006,33 @@ new DotNetToolRestore()
using HostApi;
// Creates a new test project, running a command like: "dotnet new mstest -n MyTests --force"
-var result = new DotNetNew("mstest", "-n", "MyTests", "--force")
- .Build()
- .EnsureSuccess();
+var result = new DotNetNew()
+ .WithTemplateName("mstest")
+ .WithName("MyTests")
+ .WithForce(true)
+ .Build().EnsureSuccess();
-result.ExitCode.ShouldBe(0);
+result.ExitCode.ShouldBe(0, result.ToString());
// Builds the test project, running a command like: "dotnet build -c Release" from the directory "MyTests"
result = new DotNetBuild()
.WithWorkingDirectory("MyTests")
.WithConfiguration("Release")
.WithOutput("MyOutput")
- .Build()
- .EnsureSuccess();
+ .Build().EnsureSuccess();
-result.ExitCode.ShouldBe(0);
+result.ExitCode.ShouldBe(0, result.ToString());
// Runs tests via a command like: "dotnet vstest" from the directory "MyTests"
result = new VSTest()
.AddTestFileNames(Path.Combine("MyOutput", "MyTests.dll"))
.WithWorkingDirectory("MyTests")
- .Build()
- .EnsureSuccess();
+ .Build().EnsureSuccess();
// The "result" variable provides details about a build
-result.ExitCode.ShouldBe(0);
-result.Summary.Tests.ShouldBe(1);
-result.Tests.Count(test => test.State == TestState.Finished).ShouldBe(1);
+result.ExitCode.ShouldBe(0, result.ToString());
+result.Summary.Tests.ShouldBe(1, result.ToString());
+result.Tests.Count(test => test.State == TestState.Finished).ShouldBe(1, result.ToString());
```
@@ -989,11 +1046,13 @@ result.Tests.Count(test => test.State == TestState.Finished).ShouldBe(1);
using HostApi;
// Creates a new library project, running a command like: "dotnet new classlib -n MyLib --force"
-var result = new DotNetNew("classlib", "-n", "MyLib", "--force")
- .Build()
- .EnsureSuccess();
+var result = new DotNetNew()
+ .WithTemplateName("classlib")
+ .WithName("MyLib")
+ .WithForce(true)
+ .Build().EnsureSuccess();
-result.ExitCode.ShouldBe(0);
+result.ExitCode.ShouldBe(0, result.ToString());
// Builds the library project, running a command like: "dotnet msbuild /t:Build -restore /p:configuration=Release -verbosity=detailed" from the directory "MyLib"
result = new MSBuild()
@@ -1002,12 +1061,11 @@ result = new MSBuild()
.WithRestore(true)
.AddProps(("configuration", "Release"))
.WithVerbosity(DotNetVerbosity.Detailed)
- .Build()
- .EnsureSuccess();
+ .Build().EnsureSuccess();
// The "result" variable provides details about a build
-result.Errors.Any(message => message.State == BuildMessageState.StdError).ShouldBeFalse();
-result.ExitCode.ShouldBe(0);
+result.Errors.Any(message => message.State == BuildMessageState.StdError).ShouldBeFalse(result.ToString());
+result.ExitCode.ShouldBe(0, result.ToString());
```
@@ -1028,6 +1086,32 @@ new DotNetBuildServerShutdown()
+### Run C# script
+
+
+
+``` CSharp
+// Adds the namespace "HostApi" to use .NET build API
+using HostApi;
+
+var script = Path.GetTempFileName();
+File.WriteAllText(script, "Console.WriteLine($\"Hello, {Args[0]}!\");");
+
+var stdOut = new List();
+var result = new DotNetCsi()
+ .WithScript(script)
+ .AddArgs("World")
+ .Build(message => stdOut.Add(message.Text))
+ .EnsureSuccess();
+
+result.ExitCode.ShouldBe(0);
+
+// Checks StdOut
+stdOut.Contains("Hello, World!").ShouldBeTrue(result.ToString());
+```
+
+
+
### TeamCity integration via service messages
For more details how to use TeamCity service message API please see [this](https://github.com/JetBrains/TeamCity.ServiceMessages) page. Instead of creating a root message writer like in the following example: