Myxas.ConfigStringLocalizer 0.1.1-CI-11

This is a prerelease version of Myxas.ConfigStringLocalizer.
dotnet add package Myxas.ConfigStringLocalizer --version 0.1.1-CI-11                
NuGet\Install-Package Myxas.ConfigStringLocalizer -Version 0.1.1-CI-11                
This command is intended to be used within the Package Manager Console in Visual Studio, as it uses the NuGet module's version of Install-Package.
<PackageReference Include="Myxas.ConfigStringLocalizer" Version="0.1.1-CI-11" />                
For projects that support PackageReference, copy this XML node into the project file to reference the package.
paket add Myxas.ConfigStringLocalizer --version 0.1.1-CI-11                
#r "nuget: Myxas.ConfigStringLocalizer, 0.1.1-CI-11"                
#r directive can be used in F# Interactive and Polyglot Notebooks. Copy this into the interactive tool or source code of the script to reference the package.
// Install Myxas.ConfigStringLocalizer as a Cake Addin
#addin nuget:?package=Myxas.ConfigStringLocalizer&version=0.1.1-CI-11&prerelease

// Install Myxas.ConfigStringLocalizer as a Cake Tool
#tool nuget:?package=Myxas.ConfigStringLocalizer&version=0.1.1-CI-11&prerelease                

Myxas.ConfigStringLocalizer

Build status Test status

NuGet Version NuGet Downloads

Introduction

ConfigStringLocalizer is a .NET library that provides an implementation of IStringLocalizer for loading resource-strings from an IConfiguration instance, as opposed to .NET's default of compiled RESX files. This allows you to keep your translations within a JSON file, an XML file, SQL Server, or almost any other form of storage you need (provided that an appropriate custom ConfigurationProvider is configured).

There is also out-of-the-box support for hot-reloading of changes to your resource strings if the ConfigurationProvider supports a ReloadOnChange option (as Microsoft's file-based ones do).

Although these libraries and interfaces were designed for .NET Core, they are targeting .NET Standard 1.3 and up so can also be used on .NET Framework 4.6 and newer.

Getting Started

ConfigStringLocalizer is installed from NuGet:

Installation

Install-Package Myxas.ConfigStringLocalizer

You'll also need a configuration provider, e.g. Microsoft.Extensions.Configuration.Json:

Install-Package Microsoft.Extensions.Configuration.Json

Resource-strings formats

ConfigStringLocalizer doesn't care about the exact format (or even if your configuration source holds other data too - as long as the resources are in their own sub-hierarchy of that config). The format is abstracted away by relying on the provided IConfiguration instance. What it does care about is that the resource-strings data are structured the way that it expects. See the following JSON for example:

{
  "Close": {
    "de": "Schließen",
    "de-AT-ar": "Schliessen",
    "de-DE-bb": "Schliessen",
    "en": "Close",
    "es": "Cerrar",
    "id": "Tutup",
    "it": "Chiudi",
    "ms": "Tutup",
    "pt-BR": "Fechar",
    "zh": "关"
  },
  "Colour": {
    "de": "Farbe",
    "en-AU": "Colour",
    "en-US": "Color",
    "en-GB": "Colour",
    "es": "Color",
    "id": "Warna",
    "it": "Colore",
    "ms": "Warna",
    "pt": "Cor",
    "zh-Hans": "颜色",
    "zh-Hant": "顏色"
  }
}

We have the resource-string key ("Close" and "Colour" in the above example) acting as the parent/section-group for a list of CultureInfo names and respective translations.

You can split things up into as many different config files/sources as you wish, even repeating the same resource-string key in each file but with different or overlapping translations in each - you just need to make sure you pass all of those configs into the Localizer's setup. The IConfiguration implementation takes care of merging them all together, and the ConfigStringLocalizer takes care of fallback in event of missing translations.

If multiple config sources are provided and there are translations across those "files" which conflict - a translation from a latter-loaded file overrides/takes precedence.

Have a look at this project's unit-tests and data for more examples.

Important: You cannot use a colon (:) within a resource-string key because that character has special meaning to IConfiguration implementations as a key delimiter. The workaround is to encode the character as &colon; within your key text inside the config. However, within your code, you can (and should) simply use : instead - ConfigStringLocalizer will take care of transparently encoding & decoding it for you. Note, you can easily override the encoder/decoder used if you're not happy with the existing workaround, or if the format you're using requires other characters to also be escaped/encoded.

How to register the LocalizerFactory for Dependency-Injection on ASP.NET Core

If you're using ASP.NET Core, register and configure the localizer factory with your services:

public void ConfigureServices(IServiceCollection services)
{
    var config = new ConfigurationBuilder()
        .AddJsonFile("Resources/File1.json", optional:false, reloadOnChange:true)
        .AddJsonFile("Resources/File2.json", optional:false, reloadOnChange:true)
        .Build();

    services.AddConfigLocalization(options => {
        options.Configuration = config;
    });
}

How to manually create a Localizer instance

var config = new ConfigurationBuilder()
    .AddIniFile("Resources/File1.ini")
    .Build();

var localizer = new ConfigStringLocalizer(config);

Options to further customize

You'll note that when configuring DI for ASP.NET Core in the example above, an "options" variable was configured. This object (class ConfigLocalizationOptions) has 4 properties that you can set:

IConfiguration Configuration The only mandatory thing to provide - this gives the Localizer all your resources-strings.

StringComparer KeyComparer The KeyComparer is used for looking up against resource-string keys. By default, if none is provided here, these lookups are case-sensitive and use .NET's default generic equality comparer. Note: this does not affect culture name lookups, they are always case-insensitive.

Func<string, string> KeyEncoder When you attempt to retrieve a resource-string, this is executed on the input key you're trying to lookup. The assumption is that the keys within your configuration source are already encoded. The default KeyEncoder replaces all instances of ":" with "&colon;".

Func<string, string> KeyDecoder When you call the localizer's GetAllStrings() method, the KeyDecoder is executed on every resource-string key from your configuration source. By default, it replaces all instances of ":" with ":".

You can also directly pass each of these customizations to the localizer's constructor if you're calling it manually. There's also an additional withCulture parameter that allows you to explicitly specify a culture to use when looking up resource-strings instead of defaulting to the current thread's UI culture. The full constructor signature is:

ConfigStringLocalizer(IConfiguration config,
    StringComparer keyComparer = null,
    Func<string, string> keyEncoder = null,
    Func<string, string> keyDecoder = null,
    CultureInfo withCulture = null)

Build and Test

Just build the solution; there are no special requirements apart from having the minimum .NET versions installed. The Tests project currently targets .NET 4.6, .NET Core 2.1, .NET Core 2.0 and .NET Core 1.1.

dotnet build Myxas.ConfigStringLocalizer.sln
dotnet test tests/Myxas.ConfigStringLocalizer.Tests

Contribute

Pull requests, issues and feature requests are all totally welcome and would be greatly appreciated!

Product Compatible and additional computed target framework versions.
.NET net5.0 was computed.  net5.0-windows was computed.  net6.0 was computed.  net6.0-android was computed.  net6.0-ios was computed.  net6.0-maccatalyst was computed.  net6.0-macos was computed.  net6.0-tvos was computed.  net6.0-windows was computed.  net7.0 was computed.  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.  net9.0 was computed.  net9.0-android was computed.  net9.0-browser was computed.  net9.0-ios was computed.  net9.0-maccatalyst was computed.  net9.0-macos was computed.  net9.0-tvos was computed.  net9.0-windows was computed. 
.NET Core netcoreapp1.0 was computed.  netcoreapp1.1 was computed.  netcoreapp2.0 was computed.  netcoreapp2.1 was computed.  netcoreapp2.2 was computed.  netcoreapp3.0 was computed.  netcoreapp3.1 was computed. 
.NET Standard netstandard1.3 is compatible.  netstandard1.4 was computed.  netstandard1.5 was computed.  netstandard1.6 was computed.  netstandard2.0 is compatible.  netstandard2.1 was computed. 
.NET Framework net46 is compatible.  net461 was computed.  net462 was computed.  net463 was computed.  net47 was computed.  net471 was computed.  net472 was computed.  net48 was computed.  net481 was computed. 
MonoAndroid monoandroid was computed. 
MonoMac monomac was computed. 
MonoTouch monotouch was computed. 
Tizen tizen30 was computed.  tizen40 was computed.  tizen60 was computed. 
Universal Windows Platform uap was computed.  uap10.0 was computed. 
Xamarin.iOS xamarinios was computed. 
Xamarin.Mac xamarinmac was computed. 
Xamarin.TVOS xamarintvos was computed. 
Xamarin.WatchOS xamarinwatchos was computed. 
Compatible target framework(s)
Included target framework(s) (in package)
Learn more about Target Frameworks and .NET Standard.

NuGet packages

This package is not used by any NuGet packages.

GitHub repositories

This package is not used by any popular GitHub repositories.

Version Downloads Last updated
0.1.1-CI-11 656 7/25/2018
0.1.0-CI-9 699 6/21/2018
0.1.0-CI-8 699 6/19/2018
0.1.0-CI-7 689 6/18/2018
0.1.0-CI-3 697 6/16/2018
0.1.0-CI-2 698 6/15/2018
0.1.0-CI-10 746 6/22/2018
0.1.0-CI-1 684 6/15/2018