Skip to content

Add Localization Features package for System.CommandLine #1013

New issue

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

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

Already on GitHub? Sign in to your account

Draft
wants to merge 12 commits into
base: main
Choose a base branch
from
Draft
49 changes: 47 additions & 2 deletions System.CommandLine.sln
Original file line number Diff line number Diff line change
Expand Up @@ -54,9 +54,15 @@ Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "System.CommandLine.Benchmar
EndProject
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "System.CommandLine.Hosting", "src\System.CommandLine.Hosting\System.CommandLine.Hosting.csproj", "{644C4B4A-4A32-4307-9F71-C3BF901FFB66}"
EndProject
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "System.CommandLine.Hosting.Tests", "src\System.CommandLine.Hosting.Tests\System.CommandLine.Hosting.Tests.csproj", "{39483140-BC26-4CAD-BBAE-3DC76C2F16CF}"
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "System.CommandLine.Hosting.Tests", "src\System.CommandLine.Hosting.Tests\System.CommandLine.Hosting.Tests.csproj", "{39483140-BC26-4CAD-BBAE-3DC76C2F16CF}"
EndProject
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "HostingPlayground", "samples\HostingPlayground\HostingPlayground.csproj", "{0BF6958D-9EE3-4623-B3D6-4DA77EAC1906}"
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "HostingPlayground", "samples\HostingPlayground\HostingPlayground.csproj", "{0BF6958D-9EE3-4623-B3D6-4DA77EAC1906}"
EndProject
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "System.CommandLine.Localization", "src\System.CommandLine.Localization\System.CommandLine.Localization.csproj", "{9FD1BB47-F1B9-48A2-BD54-C324357C7BEF}"
EndProject
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "System.CommandLine.Localization.Tests", "src\System.CommandLine.Localization.Tests\System.CommandLine.Localization.Tests.csproj", "{182581D0-4AAB-49EE-8713-D890A6401DCF}"
EndProject
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "LocalizationPlayground", "samples\LocalizationPlayground\LocalizationPlayground.csproj", "{8B62D16B-1CB0-40E6-810F-2FC6F8215BE0}"
EndProject
Global
GlobalSection(SolutionConfigurationPlatforms) = preSolution
Expand Down Expand Up @@ -236,6 +242,42 @@ Global
{0BF6958D-9EE3-4623-B3D6-4DA77EAC1906}.Release|x64.Build.0 = Release|Any CPU
{0BF6958D-9EE3-4623-B3D6-4DA77EAC1906}.Release|x86.ActiveCfg = Release|Any CPU
{0BF6958D-9EE3-4623-B3D6-4DA77EAC1906}.Release|x86.Build.0 = Release|Any CPU
{9FD1BB47-F1B9-48A2-BD54-C324357C7BEF}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
{9FD1BB47-F1B9-48A2-BD54-C324357C7BEF}.Debug|Any CPU.Build.0 = Debug|Any CPU
{9FD1BB47-F1B9-48A2-BD54-C324357C7BEF}.Debug|x64.ActiveCfg = Debug|Any CPU
{9FD1BB47-F1B9-48A2-BD54-C324357C7BEF}.Debug|x64.Build.0 = Debug|Any CPU
{9FD1BB47-F1B9-48A2-BD54-C324357C7BEF}.Debug|x86.ActiveCfg = Debug|Any CPU
{9FD1BB47-F1B9-48A2-BD54-C324357C7BEF}.Debug|x86.Build.0 = Debug|Any CPU
{9FD1BB47-F1B9-48A2-BD54-C324357C7BEF}.Release|Any CPU.ActiveCfg = Release|Any CPU
{9FD1BB47-F1B9-48A2-BD54-C324357C7BEF}.Release|Any CPU.Build.0 = Release|Any CPU
{9FD1BB47-F1B9-48A2-BD54-C324357C7BEF}.Release|x64.ActiveCfg = Release|Any CPU
{9FD1BB47-F1B9-48A2-BD54-C324357C7BEF}.Release|x64.Build.0 = Release|Any CPU
{9FD1BB47-F1B9-48A2-BD54-C324357C7BEF}.Release|x86.ActiveCfg = Release|Any CPU
{9FD1BB47-F1B9-48A2-BD54-C324357C7BEF}.Release|x86.Build.0 = Release|Any CPU
{182581D0-4AAB-49EE-8713-D890A6401DCF}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
{182581D0-4AAB-49EE-8713-D890A6401DCF}.Debug|Any CPU.Build.0 = Debug|Any CPU
{182581D0-4AAB-49EE-8713-D890A6401DCF}.Debug|x64.ActiveCfg = Debug|Any CPU
{182581D0-4AAB-49EE-8713-D890A6401DCF}.Debug|x64.Build.0 = Debug|Any CPU
{182581D0-4AAB-49EE-8713-D890A6401DCF}.Debug|x86.ActiveCfg = Debug|Any CPU
{182581D0-4AAB-49EE-8713-D890A6401DCF}.Debug|x86.Build.0 = Debug|Any CPU
{182581D0-4AAB-49EE-8713-D890A6401DCF}.Release|Any CPU.ActiveCfg = Release|Any CPU
{182581D0-4AAB-49EE-8713-D890A6401DCF}.Release|Any CPU.Build.0 = Release|Any CPU
{182581D0-4AAB-49EE-8713-D890A6401DCF}.Release|x64.ActiveCfg = Release|Any CPU
{182581D0-4AAB-49EE-8713-D890A6401DCF}.Release|x64.Build.0 = Release|Any CPU
{182581D0-4AAB-49EE-8713-D890A6401DCF}.Release|x86.ActiveCfg = Release|Any CPU
{182581D0-4AAB-49EE-8713-D890A6401DCF}.Release|x86.Build.0 = Release|Any CPU
{8B62D16B-1CB0-40E6-810F-2FC6F8215BE0}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
{8B62D16B-1CB0-40E6-810F-2FC6F8215BE0}.Debug|Any CPU.Build.0 = Debug|Any CPU
{8B62D16B-1CB0-40E6-810F-2FC6F8215BE0}.Debug|x64.ActiveCfg = Debug|Any CPU
{8B62D16B-1CB0-40E6-810F-2FC6F8215BE0}.Debug|x64.Build.0 = Debug|Any CPU
{8B62D16B-1CB0-40E6-810F-2FC6F8215BE0}.Debug|x86.ActiveCfg = Debug|Any CPU
{8B62D16B-1CB0-40E6-810F-2FC6F8215BE0}.Debug|x86.Build.0 = Debug|Any CPU
{8B62D16B-1CB0-40E6-810F-2FC6F8215BE0}.Release|Any CPU.ActiveCfg = Release|Any CPU
{8B62D16B-1CB0-40E6-810F-2FC6F8215BE0}.Release|Any CPU.Build.0 = Release|Any CPU
{8B62D16B-1CB0-40E6-810F-2FC6F8215BE0}.Release|x64.ActiveCfg = Release|Any CPU
{8B62D16B-1CB0-40E6-810F-2FC6F8215BE0}.Release|x64.Build.0 = Release|Any CPU
{8B62D16B-1CB0-40E6-810F-2FC6F8215BE0}.Release|x86.ActiveCfg = Release|Any CPU
{8B62D16B-1CB0-40E6-810F-2FC6F8215BE0}.Release|x86.Build.0 = Release|Any CPU
EndGlobalSection
GlobalSection(SolutionProperties) = preSolution
HideSolutionNode = FALSE
Expand All @@ -255,6 +297,9 @@ Global
{644C4B4A-4A32-4307-9F71-C3BF901FFB66} = {E5B1EC71-0FC4-4FAA-9C65-32D5016FBC45}
{39483140-BC26-4CAD-BBAE-3DC76C2F16CF} = {E5B1EC71-0FC4-4FAA-9C65-32D5016FBC45}
{0BF6958D-9EE3-4623-B3D6-4DA77EAC1906} = {6749FB3E-39DE-4321-A39E-525278E9408D}
{9FD1BB47-F1B9-48A2-BD54-C324357C7BEF} = {E5B1EC71-0FC4-4FAA-9C65-32D5016FBC45}
{182581D0-4AAB-49EE-8713-D890A6401DCF} = {E5B1EC71-0FC4-4FAA-9C65-32D5016FBC45}
{8B62D16B-1CB0-40E6-810F-2FC6F8215BE0} = {6749FB3E-39DE-4321-A39E-525278E9408D}
EndGlobalSection
GlobalSection(ExtensibilityGlobals) = postSolution
SolutionGuid = {5C159F93-800B-49E7-9905-EE09F8B8434A}
Expand Down
1 change: 1 addition & 0 deletions samples/LocalizationPlayground/.gitignore
Original file line number Diff line number Diff line change
@@ -0,0 +1 @@
xlf
Original file line number Diff line number Diff line change
@@ -0,0 +1,30 @@
using System;
using System.CommandLine.Builder;
using System.CommandLine.Invocation;
using System.Globalization;
using System.Threading;

namespace LocalizationPlayground
{
internal static class CultureEnvironmentCommandLineExtensions
{
internal static CommandLineBuilder UseCultureEnvironment(
this CommandLineBuilder builder)
{
return builder.UseMiddleware(async (context, next) =>
{
if (Environment.GetEnvironmentVariable("DOTNET_SYSTEM_GLOBALIZATION_CULTURE") is string culture &&
!string.IsNullOrEmpty(culture))
CultureInfo.CurrentCulture = CultureInfo.CurrentUICulture = CultureInfo.GetCultureInfo(culture);

await next(context).ConfigureAwait(false);

if (context.InvocationResult is IInvocationResult innerResult)
{
var execCtx = ExecutionContext.Capture();
context.InvocationResult = new ExecutionContextRestoringInvocationResult(execCtx, innerResult);
}
}, MiddlewareOrder.ExceptionHandler);
}
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,35 @@
using System;
using System.CommandLine.Invocation;
using System.Threading;

namespace LocalizationPlayground
{
internal class ExecutionContextRestoringInvocationResult : IInvocationResult
{
private static readonly ContextCallback executionContextApplyCallback = state =>
{
var (@this, context) = (ValueTuple<IInvocationResult, InvocationContext>)state!;
@this.Apply(context);
};

private readonly ExecutionContext? executionContext;
private readonly IInvocationResult innerResult;

public ExecutionContextRestoringInvocationResult(ExecutionContext? executionContext, IInvocationResult innerResult)
{
this.executionContext = executionContext;
this.innerResult = innerResult;
}

public void Apply(InvocationContext context)
{
if (executionContext is null)
innerResult.Apply(context);
else
ExecutionContext.Run(executionContext,
executionContextApplyCallback,
(innerResult, context)
);
}
}
}
14 changes: 14 additions & 0 deletions samples/LocalizationPlayground/LocalizationPlayground.csproj
Original file line number Diff line number Diff line change
@@ -0,0 +1,14 @@
<Project Sdk="Microsoft.NET.Sdk">

<PropertyGroup>
<Nullable>enable</Nullable>
<OutputType>Exe</OutputType>
<TargetFramework>netcoreapp3.1</TargetFramework>
</PropertyGroup>

<ItemGroup>
<ProjectReference Include="..\..\src\System.CommandLine\System.CommandLine.csproj" />
<ProjectReference Include="..\..\src\System.CommandLine.Localization\System.CommandLine.Localization.csproj" />
</ItemGroup>

</Project>
74 changes: 74 additions & 0 deletions samples/LocalizationPlayground/Program.cs
Original file line number Diff line number Diff line change
@@ -0,0 +1,74 @@
using System;
using System.CommandLine;
using System.CommandLine.Builder;
using System.CommandLine.Invocation;
using System.CommandLine.Localization;
using System.CommandLine.Parsing;
using System.Globalization;
using System.Threading.Tasks;
using Microsoft.Extensions.Localization;

namespace LocalizationPlayground
{
public static class Program
{
public static Task<int> Main(string[] args)
{
CultureInfo.CurrentCulture = CultureInfo.InvariantCulture;
var parser = new CommandLineBuilder(
new RootCommand
{
Description = "Playground for localized CommandLine",
Handler = CommandHandler.Create((int count, string name, InvocationContext invocation, IStringLocalizerFactory localizerFactory) =>
{
var cult = CultureInfo.CurrentUICulture;
var germanResourceNames = typeof(Program).Assembly
.GetSatelliteAssembly(CultureInfo.GetCultureInfo("de"))
.GetManifestResourceNames();

var localizer = localizerFactory.Create(typeof(Program));
var locCultureInfo = localizer.GetString("Current culture: {0}", cult.NativeName);
Console.WriteLine(locCultureInfo);
var locLine = localizer.GetString("Hello {0}!", name);

var availableStrings = localizer.GetAllStrings(true);

_ = germanResourceNames;
_ = availableStrings;

for (int i = 0; i < count; i++)
{
Console.WriteLine(locLine);
}

Console.WriteLine();
invocation.InvocationResult = new HelpResult();
}),
})
.AddOption(new Option<int>(new[] { "--count", "-c" }, () => 1)
{
Name = "count",
Description = "Count of lines to print",
Argument =
{
Name = "COUNT",
Description = "An integer value",
Arity = ArgumentArity.ZeroOrOne,
}
})
.AddArgument(new Argument<string>("NAME")
{
Description = "The name to display",
Arity = ArgumentArity.ExactlyOne,
})
.UseEnvironmentVariableDirective()
.UseDebugDirective()
.UseHelp()
.UseVersionOption()
.UseCultureEnvironment()
.UseLocalization()
.Build();
return parser.InvokeAsync(args ?? Array.Empty<string>());
}
}
}
138 changes: 138 additions & 0 deletions samples/LocalizationPlayground/Program.resx
Original file line number Diff line number Diff line change
@@ -0,0 +1,138 @@
<?xml version="1.0" encoding="utf-8"?>
<root>
<!--
Microsoft ResX Schema

Version 2.0

The primary goals of this format is to allow a simple XML format
that is mostly human readable. The generation and parsing of the
various data types are done through the TypeConverter classes
associated with the data types.

Example:

... ado.net/XML headers & schema ...
<resheader name="resmimetype">text/microsoft-resx</resheader>
<resheader name="version">2.0</resheader>
<resheader name="reader">System.Resources.ResXResourceReader, System.Windows.Forms, ...</resheader>
<resheader name="writer">System.Resources.ResXResourceWriter, System.Windows.Forms, ...</resheader>
<data name="Name1"><value>this is my long string</value><comment>this is a comment</comment></data>
<data name="Color1" type="System.Drawing.Color, System.Drawing">Blue</data>
<data name="Bitmap1" mimetype="application/x-microsoft.net.object.binary.base64">
<value>[base64 mime encoded serialized .NET Framework object]</value>
</data>
<data name="Icon1" type="System.Drawing.Icon, System.Drawing" mimetype="application/x-microsoft.net.object.bytearray.base64">
<value>[base64 mime encoded string representing a byte array form of the .NET Framework object]</value>
<comment>This is a comment</comment>
</data>

There are any number of "resheader" rows that contain simple
name/value pairs.

Each data row contains a name, and value. The row also contains a
type or mimetype. Type corresponds to a .NET class that support
text/value conversion through the TypeConverter architecture.
Classes that don't support this are serialized and stored with the
mimetype set.

The mimetype is used for serialized objects, and tells the
ResXResourceReader how to depersist the object. This is currently not
extensible. For a given mimetype the value must be set accordingly:

Note - application/x-microsoft.net.object.binary.base64 is the format
that the ResXResourceWriter will generate, however the reader can
read any of the formats listed below.

mimetype: application/x-microsoft.net.object.binary.base64
value : The object must be serialized with
: System.Runtime.Serialization.Formatters.Binary.BinaryFormatter
: and then encoded with base64 encoding.

mimetype: application/x-microsoft.net.object.soap.base64
value : The object must be serialized with
: System.Runtime.Serialization.Formatters.Soap.SoapFormatter
: and then encoded with base64 encoding.

mimetype: application/x-microsoft.net.object.bytearray.base64
value : The object must be serialized into a byte array
: using a System.ComponentModel.TypeConverter
: and then encoded with base64 encoding.
-->
<xsd:schema id="root" xmlns="" xmlns:xsd="http://www.w3.org/2001/XMLSchema" xmlns:msdata="urn:schemas-microsoft-com:xml-msdata">
<xsd:import namespace="http://www.w3.org/XML/1998/namespace" />
<xsd:element name="root" msdata:IsDataSet="true">
<xsd:complexType>
<xsd:choice maxOccurs="unbounded">
<xsd:element name="metadata">
<xsd:complexType>
<xsd:sequence>
<xsd:element name="value" type="xsd:string" minOccurs="0" />
</xsd:sequence>
<xsd:attribute name="name" use="required" type="xsd:string" />
<xsd:attribute name="type" type="xsd:string" />
<xsd:attribute name="mimetype" type="xsd:string" />
<xsd:attribute ref="xml:space" />
</xsd:complexType>
</xsd:element>
<xsd:element name="assembly">
<xsd:complexType>
<xsd:attribute name="alias" type="xsd:string" />
<xsd:attribute name="name" type="xsd:string" />
</xsd:complexType>
</xsd:element>
<xsd:element name="data">
<xsd:complexType>
<xsd:sequence>
<xsd:element name="value" type="xsd:string" minOccurs="0" msdata:Ordinal="1" />
<xsd:element name="comment" type="xsd:string" minOccurs="0" msdata:Ordinal="2" />
</xsd:sequence>
<xsd:attribute name="name" type="xsd:string" use="required" msdata:Ordinal="1" />
<xsd:attribute name="type" type="xsd:string" msdata:Ordinal="3" />
<xsd:attribute name="mimetype" type="xsd:string" msdata:Ordinal="4" />
<xsd:attribute ref="xml:space" />
</xsd:complexType>
</xsd:element>
<xsd:element name="resheader">
<xsd:complexType>
<xsd:sequence>
<xsd:element name="value" type="xsd:string" minOccurs="0" msdata:Ordinal="1" />
</xsd:sequence>
<xsd:attribute name="name" type="xsd:string" use="required" />
</xsd:complexType>
</xsd:element>
</xsd:choice>
</xsd:complexType>
</xsd:element>
</xsd:schema>
<resheader name="resmimetype">
<value>text/microsoft-resx</value>
</resheader>
<resheader name="version">
<value>2.0</value>
</resheader>
<resheader name="reader">
<value>System.Resources.ResXResourceReader, System.Windows.Forms, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089</value>
</resheader>
<resheader name="writer">
<value>System.Resources.ResXResourceWriter, System.Windows.Forms, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089</value>
</resheader>
<data name="An integer value" xml:space="preserve">
<value>An integer value</value>
</data>
<data name="Count of lines to print" xml:space="preserve">
<value>Count of lines to print</value>
</data>
<data name="Current culture: {0}" xml:space="preserve">
<value>Current language settings: {0}</value>
</data>
<data name="Hello {0}!" xml:space="preserve">
<value>Hello {0}!</value>
</data>
<data name="Playground for localized CommandLine" xml:space="preserve">
<value>Playground for localized command-line applications</value>
</data>
<data name="The name to display" xml:space="preserve">
<value>The name to display</value>
</data>
</root>
8 changes: 8 additions & 0 deletions samples/LocalizationPlayground/Properties/launchSettings.json
Original file line number Diff line number Diff line change
@@ -0,0 +1,8 @@
{
"profiles": {
"LocalizationPlayground": {
"commandName": "Project",
"commandLineArgs": "[env:DOTNET_SYSTEM_GLOBALIZATION_CULTURE=de-DE] -c 3 .NET"
Copy link

Choose a reason for hiding this comment

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

This env should not be needed

Copy link

Choose a reason for hiding this comment

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

There should be a .net core leveled env to do the same thing

Copy link
Contributor

Choose a reason for hiding this comment

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

@wli3 We were hoping to align to whatever environment variables exist already in the SDK but the only one I could find was for specifying the invariant culture, not for specifying other cultures. https://docs.microsoft.com/en-us/dotnet/core/run-time-config/globalization

Copy link

Choose a reason for hiding this comment

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

my memory is wrong. https://github.com/dotnet/sdk/blob/b1223209644d900702287faea8e9b71f95ec49f8/src/Cli/dotnet/UILanguageOverride.cs#L38

we had a CLI specific flag. If we need a flag for this library that's fine.

}
}
}
Loading