Aiursoft CommandFramework is a framework for building command line tools.
- Auto argument parsing
- Auto help page generation
- Auto version page generation
- Run as service
- Auto dependency injection
- Auto command completion
- Auto logger with
--verbose
option support - Run as a single command app or a nested command app
With this framework, you can build a modern command line tool with just a few lines of code.
C:\workspace> ninja.exe
Description:
Nuget Ninja, a tool for detecting dependencies of .NET projects.
Usage:
Microsoft.NugetNinja [command] [options]
Options:
-p, --path <path> (REQUIRED) Path of the projects to be changed.
--nuget-server <nuget-server> If you want to use a customized nuget server instead of the official nuget.org,
--token <token> The PAT token which has privilege to access the nuget server.
-d, --dry-run Preview changes without actually making them
-v, --verbose Show detailed log
-?, -h, --help Show help and usage information
Commands:
all, all-officials The command to run all officially supported features.
remove-deprecated The command to replace all deprecated packages to new packages.
upgrade-pkg The command to upgrade all package references to possible latest and avoid conflicts.
clean-pkg The command to clean up possible useless package references.
clean-prj The command to clean up possible useless project references.
Command-line applications are a great way to automate repetitive tasks or even to be your own productivity tool. But building a command-line application in .NET is not easy. You need to parse the arguments, generate help pages, and so on. This project is designed to help you build a command-line application with just a few lines of code.
Run the following command to install Aiursoft.CommandFramework
to your project from nuget.org:
dotnet add package Aiursoft.CommandFramework
To get started, create a new .NET console application:
mkdir LearnCommandFramework
cd LearnCommandFramework
dotnet new console
Install the Aiursoft.CommandFramework
package:
dotnet add package Aiursoft.CommandFramework
In Aiursoft.CommandFramework
, a command handler is a class that can be executed as a command.
To build an executable command, you can do:
using System.CommandLine;
using System.CommandLine.Invocation;
using Aiursoft.CommandFramework;
using Aiursoft.CommandFramework.Framework;
public class DownloadHandler : ExecutableCommandHandlerBuilder
{
public static readonly Option<string> Url =
new(
aliases: new[] { "-u", "--url" },
description: "The target url to download.")
{
IsRequired = true
};
protected override string Name => "download";
protected override string Description => "Download an HTTP Url.";
protected override Option[] GetCommandOptions() => new Option[]
{
// Configure your options here.
DownloadHandler.Url
};
protected override Task Execute(InvocationContext context)
{
// Your code entry:
var url = context.ParseResult.GetValueForOption(DownloadHandler.Url);
Console.WriteLine($"Downloading file from: {url}...");
return Task.CompletedTask;
}
}
// Now you can start your app! Finish your `Program.cs` entry code!
public class Program
{
public static async Task<int> Main(string[] args)
{
return await new SingleCommandApp<DownloadHandler>()
.WithDefaultOption(DownloadHandler.Url)
.RunAsync(args);
}
}
Build and run you app!
$ your-downloader.exe --url https://www.aiursoft.cn
# outputs:
Downloading file from: https://www.aiursoft.cn...
It it super simple to test a command handler.
Assuming you are using Microsoft.VisualStudio.TestTools.UnitTesting
, you can do:
using Aiursoft.CommandFramework;
using Microsoft.VisualStudio.TestTools.UnitTesting;
[TestClass]
public class IntegrationTests
{
private readonly SingleCommandApp<DownloadHandler> _program = new SingleCommandApp<DownloadHandler>()
.WithDefaultOption(DownloadHandler.Url);
[TestMethod]
public async Task InvokeHelp()
{
var result = await _program.TestRunAsync(new[] { "--help" });
Assert.AreEqual(0, result.ProgramReturn);
}
[TestMethod]
public async Task InvokeVersion()
{
var result = await _program.TestRunAsync(new[] { "--version" });
Assert.AreEqual(0, result.ProgramReturn);
}
[TestMethod]
public async Task InvokeUnknown()
{
var result = await _program.TestRunAsync(new[] { "--wtf" });
Assert.AreEqual(1, result.ProgramReturn);
}
}
Now write you UT, you can follow this practice:
- Prepare some environment
- Run your command handler
- Assert the result
- Clean up the environment
In the previous steps, we have learned how to build a single command app. A single command app is a command line tool that only has one command.
This is useful for something with limited function, like:
- Ping tool
- File download tool
- Web server tool
But in more scenarios, we usually need a command line tool with multiple commands. For example, git
has many commands like git clone
, git commit
, git push
, etc.
A nested command app is a command line tool that has multiple commands. like:
To do that, first you need to build several executable command handlers. And then wrap them with a NavigationCommandHandlerBuilder
:
public class NetworkHandler : NavigationCommandHandlerBuilder
{
protected override string Name => "network";
protected override string Description => "Network related commands.";
protected override CommandHandlerBuilder[] GetSubCommandHandlers()
{
return
[
// Where the `DownloadHandler` is an executable command handler.
new DownloadHandler()
];
}
}
And it's very similar to build a nested command app:
using System.CommandLine;
using System.CommandLine.Invocation;
using Aiursoft.CommandFramework;
using Aiursoft.CommandFramework.Framework;
using Aiursoft.CommandFramework.Models;
// Program.cs of the nested command app.
public class Program
{
public static async Task<int> Main(string[] args)
{
return await new NestedCommandApp()
.WithGlobalOptions(CommonOptionsProvider.DryRunOption)
.WithGlobalOptions(CommonOptionsProvider.VerboseOption)
.WithFeature(new NetworkHandler())
.RunAsync(args);
}
}
Now you can try:
your-app network download --url https://www.aiursoft.cn
It's easy to build a command app with dependency injection.
Of course, you need to register your services in your Startup
class first:
using Aiursoft.CommandFramework.Abstracts;
using Microsoft.Extensions.DependencyInjection;
namespace Aiursoft.DotDownload.Http;
public class Downloader
{
private readonly HttpClient _client;
public Downloader(HttpClient client)
{
_client = client;
}
public async Task DownloadWithWatchAsync(string url, string savePath, int blockSize, int threads, bool showProgressBar)
{
// Your download logic here.
}
}
public class Startup : IStartUp
{
public void ConfigureServices(IServiceCollection services)
{
services.AddHttpClient(nameof(Downloader)).ConfigurePrimaryHttpMessageHandler(() =>
{
return new HttpClientHandler()
{
AllowAutoRedirect = true,
};
});
services.AddTransient<Downloader>();
}
}
Then in your Execute(InvocationContext context)
function:
// This code is inside an `ExecutableCommandHandlerBuilder`.
protected override async Task Execute(InvocationContext context)
{
var verbose = context.ParseResult.GetValueForOption(CommonOptionsProvider.VerboseOption);
var host = ServiceBuilder
.CreateCommandHostBuilder<Startup>(verbose) // Your own startup class.
.Build();
var downloader = host.Services.GetRequiredService<Downloader>(); // Get a service from dependency injection
await downloader.DownloadWithWatchAsync(url, savePath, blockSize, threads, showProgressBar: !verbose);
}
That's it!
You can even start a background service in your command line tool!
To build a background service, you need to implement an IHostedService
. Here use ServerMonitor
as an example:
public class Startup : IStartUp
{
public void ConfigureServices(IServiceCollection services)
{
services.AddTransient<ServerInitializer>();
services.AddSingleton<IHostedService, ServerMonitor>();
}
}
// In your `ExecutableCommandHandlerBuilder`:
protected override async Task Execute(InvocationContext context)
{
var verbose = context.ParseResult.GetValueForOption(CommonOptionsProvider.VerboseOption);
var profile = context.ParseResult.GetValueForOption(_profile);
var host = ServiceBuilder
.CreateCommandHostBuilder<Startup>(verbose)
.ConfigureServices((hostBuilderContext, services)=>
{
// You can configure your services here.
services.Configure<ProfileConfig>(config =>
{
config.Profile = profile;
});
})
.Build();
await host.StartAsync(); // Now your background service is running!
await host.WaitForShutdownAsync();
}
To read more about how to write an IHostedService
, please read this doc from Microsoft.
It's simple. You can use the ProgressBar
class from Aiursoft.CommandFramework
:
using Aiursoft.CommandFramework.Models;
ProgressBar? bar = null;
if (showProgress) bar = new ProgressBar();
var completedTasks = 0;
var totalTasks = tasks.Length;
foreach (var task in tasks)
{
if (showProgress)
{
Interlocked.Increment(ref completedTasks);
// ReSharper disable once AccessToDisposedClosure
bar?.Report((double)completedTasks / totalTasks);
}
}
bar?.Dispose();
That's it! When running, you may see:
[########################-----------------------] 50%
If you want to explore a real project built with this framework, please download the following project:
Single command app:
- DotDownload as an example.
- Httping as an example.
- Static as an example.
- Dotlang as an example.
Nested command app:
- NiBot as an example.
- Nuget Ninja as an example.
- Happy Recorder as an example.
- Parser as an example.
Background service:
- IPMI Controller as an example.
There are many ways to contribute to the project: logging bugs, submitting pull requests, reporting issues, and creating suggestions.
Even if you with push rights on the repository, you should create a personal fork and create feature branches there when you need them. This keeps the main repository clean and your workflow cruft out of sight.
We're also interested in your feedback on the future of this project. You can submit a suggestion or feature request through the issue tracker. To make this process more effective, we're asking that these include more information to help define them more clearly.