Danyloff

PHP Annotations улучшают ваш Laravel проект


  • ·

· ·

С развитием PHP потребность в аннотациях стремительно падает. Раньше типы возвращаемых значений, свойств, параметров и других неочевидных вещей объявлялись аннотациями в комментарии стиля PHPDoc.

На сегодняшний день в текущей версии PHP(8.2) уже можно объявлять типы свойств, параметров, возвращаемых значений методов/функций, есть поддержка типа Enum и других полезные улучшения но, в PHP все так же много осталось “магии” которую с ловкостью используют различные фреймворки. Laravel не исключение. Так как Laravel использует слишком много “магии” порог вхождения в проект для новых разработчиков повышается. По этому в данной статье, мы, с помощью аннотаций, будем понижать тот самый порог.

Для тех кто не знаком что такое аннотации, читайте статью ниже

Модели(Model)

В Laravel для аттрибутов модели используется магические методы PHP __get(), __set(). Взглянем на типичную модель User

class User extends Model
{
    protected $table = 'users';
    protected $guarded = ['id'];
}

В данной модели User мы видим… ничего… абсолютно никакой информации по атрибутам этой модели. Чтобы узнать атрибуты новому разработчику приходится идти в таблицу, смотреть схему и держать в голове эти знания. Все это занимает время, и не мало, особенно когда моделей много. Но что же делать?! Давайте попробуем применить аннотации

/**
 * @property int $id
 * @property string $first_name
 * @property string $surname
 * @property int $age
 * @property string $password
 * @property bool $active
 * @property string $role admin|user
 * @property DateTime|null $created_at
 * @property DateTime|null $updated_at
 */
class User extends Model
{
    protected $table = 'users';
    protected $guarded = ['id'];
}

Вот теперь стало лучше. Главное не забывать редактировать аннотации если модифицируете схему. А что на счет методов?! И для методов есть свои аннотации

/**
 * @property int $id
 * @property string $first_name
 * @property string $surname
 * @property int $age
 * @property string $password
 * @property bool $active
 * @property string $role admin|user
 * @property DateTime|null $created_at
 * @property DateTime|null $updated_at
 * 
 * @method static Builder byAge(int $age)
 * @method static Builder byActive(bool $active = true)
 */
class User extends Model
{
    protected $table = 'users';
    protected $guarded = ['id'];

    public function scopeByAge(Builder $query, int $age): void
    {
        $query->where('age', $age);
    }

    public function scopeActive(Builder $query, bool $active = true): void
    {
        $query->where('active', $active);
    }
}

Вот теперь стало намного лучше. Когда ваши коллеги откроют вашу модель, они сразу увидят аттрибуты и возможные локальные scopes.

Добавим в модель отношений чтобы приблизиться к реальной ситуации.

/**
 * @property int $id
 * @property string $first_name
 * @property string $surname
 * @property int $age
 * @property string $password
 * @property bool $active
 * @property string $role admin|user
 * @property DateTime|null $created_at
 * @property DateTime|null $updated_at
 *
 * @property Collection<Photo> $photos
 * @property Collection<File> $files
 * @property Document|null $document
 *
 * @method static Builder byAge(int $age)
 * @method static Builder byActive(bool $active = true)
 */
class User extends Model
{
    protected $table = 'users';
    protected $guarded = ['id'];

    public function photos(): HasMany
    {
        return $this->hasMany(Photo::class);
    }

    public function files(): HasMany
    {
        return $this->hasMany(File::class);
    }

    public function document(): HasOne
    {
        return $this->hasOne(Document::class);
    }

    public function scopeByAge(Builder $query, int $age): void
    {
        $query->where('age', $age);
    }

    public function scopeActive(Builder $query, bool $active = true): void
    {
        $query->where('active', $active);
    }
}

Давайте теперь рассмотрим что может быть еще не очевидно в вашем коде что можно улучшить с помощью аннотаций. Допустим у вас есть UserService, с методом getAll() который возвращает массив пользователей.

final class UserService
{
    public function getAll(): array
    {
        return User::all()->toArray();
    }
}

Для массивов есть специальный синтасис

final class UserService
{
    /**
     * @return User[]
     */
    public function getAll(): array
    {
        return User::all()->toArray();
    }
}

Теперь станет понятно клиентам вашего сервиса и вашей IDE что находится внутри возвращаемого массива. Для итераторов применяется синтаксис обобщения.

final class UserService
{
    /**
     * @return Iterator<int, User>
     */
    public function getAll(): Iterator
    {
        return User::all()->getIterator();
    }
}

Для кастомных схем массивов тоже есть определенный синтаксис

final class UserService
{
    /**
     * @return array<array{
     *     id: int,
     *     name: string
     * }>
     */
    public function getAll(): array
    {
        return User::all()->map(fn (User $user) => [
            'id' => $user->getKey(),
            'name' => $user->name
        ])->toArray();
    }

    /**
     * @return array{
     *     id: int,
     *     name: string,
     *     age: int
     * }
     */
    public function find(int $id): array
    {
        return User::find($id)->only('id', 'name', 'age');
    }
}

В этом примере мы описали схему возвращаемого массива для методов getAll() и find().

Заключение

Аннотации это мощный инструмент который экономит время, помогает отлавливать больше типичных ошибок на этапе разработки и делает ваш код приятней для чтения. Данные аннотации поддерживают практически все современные IDE, и будут вам давать полезные советы при разработке. Аннотации так же используются для автоматического документирования приложения. Вывод один, используйте аннотации и не пренебрегайте этим.


Так же интересно

Generator в PHP

Generator в PHP - это функция, которая позволяет создавать итерируемые объекты (объекты, которые могут быть использованы в цикле foreach) без необходимости...

Согласование Laravel и DDD (часть 2)

В предыдущей статье мы пришли к выводу, что при реализации DDD с помощью Laravel, сам фреймворк должен стать нашей новой парадигмой программирования, чтобы...