در قسمت قبل، چون متد Validate سرویس تصدیق هویت استفاده شده، همواره مقدار false را بر میگرداند:
_identityVerifier.Initialize();
var isValidIdentity = _identityVerifier.Validate(
application.Applicant.Name, application.Applicant.Age, application.Applicant.Address);
شیء Mock آنرا طوری تنظیم کردیم که بر اساس یک applicant مشخص، خروجی true را بازگشت دهد. اما در این بین، کدهای بررسی سرویس creditScorer را کامنت کردیم:
_creditScorer.CalculateScore(application.Applicant.Name, application.Applicant.Address);
if (_creditScorer.Score < MinimumCreditScore)
{
return application.IsAccepted;
}
تا آزمایش واحد ما با موفقیت به پایان برسد. در این قسمت، کار تنظیم مقادیر خواص آنرا در آزمون واحد، به کمک Mocked objects انجام میدهیم تا این قسمت از کد نیز پوشش داده شود. برای این منظور به کلاس LoanApplicationProcessor مراجعه کرده و در متد Process آن، ابتدا مجددا از همان overload سادهی فوق متد Validate بجای نمونهی ref دار استفاده کرده و سپس کدهای creditScorer را نیز از حالت کامنت خارج میکنیم.
تنظیم مقدار خاصیت Score شیء Mock شده
اینترفیس ICreditScorer به صورت زیر تعریف شدهاست و دارای خاصیت Score میباشد که مقدار عددی آن با مقدار حداقل اعتبار تنظیم شدهی در کلاس LoanApplicationProcessor مقایسه خواهد شد (MinimumCreditScore = 100_000):
namespace Loans.Services.Contracts
{
public interface ICreditScorer
{
int Score { get; }
void CalculateScore(string applicantName, string applicantAddress);
}
}
برای تنظیم مقدار خاصیت Score، در متد Accept آزمونهای واحد تهیه شده، میتوان به صورت زیر عمل کرد:
var mockCreditScorer = new Mock<ICreditScorer>();
mockCreditScorer.Setup(x => x.Score).Returns(110_000);
که بسیار شبیه به نحوهی تنظیم مقادیر بازگشتی متدها است. در متد Setup میتوان به صورت strongly typed به تمام خواص اینترفیس ICreditScorer دسترسی یافت و سپس توسط متد Returns، مقدار بازگشتی آنها را تنظیم نمود.
اکنون اگر متد آزمایش واحد Accept را بررسی کنیم، چون شخص درخواست دهنده، دارای اعتبار بیشتری از حداقل اعتبار مورد نیاز است، این آزمایش با موفقیت به پایان خواهد رسید. اگر این تنظیم صورت نمیگرفت، شیء mockCreditScorer، مقدار پیشفرض int یا همان صفر را به عنوان مقدار Score بازگشت میداد.
تنظیم مقادیر خواص تو در تو و سلسله مراتبی اشیاء Mock شده
برای کار با خواص تو در تو، ابتدا دو مدل زیر را ایجاد میکنیم:
namespace Loans.Models
{
public class ScoreResult
{
public ScoreValue ScoreValue { get; }
}
public class ScoreValue
{
public int Score { get; }
}
}
اکنون بجای مقدار سادهی int Score { get; }، از نمونهی ScoreResult فوق، در اینترفیس ICreditScorer استفاده خواهیم کرد:
using Loans.Models;
namespace Loans.Services.Contracts
{
public interface ICreditScorer
{
int Score { get; }
void CalculateScore(string applicantName, string applicantAddress);
ScoreResult ScoreResult { get; }
}
}
در ادامه برای استفادهی از ScoreResult، به کلاس LoanApplicationProcessor مراجعه کرده و در انتهای متد Process آن، این تغییر را ایجاد میکنیم:
//if (_creditScorer.Score < MinimumCreditScore)
if (_creditScorer.ScoreResult.ScoreValue.Score < MinimumCreditScore)
اینبار اگر متد آزمون واحد Accept را اجرا کنیم، با یک null reference exception به پایان میرسد؛ چون اولین سطح این شیء تو در تو، یعنی ScoreResult، مساوی نال است.
برای رفع این مشکل در متد آزمون واحد Accept، باید به صورت زیر عمل کرد:
var mockCreditScorer = new Mock<ICreditScorer>();
mockCreditScorer.Setup(x => x.Score).Returns(110_000);
var mockScoreValue = new Mock<ScoreValue>();
mockScoreValue.Setup(x => x.Score).Returns(110_000);
var mockScoreResult = new Mock<ScoreResult>();
mockScoreResult.Setup(x => x.ScoreValue).Returns(mockScoreValue.Object);
mockCreditScorer.Setup(x => x.ScoreResult).Returns(mockScoreResult.Object);
ابتدا از پایینترین سطح یعنی ScoreValue شروع و مقدار خاصیت Score آنرا تنظیم میکنیم.
سپس یک سطح بالاتر را یعنی ScoreResult را تنظیم خواهیم کرد. در اینجا نیاز است خاصیت ScoreValue آن به mock object قبلی تنظیم شود. به همین جهت Returns آن به خاصیت Object شیء mockScoreValue، تنظیم شدهاست.
در آخر برای تنظیم خاصیت ScoreResult شیء mockCreditScorer اصلی، از شیء mockScoreResult استفاده خواهیم کرد.
در این حالت اگر متد آزمون واحد Accept را اجرا کنیم، اینبار به خطای زیر برخواهیم خورد:
Test method Loans.Tests.LoanApplicationProcessorShould.Accept threw exception:
System.NotSupportedException: Unsupported expression: x => x.Score
Non-overridable members (here: ScoreValue.get_Score) may not be used in setup / verification expressions.
عنوان میکند که خاصیت Score شیء ScoreValue، قابل بازنویسی نیست (Non-overridable). منظورش این است که برای mocking آن خاصیت، باید آنرا virtual تعریف کنیم تا کتابخانهی Moq بتواند آنرا بازنویسی کند. به همین جهت، هر دو خاصیتی را که در اینجا قصد بازنویسی آنها را داریم، به صورت virtual تعریف میکنیم:
namespace Loans.Models
{
public class ScoreResult
{
public virtual ScoreValue ScoreValue { get; }
}
public class ScoreValue
{
public virtual int Score { get; }
}
}
اکنون اگر متد آزمایش واحد Accept را بررسی کنیم با موفقیت به پایان خواهد رسید.
ساده سازی روش تنظیم مقادیر خواص تو در تو و سلسله مراتبی اشیاء Mock شده
روش دیگری نیز برای تنظیم مقادیر خواص تو در تو در کتابخانهی Moq وجود دارد:
mockCreditScorer.Setup(x => x.ScoreResult.ScoreValue.Score).Returns(110_000);
کتابخانهی Moq قادر است به نحوی که مشاهده میکنید، سلسله مراتب اشیاء را به صورت strongly typed ایجاد کرده و در نهایت خاصیت Score آنرا به 110_000 تنظیم کند.
بدیهی است در این حالت نیز باید شرط virtual بودن این خواص، برقرار باشد؛ در غیراینصورت همان استثنای NotSupportedException را دریافت خواهیم کرد.
یک نکته: اگر در زمان تشکیل یک Mock object، مقدار خاصیت DefaultValue آنرا به صورت زیر تنظیم کنیم:
var mockCreditScorer = new Mock<ICreditScorer> { DefaultValue = DefaultValue.Mock };
تمام خواص تو در توی موجود در ICreditScorer، به صورت خودکار با نمونههای پیشفرض آنها مقدار دهی و آمادهی استفاده خواهند شد. اگر بجای مقدار DefaultValue.Mock از DefaultValue.Empty استفاده شود، این مقادیر پیشفرض، نال خواهد بود (که همان حالت پیشفرض new Mock است).
بررسی تغییرات مقادیر خواص اشیاء Mock شده
کتابخانهی Moq، امکان ردیابی تغییرات مقادیر خواص اشیاء Mock شده را نیز داراست. برای نمایش آن، فرض کنید خاصیت جدید Count را به اینترفیس ICreditScorer اضافه کردهایم:
using Loans.Models;
namespace Loans.Services.Contracts
{
public interface ICreditScorer
{
int Score { get; }
void CalculateScore(string applicantName, string applicantAddress);
ScoreResult ScoreResult { get; }
int Count { get; set; }
}
}
سپس در کلاس LoanApplicationProcessor و متد Process آن، هربار که CalculateScore فراخوانی میشود، یکبار مقدار Count را افزایش میدهیم:
_creditScorer.CalculateScore(application.Applicant.Name, application.Applicant.Address);
_creditScorer.Count++;
اکنون در متد آزمون واحد Accept، بررسی میکنیم که آیا پس از یکبار فراخوانی متد CalculateScore، مقدار Count برای مثال 1 شدهاست یا خیر؟
Assert.AreEqual(1, mockCreditScorer.Object.Count);
تا اینجا اگر آزمون واحد را اجرا کنیم، با شکست مواجه خواهد شد. چون کتابخانهی Moq تغییرات مقادیر خواص شیء mockCreditScorer.Object را ردیابی نمیکند و مقدار mockCreditScorer.Object.Count، همان مقدار پیشفرض نوع int، یعنی صفر میباشد.
برای فعال سازی ردیابی تغییرات مقادیر خاصیت Count، تنها کافی است آنرا توسط متد SetupProperty، معرفی کنیم:
mockCreditScorer.SetupProperty(x => x.Count);
پس از این تغییر، بررسی متد آزمون واحد Accept با موفقیت به پایان میرسد.
در اینجا میتوان یک مقدار اولیه را هم درنظر گرفت:
mockCreditScorer.SetupProperty(x => x.Count, 10);
بدیهی است در این صورت Assert.AreEqual ما با شکست مواجه میشود؛ چون اینبار مقدار Count نهایی، بر اساس این مقدار اولیه، 11 خواهد بود.
فعالسازی بررسی تغییرات تمام مقادیر خواص اشیاء Mock شده
اگر تعداد خواصی که قرار است مورد ردیابی قرارگیرند زیاد است، بجای فراخوانی متد SetupProperty بر روی تک تک آنها، میتوان تمام آنها را به صورت زیر تحت کنترل قرار داد:
mockCreditScorer.SetupAllProperties();
نکتهی مهم: محل قرارگیری SetupAllProperties مهم است. برای مثال اگر این سطر را پس از سطر تنظیم مقدار پیشفرض x.ScoreResult.ScoreValue.Score قرار دهید، آزمایش با شکست مواجه میشود؛ چون تنظیمات بازگشت مقادیر پیشفرض خواص را به طور کامل بازنویسی میکند. بنابراین این سطر باید پیش از سطر تنظیم مقادیر پیشفرض خواص Mock شده، فراخوانی شود تا بر روی این مقادیر تنظیمی، تاثیری نداشته باشد.
کدهای کامل این قسمت را از اینجا میتوانید دریافت کنید: MoqSeries-03.zip