首页 >后端开发 >php教程 >使用 queryFi 在 Laravel 中设计可扩展的存储库结构

使用 queryFi 在 Laravel 中设计可扩展的存储库结构

Linda Hamilton
Linda Hamilton原创
2024-12-25 18:03:17567浏览

Designing a Scalable Repository Structure in Laravel with queryFi

介绍

当我开始一个新的 Laravel 项目时,我经常想知道我应该使用什么结构。
Laravel 的结构非常好,不要误会我的意思,它很难把事情搞砸。
然而,有一个问题,我们如何避免将我们需要的所有逻辑放入控制器中,例如 UserController.php?
在这篇文章中,我们将重点讨论这个问题。我将分享我的首选结构来涵盖我的用例,以及一些最佳实践,并解释如何集成 queryFi。


入门

既然你在这里,我假设你已经有一个 Laravel 项目或知道如何创建一个项目,所以我们可以跳过这一步。只需确保 API 已设置即可。

我们将使用以下结构:

app
├── Repositories
|   ├── Interfaces
|   |   ├── EloquentRepositoryInterface.php
|   └── Eloquent
|       ├── BaseRepository.php
|       └── UserRepository.php
├── Resources
    ├── BaseResource.php
    └── ExtendedJsonResource.php

您可以在此处找到资源文件夹,以及自定义记录器。


1.安装queryFi

# Backend
composer require z3rka/queryfi

# Frontend
npm i queryfi

2.添加存储库接口

<?php

namespace App\Repositories\Interfaces;

use Illuminate\Http\Resources\Json\JsonResource;
use Illuminate\Database\Eloquent\Model;
use Illuminate\Http\Request;

interface EloquentRepositoryInterface
{
    public function create(array $attributes = [], ?string $successMessage, ?string $errorMessage): JsonResource;

    public function all(Request $request, ?string $errorMessage): JsonResource;

    public function one(Request $request, Model $model, ?string $errorMessage): JsonResource;

    public function update(Request $request, Model $model, ?string $successMessage, ?string $errorMessage): JsonResource;

    public function destroy(Request $request, ?string $successMessage, ?string $errorMessage): JsonResource;
}


3.添加存储库基类

我将在此处仅添加三个方法,以保持内容整洁并避免重复代码使空间混乱。
您可以在这里找到其余的内容。

<?php

namespace App\Repositories\Eloquent;

use Z3rka\Queryfi\HasRelations;

class BaseRepository implements EloquentRepositoryInterface {
    use HasRelations;

    public function __construct(protected Model $model)
    {
        //
    }

    public function create(array $attributes = [], ?string $successMessage, ?string $errorMessage): BaseResource
    {
        try {
            return new BaseResource(
                true,
                $successMessage,
                $this->model->create($attributes)
            );
        } catch (Exception $e) {
            return new BaseResource(
                false,
                $errorMessage,
                [
                    'error' => $e
                ]
            );
        }
    }

    public function all(Request $request, ?string $errorMessage): BaseResource
    {
        try {
            return new BaseResource(
              true, 
              'Success', 
              $this->processModel($request, $this->model)
            );
        } catch (Exception $e) {
            $this->spitError($e);
            return new BaseResource(
                false,
                "Fail",
                $this->model->get(),
                [
                    "error" => $e,
                ]
            );
        }
    }

    public function one(Request $request, Model $model, ?string $errorMessage): BaseResource
    {
        try {
            return new BaseResource(
              true, 
              "success", 
              $this->processModel($request, $model)
            )
        } catch (Exception $e) {
            $this->spitError($e);
            return new BaseResource(
                false,
                "Fail",
                $model,
                [
                    "error" => $e,
                ]
            );
        }
    }
}

通常,要从模型返回某些内容,您必须 ->get() 或 find 和 ->first(),这里的情况并非如此,因为我们使用的是 queryFi 包,我们可以这样做:

$this->processModel($request, $this->model)

如果查询中没有配置 getter(在此处检查 getter),默认情况下会自动覆盖 ->get()。


4. 用户存储库

<?php

namespace App\Repositories\Eloquent;
use App\Models\User;
use Z3rka\Queryfi\HasRelations;

class UserRepository extends BaseRepository
{
    use HasRelations;

    public function __construct(User $model)
    {
        parent::__construct($model);
    }
}

5. 用户控制器

<?php

namespace App\Http\Controllers;

use Illuminate\Http\Request;
use App\Repositories\Eloquent\UserRepository;

class UserController extends Controller
{
    public function __construct(public UserRepository $userRepository)
    {
    }

    public function index(Request $request)
    {
      return $this->userRepository
                  ->all($request, "Ooops, there was an error fetching all the users.")
    }

    public function show(Request $request, User $user)
    {
      return $this->userRepository->one(
            $request,
            $user,
            "Ooops, there was an error fetching the user."
        );
    }
}

6.api路由

Route::resource('users', UserController::class)->only(['index', 'show'])

一旦我们完成了所有设置,我们就可以通过直接传递查询参数来从浏览器发出请求,例如:

url?where[name]=john&select=name&getter=first

或者,我们可以在前端使用 queryFi TypeScript 包以获得更简洁的方法。

import { createQuery } from "queryfi";

const query = createQuery('/api/users', {
    baseUrl: "http://localhost:8000"
  })
  .where({
    id: 1
  })
  .first() // will return an object with the user if it exists.

生成的查询将如下所示

http://localhost:8000/api/users?where[id]=1&getter=first

这段代码将返回 id = 1 的用户。
当您使用 .find() 时,响应将自动成为一个对象。
如果使用 .get(),它将返回一个数组。

您可以在 Playground 中尝试一下,看看响应结果如何。

由于我们使用 HasRelations 特征,因此我们可以链接前端查询以在返回数据之前直接在后端转换数据,如下所示。

import { createQuery } from "queryfi";

const query = createQuery('/api/users', {
    baseUrl: "http://localhost:8000"
  })
  .where({
    id: 1
  })
  .select(['email', 'id', 'name'])
  .get() // will return an array with one entry if it exists.

有关可以使用的更多方法,请查看 queryFi 文档


结尾

就是这样! ?现在,您已经有了一个坚实而强大的起点,可以像专业人士一样构建您的 API。
来吧,发挥你的编码能力,做出一些很棒的东西! ??

联系 github、Linkedin、queryFi

以上是使用 queryFi 在 Laravel 中设计可扩展的存储库结构的详细内容。更多信息请关注PHP中文网其他相关文章!

声明:
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn