در این مقاله از راکت، به بررسی نحوه مدیریت دادههای دامنه برای تست آنها میپردازیم. Test factory ها در لاراول یک مفهوم شناخته شده هستند؛ اگر چه در بسیاری از زمینهها کمبود دارند: آنها خیلی انعطافپذیر نیستند و همچنین نوعی جعبه سیاه برای کاربر به حساب میآیند.
مثالی از factory states برای شما آوردهام؛ یک الگوی بسیار قدرتمند که البته در لاراول ضعیف اجرا شده است.
$factory->state(Invoice::class, 'pending', [
'status' => PaidInvoiceState::class,
]);
اول از همه: IDE شما هیچ سرنخی از اینکه آبجکت $factory واقعاً چه نوعی است ندارد. و این به طرز شگفتانگیزی در فکتوری فایلها وجود دارد، با وجود این هیچ autocompletion (تکمیل خودکاری) در آن وجود ندارد. یک راهحل سریع این است که این docblock را اضافه کنید، اگر چه که این کار دست و پاگیر و مایه زحمت است.
/** @var \Illuminate\Database\Eloquent\Factory $factory */
$factory->state(/* … */);
دوم،state ها به عنوان رشته تعریف میشوند، و این کار آنها را به یک جعبه سیاه در هنگام استفاده از factory در تستها تبدیل میکند.
public function test_case()
{
$invoice = factory(Invoice::class)
->states(/* what states are actually available here? */)
->create();
}
سوم، هیچ type hinting در نتیجهی یکfactory وجود ندارد، IDE شما نمیداند که $invoice درواقع یک Invoice model است؛ و دوباره میگویم: یک جعبه سیاه.
و سرانجام، با توجه به دامنه کافی، شما ممکن است به stateهای بیشتری در مجموعه تستهای خود نیاز داشته باشید که در این صورت مدیریت آنها باگذشت زمان دشوار میشود.
در این مقاله ما به یک روش دیگر از پیادهسازی این الگوی Factory خواهیم پرداخت تا انعطافپذیری بیشتری داشته باشد و تجربهکاربری آنها را به میزان قابل توجهی بهبود ببخشد. هدف اصلی این factory classها،کمک به شما در نوشتن integration tests است، بدون اینکه وقت زیادی را صرف راهاندازی سیستمی برای آن کنید.
توجه داشته باشید که من میگویم "integration tests" نه "unit tests": وقتی که ما دامنه کد خود را تست میکنیم، منطق اصلی تجارتمان را مورد آزمایش قرار میدهیم. بیشتر اوقات، این آزمایش به این معنی نیست که ما یک قطعه ایزوله شده از یک کلاس را تست خواهیم کرد، بلکه یک قانون تجاری پیچیده است که نیاز به وجود کمی ( یا مقدار زیادی) داده، در پایگاه داده دارد.
همانطور که قبلاً اشاره کردم: ما درمورد سیستمهای بزرگ و پیچیده در این مقاله صحبت میکنیم؛ این مسأله مهمیاست پس به خاطر داشته باشید. به همین خاطر تصمیم گرفتم در این مقاله این تستها را integration tests بنامم، و این به منظور جلوگیری از سؤالات یا بحثهایی در رابطه با اینکه یونیت تستها چه هستند یا نیستند بود.
basic factory
یک تست فکتوری چیزی بیش از یک کلاس ساده نیست. هیچ پکیجی مورد نیاز نیست ، هیچ اینترفیسی برای پیادهسازی وجود ندارد و یا حتی abstractclasseها برای گسترش نیستند. قدرت یک factory به پیچیدگی کد نیست، بلکه باید با یک یا دو الگو به درستی اعمال شود.
در اینجا به نظر میرسد که چنین کلاسی ساده است:
class InvoiceFactory
{
public static function new(): self
{
return new self();
}
public function create(array $extra = []): Invoice
{
return Invoice::create(array_merge(
[
'number' => 'I-1',
'status' => PendingInvoiceState::class,
// …
],
$extra
));
}
}
بیایید درباره چند تصمیم در طراحی بحث کنیم.
اول از همه، دربارهstatic constructor new.ممکن است با دلایل نیاز ما به آن، شما گیج شوید، چنانکه ما میتوانیم یک متد استاتیک را به سادگی بسازیم. من این سؤال را بعداً به صورت کامل و عمیق توضیح میدهیم، اما الان باید بدانید که میخواهیم این فکتوری قبل از ایجاد یک invoice قابلیت تنظیم بالایی داشته باشد. بنابراین درباره اینها نگران نباشید چون به زودی برای شما واضحتر خواهد شد.
ثانیا، چرا new برای static constructor به کار رفته؟ پاسخ یک جواب کاربردی است:داخل context فکتوریها، make و create اغلب با یک فکتوری مرتبط هستند که درواقع نتیجهای را تولید میکنند. new به ما کمک میکند تا از بعضی سردرگمیهای غیر ضروری جلوگیری کنیم.
سرانجام،متد create :به یک آرایه اختیاری از دادههای اضافی نیاز دارد تا اطمینان حاصل شود که همیشه میتوانیم در تستهای خود، در لحظات آخر تغییراتی ایجاد کنیم.
اکنون با یک مثال ساده میتوانیم invoice (فاکتور) هایی مثل این را ایجاد کنیم:
public function test_case()
{
$invoice = InvoiceFactory::new()->create();
}
قبل از بررسی قابلیت تنظیم، بیایید کمی به کار خود بهبود ببخشیم :شمارههایinvoice باید منحصر به فرد باشد ،بنابراین اگر دو invoice در یک تست ایجاد کنیم، خراب میشود. اگر چه در بیشتر موارد نمیخواهیم نگران پیگیری شمارههای invoice باشیم، بنابراین اجازه دهید فکتوری این موارد را انجام دهد:
class InvoiceFactory
{
private static int $number = 0;
public function create(array $extra = []): Invoice
{
self::$number += 1;
return Invoice::create(array_merge(
[
'number' => 'I-' . self::$number,
// …
],
$extra
));
}
}
فکتوریها در فکتوریها
در مثال اصلی، من به شما نشان دادم که ممکن است بخواهیم یک paid invoice (فاکتور پرداختی) ایجاد کنیم. قبلا تصور میکردم این به معنای تغییر وضعیت فیلد در invoice model است؛ ولی خب کمی ساده لوح بودم که اینگونه فکر میکردم. برای ذخیره در پایگاه داده به یک پرداخت واقعی نیز نیاز داریم! فکتوریهای پیشفرض لاراول میتوانند این callback ها را که پس از ایجاد یک مدل شروع میشوند، هندل کنند. هر چند تصور کنید اگر چندین یا حتی شاید دهها state برای مدیریت داشته باشید، هر کدام عوارض جانبی خاص خودشان را دارند. یک $factory→afterCreating ساده کافی نیست که بتواند همهی اینها را به روشی سالم مدیریت کند.
بنابراین، بیایید همه چیز را برگردانیم؛ و قبل از ایجاد یک invoice واقعی، invoice factory خود را به درستی پیکربندی کنیم.
class InvoiceFactory
{
private string $status = null;
public function create(array $extra = []): Invoice
{
$invoice = Invoice::create(array_merge(
[
'status' => $this->status ?? PendingInvoiceState::class
],
$extra
));
if ($invoice->status->isPaid()) {
PaymentFactory::new()->forInvoice($invoice)->create();
}
return $invoice;
}
public function paid(): self
{
$clone = clone $this;
$clone->status = PaidInvoiceState::class;
return $clone;
}
}
درضمن اگر درباره clone تعجب کردید، بعداً آن را بررسی خواهیم کرد.
چیزی که ما قابل تنظیم کردهایم وضعیت invoice است، دقیقاً مثل factory stateها در لاراول؛ اما در موردی که ما داریم این مزیت وجود دارد که IDE ما درواقع میداند با چه کاری روبهرو هستیم :
public function test_case()
{
$invoice = InvoiceFactory::new()
->paid()
->create();
}
هنوز هم جای پیشرفت وجود دارد، آیا دقت کردید که بعد از invoice، ما create را چک کردیم؟
if ($invoice->status->isPaid()) {
PaymentFactory::new()->forInvoice($invoice)->create();
}
هنوز هم میتوان انعطافپذیری بیشتری انجام داد. ما از PaymentFactory در زیر استفاده میکنیم، اما اگر بخواهیم کنترل دقیقتری درباره نحوه انجام این payment (پرداخت) داشته باشیم، چه میکنید؟ میتوانید تصور کنید قوانینی در تجارت درباره paid invoices (فاکتورهای پرداخت) وجود دارد که بسته به نوع پرداخت، رفتار متفاوتی دارند.
همچنین ما میخواهیم از پیکربندی مستقیم در InvoiceFactory جلوگیری کنیم، چرا که خیلی زود به یک خرابکاری تبدیل میشود. بنابراین چگونه این کار را حل کنیم؟!
جواب اینجاست :ما به توسعهدهنده اجازه میدهیم که به صورت اختیاری یک PaymentFactory را به InvoiceFactory منتقل (pass )کند تا این فکتوری هر طور که توسعهدهنده میخواهد پیکربندی شود. در اینجا به اینگونه به نظر میرسد:
public function paid(PaymentFactory $paymentFactory = null): self
{
$clone = clone $this;
$clone->status = PaidInvoiceState::class;
$clone->paymentFactory = $paymentFactory ?? PaymentFactory::new();
return $clone;
}
و نحوه استفاده آن در متد create آمده است:
if ($this->paymentFactory) {
$this->paymentFactory->forInvoice($invoice)->create();
}
با این کار، امکانات بسیاری به وجود میآیند. در این مثال ما در حال ساخت یک invoice (فاکتور) برای پرداخت هستیم، بهخصوص با یکBancontact payment .
public function test_case()
{
$invoice = InvoiceFactory::new()
->paid(
PaymentFactory::new()->type(BancontactPaymentType::class)
)
->create();
}
مثال دیگر: میخواهیم نحوه پرداخت فاکتور را هنگام پرداخت آن آزمایش کنیم، اما فقط پس از انقضای فاکتور :
public function test_case()
{
$invoice = InvoiceFactory::new()
->expiresAt('2020-01-01')
->paid(
PaymentFactory::new()->at('2020-01-20')
)
->create();
}
فقط با چند خط کد، انعطاف بیشتری پیدا میکنیم.
Immutable factories
خب درباره cloning که جلوتر حرفش را زدیم چه فکر میکنید؟ چرا مهم است که فکتوریهای immutable یا غیرقابل تغییر بسازیم؟ ببینید، گاهی شما نیاز به ساخت چندین مدل یا یک فکتوری یکسان دارید، اما با تفاوتهای کوچک. به جای ساخت یک factory object جدید برای هر model ، میتوانید از factory object اصلی مجدد استفاده کنید و فقط موارد مورد نیاز خود را تغییر دهید.
اگر چه از immutable factorie ها استفاده نمیکنید، اما این احتمال وجود دارد که در نهایت به دادههایی برسید که درواقع آنها را نمیخواستید. مثالی درباره invoice payments یا فاکتورهای پرداخت میزنیم: فرض کنید ما در یک تاریخ به دو فاکتور نیاز داریم، یکی پرداخت شده و دیگری معلق مانده.
$invoiceFactory = InvoiceFactory::new()
->expiresAt(Carbon::make('2020-01-01'));
$invoiceA = $invoiceFactory->paid()->create();
$invoiceB = $invoiceFactory->create();
اگر متد paid ما immutable یا غیرقابل تغییر بود، به این معنی میبود که $invoiceB نیز یک فاکتور پرداخت شده خواهد بود! مطمئناً ما میتوانیم روی ایجاد هر مدلی یک مدیریت همه جانبه داشته باشیم، اما این، انعطافپذیری این الگو را میگیرد. به این دلیل است که فانکشنهای تغییر ناپذیر عالی هستند: شما میتوانید بدون نگرانی درباره عوارض جانبی یک base factory را راهاندازی کنید و از آن در طول تستهای خود استفاده مجدد کنید!
با استفاده از این دو اصل ( پیکربندی فکتوریها داخل فکتوریها و تغییر ناپذیر ساختن آنها)، امکانات بسیار زیادی به وجود میآیند. مطمئنا نوشتن این فکتوریها مدتی طول میکشد، اما در طول توسعه نیز در وقتمان به طور قابل توجهی صرفهجویی میشود. به نظر من آنها کاملاً ارزش دارند چرا که بیشتر از هزینهای که برای آنها میکنید، دریافت خواهید کرد.
از زمان استفاده از این الگو من اصلاً و هرگز به فکتورهای داخلی لاراول نگاه نکرده ام.
یک نکته منفی که میتوانم بگوییم این است که برای ایجاد همزمان چندین مدل، به کمی کد اضافه تر نیاز دارید. اگر بخواهید، به راحتی میتوانید قعطه کوچکی از کد را در یک کلاس base factory مثل این اضافه کنید:
abstract class Factory
{
// Concrete factory classes should add a return type
abstract public function create(array $extra = []);
public function times(int $times, array $extra = []): Collection
{
return collect()
->times($times)
->map(fn() => $this->create($extra));
}
}
همچنین به خاطر داشته باشید که میتوانید از این فکتوریها برای موارد دیگر نیز استفاده کنید،نه صرفاً برای مدلها. من همچنین از آنها به طور گسترده برای راهاندازی DTOs استفاده کردهام.
پیشنهاد میکنم هر موقع که به فکتوریها نیاز دارید با آنها بازی کنید، به شما اطمینان میدهم که از آنها ناامید نخواهید شد. امیدوارم از این مقاله لذت برده باشید.
دیدگاه و پرسش
در حال دریافت نظرات از سرور، لطفا منتظر بمانید
در حال دریافت نظرات از سرور، لطفا منتظر بمانید