Aiursoft.CommandFramework
7.0.18
See the version list below for details.
dotnet add package Aiursoft.CommandFramework --version 7.0.18
NuGet\Install-Package Aiursoft.CommandFramework -Version 7.0.18
<PackageReference Include="Aiursoft.CommandFramework" Version="7.0.18" />
paket add Aiursoft.CommandFramework --version 7.0.18
#r "nuget: Aiursoft.CommandFramework, 7.0.18"
// Install Aiursoft.CommandFramework as a Cake Addin #addin nuget:?package=Aiursoft.CommandFramework&version=7.0.18 // Install Aiursoft.CommandFramework as a Cake Tool #tool nuget:?package=Aiursoft.CommandFramework&version=7.0.18
Aiursoft CommandFramework
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.
Why this project?
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.
How to install
Run the following command to install Aiursoft.CommandFramework
to your project from nuget.org:
dotnet add package Aiursoft.CommandFramework
Learn step 1: How to build an executable command handler?
In Aiursoft.CommandFramework
, a command handler is a class that can be executed as a command.
To build an executable command, you can do:
public class DownloadHandler : ExecutableCommandHandlerBuilder
{
public static readonly Option<string> Url =
new(
aliases: new[] { "-u", "--url" },
description: "The target url to download.");
public override string Name => "download";
public override string Description => "Download an HTTP Url.";
public override Option[] GetCommandOptions() => new Option[]
{
// Configure your options here.
DownloadHandler.Url
};
protected override async Task Execute(InvocationContext context)
{
// Your code entry:
var url = context.ParseResult.GetValueForOption(DownloadHandler.Url);
Console.WriteLine($"Downloading file from: {url}...");
}
}
// Now you can start your app! Finish your `Program.cs` entry code!
public class Program
{
public static async Task Main(string[] args)
{
return await new DownloadHandler()
.RunAsync(args, defaultOption: OptionsProvider.Url);
}
}
Build and run you app!
$ your-downloader.exe --url https://www.aiursoft.cn
# outputs:
Downloading file from: https://www.aiursoft.cn...
Learn step 2: How to test your command handler?
It it super simple to test a command handler:
using Microsoft.VisualStudio.TestTools.UnitTesting;
[TestClass]
public class IntegrationTests
{
private readonly DownloadHandler _program = new();
[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
Learn step 3: Understand single command app and nested command app
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:
Learn step 4: How to build a nested command app?
To do that, first you need to build several executable command handlers. And then wrap them with a NavigationCommandHandlerBuilder
:
public class ConfigHandler : NavigationCommandHandlerBuilder
{
public override string Name => "config";
public override string Description => "Configuration management.";
public override CommandHandlerBuilder[] GetSubCommandHandlers()
{
return new CommandHandlerBuilder[]
{
new GetDbLocationHandler(), // Where the `GetDbLocationHandler` is an `ExecutableCommandHandlerBuilder`.
new SetDbLocationHandler() // Where the `SetDbLocationHandler` is an `ExecutableCommandHandlerBuilder`.
};
}
}
And it's a little different to run a nested command app:
// Program.cs of the nested command app.
return await new AiursoftCommandApp()
.Configure(command =>
{
command
.AddGlobalOptions()
.AddPlugins(
new CalendarPlugin() // One app can have multiple plugins.
);
})
.RunAsync(args);
// This is a plugin for your app
public class CalendarPlugin : IPlugin
{
public ICommandHandlerBuilder[] Install() // One plugin can provide multiple command handlers.
{
return new ICommandHandlerBuilder[] // Both `NavigationCommandHandlerBuilder` and `ExecutableCommandHandlerBuilder` can be registered.
{
new ConfigHandler() // Where the `ConfigHandler` is a `NavigationCommandHandlerBuilder`.
};
}
}
// Your app may have some global options. So you can write a global options provider.
public static class OptionsProvider
{
public static Command AddGlobalOptions(this Command command)
{
var options = new Option[]
{
CommonOptionsProvider.DryRunOption,
CommonOptionsProvider.VerboseOption
};
foreach (var option in options)
{
command.AddGlobalOption(option);
}
return command;
}
}
Now you can try:
your-app config get-db-location
Learn step 5: How to test a nested command app?
It's also simple to test a nested command app:
[TestClass]
public class IntegrationTests
{
private readonly AiursoftCommandApp _program;
public IntegrationTests()
{
_program = new AiursoftCommandApp()
.Configure(command =>
{
command
.AddGlobalOptions()
.AddPlugins(
new CalendarPlugin()
);
});
}
[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[] { "config", "get-db-location" });
Assert.AreEqual(0, result.ProgramReturn);
}
[TestMethod]
public async Task InvokeWithoutArg()
{
var result = await _program.TestRunAsync(Array.Empty<string>());
Assert.AreEqual(1, result.ProgramReturn);
}
}
Learn step 6: How to build a command app with dependency injection?
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 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!
Learn step 7: How to build and configure a background service?
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.
Learn step 8: Read more examples
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.
Nested command app:
- Nuget Ninja as an example.
- Happy Recorder as an example.
Background service:
- IPMI Controller as an example.
How to contribute
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.
Product | Versions Compatible and additional computed target framework versions. |
---|---|
.NET | net7.0 is compatible. net7.0-android was computed. net7.0-ios was computed. net7.0-maccatalyst was computed. net7.0-macos was computed. net7.0-tvos was computed. net7.0-windows was computed. net8.0 was computed. net8.0-android was computed. net8.0-browser was computed. net8.0-ios was computed. net8.0-maccatalyst was computed. net8.0-macos was computed. net8.0-tvos was computed. net8.0-windows was computed. |
-
net7.0
- Microsoft.Extensions.Hosting (>= 7.0.1)
- System.CommandLine (>= 2.0.0-beta4.22272.1)
NuGet packages (3)
Showing the top 3 NuGet packages that depend on Aiursoft.CommandFramework:
Package | Downloads |
---|---|
Anduin.HappyRecorder.PluginFramework
Package Description |
|
Aiursoft.Dotlang.Core
Package Description |
|
Aiursoft.NiBot.Dedup
Nuget package of 'Dedup' provided by Aiursoft |
GitHub repositories
This package is not used by any popular GitHub repositories.
Version | Downloads | Last updated |
---|---|---|
8.0.6 | 62 | 10/9/2024 |
8.0.5 | 51 | 10/7/2024 |
8.0.4 | 55 | 7/9/2024 |
8.0.3 | 55 | 6/28/2024 |
8.0.2 | 60 | 6/27/2024 |
8.0.1 | 87 | 2/19/2024 |
8.0.0 | 65 | 2/19/2024 |
7.0.25 | 95 | 1/30/2024 |
7.0.24 | 151 | 12/12/2023 |
7.0.23 | 79 | 12/12/2023 |
7.0.22 | 70 | 12/12/2023 |
7.0.21 | 78 | 12/12/2023 |
7.0.20 | 82 | 12/11/2023 |
7.0.19 | 83 | 12/11/2023 |
7.0.18 | 77 | 12/11/2023 |
7.0.17 | 87 | 12/6/2023 |
7.0.16 | 84 | 12/6/2023 |
7.0.15 | 83 | 12/6/2023 |
7.0.14 | 79 | 12/6/2023 |
7.0.13 | 73 | 12/6/2023 |
7.0.12 | 78 | 12/6/2023 |
7.0.11 | 86 | 12/6/2023 |
7.0.10 | 86 | 12/6/2023 |
7.0.9 | 75 | 12/6/2023 |
7.0.8 | 86 | 12/6/2023 |
7.0.7 | 78 | 12/6/2023 |
7.0.6 | 105 | 11/11/2023 |
7.0.5 | 89 | 11/2/2023 |
7.0.4 | 74 | 11/2/2023 |
7.0.3 | 78 | 11/2/2023 |
7.0.2 | 96 | 9/13/2023 |
7.0.1 | 96 | 9/13/2023 |
7.0.0 | 90 | 9/5/2023 |
6.0.0 | 205 | 7/20/2023 |
1.0.0 | 132 | 7/20/2023 |