add version 0.0.1
This commit is contained in:
@@ -3,7 +3,19 @@
|
||||
{
|
||||
"name": "C# (.NET)",
|
||||
// Or use a Dockerfile or Docker Compose file. More info: https://containers.dev/guide/dockerfile
|
||||
"image": "mcr.microsoft.com/devcontainers/dotnet:0-6.0"
|
||||
"image": "mcr.microsoft.com/devcontainers/dotnet:0-6.0",
|
||||
"features": {
|
||||
"ghcr.io/devcontainers/features/nix:1": {}
|
||||
},
|
||||
"customizations": {
|
||||
"vscode": {
|
||||
"extensions": [
|
||||
"formulahendry.dotnet-test-explorer",
|
||||
"jnoortheen.nix-ide",
|
||||
"eamodio.gitlens"
|
||||
]
|
||||
}
|
||||
},
|
||||
|
||||
// Features to add to the dev container. More info: https://containers.dev/features.
|
||||
// "features": {},
|
||||
@@ -17,10 +29,7 @@
|
||||
// }
|
||||
|
||||
// Use 'postCreateCommand' to run commands after the container is created.
|
||||
// "postCreateCommand": "dotnet restore",
|
||||
|
||||
// Configure tool-specific properties.
|
||||
// "customizations": {},
|
||||
"postCreateCommand": "nix-env -i nixpkgs-fmt"
|
||||
|
||||
// Uncomment to connect as root instead. More info: https://aka.ms/dev-containers-non-root.
|
||||
// "remoteUser": "root"
|
||||
|
||||
8
.gitignore
vendored
8
.gitignore
vendored
@@ -475,10 +475,4 @@ $RECYCLE.BIN/
|
||||
|
||||
# Windows shortcuts
|
||||
*.lnk
|
||||
|
||||
|
||||
app/
|
||||
example/
|
||||
testapp/
|
||||
testgen/
|
||||
tmp/
|
||||
result
|
||||
40
.vscode/launch.json
vendored
Normal file
40
.vscode/launch.json
vendored
Normal file
@@ -0,0 +1,40 @@
|
||||
{
|
||||
"version": "0.2.0",
|
||||
"configurations": [
|
||||
|
||||
{
|
||||
// Use IntelliSense to find out which attributes exist for C# debugging
|
||||
// Use hover for the description of the existing attributes
|
||||
// For further information visit https://github.com/OmniSharp/omnisharp-vscode/blob/master/debugger-launchjson.md
|
||||
"name": "Launch Example App",
|
||||
"type": "coreclr",
|
||||
"request": "launch",
|
||||
"preLaunchTask": "build",
|
||||
// If you have changed target frameworks, make sure to update the program path.
|
||||
"program": "${workspaceFolder}/example/Example.App/bin/Debug/net6.0/Example.App.dll",
|
||||
"args": [],
|
||||
"cwd": "${workspaceFolder}/example/Example.App",
|
||||
// For more information about the 'console' field, see https://aka.ms/VSCode-CS-LaunchJson-Console
|
||||
"console": "internalConsole",
|
||||
"stopAtEntry": false
|
||||
},
|
||||
{
|
||||
"name": "Update Resources.cs",
|
||||
"type": "coreclr",
|
||||
"request": "launch",
|
||||
"preLaunchTask": "build",
|
||||
"program": "${workspaceFolder}/ResourceString.Net.App.Console/bin/Debug/net6.0/ResourceString.Net.App.Console.dll",
|
||||
"args": [
|
||||
"${workspaceFolder}/ResourceString.Net.Logic/Properties/Resources.resx"
|
||||
],
|
||||
"cwd": "${workspaceFolder}/ResourceString.Net.App.Console",
|
||||
"console": "integratedTerminal",
|
||||
"stopAtEntry": false
|
||||
},
|
||||
{
|
||||
"name": ".NET Core Attach",
|
||||
"type": "coreclr",
|
||||
"request": "attach"
|
||||
}
|
||||
]
|
||||
}
|
||||
3
.vscode/settings.json
vendored
Normal file
3
.vscode/settings.json
vendored
Normal file
@@ -0,0 +1,3 @@
|
||||
{
|
||||
"dotnet-test-explorer.testProjectPath": "**/*Tests.csproj"
|
||||
}
|
||||
41
.vscode/tasks.json
vendored
Normal file
41
.vscode/tasks.json
vendored
Normal file
@@ -0,0 +1,41 @@
|
||||
{
|
||||
"version": "2.0.0",
|
||||
"tasks": [
|
||||
{
|
||||
"label": "build",
|
||||
"command": "nix",
|
||||
"type": "process",
|
||||
"args": [
|
||||
"--experimental-features",
|
||||
"nix-command flakes",
|
||||
"run",
|
||||
".#devTasks.buildAllAssemblies"
|
||||
],
|
||||
"problemMatcher": "$msCompile"
|
||||
},
|
||||
{
|
||||
"label": "publish",
|
||||
"command": "dotnet",
|
||||
"type": "process",
|
||||
"args": [
|
||||
"publish",
|
||||
"${workspaceFolder}/example/Example.App/Example.App.csproj",
|
||||
"/property:GenerateFullPaths=true",
|
||||
"/consoleloggerparameters:NoSummary"
|
||||
],
|
||||
"problemMatcher": "$msCompile"
|
||||
},
|
||||
{
|
||||
"label": "watch",
|
||||
"command": "dotnet",
|
||||
"type": "process",
|
||||
"args": [
|
||||
"watch",
|
||||
"run",
|
||||
"--project",
|
||||
"${workspaceFolder}/example/Example.App/Example.App.csproj"
|
||||
],
|
||||
"problemMatcher": "$msCompile"
|
||||
}
|
||||
]
|
||||
}
|
||||
21
LICENSE
Normal file
21
LICENSE
Normal file
@@ -0,0 +1,21 @@
|
||||
MIT License
|
||||
|
||||
Copyright (c) 2023 stubbfel
|
||||
|
||||
Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||
of this software and associated documentation files (the "Software"), to deal
|
||||
in the Software without restriction, including without limitation the rights
|
||||
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||||
copies of the Software, and to permit persons to whom the Software is
|
||||
furnished to do so, subject to the following conditions:
|
||||
|
||||
The above copyright notice and this permission notice shall be included in all
|
||||
copies or substantial portions of the Software.
|
||||
|
||||
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||||
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||||
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
||||
SOFTWARE.
|
||||
240
Readme.md
240
Readme.md
@@ -0,0 +1,240 @@
|
||||
# ResourceString.Net
|
||||
|
||||
## What is ResourceString.Net?
|
||||
|
||||
ResourceString.Net is a powerful .NET library that allows you to work with string resources in a type-safe manner.
|
||||
It leverages the `resx` file in your project and utilizes a [c# source code generator](https://learn.microsoft.com/en-us/dotnet/csharp/roslyn-sdk/source-generators-overview) to create a comprehensive API.
|
||||
With ResourceString.Net, you can handle resource strings as "multi-language strings" (see [The ResourceString-Classes](#the-resourcestring-classes)) instead of built-in strings.
|
||||
This provides the ability to switch languages during runtime without the need to rerun string factory methods.
|
||||
Additionally, ResourceString.Net ensures that formatted strings have methods with the correct number of expected parameters.
|
||||
|
||||
## Installation and Setup Instructions
|
||||
|
||||
To incorporate ResourceString.Net into your .NET application, follow these simple steps:
|
||||
|
||||
1. Install the NuGet package by executing the following command in the NuGet package manager or the dotnet CLI:
|
||||
|
||||
```sh
|
||||
dotnet add package ResourceString.Net
|
||||
```
|
||||
|
||||
2. Once the package is installed, you can start using the APIs in your application.
|
||||
|
||||
## Getting Started
|
||||
|
||||
To quickly get started with ResourceString.Net, follow the steps below:
|
||||
|
||||
1. Run the following script to create a new project and a resource file:
|
||||
|
||||
```sh
|
||||
dotnet new console --name MyTestConsoleApp
|
||||
cd MyTestConsoleApp
|
||||
|
||||
dotnet add package "System.Resources.Extensions"
|
||||
dotnet add package "ResourceString.Net"
|
||||
|
||||
echo "<?xml version='1.0' encoding='utf-8'?>
|
||||
<root>
|
||||
<data name='Greetings'>
|
||||
<value>Hello {0}</value>
|
||||
<comment>0 = name</comment>
|
||||
</data>
|
||||
<data name='World' type='System.String'>
|
||||
<value>World</value>
|
||||
</data>
|
||||
</root>
|
||||
" > Resources.resx
|
||||
|
||||
echo "var message = MyTestConsoleApp.Resources.Greetings.From(
|
||||
MyTestConsoleApp.Resources.World
|
||||
);
|
||||
|
||||
Console.WriteLine(message.Value);
|
||||
" > Program.cs
|
||||
```
|
||||
|
||||
2. Add the following `PropertyGroup` to the `MyTestConsoleApp.csproj` file to enable ResourceString.Net to handle the project's resource file:
|
||||
|
||||
```xml
|
||||
<PropertyGroup>
|
||||
<AdditionalFileItemNames>$(AdditionalFileItemNames);EmbeddedResource</AdditionalFileItemNames>
|
||||
</PropertyGroup>
|
||||
```
|
||||
|
||||
3. Run the project using the following command:
|
||||
|
||||
```sh
|
||||
dotnet run
|
||||
# Expected output: Hello World
|
||||
```
|
||||
|
||||
During compile time, the ResourceString.Net source code generator will automatically add the following code to the `MyTestConsoleApp.csproj` project:
|
||||
|
||||
```cs
|
||||
using ResourceString.Net.Contract;
|
||||
using System;
|
||||
using System.Globalization;
|
||||
using System.Resources;
|
||||
using System.Threading;
|
||||
|
||||
namespace MyTestConsoleApp
|
||||
{
|
||||
internal static class Resources
|
||||
{
|
||||
|
||||
#region ResourceManager
|
||||
|
||||
private static readonly Type _Type = typeof(Resources);
|
||||
|
||||
private static readonly Lazy<ResourceManager> _ResourceManager = new Lazy<ResourceManager>(
|
||||
() => new ResourceManager(_Type.FullName ?? string.Empty, _Type.Assembly),
|
||||
LazyThreadSafetyMode.PublicationOnly
|
||||
);
|
||||
|
||||
public static ResourceManager ResourceManager => _ResourceManager.Value;
|
||||
|
||||
#endregion // ResourceManager
|
||||
|
||||
|
||||
internal static class Greetings
|
||||
{
|
||||
private static readonly Lazy<IResourceString> LazyFormat = new Lazy<IResourceString>(
|
||||
() => new ResourceManagerString("Greetings", ResourceManager, CultureInfo.CurrentCulture),
|
||||
LazyThreadSafetyMode.PublicationOnly
|
||||
);
|
||||
|
||||
public static IResourceString Format => LazyFormat.Value;
|
||||
|
||||
public static IResourceString From(IResourceString name) => new FormattedResourceString(
|
||||
Format,
|
||||
name
|
||||
);
|
||||
|
||||
}
|
||||
|
||||
#region World
|
||||
|
||||
private static readonly Lazy<IResourceString> LazyWorld = new Lazy<IResourceString>(
|
||||
() => new ResourceManagerString("World", ResourceManager, CultureInfo.CurrentCulture),
|
||||
LazyThreadSafetyMode.PublicationOnly
|
||||
);
|
||||
|
||||
public static IResourceString World => LazyWorld.Value;
|
||||
|
||||
#endregion // World
|
||||
|
||||
}
|
||||
}
|
||||
|
||||
```
|
||||
|
||||
## The ResourceString-Classes
|
||||
|
||||
### IResourceString
|
||||
|
||||
- Interface that defines the contract for resource strings.
|
||||
- Contains properties: `Value`, representing the string value, and `GetValue(CultureInfo cultureInfo)`, for retrieving the value for a specific `CultureInfo`.
|
||||
|
||||
### FormattedResourceString
|
||||
|
||||
- Implements the `IResourceString` interface.
|
||||
- Represents a resource string with placeholders for parameters.
|
||||
- Allows for dynamic formatting of the string by providing a format and an array of parameters.
|
||||
- Formats the string by replacing the placeholders with the parameter values.
|
||||
|
||||
### JoinedResourceString
|
||||
|
||||
- Implements the `IResourceString` interface.
|
||||
- Represents a resource string that joins multiple elements with a separator.
|
||||
- Useful for constructing strings that involve concatenating multiple resource strings or literal strings together.
|
||||
- Allows customization of the separator between the elements.
|
||||
|
||||
### LiteralString
|
||||
|
||||
- Implements the `IResourceString` interface.
|
||||
- Represents a literal string resource.
|
||||
- Provides the actual string value as-is without any formatting or localization.
|
||||
- Can be used for static, non-localized strings.
|
||||
|
||||
### ResourceManagerString
|
||||
|
||||
- Implements the `IResourceString` interface.
|
||||
- Represents a resource string retrieved from a `ResourceManager`.
|
||||
- Provides access to resource strings stored in `resx` files or other resource sources.
|
||||
- Handles retrieving the localized value for the specified `CultureInfo`.
|
||||
|
||||
Great! Based on the provided code for the console app `ResourceString.Net.App.Console`, let's enhance the Readme.md file to include instructions on how to use the console app and generate a C# class based on a given resource file.
|
||||
|
||||
## Console App: ResourceString.Net.App.Console
|
||||
|
||||
The ResourceString.Net project also provide a console app, `ResourceString.Net.App.Console`, which allows you to generate a C# class based on a given resource file. This can be useful for automating the creation of resource classes in your projects without usage of the `source generator`.
|
||||
|
||||
### Usage
|
||||
|
||||
To use the console app, follow these steps:
|
||||
|
||||
1. Build the console app from this source repository
|
||||
2. Open a command prompt or terminal.
|
||||
3. Navigate to the directory where the `ResourceString.Net.App.Console` executable is located.
|
||||
|
||||
#### Syntax
|
||||
|
||||
```
|
||||
ResourceString.Net.App.Console <sourceFile> [namespaceString] [className]
|
||||
```
|
||||
|
||||
#### Parameters
|
||||
|
||||
- `<sourceFile>`: Required. The path to the resource file (e.g., `Resources.resx`) that you want to generate the C# class from.
|
||||
- `[namespaceString]` (optional): The namespace for the generated C# class. If not provided, the default value is `"Properties"`.
|
||||
- `[className]` (optional): The name of the generated C# class. If not provided, the default value is `"Resources"`.
|
||||
|
||||
#### Examples
|
||||
|
||||
Here are some examples of how to use the console app:
|
||||
|
||||
- Generate a C# class named `Resources.cs` in the `"Properties"` namespace based on the `Resources.resx` file:
|
||||
|
||||
```
|
||||
ResourceString.Net.App.Console Resources.resx
|
||||
```
|
||||
|
||||
- Generate a C# class named `MyResources.cs` in the `"MyNamespace"` namespace based on the `MyResources.resx` file:
|
||||
|
||||
```
|
||||
ResourceString.Net.App.Console MyResources.resx MyNamespace MyResources
|
||||
```
|
||||
|
||||
### Output
|
||||
|
||||
The console app will generate the C# class code based on the provided resource file and output it to the console.
|
||||
You can redirect the output to a file if desired.
|
||||
|
||||
## Third party packages
|
||||
|
||||
| Package | Version |
|
||||
| ------------------------------------- | ------- |
|
||||
| Fody | 6.7.0 |
|
||||
| ILMerge.Fody | 1.24.0 |
|
||||
| Microsoft.CodeAnalysis.Analyzers | 3.3.4 |
|
||||
| Microsoft.CodeAnalysis.CSharp | 4.3.0 |
|
||||
| NETStandard.Library (Auto-referenced) | 2.0.3 |
|
||||
| LanguageExt.Core | 4.4.3 |
|
||||
| System.Resources.Extensions | 7.0.0 |
|
||||
|
||||
## Development Notes
|
||||
|
||||
Here are some useful aliases for running Nix commands with experimental features:
|
||||
|
||||
- `nixe`: This alias runs the `nix` command with the experimental feature flag `nix-command flakes`.
|
||||
- `nulock`: This alias runs the `nixe` command with the argument `run .#devTasks.updateNugetLock`, which updates the NuGet lock file.
|
||||
- `fllock`: This alias runs the `nixe` command with the argument `run .#devTasks.updateFlakeLock`, which updates the Flake lock file.
|
||||
- `ulock`: This alias combines the `nulock` and `fllock` aliases to update both the NuGet and Flake lock files.
|
||||
|
||||
To load the `alias.sh` source file from the current folder, use the following command:
|
||||
|
||||
```sh
|
||||
source ./alias.sh
|
||||
```
|
||||
|
||||
By executing the command above, you'll make the aliases available for use in the current terminal session.
|
||||
|
||||
18
ResourceString.Net.App.Console/Program.cs
Normal file
18
ResourceString.Net.App.Console/Program.cs
Normal file
@@ -0,0 +1,18 @@
|
||||
using ResourceString.Net.Logic.Factories;
|
||||
using ResourceString.Net.Logic.Parsers.Resx;
|
||||
|
||||
var sourceFile = args.First();
|
||||
var namespaceString = args.Skip(1).FirstOrDefault() ?? "Properties";
|
||||
var className = args.Skip(2).FirstOrDefault() ?? "Resources";
|
||||
|
||||
var result = Parser.TryParse(System.IO.File.ReadAllText(sourceFile)).Match(
|
||||
Some: v => CodeSnippetFactory.CreateResourceClassCodeSnippet(
|
||||
namespaceString,
|
||||
className,
|
||||
CodeSnippetFactory.CreateResourceMangerMemberCodeSnippet(className),
|
||||
v.Resources
|
||||
),
|
||||
None: () => throw new InvalidOperationException()
|
||||
);
|
||||
|
||||
Console.WriteLine(result.Value.Trim());
|
||||
@@ -0,0 +1,16 @@
|
||||
<Project Sdk="Microsoft.NET.Sdk">
|
||||
|
||||
<PropertyGroup>
|
||||
<OutputType>Exe</OutputType>
|
||||
<TargetFramework>net6.0</TargetFramework>
|
||||
<ImplicitUsings>enable</ImplicitUsings>
|
||||
<Nullable>enable</Nullable>
|
||||
<LangVersion>latest</LangVersion>
|
||||
<PackAsTool>true</PackAsTool>
|
||||
</PropertyGroup>
|
||||
|
||||
<ItemGroup>
|
||||
<ProjectReference Include="..\ResourceString.Net.Logic\ResourceString.Net.Logic.csproj" />
|
||||
</ItemGroup>
|
||||
|
||||
</Project>
|
||||
@@ -1,233 +0,0 @@
|
||||
using System;
|
||||
using System.Collections;
|
||||
using System.Collections.Generic;
|
||||
using System.Globalization;
|
||||
using System.Linq;
|
||||
using System.Resources;
|
||||
using System.Threading;
|
||||
|
||||
namespace ResourceString.Net.Contract
|
||||
{
|
||||
public interface IResourceString
|
||||
{
|
||||
|
||||
string Value { get; }
|
||||
|
||||
string GetValue(CultureInfo cultureInfo);
|
||||
}
|
||||
|
||||
public class LiteralString : IResourceString
|
||||
{
|
||||
private Lazy<string> m_Value;
|
||||
|
||||
public string Value => m_Value.Value;
|
||||
|
||||
public static IResourceString Empty {get;} = Factory(string.Empty);
|
||||
public static IResourceString Factory(string source)
|
||||
{
|
||||
return new LiteralString(() => source);
|
||||
}
|
||||
|
||||
public LiteralString(Func<string> factory)
|
||||
{
|
||||
m_Value = new Lazy<string>(
|
||||
() => factory?.Invoke() ?? string.Empty,
|
||||
LazyThreadSafetyMode.PublicationOnly
|
||||
);
|
||||
}
|
||||
|
||||
public string GetValue(CultureInfo _) => Value;
|
||||
}
|
||||
|
||||
public class ResourceManagerString : IResourceString
|
||||
{
|
||||
private readonly CultureInfo m_Culture;
|
||||
|
||||
public string Id { get; }
|
||||
|
||||
public ResourceManager Manager { get; }
|
||||
|
||||
public string Value => GetValue(m_Culture);
|
||||
|
||||
public ResourceManagerString(
|
||||
string id,
|
||||
ResourceManager manager,
|
||||
CultureInfo cultureInfo)
|
||||
{
|
||||
Id = id ?? string.Empty;
|
||||
Manager = manager ?? throw new ArgumentNullException(nameof(manager));
|
||||
m_Culture = cultureInfo ?? CultureInfo.CurrentCulture;
|
||||
}
|
||||
|
||||
public string GetValue(CultureInfo cultureInfo)
|
||||
{
|
||||
return Manager.GetString(Id, cultureInfo);
|
||||
}
|
||||
}
|
||||
|
||||
public class FormattableResourceString : IResourceString
|
||||
{
|
||||
public IResourceString Format { get; }
|
||||
|
||||
public IEnumerable<IResourceString> Parameters { get; }
|
||||
|
||||
public string Value => GetValue(CultureInfo.CurrentCulture);
|
||||
|
||||
public FormattableResourceString(
|
||||
IResourceString format,
|
||||
params IResourceString[] parameters)
|
||||
{
|
||||
Format = format ?? new JoinedResourceString(
|
||||
LiteralString.Factory(","),
|
||||
parameters.Select(((p, idx) => LiteralString.Factory("{" + idx + "}"))).ToArray()
|
||||
);
|
||||
|
||||
Parameters = parameters;
|
||||
}
|
||||
|
||||
public string GetValue(CultureInfo cultureInfo)
|
||||
{
|
||||
var format = Format.GetValue(cultureInfo);
|
||||
var parameterStrings = Parameters.Select(p => p.GetValue(cultureInfo));
|
||||
return string.Format(format, parameterStrings.ToArray());
|
||||
}
|
||||
}
|
||||
|
||||
public class JoinedResourceString : IResourceString
|
||||
{
|
||||
public IResourceString Separator { get; }
|
||||
|
||||
public IEnumerable<IResourceString> Elements { get; }
|
||||
|
||||
public string Value => GetValue(CultureInfo.CurrentCulture);
|
||||
|
||||
public JoinedResourceString(
|
||||
IResourceString separator,
|
||||
params IResourceString[] elements)
|
||||
{
|
||||
Separator = separator ?? new LiteralString(
|
||||
() => ","
|
||||
);
|
||||
|
||||
Elements = elements;
|
||||
}
|
||||
|
||||
public string GetValue(CultureInfo cultureInfo)
|
||||
{
|
||||
var separator = Separator.GetValue(cultureInfo);
|
||||
var elementsStrings = Elements.Select(p => p.GetValue(cultureInfo));
|
||||
return string.Join(separator, elementsStrings);
|
||||
}
|
||||
}
|
||||
|
||||
internal static class ResourceStrings
|
||||
{
|
||||
private static readonly ResourceManager ResourceManager = new ResourceManager(typeof(int));
|
||||
|
||||
private static readonly Lazy<ResourceManagerString> LazyEntry = new Lazy<ResourceManagerString>(
|
||||
() => new ResourceManagerString("EntryId", ResourceManager, CultureInfo.CurrentCulture),
|
||||
LazyThreadSafetyMode.PublicationOnly
|
||||
);
|
||||
|
||||
public static ResourceManagerString Entry => LazyEntry.Value;
|
||||
|
||||
public static class FormattedEntry
|
||||
{
|
||||
private static readonly Lazy<ResourceManagerString> Value = new Lazy<ResourceManagerString>(
|
||||
() => new ResourceManagerString("FormatEntryId", ResourceManager, CultureInfo.CurrentCulture),
|
||||
LazyThreadSafetyMode.PublicationOnly
|
||||
);
|
||||
|
||||
public static string Id => Value.Value.Id;
|
||||
|
||||
public static ResourceManager Manager => Value.Value.Manager;
|
||||
|
||||
public static IResourceString Format => Value.Value;
|
||||
|
||||
public static IResourceString From(IResourceString p0, IResourceString p1) => new FormattableResourceString(
|
||||
Format,
|
||||
p0,
|
||||
p1
|
||||
);
|
||||
|
||||
|
||||
static IEnumerable<IResourceString> test(IResourceReader r)
|
||||
{
|
||||
var enumerator = r.GetEnumerator();
|
||||
|
||||
var dict = enumerator.ToDictionary<string, string>();
|
||||
|
||||
return dict.Select(e =>new LiteralString(() => CreateString(e)));
|
||||
|
||||
string CreateString(KeyValuePair<string, string> entry)
|
||||
{
|
||||
var key = entry.Key;
|
||||
var keyName = key.Replace(" ", "").Trim();
|
||||
return @$"
|
||||
private static readonly Lazy<ResourceString> Lazy{keyName} = new Lazy<ResourceString>(
|
||||
() => new ResourceString(""{key}"", ResourceManager, CultureInfo.CurrentCulture),
|
||||
LazyThreadSafetyMode.PublicationOnly
|
||||
);
|
||||
|
||||
public static ResourceString {keyName} => Lazy{keyName}.Value;
|
||||
";
|
||||
}
|
||||
}
|
||||
|
||||
public static class Hello
|
||||
{
|
||||
public static IResourceString Format { get; } = LiteralString.Factory("{0}, {1} !!!");
|
||||
|
||||
public static IResourceString Value { get; } = LiteralString.Factory("Hello");
|
||||
|
||||
public static IResourceString From(IResourceString name) => new FormattableResourceString(
|
||||
Format,
|
||||
Value,
|
||||
name
|
||||
);
|
||||
}
|
||||
|
||||
public static class HelloWorld
|
||||
{
|
||||
public static IResourceString Value { get; } = Hello.From(World);
|
||||
|
||||
public static IResourceString From(IResourceString name) => new FormattableResourceString(
|
||||
Hello.Format,
|
||||
Value,
|
||||
name
|
||||
);
|
||||
}
|
||||
|
||||
public static IResourceString World { get; } = LiteralString.Factory("World");
|
||||
|
||||
}
|
||||
}
|
||||
|
||||
public static class Extensions
|
||||
{
|
||||
public static IDictionary<TKey, TValue> ToDictionary<TKey, TValue>(this IDictionaryEnumerator source)
|
||||
{
|
||||
return source.Collect<TKey, TValue>().ToDictionary(k => k.Key, v => v.Value);
|
||||
}
|
||||
|
||||
public static IEnumerable<KeyValuePair<TKey, TValue>> Collect<TKey, TValue>(this IDictionaryEnumerator source)
|
||||
{
|
||||
if (source is null)
|
||||
{
|
||||
yield break;
|
||||
}
|
||||
|
||||
while (source.MoveNext())
|
||||
{
|
||||
if (source.Key is TKey key && source.Value is TValue value)
|
||||
{
|
||||
yield return new KeyValuePair<TKey, TValue>(
|
||||
key,
|
||||
value
|
||||
);
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
35
ResourceString.Net.Contract/FormattedResourceString.cs
Normal file
35
ResourceString.Net.Contract/FormattedResourceString.cs
Normal file
@@ -0,0 +1,35 @@
|
||||
using System.Collections.Generic;
|
||||
using System.Globalization;
|
||||
using System.Linq;
|
||||
|
||||
namespace ResourceString.Net.Contract
|
||||
{
|
||||
public class FormattedResourceString : IResourceString
|
||||
{
|
||||
public IResourceString Format { get; }
|
||||
|
||||
public IEnumerable<IResourceString> Parameters { get; }
|
||||
|
||||
public string Value => GetValue(CultureInfo.CurrentCulture);
|
||||
|
||||
public FormattedResourceString(
|
||||
IResourceString format,
|
||||
params IResourceString[] parameters)
|
||||
{
|
||||
Format = format ?? new JoinedResourceString(
|
||||
LiteralString.Factory(","),
|
||||
parameters.Select(((p, idx) => LiteralString.Factory("{" + idx + "}"))).ToArray()
|
||||
);
|
||||
|
||||
Parameters = parameters;
|
||||
}
|
||||
|
||||
public string GetValue(CultureInfo cultureInfo)
|
||||
{
|
||||
var format = Format.GetValue(cultureInfo);
|
||||
var parameterStrings = Parameters.Select(p => p.GetValue(cultureInfo));
|
||||
return string.Format(format, parameterStrings.ToArray());
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
13
ResourceString.Net.Contract/IResourceString.cs
Normal file
13
ResourceString.Net.Contract/IResourceString.cs
Normal file
@@ -0,0 +1,13 @@
|
||||
using System.Globalization;
|
||||
|
||||
namespace ResourceString.Net.Contract
|
||||
{
|
||||
public interface IResourceString
|
||||
{
|
||||
|
||||
string Value { get; }
|
||||
|
||||
string GetValue(CultureInfo cultureInfo);
|
||||
}
|
||||
}
|
||||
|
||||
35
ResourceString.Net.Contract/JoinedResourceString.cs
Normal file
35
ResourceString.Net.Contract/JoinedResourceString.cs
Normal file
@@ -0,0 +1,35 @@
|
||||
using System.Collections.Generic;
|
||||
using System.Globalization;
|
||||
using System.Linq;
|
||||
|
||||
namespace ResourceString.Net.Contract
|
||||
{
|
||||
|
||||
public class JoinedResourceString : IResourceString
|
||||
{
|
||||
public IResourceString Separator { get; }
|
||||
|
||||
public IEnumerable<IResourceString> Elements { get; }
|
||||
|
||||
public string Value => GetValue(CultureInfo.CurrentCulture);
|
||||
|
||||
public JoinedResourceString(
|
||||
IResourceString separator,
|
||||
params IResourceString[] elements)
|
||||
{
|
||||
Separator = separator ?? new LiteralString(
|
||||
() => ","
|
||||
);
|
||||
|
||||
Elements = elements;
|
||||
}
|
||||
|
||||
public string GetValue(CultureInfo cultureInfo)
|
||||
{
|
||||
var separator = Separator.GetValue(cultureInfo);
|
||||
var elementsStrings = Elements.Select(p => p.GetValue(cultureInfo));
|
||||
return string.Join(separator, elementsStrings);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
30
ResourceString.Net.Contract/LiteralString.cs
Normal file
30
ResourceString.Net.Contract/LiteralString.cs
Normal file
@@ -0,0 +1,30 @@
|
||||
using System;
|
||||
using System.Globalization;
|
||||
using System.Threading;
|
||||
|
||||
namespace ResourceString.Net.Contract
|
||||
{
|
||||
public class LiteralString : IResourceString
|
||||
{
|
||||
private Lazy<string> m_Value;
|
||||
|
||||
public string Value => m_Value.Value;
|
||||
|
||||
public static IResourceString Empty { get; } = Factory(string.Empty);
|
||||
public static IResourceString Factory(string source)
|
||||
{
|
||||
return new LiteralString(() => source);
|
||||
}
|
||||
|
||||
public LiteralString(Func<string> factory)
|
||||
{
|
||||
m_Value = new Lazy<string>(
|
||||
() => factory?.Invoke() ?? string.Empty,
|
||||
LazyThreadSafetyMode.PublicationOnly
|
||||
);
|
||||
}
|
||||
|
||||
public string GetValue(CultureInfo _) => Value;
|
||||
}
|
||||
}
|
||||
|
||||
33
ResourceString.Net.Contract/ResourceManagerString.cs
Normal file
33
ResourceString.Net.Contract/ResourceManagerString.cs
Normal file
@@ -0,0 +1,33 @@
|
||||
using System;
|
||||
using System.Globalization;
|
||||
using System.Resources;
|
||||
|
||||
namespace ResourceString.Net.Contract
|
||||
{
|
||||
public class ResourceManagerString : IResourceString
|
||||
{
|
||||
private readonly CultureInfo m_Culture;
|
||||
|
||||
public string Id { get; }
|
||||
|
||||
public ResourceManager Manager { get; }
|
||||
|
||||
public string Value => GetValue(m_Culture);
|
||||
|
||||
public ResourceManagerString(
|
||||
string id,
|
||||
ResourceManager manager,
|
||||
CultureInfo cultureInfo)
|
||||
{
|
||||
Id = id ?? string.Empty;
|
||||
Manager = manager ?? throw new ArgumentNullException(nameof(manager));
|
||||
m_Culture = cultureInfo ?? CultureInfo.CurrentCulture;
|
||||
}
|
||||
|
||||
public string GetValue(CultureInfo cultureInfo)
|
||||
{
|
||||
return Manager.GetString(Id, cultureInfo);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
@@ -11,8 +11,8 @@ public class LogicResourcesTests
|
||||
{
|
||||
var rm = Properties.Resources.ResourceManager;
|
||||
Assert.AreEqual(
|
||||
new FormattableResourceString(
|
||||
Properties.Resources.ResourceStringMembers,
|
||||
new FormattedResourceString(
|
||||
Properties.Resources.ResourceStringMembers.Format,
|
||||
LiteralString.Factory("ResourceStringMembers"),
|
||||
LiteralString.Factory("ResourceManager")
|
||||
).Value,
|
||||
@@ -24,6 +24,44 @@ public class LogicResourcesTests
|
||||
);
|
||||
}
|
||||
|
||||
[TestMethod]
|
||||
public void Test_Get_ResourceFormatClassMembers()
|
||||
{
|
||||
var rm = Properties.Resources.ResourceManager;
|
||||
Assert.AreEqual(
|
||||
new FormattedResourceString(
|
||||
Properties.Resources.ResourceFormatClassMembers.Format,
|
||||
LiteralString.Factory("ResourceFormatClassMembers"),
|
||||
LiteralString.Factory("ResourceManager"),
|
||||
LiteralString.Factory("Format")
|
||||
).Value,
|
||||
string.Format(
|
||||
rm.GetString("ResourceFormatClassMembers") ?? string.Empty,
|
||||
"ResourceFormatClassMembers",
|
||||
"ResourceManager",
|
||||
"Format"
|
||||
)
|
||||
);
|
||||
}
|
||||
|
||||
[TestMethod]
|
||||
public void Test_Get_ResourceFormatClassFromMethod()
|
||||
{
|
||||
var rm = Properties.Resources.ResourceManager;
|
||||
Assert.AreEqual(
|
||||
new FormattedResourceString(
|
||||
Properties.Resources.ResourceFormatClassFromMethod.Format,
|
||||
LiteralString.Factory("IResourceString name"),
|
||||
LiteralString.Factory("name")
|
||||
).Value,
|
||||
string.Format(
|
||||
rm.GetString("ResourceFormatClassFromMethod") ?? string.Empty,
|
||||
"IResourceString name",
|
||||
"name"
|
||||
)
|
||||
);
|
||||
}
|
||||
|
||||
[TestMethod]
|
||||
public void Test_Get_ResourceManagerMemberTemplate()
|
||||
{
|
||||
|
||||
@@ -1,7 +1,7 @@
|
||||
<Project Sdk="Microsoft.NET.Sdk">
|
||||
|
||||
<PropertyGroup>
|
||||
<TargetFramework>net7.0</TargetFramework>
|
||||
<TargetFramework>net6.0</TargetFramework>
|
||||
<ImplicitUsings>enable</ImplicitUsings>
|
||||
<Nullable>enable</Nullable>
|
||||
<LangVersion>latest</LangVersion>
|
||||
@@ -9,10 +9,10 @@
|
||||
</PropertyGroup>
|
||||
|
||||
<ItemGroup>
|
||||
<PackageReference Include="Microsoft.NET.Test.Sdk" Version="17.3.2" />
|
||||
<PackageReference Include="MSTest.TestAdapter" Version="2.2.10" />
|
||||
<PackageReference Include="MSTest.TestFramework" Version="2.2.10" />
|
||||
<PackageReference Include="coverlet.collector" Version="3.1.2">
|
||||
<PackageReference Include="Microsoft.NET.Test.Sdk" Version="17.6.0" />
|
||||
<PackageReference Include="MSTest.TestAdapter" Version="3.0.2" />
|
||||
<PackageReference Include="MSTest.TestFramework" Version="3.0.2" />
|
||||
<PackageReference Include="coverlet.collector" Version="3.2.0">
|
||||
<PrivateAssets>all</PrivateAssets>
|
||||
<IncludeAssets>runtime; build; native; contentfiles; analyzers; buildtransitive</IncludeAssets>
|
||||
</PackageReference>
|
||||
|
||||
@@ -11,13 +11,18 @@ public class ResxFileTests
|
||||
<root>
|
||||
<data name='Test1'>
|
||||
<value>Value1</value>
|
||||
<comment>This is a greeting message.</comment>
|
||||
</data>
|
||||
<data name='Test2' type='{typeof(int).FullName}'>
|
||||
<value>Value2</value>
|
||||
<data name='Test2' type='{typeof(string).FullName}'>
|
||||
<value>{{2}}Value2{{0}}{{{{1}}}}</value>
|
||||
<comment>2 = prefix</comment>
|
||||
</data>
|
||||
<data name='Test3' type='{typeof(byte[]).FullName}'>
|
||||
<value>0xDEADBEEF</value>
|
||||
</data>
|
||||
</root>";
|
||||
|
||||
private const string expectedClass = @"
|
||||
private const string expectedClass = @"
|
||||
using ResourceString.Net.Contract;
|
||||
using System;
|
||||
using System.Globalization;
|
||||
@@ -34,7 +39,7 @@ namespace TestNameSpace
|
||||
private static readonly Type _Type = typeof(ResourceManager);
|
||||
|
||||
private static readonly Lazy<ResourceManager> _ResourceManager = new Lazy<ResourceManager>(
|
||||
() => new ResourceManager(_Type.FullName, _Type.Assembly),
|
||||
() => new ResourceManager(_Type.FullName ?? string.Empty, _Type.Assembly),
|
||||
LazyThreadSafetyMode.PublicationOnly
|
||||
);
|
||||
|
||||
@@ -54,16 +59,22 @@ namespace TestNameSpace
|
||||
|
||||
#endregion // Test1
|
||||
|
||||
#region Test2
|
||||
|
||||
private static readonly Lazy<IResourceString> LazyTest2 = new Lazy<IResourceString>(
|
||||
internal static class Test2
|
||||
{
|
||||
private static readonly Lazy<IResourceString> LazyFormat = new Lazy<IResourceString>(
|
||||
() => new ResourceManagerString(""Test2"", ResourceManager, CultureInfo.CurrentCulture),
|
||||
LazyThreadSafetyMode.PublicationOnly
|
||||
);
|
||||
|
||||
public static IResourceString Test2 => LazyTest2.Value;
|
||||
public static IResourceString Format => LazyFormat.Value;
|
||||
|
||||
#endregion // Test2
|
||||
public static IResourceString From(IResourceString prefix, IResourceString p1) => new FormattedResourceString(
|
||||
Format,
|
||||
prefix,
|
||||
p1
|
||||
);
|
||||
|
||||
}
|
||||
|
||||
}
|
||||
}";
|
||||
@@ -77,8 +88,9 @@ namespace TestNameSpace
|
||||
{
|
||||
// Arrange
|
||||
var expectedData = new[] {
|
||||
("Test1", "Value1", string.Empty),
|
||||
("Test2", "Value2", typeof(int).FullName)
|
||||
("Test1", "Value1", string.Empty, "This is a greeting message."),
|
||||
("Test2", "{2}Value2{0}{{1}}", typeof(string).FullName, "2 = prefix"),
|
||||
("Test3", "0xDEADBEEF", typeof(byte[]).FullName, string.Empty)
|
||||
};
|
||||
|
||||
// Act
|
||||
@@ -90,7 +102,12 @@ namespace TestNameSpace
|
||||
// Assert
|
||||
CollectionAssert.AreEqual(
|
||||
expectedData,
|
||||
result.Select(i => (i.Name, i.Value, i.Type.IfNone(string.Empty))).ToList()
|
||||
result.Select(i => (
|
||||
i.Name,
|
||||
i.Value,
|
||||
i.Type.IfNone(string.Empty),
|
||||
i.Comment.IfNone(string.Empty)
|
||||
)).ToList()
|
||||
);
|
||||
}
|
||||
|
||||
@@ -144,8 +161,8 @@ namespace TestNameSpace
|
||||
);
|
||||
|
||||
Assert.AreEqual(
|
||||
"#region Test2",
|
||||
result.Select(i => i.Value.Substring(0, 25).Trim()).ToList()[1]
|
||||
"internal static class Test2",
|
||||
result.Select(i => i.Value.Substring(0, 45).Trim()).ToList()[1]
|
||||
);
|
||||
|
||||
}
|
||||
|
||||
@@ -12,5 +12,7 @@ internal record Resource
|
||||
|
||||
public Option<string> Type { get; init;}
|
||||
|
||||
public Option<string> Comment { get; init;}
|
||||
|
||||
public string Value { get; }
|
||||
}
|
||||
|
||||
@@ -9,14 +9,11 @@ internal static class CodeSnippetFactory
|
||||
{
|
||||
public static IResourceString CreateResourceMangerMemberCodeSnippet(string typeName)
|
||||
{
|
||||
var formatString = Properties.Resources.ResourceManagerMemberTemplate;
|
||||
return new FormattableResourceString(
|
||||
formatString,
|
||||
return Properties.Resources.ResourceManagerMemberTemplate.From(
|
||||
LiteralString.Factory(typeName)
|
||||
);
|
||||
}
|
||||
|
||||
|
||||
public static IEnumerable<IResourceString> CreateMemberCodeSnippets(IEnumerable<Resource> resources)
|
||||
{
|
||||
return CreateMemberCodeSnippets(
|
||||
@@ -35,12 +32,63 @@ internal static class CodeSnippetFactory
|
||||
return Enumerable.Empty<IResourceString>();
|
||||
}
|
||||
|
||||
var formatString = Properties.Resources.ResourceStringMembers;
|
||||
return resources.Select(r => new FormattableResourceString(
|
||||
formatString,
|
||||
LiteralString.Factory(r.Name),
|
||||
resourceManagerName
|
||||
var stringResourses = resources.Where(r => r.Type.Match(
|
||||
v => typeof(string).IsAssignableFrom(Type.GetType(v.Trim(), false, true)),
|
||||
() => true
|
||||
));
|
||||
|
||||
return stringResourses.Select(r =>
|
||||
{
|
||||
var openBraces = r.Value
|
||||
.Replace("{{", string.Empty)
|
||||
.Replace("{{", string.Empty)
|
||||
.Split('{')
|
||||
.Skip(1)
|
||||
.ToArray();
|
||||
|
||||
if (openBraces.Any())
|
||||
{
|
||||
var formatNames = new System.Collections.Generic.HashSet<string>(
|
||||
openBraces.Select(b => b.Split('}').First())
|
||||
);
|
||||
|
||||
var resolveParameterNames = r.Comment.Match(
|
||||
c => c.Split(',')
|
||||
.Select(str => str.Split('='))
|
||||
.Where(parts => parts.Count() == 2)
|
||||
.Select(parts => (
|
||||
parts.First().Trim().Trim('{', '}').Split(' ').First(),
|
||||
parts.Last().Trim().Split(' ').First()
|
||||
)).Where(t => uint.TryParse(t.Item1, out var _))
|
||||
.GroupBy(t => t.Item1)
|
||||
.ToDictionary(k => k.Key, v => v.First().Item2),
|
||||
() => new Dictionary<string, string>()
|
||||
);
|
||||
|
||||
var parameterNames = formatNames.Select(
|
||||
n => resolveParameterNames.TryGetValue(n.Trim(), out var paramName)
|
||||
? paramName.Substring(0, 1).ToLower() + paramName.Substring(1)
|
||||
: uint.TryParse(n, out var value) ? $"p{value + 1}" : n
|
||||
).ToArray();
|
||||
|
||||
var from = Properties.Resources.ResourceFormatClassFromMethod.From(
|
||||
new JoinedResourceString(
|
||||
LiteralString.Factory(", "),
|
||||
parameterNames.Select(n => LiteralString.Factory($"{nameof(IResourceString)} {n}")).ToArray()
|
||||
),
|
||||
new JoinedResourceString(
|
||||
LiteralString.Factory($",{System.Environment.NewLine} "),
|
||||
parameterNames.Select(n => LiteralString.Factory($"{n}")).ToArray()
|
||||
)
|
||||
);
|
||||
return Properties.Resources.ResourceFormatClassMembers.From(
|
||||
LiteralString.Factory(r.Name), resourceManagerName, from
|
||||
);
|
||||
}
|
||||
return Properties.Resources.ResourceStringMembers.From(
|
||||
LiteralString.Factory(r.Name), resourceManagerName
|
||||
);
|
||||
});
|
||||
}
|
||||
|
||||
public static IResourceString CreateResourceClassCodeSnippet(
|
||||
@@ -50,9 +98,7 @@ internal static class CodeSnippetFactory
|
||||
IEnumerable<IResourceString> memberSnippets
|
||||
)
|
||||
{
|
||||
var formatString = Properties.Resources.ResourcesClassTemplate;
|
||||
return new FormattableResourceString(
|
||||
formatString,
|
||||
return Properties.Resources.ResourcesClassTemplate.From(
|
||||
LiteralString.Factory(namespaceString),
|
||||
LiteralString.Factory(resourceClassName),
|
||||
resourceManagerSnippet,
|
||||
|
||||
@@ -23,11 +23,17 @@ internal static class Parser
|
||||
{
|
||||
var name = i.GetAttribute("name");
|
||||
var type = i.GetAttribute("type");
|
||||
var value = i.SelectSingleNode("descendant::value")?.InnerXml;
|
||||
var comment = i.SelectSingleNode("descendant::comment")?.InnerXml ?? string.Empty;
|
||||
var value = i.SelectSingleNode("descendant::value")?.InnerXml ?? string.Empty;
|
||||
|
||||
return new Resource(name, value ?? string.Empty)
|
||||
return new Resource(name, value)
|
||||
{
|
||||
Type = type
|
||||
Type = string.IsNullOrWhiteSpace(type)
|
||||
? Option<string>.None
|
||||
: Option<string>.Some(type),
|
||||
Comment = string.IsNullOrWhiteSpace(comment)
|
||||
? Option<string>.None
|
||||
: Option<string>.Some(comment)
|
||||
};
|
||||
}).ToArray();
|
||||
|
||||
|
||||
@@ -1,3 +1,4 @@
|
||||
using ResourceString.Net.Contract;
|
||||
using System;
|
||||
using System.Globalization;
|
||||
using System.Resources;
|
||||
@@ -7,12 +8,13 @@ namespace ResourceString.Net.Logic.Properties
|
||||
{
|
||||
internal static class Resources
|
||||
{
|
||||
|
||||
#region ResourceManager
|
||||
|
||||
private static readonly Type _Type = typeof(Resources);
|
||||
|
||||
private static readonly Lazy<ResourceManager> _ResourceManager = new Lazy<ResourceManager>(
|
||||
() => new ResourceManager(_Type.FullName, _Type.Assembly),
|
||||
() => new ResourceManager(_Type.FullName ?? string.Empty, _Type.Assembly),
|
||||
LazyThreadSafetyMode.PublicationOnly
|
||||
);
|
||||
|
||||
@@ -20,38 +22,93 @@ namespace ResourceString.Net.Logic.Properties
|
||||
|
||||
#endregion // ResourceManager
|
||||
|
||||
#region ResourceStringMembers
|
||||
|
||||
private static readonly Lazy<IResourceString> LazyResourceStringMembers = new Lazy<IResourceString>(
|
||||
internal static class ResourceStringMembers
|
||||
{
|
||||
private static readonly Lazy<IResourceString> LazyFormat = new Lazy<IResourceString>(
|
||||
() => new ResourceManagerString("ResourceStringMembers", ResourceManager, CultureInfo.CurrentCulture),
|
||||
LazyThreadSafetyMode.PublicationOnly
|
||||
);
|
||||
|
||||
public static IResourceString ResourceStringMembers => LazyResourceStringMembers.Value;
|
||||
public static IResourceString Format => LazyFormat.Value;
|
||||
|
||||
#endregion // ResourceStringMembers
|
||||
public static IResourceString From(IResourceString resourceId, IResourceString resourceManagerPropertyName) => new FormattedResourceString(
|
||||
Format,
|
||||
resourceId,
|
||||
resourceManagerPropertyName
|
||||
);
|
||||
|
||||
#region ResourceManagerMemberTemplate
|
||||
}
|
||||
|
||||
private static readonly Lazy<IResourceString> LazyResourceManagerMemberTemplate = new Lazy<IResourceString>(
|
||||
() => new ResourceManagerString("ResourceManagerMemberTemplate", ResourceManager, CultureInfo.CurrentCulture),
|
||||
internal static class ResourceFormatClassMembers
|
||||
{
|
||||
private static readonly Lazy<IResourceString> LazyFormat = new Lazy<IResourceString>(
|
||||
() => new ResourceManagerString("ResourceFormatClassMembers", ResourceManager, CultureInfo.CurrentCulture),
|
||||
LazyThreadSafetyMode.PublicationOnly
|
||||
);
|
||||
|
||||
public static IResourceString ResourceManagerMemberTemplate => LazyResourceManagerMemberTemplate.Value;
|
||||
public static IResourceString Format => LazyFormat.Value;
|
||||
|
||||
#endregion // ResourceManagerMemberTemplate
|
||||
public static IResourceString From(IResourceString resourceId, IResourceString resourceManagerPropertyName, IResourceString fromMethodDefinition) => new FormattedResourceString(
|
||||
Format,
|
||||
resourceId,
|
||||
resourceManagerPropertyName,
|
||||
fromMethodDefinition
|
||||
);
|
||||
|
||||
#region ResourcesClassTemplate
|
||||
}
|
||||
|
||||
private static readonly Lazy<IResourceString> LazyResourcesClassTemplate = new Lazy<IResourceString>(
|
||||
internal static class ResourceFormatClassFromMethod
|
||||
{
|
||||
private static readonly Lazy<IResourceString> LazyFormat = new Lazy<IResourceString>(
|
||||
() => new ResourceManagerString("ResourceFormatClassFromMethod", ResourceManager, CultureInfo.CurrentCulture),
|
||||
LazyThreadSafetyMode.PublicationOnly
|
||||
);
|
||||
|
||||
public static IResourceString Format => LazyFormat.Value;
|
||||
|
||||
public static IResourceString From(IResourceString fromMethodSignature, IResourceString parameterNames) => new FormattedResourceString(
|
||||
Format,
|
||||
fromMethodSignature,
|
||||
parameterNames
|
||||
);
|
||||
|
||||
}
|
||||
|
||||
internal static class ResourcesClassTemplate
|
||||
{
|
||||
private static readonly Lazy<IResourceString> LazyFormat = new Lazy<IResourceString>(
|
||||
() => new ResourceManagerString("ResourcesClassTemplate", ResourceManager, CultureInfo.CurrentCulture),
|
||||
LazyThreadSafetyMode.PublicationOnly
|
||||
);
|
||||
|
||||
public static IResourceString ResourcesClassTemplate => LazyResourcesClassTemplate.Value;
|
||||
public static IResourceString Format => LazyFormat.Value;
|
||||
|
||||
#endregion // ResourcesClassTemplate
|
||||
public static IResourceString From(IResourceString ns, IResourceString className, IResourceString resourceManagerRegion, IResourceString resourceRegions) => new FormattedResourceString(
|
||||
Format,
|
||||
ns,
|
||||
className,
|
||||
resourceManagerRegion,
|
||||
resourceRegions
|
||||
);
|
||||
|
||||
}
|
||||
|
||||
internal static class ResourceManagerMemberTemplate
|
||||
{
|
||||
private static readonly Lazy<IResourceString> LazyFormat = new Lazy<IResourceString>(
|
||||
() => new ResourceManagerString("ResourceManagerMemberTemplate", ResourceManager, CultureInfo.CurrentCulture),
|
||||
LazyThreadSafetyMode.PublicationOnly
|
||||
);
|
||||
|
||||
public static IResourceString Format => LazyFormat.Value;
|
||||
|
||||
public static IResourceString From(IResourceString resourceManagerTypeName) => new FormattedResourceString(
|
||||
Format,
|
||||
resourceManagerTypeName
|
||||
);
|
||||
|
||||
}
|
||||
|
||||
#region DefaultPropertyName_ResourceManager
|
||||
|
||||
@@ -63,5 +120,6 @@ namespace ResourceString.Net.Logic.Properties
|
||||
public static IResourceString DefaultPropertyName_ResourceManager => LazyDefaultPropertyName_ResourceManager.Value;
|
||||
|
||||
#endregion // DefaultPropertyName_ResourceManager
|
||||
|
||||
}
|
||||
}
|
||||
|
||||
@@ -13,6 +13,31 @@
|
||||
|
||||
#endregion // {0}
|
||||
</value>
|
||||
<comment>0 = ResourceId, 1 = ResourceManagerPropertyName</comment>
|
||||
</data>
|
||||
<data name="ResourceFormatClassMembers" xml:space="preserve">
|
||||
<value>
|
||||
internal static class {0}
|
||||
{{
|
||||
private static readonly Lazy<IResourceString> LazyFormat = new Lazy<IResourceString>(
|
||||
() => new ResourceManagerString("{0}", {1}, CultureInfo.CurrentCulture),
|
||||
LazyThreadSafetyMode.PublicationOnly
|
||||
);
|
||||
|
||||
public static IResourceString Format => LazyFormat.Value;
|
||||
{2}
|
||||
}}
|
||||
</value>
|
||||
<comment>0 = ResourceId, 1 = ResourceManagerPropertyName, 2 = FromMethodDefinition</comment>
|
||||
</data>
|
||||
<data name="ResourceFormatClassFromMethod" xml:space="preserve">
|
||||
<value>
|
||||
public static IResourceString From({0}) => new FormattedResourceString(
|
||||
Format,
|
||||
{1}
|
||||
);
|
||||
</value>
|
||||
<comment>0 = FromMethodSignature, 1 = ParameterNames</comment>
|
||||
</data>
|
||||
<data name="ResourcesClassTemplate" xml:space="preserve">
|
||||
<value>
|
||||
@@ -31,6 +56,7 @@ namespace {0}
|
||||
}}
|
||||
}}
|
||||
</value>
|
||||
<comment>0 = ns, 1 = ClassName, 2 = ResourceManagerRegion, 3 = ResourceRegions</comment>
|
||||
</data>
|
||||
<data name="ResourceManagerMemberTemplate" xml:space="preserve">
|
||||
<value>
|
||||
@@ -39,7 +65,7 @@ namespace {0}
|
||||
private static readonly Type _Type = typeof({0});
|
||||
|
||||
private static readonly Lazy<ResourceManager> _ResourceManager = new Lazy<ResourceManager>(
|
||||
() => new ResourceManager(_Type.FullName, _Type.Assembly),
|
||||
() => new ResourceManager(_Type.FullName ?? string.Empty, _Type.Assembly),
|
||||
LazyThreadSafetyMode.PublicationOnly
|
||||
);
|
||||
|
||||
@@ -47,6 +73,7 @@ namespace {0}
|
||||
|
||||
#endregion // ResourceManager
|
||||
</value>
|
||||
<comment>0 = ResourceManagerTypeName</comment>
|
||||
</data>
|
||||
<data name="DefaultPropertyName_ResourceManager">
|
||||
<value>ResourceManager</value>
|
||||
|
||||
@@ -7,7 +7,7 @@
|
||||
</PropertyGroup>
|
||||
|
||||
<ItemGroup>
|
||||
<PackageReference Include="LanguageExt.Core" Version="4.4.2" />
|
||||
<PackageReference Include="LanguageExt.Core" Version="4.4.3" />
|
||||
<PackageReference Include="System.Resources.Extensions" Version="7.0.0" />
|
||||
</ItemGroup>
|
||||
|
||||
@@ -19,5 +19,11 @@
|
||||
<AssemblyAttribute Include="System.Runtime.CompilerServices.InternalsVisibleTo">
|
||||
<_Parameter1>$(MSBuildProjectName).Tests</_Parameter1>
|
||||
</AssemblyAttribute>
|
||||
<AssemblyAttribute Include="System.Runtime.CompilerServices.InternalsVisibleTo">
|
||||
<_Parameter1>ResourceString.Net.App.Console</_Parameter1>
|
||||
</AssemblyAttribute>
|
||||
<AssemblyAttribute Include="System.Runtime.CompilerServices.InternalsVisibleTo">
|
||||
<_Parameter1>ResourceString.Net</_Parameter1>
|
||||
</AssemblyAttribute>
|
||||
</ItemGroup>
|
||||
</Project>
|
||||
|
||||
3
ResourceString.Net/FodyWeavers.xml
Normal file
3
ResourceString.Net/FodyWeavers.xml
Normal file
@@ -0,0 +1,3 @@
|
||||
<Weavers xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:noNamespaceSchemaLocation="FodyWeavers.xsd">
|
||||
<ILMerge FullImport='true' HideImportedTypes='false' IncludeAssemblies='ResourceString.Net*|LanguageExt*'/>
|
||||
</Weavers>
|
||||
50
ResourceString.Net/Generator.cs
Normal file
50
ResourceString.Net/Generator.cs
Normal file
@@ -0,0 +1,50 @@
|
||||
using Microsoft.CodeAnalysis;
|
||||
using ResourceString.Net.Logic.Factories;
|
||||
using ResourceString.Net.Logic.Parsers.Resx;
|
||||
|
||||
namespace ResourceString.Net;
|
||||
|
||||
[Generator]
|
||||
public class Generator : IIncrementalGenerator
|
||||
{
|
||||
public void Initialize(IncrementalGeneratorInitializationContext initContext)
|
||||
{
|
||||
// find all additional files that end with .txt
|
||||
var resxFiles = initContext.AdditionalTextsProvider.Where(static file => file.Path.EndsWith(".resx"));
|
||||
var assemblies = initContext.CompilationProvider.Select(static (c, _) => c.Assembly);
|
||||
|
||||
// read their contents and save their name
|
||||
var config = resxFiles.Combine(assemblies).Select(
|
||||
(t, cancellationToken) => (
|
||||
name: System.IO.Path.GetFileNameWithoutExtension(t.Left.Path),
|
||||
content: t.Left.GetText(cancellationToken)!.ToString(),
|
||||
assembly: t.Right,
|
||||
path: t.Left.Path
|
||||
)
|
||||
);
|
||||
|
||||
// generate a class that contains their values as const strings
|
||||
initContext.RegisterSourceOutput(config, (spc, t) =>
|
||||
{
|
||||
Parser.TryParse(t.content).IfSome(v =>
|
||||
{
|
||||
var relativeNamespace = System.IO.Path.GetDirectoryName(t.path).Substring(
|
||||
System.IO.Path.GetDirectoryName(t.assembly.Locations[0].GetLineSpan().Path).Length
|
||||
).Trim(System.IO.Path.DirectorySeparatorChar).Replace(System.IO.Path.DirectorySeparatorChar, '.');
|
||||
|
||||
var ns = string.IsNullOrWhiteSpace(relativeNamespace)
|
||||
? t.assembly.Name
|
||||
: $"{t.assembly.Name}.{relativeNamespace}";
|
||||
|
||||
var snippet = CodeSnippetFactory.CreateResourceClassCodeSnippet(
|
||||
ns,
|
||||
t.name,
|
||||
CodeSnippetFactory.CreateResourceMangerMemberCodeSnippet(t.name),
|
||||
v.Resources
|
||||
);
|
||||
|
||||
spc.AddSource(ns + t.name, snippet.Value);
|
||||
});
|
||||
});
|
||||
}
|
||||
}
|
||||
41
ResourceString.Net/ResourceString.Net.csproj
Normal file
41
ResourceString.Net/ResourceString.Net.csproj
Normal file
@@ -0,0 +1,41 @@
|
||||
<Project Sdk="Microsoft.NET.Sdk">
|
||||
|
||||
<PropertyGroup>
|
||||
<TargetFramework>netstandard2.0</TargetFramework>
|
||||
<LangVersion>latest</LangVersion>
|
||||
<Nullable>enable</Nullable>
|
||||
<EnforceExtendedAnalyzerRules>true</EnforceExtendedAnalyzerRules>
|
||||
<CopyLocalLockFileAssemblies>true</CopyLocalLockFileAssemblies>
|
||||
<Authors>stubbfel</Authors>
|
||||
<PackageRequireLicenseAcceptance>false</PackageRequireLicenseAcceptance>
|
||||
<PackageTags>generator</PackageTags>
|
||||
<NoPackageAnalysis>true</NoPackageAnalysis>
|
||||
</PropertyGroup>
|
||||
|
||||
<!-- It puts the dll in the expected folder of the NuGet package to be recognized as a C# analyzer -->
|
||||
<ItemGroup>
|
||||
<None Include="$(OutputPath)\$(AssemblyName).dll" Pack="true" PackagePath="analyzers/dotnet/cs" Visible="false" />
|
||||
</ItemGroup>
|
||||
|
||||
<ItemGroup>
|
||||
<PackageReference Include="Fody" Version="6.7.0">
|
||||
<IncludeAssets>runtime; build; native; contentfiles; analyzers; buildtransitive</IncludeAssets>
|
||||
<PrivateAssets>all</PrivateAssets>
|
||||
</PackageReference>
|
||||
<PackageReference Include="ILMerge.Fody" Version="1.24.0">
|
||||
<PrivateAssets>all</PrivateAssets>
|
||||
</PackageReference>
|
||||
<PackageReference Include="Microsoft.CodeAnalysis.Analyzers" Version="3.3.4">
|
||||
<PrivateAssets>all</PrivateAssets>
|
||||
</PackageReference>
|
||||
<PackageReference Include="Microsoft.CodeAnalysis.CSharp" Version="4.3.0">
|
||||
<PrivateAssets>all</PrivateAssets>
|
||||
</PackageReference>
|
||||
</ItemGroup>
|
||||
|
||||
<ItemGroup>
|
||||
<ProjectReference Include="..\ResourceString.Net.Logic\ResourceString.Net.Logic.csproj">
|
||||
<PrivateAssets>all</PrivateAssets>
|
||||
</ProjectReference>
|
||||
</ItemGroup>
|
||||
</Project>
|
||||
48
ResourceString.Net/deps.nix
Normal file
48
ResourceString.Net/deps.nix
Normal file
@@ -0,0 +1,48 @@
|
||||
{ fetchNuGet }: [
|
||||
(fetchNuGet { pname = "Fody"; version = "6.7.0"; sha256 = "0fv0zrffa296qjyi11yk31vfqh6gm1nxsx8g5zz380jcsrilnp3h"; })
|
||||
(fetchNuGet { pname = "ILMerge.Fody"; version = "1.24.0"; sha256 = "1gibwcl8ngbvwlcqzd9clysrhsjb8g4gwry7n8ifw1mrw7sjjk6x"; })
|
||||
(fetchNuGet { pname = "LanguageExt.Core"; version = "4.4.3"; sha256 = "1pd7wx4c21v56y6i75sxbs990mjrs6bp9h8c48a5w79s1zpbinw5"; })
|
||||
(fetchNuGet { pname = "Microsoft.Bcl.AsyncInterfaces"; version = "7.0.0"; sha256 = "1waiggh3g1cclc81gmjrqbh128kwfjky3z79ma4bd2ms9pa3gvfm"; })
|
||||
(fetchNuGet { pname = "Microsoft.CodeAnalysis.Analyzers"; version = "3.3.4"; sha256 = "0wd6v57p53ahz5z9zg4iyzmy3src7rlsncyqpcag02jjj1yx6g58"; })
|
||||
(fetchNuGet { pname = "Microsoft.CodeAnalysis.Common"; version = "4.3.0"; sha256 = "0qpxygiq53v2d2wl6hccnkjf1lhlxjh4q3w5b6d23aq9pw5qj626"; })
|
||||
(fetchNuGet { pname = "Microsoft.CodeAnalysis.CSharp"; version = "4.3.0"; sha256 = "0m9qqn391ayfi1ffkzvhpij790hs96q6dbhzfkj2ahvw6qx47b30"; })
|
||||
(fetchNuGet { pname = "Microsoft.CSharp"; version = "4.7.0"; sha256 = "0gd67zlw554j098kabg887b5a6pq9kzavpa3jjy5w53ccjzjfy8j"; })
|
||||
(fetchNuGet { pname = "Microsoft.NETCore.Platforms"; version = "1.1.0"; sha256 = "08vh1r12g6ykjygq5d3vq09zylgb84l63k49jc4v8faw9g93iqqm"; })
|
||||
(fetchNuGet { pname = "Microsoft.NETCore.Targets"; version = "1.1.0"; sha256 = "193xwf33fbm0ni3idxzbr5fdq3i2dlfgihsac9jj7whj0gd902nh"; })
|
||||
(fetchNuGet { pname = "NETStandard.Library"; version = "2.0.3"; sha256 = "1fn9fxppfcg4jgypp2pmrpr6awl3qz1xmnri0cygpkwvyx27df1y"; })
|
||||
(fetchNuGet { pname = "System.Buffers"; version = "4.5.1"; sha256 = "04kb1mdrlcixj9zh1xdi5as0k0qi8byr5mi3p3jcxx72qz93s2y3"; })
|
||||
(fetchNuGet { pname = "System.Collections"; version = "4.3.0"; sha256 = "19r4y64dqyrq6k4706dnyhhw7fs24kpp3awak7whzss39dakpxk9"; })
|
||||
(fetchNuGet { pname = "System.Collections.Immutable"; version = "6.0.0"; sha256 = "1js98kmjn47ivcvkjqdmyipzknb9xbndssczm8gq224pbaj1p88c"; })
|
||||
(fetchNuGet { pname = "System.Diagnostics.Contracts"; version = "4.3.0"; sha256 = "1gxawcr4d2y5jmc6y7iv8c1q83hm22f6savcvspvhmpl974jigib"; })
|
||||
(fetchNuGet { pname = "System.Diagnostics.Debug"; version = "4.3.0"; sha256 = "00yjlf19wjydyr6cfviaph3vsjzg3d5nvnya26i2fvfg53sknh3y"; })
|
||||
(fetchNuGet { pname = "System.Globalization"; version = "4.3.0"; sha256 = "1cp68vv683n6ic2zqh2s1fn4c2sd87g5hpp6l4d4nj4536jz98ki"; })
|
||||
(fetchNuGet { pname = "System.IO"; version = "4.3.0"; sha256 = "05l9qdrzhm4s5dixmx68kxwif4l99ll5gqmh7rqgw554fx0agv5f"; })
|
||||
(fetchNuGet { pname = "System.Linq"; version = "4.3.0"; sha256 = "1w0gmba695rbr80l1k2h4mrwzbzsyfl2z4klmpbsvsg5pm4a56s7"; })
|
||||
(fetchNuGet { pname = "System.Linq.Expressions"; version = "4.3.0"; sha256 = "0ky2nrcvh70rqq88m9a5yqabsl4fyd17bpr63iy2mbivjs2nyypv"; })
|
||||
(fetchNuGet { pname = "System.Linq.Queryable"; version = "4.3.0"; sha256 = "0vidv9cjwy8scabxd33mm4zl5vql695rz56ydc42m9b731xi2ahj"; })
|
||||
(fetchNuGet { pname = "System.Memory"; version = "4.5.4"; sha256 = "14gbbs22mcxwggn0fcfs1b062521azb9fbb7c113x0mq6dzq9h6y"; })
|
||||
(fetchNuGet { pname = "System.Memory"; version = "4.5.5"; sha256 = "08jsfwimcarfzrhlyvjjid61j02irx6xsklf32rv57x2aaikvx0h"; })
|
||||
(fetchNuGet { pname = "System.Numerics.Vectors"; version = "4.4.0"; sha256 = "0rdvma399070b0i46c4qq1h2yvjj3k013sqzkilz4bz5cwmx1rba"; })
|
||||
(fetchNuGet { pname = "System.ObjectModel"; version = "4.3.0"; sha256 = "191p63zy5rpqx7dnrb3h7prvgixmk168fhvvkkvhlazncf8r3nc2"; })
|
||||
(fetchNuGet { pname = "System.Reflection"; version = "4.3.0"; sha256 = "0xl55k0mw8cd8ra6dxzh974nxif58s3k1rjv1vbd7gjbjr39j11m"; })
|
||||
(fetchNuGet { pname = "System.Reflection.Emit"; version = "4.7.0"; sha256 = "121l1z2ypwg02yz84dy6gr82phpys0njk7yask3sihgy214w43qp"; })
|
||||
(fetchNuGet { pname = "System.Reflection.Emit.ILGeneration"; version = "4.3.0"; sha256 = "0w1n67glpv8241vnpz1kl14sy7zlnw414aqwj4hcx5nd86f6994q"; })
|
||||
(fetchNuGet { pname = "System.Reflection.Emit.ILGeneration"; version = "4.7.0"; sha256 = "0l8jpxhpgjlf1nkz5lvp61r4kfdbhr29qi8aapcxn3izd9wd0j8r"; })
|
||||
(fetchNuGet { pname = "System.Reflection.Emit.Lightweight"; version = "4.7.0"; sha256 = "0mbjfajmafkca47zr8v36brvknzks5a7pgb49kfq2d188pyv6iap"; })
|
||||
(fetchNuGet { pname = "System.Reflection.Extensions"; version = "4.3.0"; sha256 = "02bly8bdc98gs22lqsfx9xicblszr2yan7v2mmw3g7hy6miq5hwq"; })
|
||||
(fetchNuGet { pname = "System.Reflection.Metadata"; version = "5.0.0"; sha256 = "17qsl5nanlqk9iz0l5wijdn6ka632fs1m1fvx18dfgswm258r3ss"; })
|
||||
(fetchNuGet { pname = "System.Reflection.Primitives"; version = "4.3.0"; sha256 = "04xqa33bld78yv5r93a8n76shvc8wwcdgr1qvvjh959g3rc31276"; })
|
||||
(fetchNuGet { pname = "System.Reflection.TypeExtensions"; version = "4.3.0"; sha256 = "0y2ssg08d817p0vdag98vn238gyrrynjdj4181hdg780sif3ykp1"; })
|
||||
(fetchNuGet { pname = "System.Resources.Extensions"; version = "7.0.0"; sha256 = "0d5gk5g5qqkwa728jwx9yabgjvgywsy6k8r5vgqv2dmlvjrqflb4"; })
|
||||
(fetchNuGet { pname = "System.Resources.ResourceManager"; version = "4.3.0"; sha256 = "0sjqlzsryb0mg4y4xzf35xi523s4is4hz9q4qgdvlvgivl7qxn49"; })
|
||||
(fetchNuGet { pname = "System.Runtime"; version = "4.3.0"; sha256 = "066ixvgbf2c929kgknshcxqj6539ax7b9m570cp8n179cpfkapz7"; })
|
||||
(fetchNuGet { pname = "System.Runtime.CompilerServices.Unsafe"; version = "4.5.3"; sha256 = "1afi6s2r1mh1kygbjmfba6l4f87pi5sg13p4a48idqafli94qxln"; })
|
||||
(fetchNuGet { pname = "System.Runtime.CompilerServices.Unsafe"; version = "6.0.0"; sha256 = "0qm741kh4rh57wky16sq4m0v05fxmkjjr87krycf5vp9f0zbahbc"; })
|
||||
(fetchNuGet { pname = "System.Runtime.Extensions"; version = "4.3.0"; sha256 = "1ykp3dnhwvm48nap8q23893hagf665k0kn3cbgsqpwzbijdcgc60"; })
|
||||
(fetchNuGet { pname = "System.Text.Encoding"; version = "4.3.0"; sha256 = "1f04lkir4iladpp51sdgmis9dj4y8v08cka0mbmsy0frc9a4gjqr"; })
|
||||
(fetchNuGet { pname = "System.Text.Encoding.CodePages"; version = "6.0.0"; sha256 = "0gm2kiz2ndm9xyzxgi0jhazgwslcs427waxgfa30m7yqll1kcrww"; })
|
||||
(fetchNuGet { pname = "System.Threading"; version = "4.3.0"; sha256 = "0rw9wfamvhayp5zh3j7p1yfmx9b5khbf4q50d8k5rk993rskfd34"; })
|
||||
(fetchNuGet { pname = "System.Threading.Tasks"; version = "4.3.0"; sha256 = "134z3v9abw3a6jsw17xl3f6hqjpak5l682k2vz39spj4kmydg6k7"; })
|
||||
(fetchNuGet { pname = "System.Threading.Tasks.Extensions"; version = "4.5.4"; sha256 = "0y6ncasgfcgnjrhynaf0lwpkpkmv4a07sswwkwbwb5h7riisj153"; })
|
||||
(fetchNuGet { pname = "System.ValueTuple"; version = "4.5.0"; sha256 = "00k8ja51d0f9wrq4vv5z2jhq8hy31kac2rg0rv06prylcybzl8cy"; })
|
||||
]
|
||||
6
alias.sh
Normal file
6
alias.sh
Normal file
@@ -0,0 +1,6 @@
|
||||
#!/bin/env sh
|
||||
|
||||
alias nixe='nix --experimental-features "nix-command flakes"'
|
||||
alias nulock='nixe run .#devTasks.updateNugetLock'
|
||||
alias fllock='nixe run .#devTasks.updateFlakeLock'
|
||||
alias ulock='nulock && fllock'
|
||||
23
example/Example.App/Example.App.csproj
Normal file
23
example/Example.App/Example.App.csproj
Normal file
@@ -0,0 +1,23 @@
|
||||
<Project Sdk="Microsoft.NET.Sdk">
|
||||
|
||||
<ItemGroup>
|
||||
<ProjectReference Include="..\..\ResourceString.Net\ResourceString.Net.csproj" OutputItemType="Analyzer" />
|
||||
</ItemGroup>
|
||||
|
||||
<PropertyGroup>
|
||||
<OutputType>Exe</OutputType>
|
||||
<TargetFramework>net6.0</TargetFramework>
|
||||
<ImplicitUsings>enable</ImplicitUsings>
|
||||
<Nullable>enable</Nullable>
|
||||
<LangVersion>latest</LangVersion>
|
||||
<IsPackable>false</IsPackable>
|
||||
</PropertyGroup>
|
||||
|
||||
<PropertyGroup>
|
||||
<AdditionalFileItemNames>$(AdditionalFileItemNames);EmbeddedResource</AdditionalFileItemNames>
|
||||
</PropertyGroup>
|
||||
|
||||
<ItemGroup>
|
||||
<PackageReference Include="System.Resources.Extensions" Version="7.0.0" />
|
||||
</ItemGroup>
|
||||
</Project>
|
||||
22
example/Example.App/Program.cs
Normal file
22
example/Example.App/Program.cs
Normal file
@@ -0,0 +1,22 @@
|
||||
namespace Test
|
||||
{
|
||||
|
||||
internal class Program
|
||||
{
|
||||
private static void Main(string[] args)
|
||||
{
|
||||
Console.WriteLine("Hello");
|
||||
|
||||
var foo = Example.App.Resources.Test1;
|
||||
Console.WriteLine(foo.Value);
|
||||
|
||||
var bar = Example.App.Properties.SubProperties.Properties.Test2.From(
|
||||
ResourceString.Net.Contract.LiteralString.Factory("1"),
|
||||
ResourceString.Net.Contract.LiteralString.Factory("2")
|
||||
);
|
||||
|
||||
Console.WriteLine(bar.Value);
|
||||
}
|
||||
}
|
||||
|
||||
}
|
||||
14
example/Example.App/Properties/SubProperties/Properties.resx
Normal file
14
example/Example.App/Properties/SubProperties/Properties.resx
Normal file
@@ -0,0 +1,14 @@
|
||||
<?xml version='1.0' encoding='utf-8'?>
|
||||
<root>
|
||||
<data name='Test1'>
|
||||
<value>Value1</value>
|
||||
<comment>This is a greeting message.</comment>
|
||||
</data>
|
||||
<data name='Test2' type='System.String'>
|
||||
<value>{1}Value2{0}{{2}}</value>
|
||||
<comment>2 = prefix</comment>
|
||||
</data>
|
||||
<data name='Test3' type='System.Byte[]'>
|
||||
<value>0xDEADBEEF</value>
|
||||
</data>
|
||||
</root>
|
||||
14
example/Example.App/Resources.resx
Normal file
14
example/Example.App/Resources.resx
Normal file
@@ -0,0 +1,14 @@
|
||||
<?xml version='1.0' encoding='utf-8'?>
|
||||
<root>
|
||||
<data name='Test1'>
|
||||
<value>Value1</value>
|
||||
<comment>This is a greeting message.</comment>
|
||||
</data>
|
||||
<data name='Test2' type='System.String'>
|
||||
<value>{1}Value2{0}{{2}}</value>
|
||||
<comment>2 = prefix</comment>
|
||||
</data>
|
||||
<data name='Test3' type='System.Byte[]'>
|
||||
<value>0xDEADBEEF</value>
|
||||
</data>
|
||||
</root>
|
||||
26
flake.lock
generated
Normal file
26
flake.lock
generated
Normal file
@@ -0,0 +1,26 @@
|
||||
{
|
||||
"nodes": {
|
||||
"nixpkgs": {
|
||||
"locked": {
|
||||
"lastModified": 1684364070,
|
||||
"narHash": "sha256-+bmqPSEQBePWwmfwxUX8kvJLyg8OM9mRKnDi5qB+m1s=",
|
||||
"owner": "NixOS",
|
||||
"repo": "nixpkgs",
|
||||
"rev": "2e6eb88c9ab70147e6087d37c833833fd4a907e5",
|
||||
"type": "github"
|
||||
},
|
||||
"original": {
|
||||
"id": "nixpkgs",
|
||||
"ref": "nixos-22.11-small",
|
||||
"type": "indirect"
|
||||
}
|
||||
},
|
||||
"root": {
|
||||
"inputs": {
|
||||
"nixpkgs": "nixpkgs"
|
||||
}
|
||||
}
|
||||
},
|
||||
"root": "root",
|
||||
"version": 7
|
||||
}
|
||||
158
flake.nix
Normal file
158
flake.nix
Normal file
@@ -0,0 +1,158 @@
|
||||
{
|
||||
description = "";
|
||||
inputs.nixpkgs.url = "nixpkgs/nixos-22.11-small";
|
||||
outputs = { self, nixpkgs }:
|
||||
let
|
||||
name = "ResourceString.Net";
|
||||
version = "0.0.1";
|
||||
supportedSystems =
|
||||
[ "x86_64-linux" "x86_64-darwin" "aarch64-linux" "aarch64-darwin" ];
|
||||
forAllSystems = nixpkgs.lib.genAttrs supportedSystems;
|
||||
nixpkgsFor = forAllSystems (system: import nixpkgs { inherit system; });
|
||||
scripts = forAllSystems (system: {
|
||||
updateFlakeLockFile = nixpkgsFor.${system}.writeScript "update_flake_lock.sh" ''
|
||||
nix --experimental-features 'nix-command flakes' flake lock --update-input nixpkgs
|
||||
nix --experimental-features 'nix-command flakes' build
|
||||
'';
|
||||
|
||||
updateNugetLockFile = nixpkgsFor.${system}.writeScript "update_nuget_lock.sh" ''
|
||||
temp_package_folder=$(mktemp -d)
|
||||
dotnet restore ResourceString.Net --packages "$temp_package_folder"
|
||||
${nixpkgsFor.${system}.nuget-to-nix}/bin/nuget-to-nix "$temp_package_folder" > ResourceString.Net/deps.nix
|
||||
rm -rf "$temp_package_folder"
|
||||
'';
|
||||
|
||||
autoTag = nixpkgsFor.${system}.writeScript "auto_tag.sh" ''
|
||||
git tag --force v${version}
|
||||
git push origin v${version}
|
||||
'';
|
||||
updateLogicResourceFile = nixpkgsFor.${system}.writeScript "update_logic_resource_file.sh" ''
|
||||
dotnet run --project ResourceString.Net.App.Console ResourceString.Net.Logic/Properties/Resources.resx ResourceString.Net.Logic.Properties > ResourceString.Net.Logic/Properties/Resources.cs.tmp
|
||||
mv ResourceString.Net.Logic/Properties/Resources.cs.tmp ResourceString.Net.Logic/Properties/Resources.cs
|
||||
'';
|
||||
runAllTests = nixpkgsFor.${system}.writeScript "run_all_tests.sh" ''
|
||||
dotnet test *.Tests
|
||||
'';
|
||||
buildAllAssemblies = nixpkgsFor.${system}.writeScript "build_all_assemblies.sh" ''
|
||||
# Find all .csproj files below the current folder
|
||||
csproj_files=$(find . -name '*.csproj')
|
||||
|
||||
# Create a temporary file to store the number of dependencies for each project
|
||||
temp_file=$(mktemp)
|
||||
|
||||
# Loop through each .csproj file and count its number of ResourceString dependencies
|
||||
for file in $csproj_files
|
||||
do
|
||||
count=$(cat "$file" | grep -c "<ProjectReference.*ResourceString")
|
||||
depth=$(echo "$file" | tr -cd '/' | wc -c)
|
||||
dot_count=$(echo "$file" | tr -cd '.' | wc -c)
|
||||
echo "$depth $count $dot_count $file" >> "$temp_file"
|
||||
done
|
||||
|
||||
# Sort the projects by their count of dependencies and path depth, in ascending order
|
||||
sorted_projects=$(sort -nk1 -nk2 -nk3 -t' ' "$temp_file" | awk '{print $4}')
|
||||
|
||||
# Loop through the sorted projects and build them
|
||||
while read -r file
|
||||
do
|
||||
echo "Building project: $file"
|
||||
dotnet build "$file"
|
||||
done << EOF
|
||||
$sorted_projects
|
||||
EOF
|
||||
|
||||
# Remove the temporary file
|
||||
rm "$temp_file"
|
||||
|
||||
'';
|
||||
|
||||
cleanAllAssemblies = nixpkgsFor.${system}.writeScript "clean_all_assemblies.sh" ''
|
||||
csproj_files=$(find . -name '*.csproj')
|
||||
for file in $csproj_files
|
||||
do
|
||||
dotnet clean "$file"
|
||||
done
|
||||
'';
|
||||
runDefaultApp = nixpkgsFor.${system}.writeScript "run_default_app.sh" ''
|
||||
dotnet run --project ResourceString.Net.App.Console $@
|
||||
'';
|
||||
createNugetPackage = nixpkgsFor.${system}.writeScript "run_default_app.sh" ''
|
||||
dotnet run --project ResourceString.Net.App.Console $@
|
||||
'';
|
||||
});
|
||||
|
||||
in
|
||||
rec {
|
||||
packages = forAllSystems (system: {
|
||||
default = nixpkgsFor.${system}.buildDotnetModule {
|
||||
pname = name;
|
||||
version = version;
|
||||
|
||||
src = self;
|
||||
|
||||
projectFile = [
|
||||
"ResourceString.Net/ResourceString.Net.csproj"
|
||||
"ResourceString.Net.App.Console/ResourceString.Net.App.Console.csproj"
|
||||
"example/Example.App/Example.App.csproj"
|
||||
];
|
||||
|
||||
nugetDeps = ./ResourceString.Net/deps.nix;
|
||||
projectReferences = [ ];
|
||||
|
||||
dotnet-sdk = nixpkgsFor.${system}.dotnet-sdk;
|
||||
dotnet-runtime = nixpkgsFor.${system}.dotnet-runtime;
|
||||
executables = [ "ResourceString.Net.App.Console" "Example.App" ];
|
||||
packNupkg = true;
|
||||
runtimeDeps = [ ];
|
||||
};
|
||||
});
|
||||
|
||||
apps = forAllSystems (system: {
|
||||
default = {
|
||||
type = "app";
|
||||
program = "${scripts.${system}.runDefaultApp}";
|
||||
};
|
||||
devTasks = {
|
||||
updateFlakeLock = {
|
||||
type = "app";
|
||||
program = "${scripts.${system}.updateFlakeLockFile}";
|
||||
};
|
||||
updateNugetLock = {
|
||||
type = "app";
|
||||
program = "${scripts.${system}.updateNugetLockFile}";
|
||||
};
|
||||
autoTag = {
|
||||
type = "app";
|
||||
program = "${scripts.${system}.autoTag}";
|
||||
};
|
||||
updateLogicResourceFile = {
|
||||
type = "app";
|
||||
program = "${scripts.${system}.updateLogicResourceFile}";
|
||||
};
|
||||
runAllTests = {
|
||||
type = "app";
|
||||
program = "${scripts.${system}.runAllTests}";
|
||||
};
|
||||
buildAllAssemblies = {
|
||||
type = "app";
|
||||
program = "${scripts.${system}.buildAllAssemblies}";
|
||||
};
|
||||
cleanAllAssemblies = {
|
||||
type = "app";
|
||||
program = "${scripts.${system}.cleanAllAssemblies}";
|
||||
};
|
||||
};
|
||||
});
|
||||
|
||||
devShells = forAllSystems (system: {
|
||||
default = nixpkgsFor.${system}.mkShell {
|
||||
name = "dev-shell";
|
||||
packages =
|
||||
[ nixpkgsFor.${system}.dotnet-sdk nixpkgsFor.${system}.nixfmt ];
|
||||
shellHook = ''
|
||||
alias nixe="nix --experimental-features 'nix-command flakes'"
|
||||
'';
|
||||
};
|
||||
});
|
||||
};
|
||||
}
|
||||
Reference in New Issue
Block a user