mirror of
https://github.com/xcp-ng/xenadmin.git
synced 2025-01-22 00:00:40 +01:00
b16eaaae4e
Signed-off-by: Konstantina Chremmou <konstantina.chremmou@citrix.com>
153 lines
6.2 KiB
C#
153 lines
6.2 KiB
C#
/* Copyright (c) Citrix Systems, Inc.
|
|
* All rights reserved.
|
|
*
|
|
* Redistribution and use in source and binary forms,
|
|
* with or without modification, are permitted provided
|
|
* that the following conditions are met:
|
|
*
|
|
* * Redistributions of source code must retain the above
|
|
* copyright notice, this list of conditions and the
|
|
* following disclaimer.
|
|
* * Redistributions in binary form must reproduce the above
|
|
* copyright notice, this list of conditions and the
|
|
* following disclaimer in the documentation and/or other
|
|
* materials provided with the distribution.
|
|
*
|
|
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND
|
|
* CONTRIBUTORS "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES,
|
|
* INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
|
|
* MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
|
* DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR
|
|
* CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
|
|
* SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING,
|
|
* BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
|
|
* SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
|
|
* INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY,
|
|
* WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
|
|
* NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
|
|
* OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
|
|
* SUCH DAMAGE.
|
|
*/
|
|
|
|
using System.Globalization;
|
|
using System.Linq;
|
|
using System.Collections.Generic;
|
|
using System.Configuration;
|
|
using System.Reflection;
|
|
using NUnit.Framework;
|
|
|
|
|
|
namespace XenAdminTests.CodeTests
|
|
{
|
|
[TestFixture, Category(TestCategories.Unit)]
|
|
public class AssemblyTests
|
|
{
|
|
[Test, Combinatorial]
|
|
[Description("Checks all resx files in the project have their i18n counterparts in place")]
|
|
public void TestEnsureI18NFilesInPlace(
|
|
[Values("ja", "zh-CN")] string locale,
|
|
[Values("XenCenterMain", "XenModel", "XenOvf", "XenOvfTransport")] string assemblyName)
|
|
{
|
|
var assembly = FindAssemblyByNameRecursively(assemblyName);
|
|
Assert.NotNull($"Assembly {assemblyName} was not found.");
|
|
|
|
var excludeFromCheck = new[] {"XenAdmin.Help.HelpManager", "XenAdmin.Branding"};
|
|
var missing = new List<string>();
|
|
var extra = new List<string>();
|
|
|
|
List<string> defaultResx = new List<string>(assembly.GetManifestResourceNames().Where(resource => resource.EndsWith("resources")));
|
|
|
|
CultureInfo cultureInfo = new CultureInfo(locale);
|
|
Assembly localeDll = assembly.GetSatelliteAssembly(cultureInfo);
|
|
List<string> localeResx = new List<string>(localeDll.GetManifestResourceNames());
|
|
|
|
foreach (string def in defaultResx)
|
|
{
|
|
var name = def.Substring(0, def.Length - ".resources".Length);
|
|
var exclude = excludeFromCheck.Contains(name);
|
|
|
|
string localName = $"{name}.{locale}.resources";
|
|
var localized = localeResx.Contains(localName);
|
|
|
|
if (localized && exclude)
|
|
extra.Add(localName);
|
|
else if (!localized && !exclude)
|
|
missing.Add(localName);
|
|
}
|
|
|
|
Assert.IsEmpty(missing, "Missing resources detected.");
|
|
Assert.IsEmpty(extra, "Unnecessary resources detected.");
|
|
}
|
|
|
|
[Test]
|
|
[Description("Checks that if there are user-scoped settings in an assembly, these have roaming=true")]
|
|
public void TestUserSettingsAreRoaming(
|
|
[Values("XenCenterMain", "XenOvf", "XenOvfTransport", "XenServerHealthCheck")] string assemblyName)
|
|
{
|
|
var assembly = FindAssemblyByNameRecursively(assemblyName);
|
|
Assert.NotNull($"Assembly {assemblyName} was not found.");
|
|
|
|
var nonRoaming = new List<string>();
|
|
|
|
foreach (var type in assembly.GetTypes())
|
|
{
|
|
if (type.Name == "Settings" && typeof(SettingsBase).IsAssignableFrom(type))
|
|
{
|
|
var properties = ((SettingsBase)type.GetProperty("Default").GetValue(null, null)).Properties;
|
|
|
|
foreach (SettingsProperty prop in properties)
|
|
{
|
|
bool userScope = false;
|
|
bool roaming = false;
|
|
|
|
foreach (var val in prop.Attributes.Values)
|
|
{
|
|
if (val is SettingsManageabilityAttribute sma && sma.Manageability == SettingsManageability.Roaming)
|
|
roaming = true;
|
|
else if (val is UserScopedSettingAttribute)
|
|
userScope = true;
|
|
}
|
|
|
|
if (userScope && !roaming)
|
|
nonRoaming.Add(prop.Name);
|
|
}
|
|
}
|
|
}
|
|
|
|
Assert.IsEmpty(nonRoaming, "Detected user-scoped settings that are not roaming.");
|
|
}
|
|
|
|
#region Auxiliary private methods
|
|
|
|
/// <summary>
|
|
/// Some of the assemblies we need to check may not be in the manifest of the assembly we're
|
|
/// looking at, but are referenced by assemblies in the manifest. This method retrieves them.
|
|
/// </summary>
|
|
private Assembly FindAssemblyByNameRecursively(string name, Assembly startAssembly = null)
|
|
{
|
|
if (startAssembly == null)
|
|
startAssembly = Assembly.GetExecutingAssembly();
|
|
|
|
var assemblyNames = startAssembly.GetReferencedAssemblies();
|
|
var assemblyName = assemblyNames.FirstOrDefault(an => an.Name == name);
|
|
|
|
if (assemblyName == null)
|
|
{
|
|
foreach (var temp in assemblyNames)
|
|
{
|
|
var found = FindAssemblyByNameRecursively(name, Assembly.Load(temp));
|
|
if (found != null)
|
|
return found;
|
|
}
|
|
}
|
|
else
|
|
return Assembly.Load(assemblyName);
|
|
|
|
return null;
|
|
}
|
|
|
|
#endregion
|
|
}
|
|
}
|
|
|