Configuration.Manger в модульном тестировании
0 Gilad [2016-10-23 00:32:00]
Я уже прочитал несколько замечательных вопросов и отвечу на эту проблему, например, Can not Use ConfigurationManager в проекте Unit Test
и как издеваться над ConfigurationManager.AppSettings с moq
На данный момент решение, которое я использую в своих модульных тестах, как и файл App.config для проекта модульных тестов. и он работает нормально.
моя цель - динамически загружать плагин (*.dll) в мой проект С#.
вот мой оригинальный App.config
<?xml version="1.0"?>
<configuration>
<configSections>
<sectionGroup name="applicationSettings" type="System.Configuration.ApplicationSettingsGroup, System, Version=4.0.0.0, Culture=neutral, PublicFormattedKeyToken=b77a5c561934e089">
<section name="IQCMain.Properties.Settings" type="System.Configuration.ClientSettingsSection, System, Version=4.0.0.0, Culture=neutral, PublicFormattedKeyToken=b77a5c561934e089" requirePermission="false" />
</sectionGroup>
<section name="CalibrationToolsSection" type="sec.Calibration.Configuration.CalibrationToolsSection,sec" />
</configSections>
<CalibrationToolsSection>
<!-- This section contains the calibration tools' loading-enabling options, and versioning parameters as follows:
toolname - the calibration tool name as returned by GetName() in ICalibrationPlugin (for static tool) or as appears in the records (for auto-gen tool)
isvisible - if set to "false" the tool will not be loaded. Deafult value: true
version - if set to "false" the tool will not be loaded. Deafult value: true /-->
<!--IpuToolCalibration-->
<IpuToolCalibrations>
<add ipu="p1">
<CalibrationToolsLoadingSettings>
<add toolname="AE" version="1" isvisible="false"/>
<add toolname="AE2" version="1" isvisible="true"/>
</CalibrationToolsLoadingSettings>
</add>
<add ipu="p2">
<CalibrationToolsLoadingSettings>
<add toolname="AE" version="1" isvisible="false"/>
<add toolname="AE2" version="1" isvisible="true"/>
</CalibrationToolsLoadingSettings>
</add>
<add ipu="p3">
<CalibrationToolsLoadingSettings>
<add toolname="AE" version="1" isvisible="false"/>
<add toolname="AE2" version="1" isvisible="true"/>
</CalibrationToolsLoadingSettings>
</add>
<add ipu="p4">
<CalibrationToolsLoadingSettings>
<add toolname="AE" version="1" isvisible="false"/>
<add toolname="AE2" version="1" isvisible="true"/>
</CalibrationToolsLoadingSettings>
<PalAteConnectivitySettings>
<add pal-uuid="1111" connect-ate="true" name="p4_wb" />
<add pal-uuid="2222" connect-ate="true" name="p4_lsc" />
<add pal-uuid="3333" connect-ate="true" name="p4_blc" />
<add pal-uuid="4343" connect-ate="true" name="p4_disparity" />
<add pal-uuid="9999" connect-ate="true" name="p4_gridbaseob" />
</PalAteConnectivitySettings>
</add>
<add ipu="p5">
<CalibrationToolsLoadingSettings>
<add toolname="ACMCmc" version="1" isvisible="true"/>
<add toolname="ACM3A" version="1" isvisible="true"/>
<add toolname="AE" version="1" isvisible="false"/>
<add toolname="AE2" version="1" isvisible="true"/>
<add toolname="llolo" version="1" isvisible="true" />
<add toolname="lalla" version="1" isvisible="true" />
</CalibrationToolsLoadingSettings>
<PalAteConnectivitySettings>
<add pal-uuid="32398" connect-ate="true" name="p5_wb" />
<add pal-uuid="53711" connect-ate="true" name="p5_lsc" />
<add pal-uuid="40661" connect-ate="true" name="p5_blc" />
<add pal-uuid="55093" connect-ate="true" name="p5_disparity" />
<add pal-uuid="46517" connect-ate="true" name="p5_gridbaseob" />
</PalAteConnectivitySettings>
</IpuToolCalibrations>
</CalibrationToolsSection>
<appSettings>
<add key="DisableCalibrationCheck" value="false" />
<add key="ClientSettingsProvider.ServiceUri" value="" />
</appSettings>
Я видел, что вы можете динамически создавать коллекции, например
ConfigurationManager.AppSettings["mykey"] = "myvalue";
прямо сейчас я использую такие классы, используя System.Configuration;
namespace Manager.Calibration.Configuration
{
public class CalibrationToolsSection : ConfigurationSection
{
public static readonly string CALIBRATION_SECTION_NAME = "CalibrationToolsSection";
[ConfigurationProperty("IpuToolCalibrations")]
public IpuToolCalibrationCollection IpuToolCalibrations
{
get
{
return ((IpuToolCalibrationCollection)this["IpuToolCalibrations"]);
}
set
{
this["IpuToolCalibrations"] = value;
}
}
}
}
а также
public class IpuToolCalibrationCollection : ConfigurationElementCollection
{
public IpuToolCalibration this[int index]
{
get { return base.BaseGet(index) as IpuToolCalibration; }
set
{
if (base.BaseGet(index) != null)
{
base.BaseRemoveAt(index);
}
this.BaseAdd(index, value);
}
}
protected override ConfigurationElement CreateNewElement()
{
return new IpuToolCalibration();
}
protected override object GetElementKey(ConfigurationElement element)
{
return ((IpuToolCalibration)element).Ipu;
}
}
и так далее
public class IpuToolCalibration : ConfigurationElement
{
public static readonly string IPU_TOOL_CALIBRATION_SECTION_NAME = "IpuToolCalibration";
[ConfigurationProperty("ipu", IsRequired = true)]
public string Ipu
{
get { return (string)this["ipu"]; }
set { this["ipu"] = value; }
}
[ConfigurationProperty("CalibrationToolsLoadingSettings")]
public ToolSettingElementCollection ToolsLoadingSettingsCollection
{
get
{
return ((ToolSettingElementCollection)this["CalibrationToolsLoadingSettings"]);
}
set
{
this["CalibrationToolsLoadingSettings"] = value;
}
}
[ConfigurationProperty("PalAteConnectivitySettings")]
public FilterAteConnectivityElementCollection PalAteConnectivityCollection
{
get
{
return ((FilterAteConnectivityElementCollection)this["PalAteConnectivitySettings"]);
}
set
{
this["PalAteConnectivitySettings"] = value;
}
}
}
}
Может ли кто-нибудь объяснить, как я могу создать их из кода, так как все они являются статическими классами без конструкторов, как я могу их высмеять, я использую NSubstitute. если, например, я хочу загрузить часть плагинов, а не все из них?
c# unit-testing
1 ответ
-1 TheRock [2016-10-24 18:07:00]
Я думаю, что в любое время, когда вы сталкиваетесь с попыткой модульного тестирования любой части кода, которая имеет статическую реализацию, вам нужно создать какую-либо оболочку, которая затем может быть впрыснута там, где это необходимо - в некотором смысле это вариант "узор декоратора". Образец кода, который я предоставляю, - это просто проиллюстрировать суть:
public class FileReaderService : IFileReaderService
{
public string GetFileAsString(string fileName)
{
if(!File.Exists(fileName))
throw new ArgumentException("File Path does not exist.");
return File.ReadAllText(fileName);
}
}
public interface IFileReaderService
{
string GetFileAsString(string fileName);
}
Я обертываю статическую функцию File.ReadAllText с помощью службы FileReaderService и имею интерфейс IFileReaderService. Теперь я могу вставлять службу в любом месте, где она мне нужна, а затем иметь возможность макетировать и писать модульные тесты соответственно. Надеюсь, это даст вам представление о том, как вы можете реорганизовать свой код практически так же.