Back to Blog

Laravel 9.5 Released

Julian Beaujardin
Julian Beaujardin March 16th, 2022

The Laravel team released 9.5 with partial queue faking, a freezeTime() test helper, a storage assertDirectoryEmpty() assertion, closures in assertJsonPath(), and more:

Callback Support on Collections Implode Method

@Lito contributed callback support on Collect::implode() to simplify ->map()->implode() calls:

{{-- Before --}}
<span>{{ $user->cities->map(fn ($city) => $city->name.' ('.$city->state->name.')')->implode(', ') }}</span>

{{-- Using a callback --}}
<span>{{ $user->cities->implode(fn ($city) => $city->name.' ('.$city->state->name.')', ', ') }}</span>

Assert an Empty Directory With the Storage Fake

Mark Beech contributed the ability to assert an empty directory with a Storage::fake() instance:

// Before 9.5
$this->assertEmpty(Storage::disk('temp')->allFiles('/foo'));

// +9.5
Storage::disk('temp')->assertDirectoryEmpty('/foo');

If there are no files in the directory but other subdirectories, the assertion will fail because it contains other folders/files. Here's an example from the pull request discussion:

Storage::fake('temp');
Storage::disk('temp')->put('/foo/bar.txt', 'string');
Storage::disk('temp')->assertDirectoryEmpty('/'); // fail

JSON Assertion "assertJsonPath()" Now Accepts a Closure

Fabien Villepinte contributed the ability to pass a closure to assertJsonPath without any backwards-compatible breaks:

$response = TestResponse::fromBaseResponse(new Response([
    'data' => ['foo' => 'bar'],
]));

$response->assertJsonPath('data.foo', 'bar');
$response->assertJsonPath('data.foo', fn ($value) => $value === 'bar');

While the example above seems more straightforward with the string version, you can now reach for a Closure if you need more complex logic around the path assertion.

Partial Queue Faking

Taylor Otwell contributed partial faking for queue jobs in tests:

Queue::fake([JobsToFake::class, /* ... */]);

New Method to Create "through" Models

Hafez Divandari contributed the ability to create a new "through" model without overriding the whole hasOneThroughor hasManyThrough methods:

// Define a `newThroughInstance` method
protected function newThroughInstance($resource)
{
    return (new \App\Models\ExampleEntity)->setTable($resource);
}

New Wrap String Helper

Markus Hebenstreit contributed a wrap() string helper. Here's example usages from the pull request description:

Str:wrap('value')->wrap('"');
Str::of('value')->wrap('"');
str('value')->wrap('"');

// Outputs: "value"

Str:wrap('is', 'This ', ' me!');
Str::of('is')->wrap('This ', ' me!');
str('is')->wrap('This ', ' me!');

// Outputs: This is me!

Freeze Time Helper for Tests

@Italo: contributed a freezeTime() test method which will freeze the current time in a test:

public function test_something()
{
    $this->freezeTime();

    // Or set the time at the current second for dates
    // that have no sub-second precision.
    $this->freezeSecond();
}

The freezeTime() method is syntactic sugar for the following:

$this->travelTo(Carbon::now());

Allow Callables in Http::beforeSending()

Dries Vints: contributed to accepting callables in the Http::beforeSending() method instead of only invokable classes. Now, the following example will work instead of getting a "Call to member function __invoke() on array":

Http::baseUrl('https://api.example.org')
    ->beforeSending([ $this, 'prepareRequest' ])
    ->asJson()
    ->withoutVerifying();

Release Notes

You can see the complete list of new features and updates below and the diff between 9.4.0 and 9.5.0 on GitHub. The following release notes are directly from the changelog:

v9.5.1 - 2022-03-15

Reverted

  • Revert "Fix the guard instance used." (#41491)

v9.5.0 - 2022-03-15

Added

  • Added callback support on implode Collection method. (#41405)
  • Added Illuminate/Filesystem/FilesystemAdapter::assertDirectoryEmpty() (#41398)
  • Implement email "metadata" for SesTransport (#41422)
  • Make assertPath() accepts Closure (#41409)
  • Added callable support to operatorForWhere on Collection (#41414, #41424)
  • Added partial queue faking (#41425)
  • Added --name option to schedule:test command (#41439)
  • Define Illuminate/Database/Eloquent/Concerns/HasRelationships::newRelatedThroughInstance() (#41444)
  • Added Illuminate/Support/Stringable::wrap() (#41455)
  • Adds "freezeTime" helper for tests (#41460)
  • Allow for callables with beforeSending inIlluminate/Http/Client/PendingRequest.php::runBeforeSendingCallbacks() (#41489)

Fixed

  • Fixed deprecation warnings from route:list when filtering on name or domain (#41421)
  • Fixes HTTP::pool response when a URL returns a null status code (#41412)
  • Fixed recaller name resolution in Illuminate/Session/Middleware/AuthenticateSession.php (#41429)
  • Fixed the guard instance used in /Illuminate/Session/Middleware/AuthenticateSession.php (#41447)
  • Fixed route:list --except-vendor hiding Route::view() & Route::redirect() (#41465)

Changed

  • Add null typing to connection property in \Illuminate\Database\Eloquent\Factories\Factory (#41418)
  • Update reserved names in GeneratorCommand (#41441)
  • Redesign php artisan schedule:list Command (#41445)
  • Extend eloquent higher order proxy properties (#41449)
  • Allow passing named arguments to dynamic scopes (#41478)
  • Throw if tag is passed but is not supported in Illuminate/Encryption/Encrypter.php (#41479)
  • Update PackageManifest::$vendorPath initialisation for cases, when composer vendor dir is not in project director (#41463)