2017-08-21 15:34:15 +00:00
|
|
|
//
|
|
|
|
// resx2sr.cs
|
|
|
|
//
|
|
|
|
// Authors:
|
|
|
|
// Marek Safar <marek.safar@gmail.com>
|
|
|
|
//
|
|
|
|
// Copyright (C) 2016 Xamarin Inc (http://www.xamarin.com)
|
|
|
|
//
|
|
|
|
// 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.
|
|
|
|
//
|
|
|
|
|
2019-04-12 14:10:50 +00:00
|
|
|
/*
|
|
|
|
* When making any changes to this tool, please be aware that these are not
|
|
|
|
* automatically tested by a Jenkins job, so you need to do that manually.
|
|
|
|
*
|
|
|
|
* You need to do the following:
|
|
|
|
*
|
|
|
|
* 1.) Build the tool with the `build` profile
|
|
|
|
* $ make PROFILE=build -C mcs/tools/resx2sr
|
|
|
|
*
|
|
|
|
* 2.) Update the SR in corlib:
|
|
|
|
* $ make -C mcs/class/corlib update-corefx-sr
|
|
|
|
*
|
|
|
|
* 3.) Look at the output by doing a `git diff` and also building the code.
|
|
|
|
*
|
|
|
|
* 4.) Search for 'RESX_RESOURCE_STRING' and `RESX_EXTRA_ARGUMENTS` in all BCL Makefiles:
|
|
|
|
* $ git grep RESX_RESOURCE_STRING mcs/class
|
|
|
|
* $ git grep 'RESX_EXTRA_ARGUMENTS' mcs/class
|
|
|
|
*
|
|
|
|
* 5.) As a minimum, repeat steps 2.) and 3.) for each directory containing `RESX_EXTRA_ARGUMENTS`.
|
|
|
|
* (at the moment, this is only corlib, but please check with `git grep` to make sure).
|
|
|
|
*
|
|
|
|
* 6.) To test the tool in `--existing` mode, check out `System.Web.Services`, you can do something
|
|
|
|
* like this:
|
|
|
|
*
|
|
|
|
* RESX_EXTRA_ARGUMENTS = --in=corefx/SR.template.cs --name=System.Web.Services.Res --existing
|
|
|
|
* RESX_RESOURCE_STRING = ../referencesource/System.Web.Services/System.Web.Services.txt
|
|
|
|
*
|
|
|
|
* If you have any questions about the process, please ask me and I'll be glad to help.
|
|
|
|
*
|
|
|
|
* December 7th, 2018
|
|
|
|
* Martin Baulig (mabaul@microsoft.com)
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
2017-08-21 15:34:15 +00:00
|
|
|
using System;
|
|
|
|
using System.IO;
|
|
|
|
using System.Collections.Generic;
|
|
|
|
using System.Resources;
|
|
|
|
using System.ComponentModel.Design;
|
2018-08-07 15:19:03 +00:00
|
|
|
using System.Text.RegularExpressions;
|
2017-08-21 15:34:15 +00:00
|
|
|
using Mono.Options;
|
|
|
|
|
|
|
|
public class Program
|
|
|
|
{
|
|
|
|
class CmdOptions
|
|
|
|
{
|
|
|
|
public bool ShowHelp { get; set; }
|
|
|
|
public string OutputFile { get; set; }
|
2019-04-12 14:10:50 +00:00
|
|
|
public bool ExistingOnly { get; set; }
|
|
|
|
public bool WarnConstantMismatch { get; set; }
|
2017-08-21 15:34:15 +00:00
|
|
|
}
|
|
|
|
|
2018-08-07 15:19:03 +00:00
|
|
|
internal static List<string> InputFiles;
|
|
|
|
internal static Dictionary<string, object> ExistingKeys;
|
|
|
|
|
2017-08-21 15:34:15 +00:00
|
|
|
public static int Main (string[] args)
|
|
|
|
{
|
|
|
|
var options = new CmdOptions ();
|
|
|
|
|
2018-08-07 15:19:03 +00:00
|
|
|
InputFiles = new List<string> ();
|
|
|
|
ExistingKeys = new Dictionary<string, object> ();
|
|
|
|
|
2019-04-12 14:10:50 +00:00
|
|
|
string className = "SR";
|
|
|
|
|
2017-08-21 15:34:15 +00:00
|
|
|
var p = new OptionSet () {
|
|
|
|
{ "o|out=", "Specifies output file name",
|
|
|
|
v => options.OutputFile = v },
|
2018-08-07 15:19:03 +00:00
|
|
|
{ "i|in=", "Specifies input file name",
|
|
|
|
v => InputFiles.Add (v) },
|
2019-04-12 14:10:50 +00:00
|
|
|
{ "n|name=", "Name for generated class, default is 'SR'",
|
|
|
|
v => className = v },
|
|
|
|
{ "h|help", "Display available options",
|
2017-08-21 15:34:15 +00:00
|
|
|
v => options.ShowHelp = v != null },
|
2019-04-12 14:10:50 +00:00
|
|
|
{ "e|existing", "Only update existing values, do not add keys",
|
|
|
|
v => options.ExistingOnly = true
|
|
|
|
}, { "warn-mismatch", "Warn about constant mismatches",
|
|
|
|
v => options.WarnConstantMismatch = true
|
|
|
|
}
|
2017-08-21 15:34:15 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
List<string> extra;
|
|
|
|
try {
|
|
|
|
extra = p.Parse (args);
|
|
|
|
} catch (OptionException e) {
|
|
|
|
Console.WriteLine (e.Message);
|
|
|
|
Console.WriteLine ("Try 'resx2sr -help' for more information.");
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (options.ShowHelp) {
|
|
|
|
ShowHelp (p);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (extra.Count < 1) {
|
|
|
|
ShowHelp (p);
|
|
|
|
return 2;
|
|
|
|
}
|
|
|
|
|
2018-08-07 15:19:03 +00:00
|
|
|
if (!LoadInputFiles ())
|
|
|
|
return 4;
|
|
|
|
|
2017-08-21 15:34:15 +00:00
|
|
|
var resxStrings = new List<Tuple<string, string, string>> ();
|
|
|
|
if (!LoadStrings (resxStrings, extra))
|
|
|
|
return 3;
|
|
|
|
|
2019-04-12 14:10:50 +00:00
|
|
|
GenerateFile (className, resxStrings, options);
|
2017-08-21 15:34:15 +00:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void ShowHelp (OptionSet p)
|
|
|
|
{
|
|
|
|
Console.WriteLine ("Usage: resx2sr [options] input-files");
|
|
|
|
Console.WriteLine ("Generates C# file with string constants from resource file");
|
|
|
|
Console.WriteLine ();
|
|
|
|
Console.WriteLine ("Options:");
|
|
|
|
p.WriteOptionDescriptions (Console.Out);
|
|
|
|
}
|
|
|
|
|
2019-04-12 14:10:50 +00:00
|
|
|
static void GenerateFile (string className, List<Tuple<string, string, string>> txtStrings, CmdOptions options)
|
2017-08-21 15:34:15 +00:00
|
|
|
{
|
|
|
|
// var outputFile = options.OutputFile ?? "SR.cs";
|
|
|
|
|
|
|
|
using (var str = options.OutputFile == null ? Console.Out : new StreamWriter (options.OutputFile)) {
|
|
|
|
str.WriteLine ("//");
|
|
|
|
str.WriteLine ("// This file was generated by resx2sr tool");
|
|
|
|
str.WriteLine ("//");
|
|
|
|
str.WriteLine ();
|
|
|
|
|
2019-04-12 14:10:50 +00:00
|
|
|
int nsIdx = className.LastIndexOf ('.');
|
|
|
|
if (nsIdx > 0) {
|
|
|
|
str.WriteLine ($"namespace {className.Substring (0, nsIdx)}");
|
|
|
|
str.WriteLine ("{");
|
|
|
|
className = className.Substring (nsIdx+1);
|
|
|
|
}
|
|
|
|
|
|
|
|
str.WriteLine ($"partial class {className}");
|
2017-08-21 15:34:15 +00:00
|
|
|
str.WriteLine ("{");
|
|
|
|
|
|
|
|
var dict = new Dictionary<string, string> ();
|
|
|
|
|
|
|
|
foreach (var entry in txtStrings) {
|
|
|
|
|
|
|
|
var value = ToCSharpString (entry.Item2);
|
|
|
|
string found;
|
|
|
|
if (dict.TryGetValue (entry.Item1, out found)) {
|
2019-04-12 14:10:50 +00:00
|
|
|
if (found == value || !options.WarnConstantMismatch)
|
2017-08-21 15:34:15 +00:00
|
|
|
continue;
|
|
|
|
|
2019-04-12 14:10:50 +00:00
|
|
|
// The same entry was found with different values in multiple input files.
|
|
|
|
Console.Error.WriteLine ($"Constant value mismatch for {entry.Item1}:\n\tOld: {found}\n\tNew: {value}");
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Always add to list of seen keys.
|
|
|
|
dict.Add (entry.Item1, value);
|
|
|
|
|
|
|
|
// The following conditional could be simplified to one line, but I belive
|
|
|
|
// it is easier to understand what it does by writing it verbosely like this.
|
|
|
|
if (options.ExistingOnly) {
|
|
|
|
// We read in all entries, then update the strings of the existing ones,
|
|
|
|
// so skip the non-existing ones.
|
|
|
|
if (!ExistingKeys.ContainsKey (entry.Item1))
|
|
|
|
continue;
|
2017-08-21 15:34:15 +00:00
|
|
|
} else {
|
2019-04-12 14:10:50 +00:00
|
|
|
// In this mode of operation, we pass some existing files via `--in=`,
|
|
|
|
// so we skip the existing ones.
|
|
|
|
if (ExistingKeys.ContainsKey (entry.Item1))
|
|
|
|
continue;
|
2017-08-21 15:34:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
str.Write ($"\tpublic const string {entry.Item1} = \"{value}\";");
|
|
|
|
|
|
|
|
if (!string.IsNullOrEmpty (entry.Item3))
|
|
|
|
str.Write (" // {entry.Item3}");
|
|
|
|
|
|
|
|
str.WriteLine ();
|
|
|
|
}
|
2019-04-12 14:10:50 +00:00
|
|
|
|
|
|
|
if (options.ExistingOnly) {
|
|
|
|
// Add any keys that we missed from the input files.
|
|
|
|
foreach (var v in ExistingKeys.Keys) {
|
|
|
|
if (!dict.ContainsKey (v)) {
|
|
|
|
str.WriteLine ($"\tinternal const string {v} = \"{v}\";");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-08-21 15:34:15 +00:00
|
|
|
str.WriteLine ("}");
|
2019-04-12 14:10:50 +00:00
|
|
|
|
|
|
|
if (nsIdx > 0) {
|
|
|
|
str.WriteLine ("}");
|
|
|
|
}
|
2017-08-21 15:34:15 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static string ToCSharpString (string str)
|
|
|
|
{
|
|
|
|
str = str.Replace ("\n", "\\n");
|
|
|
|
|
|
|
|
return str.Replace ("\\", "\\\\").Replace ("\"", "\\\"");
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool LoadStrings (List<Tuple<string, string, string>> resourcesStrings, List<string> files)
|
|
|
|
{
|
|
|
|
var keys = new Dictionary<string, string> ();
|
|
|
|
foreach (var fileName in files) {
|
|
|
|
if (!File.Exists (fileName)) {
|
|
|
|
Console.Error.WriteLine ($"Error reading resource file '{fileName}'");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2019-04-12 14:10:50 +00:00
|
|
|
if (string.Equals (Path.GetExtension (fileName), ".txt", StringComparison.OrdinalIgnoreCase)) {
|
|
|
|
resourcesStrings.AddRange (ReadTextResources (fileName));
|
|
|
|
} else {
|
|
|
|
resourcesStrings.AddRange (ReadResxFile (fileName));
|
2017-08-21 15:34:15 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
2018-08-07 15:19:03 +00:00
|
|
|
|
2019-04-12 14:10:50 +00:00
|
|
|
static IEnumerable<Tuple<string,string,string>> ReadResxFile (string fileName)
|
|
|
|
{
|
|
|
|
var rr = new ResXResourceReader (fileName);
|
|
|
|
rr.UseResXDataNodes = true;
|
|
|
|
var dict = rr.GetEnumerator ();
|
|
|
|
while (dict.MoveNext ()) {
|
|
|
|
var node = (ResXDataNode)dict.Value;
|
|
|
|
yield return Tuple.Create (node.Name, (string) node.GetValue ((ITypeResolutionService)null), node.Comment);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static IEnumerable<Tuple<string,string,string>> ReadTextResources (string fileName)
|
|
|
|
{
|
|
|
|
foreach (var line in File.ReadAllLines (fileName)) {
|
|
|
|
if (line.Length == 0 || line[0] == ';') {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
var idx = line.IndexOf ('=');
|
|
|
|
if (idx < 1) {
|
|
|
|
Console.Error.WriteLine ($"Error reading resource file '{fileName}'");
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
yield return Tuple.Create (line.Substring (0, idx), line.Substring (idx+1), (string)null);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-08-07 15:19:03 +00:00
|
|
|
static bool LoadInputFiles ()
|
|
|
|
{
|
|
|
|
var reg = new Regex (@"\s*public const string (\w+)\s+=\s+");
|
|
|
|
var keys = new Dictionary<string, string> ();
|
|
|
|
foreach (var fileName in InputFiles) {
|
|
|
|
if (!File.Exists (fileName)) {
|
|
|
|
Console.Error.WriteLine ($"Error reading input file '{fileName}'");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
using (var reader = new StreamReader (fileName)) {
|
|
|
|
string line;
|
|
|
|
while ((line = reader.ReadLine ()) != null) {
|
|
|
|
var match = reg.Match (line);
|
|
|
|
if (!match.Success)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
var key = match.Groups[1].Value;
|
2019-04-12 14:10:50 +00:00
|
|
|
ExistingKeys[key] = null;
|
2018-08-07 15:19:03 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
2019-04-12 14:10:50 +00:00
|
|
|
}
|