Uncovering PHP bugs with @template

January 30, 2019 by Matt Brown - original article

This guide discusses functionality available in two PHP static analysis tools: Psalm (from Vimeo), and Phan. PHPStan has plans to support templated types in an upcoming version.


At a former job I wrote a load of C# code, and I really enjoyed playing around with generic classes. You could write code like…

class LazyList<T> : IEnumerable<T> { ... }

...

var list = new LazyList<Foo>(ids);  
foreach (var value in list) {}

…and know that value inside the loop has the type Foo. Having that guarantee (and knowing it would be enforced by the type checker) left you free to worry about Everything Else.

If you’ve written C#, Java, TypeScript, or any number of other languages, this feature—generic type parameters — may be familiar. It enables you to write very simple equations, in the form of type annotations, that a type checker will solve to produce useful type information.

When I arrived at Vimeo (and started using PHP), I left those syntactic niceties behind. PHP lacks type parameters, so the equivalent of the above code looks something like this:

class LazyList extends \ArrayObject { ... }

...

$list = new LazyList(Foo::class, $ids);foreach ($list as $value) {}

That code might make sense to a human (and work at runtime), but it leaves the type checker in the lurch. Type checkers like Psalm rely on PhpDoc annotations to provide information that the language can’t, but there’s no standardized annotation you can give LazyList that will tell you what type $value has.

At this point developers normally pick one of three options:

Our codebase mostly used the first approach, but the mixed type reduced our type checker’s ability to find bugs¹. For example, there were a few places where we’d iterate over $list and call $value->methodWithTypo() inside the loop. Since $value was typed to mixed our type checker could not see the bug.

We needed a way to tell the type checker exactly how LazyList behaves. After going through a number of alternative docblock tags, I settled on one already used for the purpose by Phan — @template — and added some additional behavior (since adopted by that type checker too).

With @template and a few other annotations I’ll explain we fill out the above class’s definition:

<?php

/**
 * @template T
 * @extends \ArrayObject<int, T>
 */
class LazyList extends \ArrayObject
{
  /**
   * @var class-string<T>
   */
  public $type;

  /**
   * @param class-string<T> $type
   */
  public function __construct(string $type, array $ids)
  {
    $this->type = $type;
    // more
  }

  // more
}

Here’s what the different annotations mean:

By adding that single @template tag to LazyList, we help the type checker understand the use of that object in hundreds of places throughout our codebase.

We use @template tags in a bunch of other scenarios:

In each case we’re adding information that the typechecker didn’t know previously, meaning fewer false positives and more bugs found.

In general, @template can be used in most places that you’d use a templated/type parameter in one of the languages mentioned above.

Going Deeper

Since adding initial support for @template, a number of people have requested more comprehensive templating rules that are present in other languages. The following features are available in more recent releases of Psalm.

Template type constraints (Psalm-only)

What if we want to restrict the creation of LazyList instances only to class strings corresponding to a certain type?

We want a situation in which LazyList(Foo::class ...) is permitted, as is LazyList(SubclassOfFoo::class ...), but not new LazyList(Bar::class ...).

We can accomplish this by appending of SomeClassOrInterface to our tag:

/**
 * @template T of Foo
 */
class LazyList extends \ArrayObject {...}

Phan has plans for a similar feature.

Combining with @psalm-assert/@phan-assert

Psalm and Phan support assertions in function docblocks in the form of @psalm-assert/@phan-assert. You can use this annotation to write your own templated assertInstanceOf checks like so:

<?php
/**
 * @template T
 *
 * @param class-string<T> $class
 *
 * @psalm-assert T $input
 * @phan-assert T $input
 */
function assertInstanceOf(string $class, object $input) : void {
  if (!is_a($input, $class)) {
    throw new \UnexpectedValueException('Bad');
  }
}

Combining with @param-out&nbsp;(Psalm-only)

Psalm now supports out params, in the form of a @param-out declaration.

This means we can type check the function [usort](http://php.net/manual/en/function.usort.php) without changing Psalm’s internals — we just define usort using template types:

/**
 * @template T
 *
 * @param array<mixed, T> $arr
 * @param callable(T, T):int $callback
 *
 * @param-out array<int, T> $arr
 */
function usort(array &$arr, callable $callback): bool {}

Now Psalm detects that the following is an error:

<?php

$a = [[1], [2], [3]];
usort($a, "strcmp");

Going wild with templates

We can do some weird and wonderful things with the power of templated types.

In the following code, the first parameter of the function foo expects a closure that itself takes a closure and returns whatever type that inner closure returns. We then call it with a closure whose inner closure returns a different type than the one it returns. Whether or not you understood that, Psalm sees the bug.

<?php

/**
 * @template T
 * @param Closure(Closure():T):T $s
 */
function foo(Closure $s) : void {}

foo(
  /** @param Closure():int $s */
  function(Closure $s) : string {
    return '#' . $s();
  }
);

Towards a future full of types

At Vimeo we use a lot of TypeScript, and our developers love it. While part of TypeScript’s success lies in its excellent type checker, I’d argue that its real breakthrough has been the way it enables developers, through its expansive type system, to eloquently describe how their code operates.

I want PHP to have a similarly expressive type system, and, with the addition of @template, I hope developers will write not just type-safe code, but better structured, more logical code.

If you have ideas for other advanced typing features, please open a ticket!


¹ Vimeo’s own Psalm config now prevents developers from adding untyped PHP code. Existing issues have been grandfathered in via Psalm’s baseline functionality.