Není to sice ideální, ale někdy tomu situace dá, že potřebuji testovat metodu, která by jinak byla private. V takovém případě ji dělám internal a do testovacího projektu ji zpřístupňuji pomocí global atributu InternalsVisibleTo z AssemblyInfo.cs:
[assembly: InternalsVisibleTo("Havit.Xy.ServicesTests")]
To všechno funguje celkem v pohodě až do okamžiku, než se pustíte do partial-mockingu. Máme například následující třídu, kterou chceme testovat
public class TestedClass { public virtual void Process() { foreach (var item in ...) { ProcessItem(item); } } internal virtual void ProcessItem(string item) { ... } }
Dokud testujeme samotnou třídu, je vše v pohodě:
var sut = new TestedClass(); var output = sut.ProcessItem(input);
Zajímavé to začne být, pokud začneme tvořit partial-mock, tedy chceme část funkčnosti třídy nasetupovat jinak, abychom testovali jen určitou část. Třeba chceme testovat Process() bez ProcessItem():
// arrange var sut = new Mock(); sut.Setup(sut => sut.ProcessItem(It.IsAny()).Returns(String.Empty); sut.CallBase = true; // act sut.Process(); // assert ...
Najednou zjistíte, že Moq nový setup metody ProcessItem ignoruje a volá stále původní implementaci. Žádný problém navíc nehlásí a já se jen nestačím divit, co se kde hnojí.
Pak vám to najednou docvakne. Vše začne fungovat, pokud setupovanou metodu ProcessItem uděláte protected internal:
public class TestedClass { public virtual void Process() { foreach (var item in ...) { ProcessItem(item); } } protected internal virtual void ProcessItem(string item) { ... } }
…najednou vše funguje.