Writing Testable Code in Laravel: A Beginner’s Guide

Abu Sayed
7 min readFeb 25, 2024

--

Struggling to write testable code in Laravel? Learn techniques like dependency injection, mocking, and decoupling to make your code easy to test. This beginner’s guide covers SOLID principles, avoiding antipatterns, and using Laravel tools like interfaces and factories to enable fast, maintainable tests.

Learn Laravel — Writing Testable Code in Laravel: A Beginner’s Guide
Learn Laravel — Writing Testable Code in Laravel: A Beginner’s Guide

Testing your code is super important if you want to build robust, maintainable applications in Laravel. But it can be tricky to write testable code if you’re just starting out. Not to worry — I’m here to break it down for you in plain English!

See, when code is “testable”, it means you can easily verify that it works as expected by writing automated tests. This saves you time down the road when making changes, because your tests will catch any regressions or broken functionality. Pretty handy, right?

Now I won’t lie: writing testable code takes some extra thought and effort up front. But it’s worth it! Here are some tips to get you on the right track…

Depend on Interfaces, Not Concrete Classes

When your classes depend on interfaces instead of concrete classes, it makes them way easier to fake or mock during testing. Let me explain…

Imagine your UserController depends directly on the UserRepository class. That would be pretty hard to test, because UserRepository hits the real database!

Instead, have your UserController depend only on the UserRepositoryInterface. Then you can mock that interface and avoid hitting the real database during tests.

It’s a similar idea for dependencies like the cache, file storage, sending email, etc. Code against their interfaces so you can fake ’em during tests!

Avoid Singleton Classes

The Singleton pattern can be handy sometimes, but it’s enemy #1 for testable code.

That’s because Singleton classes can’t be faked or replaced during testing since there can be only one!

So if your Singleton class hits the database, talks to an API, or does anything besides plain logic, it’ll be difficult to test classes that depend on it.

Avoid using the Singleton pattern unless absolutely necessary. And if you do need it, consider refactoring the test-unfriendly parts into another class.

Leverage Laravel’s Contracts

Laravel ships with a bunch of super handy contracts and interfaces like:

  • Illuminate\Contracts\Cache\Repository
  • Illuminate\Contracts\Queue\Queue
  • Illuminate\Contracts\Mail\Mailer

Instead of coupling your code directly to Laravel’s implementations, depend on the contracts.

For example, typehint the CacheRepository contract instead of the concrete CacheManager class. This makes it a breeze to mock the cache during testing!

Prefer Dependency Injection Over Facades

Laravel’s facades provide easy access to internal classes, but can make testing harder.

That’s because facades don’t expose which concrete class is actually being used. Plus, it’s impossible to mock a facade without ugly hacks.

Use dependency injection instead for classes you need to test:

// Avoid this in testable code
Cache::put('key', 'value');
// Do this instead
public function __construct(CacheRepository $cache) {
$this->cache = $cache;
}
public function someMethod() {
$this->cache->put('key', 'value');
}

Now you can easily mock or swap the CacheRepository during testing!

Fetch Data in Controllers, Not Views

A common mistake is querying the database directly from the view.

This makes the view impossible to test in isolation since it hits the real DB. Not cool!

Instead, query for data in the controller and pass it to the view:

// Bad and hard to test
public function index() {
return view('reports.index')
->with('users', User::all());
}
// Better and testable!
public function index() {
$users = User::all();

return view('reports.index', compact('users'));
}

Now you can test the controller without touching the real database. And the view can be tested in isolation too. Win-win!

Avoid God Classes

Sometimes developers cram tons of different logic into model classes — a classic “God object” anti-pattern.

This makes models nearly impossible to test. Every method hits the database and has 15 responsibilities!

Instead, break code into smaller classes with single responsibilities. For example:

  • Data access objects (DAOs) for databases queries
  • Services for specific use cases and logic
  • Value objects like Entities and DTOs

Each class will be focused and way easier to fake, mock, and test.

Don’t Share State Between Tests

Your tests should be isolated without shared state.

For example, don’t rely on data in the database or cache from a previous test.

Laravel makes this easy with transactions for database tests and mocking for caches, sessions, etc.

Make sure each test can run independently. You want fast, focused tests that only fail when there’s an actual bug.

Prefer Feature Tests Over Unit Tests

Feature tests verify the full stack — database, queues, facades, routing, etc.

This gives you more confidence that features will work in production.

Of course, unit tests have their place too. But aim for more feature tests that test the app as a whole.

A Beginner’s Guide to Writing Testable Code in Laravel
A Beginner’s Guide to Writing Testable Code in Laravel

Now Dive into this detailed walkthrough on how to write testable code in Laravel. Learn about different use cases, examples, and small programs that can help you understand and perfect this critical aspect of coding!

Why is Writing Testable Code Important?

Imagine you’re constructing a building. Would you want to live in it without ensuring it’s safe and secure? The same concept applies to coding. Writing testable code is like conducting a safety check. It’s the process of making sure your code doesn’t crumble under pressure.

Laravel: An Ally in Writing Testable Code

Laravel, the popular PHP framework, is your ally in this mission. It ensures you’re not just writing code, but crafting high-quality, testable code.

Laravel’s Testing Tools: Your Coding Companions

Laravel comes equipped with several in-built testing tools that make your life easier. PHPUnit, Laravel Dusk, and HTTP tests are some of the big guns in Laravel’s testing arsenal. They’re here to help, so don’t shy away from using them.

PHPUnit: Your First Line of Defense

PHPUnit is like your first line of defense. It’s a tool that helps you write unit tests for your code. Here’s how it works:

<?php
use PHPUnit\Framework\TestCase;
class ExampleTest extends TestCase
{
public function testBasicTest()
{
$this->assertTrue(true);
}
}

In this example, `assertTrue(true)` is a simple assertion that checks if the given condition is true.

Laravel Dusk: Your Browser Testing Buddy

When it comes to browser testing, Laravel Dusk has got your back. It’s like a friendly ghost who can interact with your web application just like a human. Here’s a glimpse of it in action:

<?php
use Laravel\Dusk\Browser;
class ExampleTest extends DuskTestCase
{
public function testBasicExample()
{
$this->browse(function (Browser $browser) {
$browser->visit('/')
->assertSee('Laravel');
});
}
}

In this example, Laravel Dusk visits the root URL (`/`) and checks if it can see the word ‘Laravel’.

Writing Testable Code: The Essential Steps

Now that we’ve got our tools, let’s get into the heart of the matter: writing testable code.

Keep It Simple, Silly (KISS)

Remember the KISS principle? It applies to coding too!

The simpler your code, the easier it is to test. So, try to keep your functions and classes short and sweet.

Don’t Repeat Yourself (DRY)

Repetition is a coder’s worst enemy. It makes your code harder to maintain and test. So, always adhere to the DRY principle.

Single Responsibility Principle (SRP)

Each class or method should have a single responsibility. This makes testing a breeze. Think of it like this: if each person in a team knows their role and does it well, the team performs better. The same logic applies to your code.

Dependency Injection: A Lifesaver

Dependency Injection is like a lifesaver in the sea of coding. It allows you to inject objects into a class, instead of creating them there. This makes your code more flexible and easier to test.

Wrapping Up

Writing testable code in Laravel might seem like a daunting task, but it doesn’t have to be. With the right tools and techniques, you can create code that’s not only functional but also robust and reliable. So, go ahead, give it a try! Who knows, you might just find yourself enjoying the process.

Remember, coding is an art, and like any artist, you should take pride in your creation. So, let’s ensure that our code is not just good, but great. After all, quality matters, right?

Bonus: Use Factories for Test Data

Manually defining test data is a pain. Laravel’s model factories make it a cinch to generate stubs with dummy data.

Just define a factory for each model:

UserFactory::define(User::class, function (Generator $faker) {
return [
'name' => $faker->name,
'email' => $faker->email
];
});

Then you can easily create stubs in tests:

public function test_it_creates_a_user() {
$user = User::factory()->make();

// Assert on user
}

Factories will save you tons of time and keep your tests focused on behavior, not test data setup!

There you have it!

Those are my top tips for writing testable code in Laravel. Here’s a quick recap:

  • Depend on interfaces/contracts
  • Avoid singletons
  • Leverage Laravel contracts
  • Use dependency injection over facades
  • Fetch data in controllers
  • Break code into small, focused classes
  • Isolate tests from each other
  • Write feature tests when possible
  • Use factories for test data

It takes practice, but you’ll get the hang of it! Writing testable code pays off big time by giving you the confidence to ship quality software. Your future self will thank you!

Let me know if you have any other questions. Happy testing!

--

--

Abu Sayed
Abu Sayed

Written by Abu Sayed

Bangladeshi Full Stack Web Dev, Sys Admin & DevOps Engineer. Skills: Data analysis, SQL, Unity, C#. Python, PHP & Laravel. Me on: bd.linkedin.com/in/imabusayed

No responses yet