API 说明

jsonapi.org

Discuz! Q 使用的是 jsonapi.orgAPI 说明 - 图1 定义的格式,使用 tobscure/json-apiAPI 说明 - 图2 包的实现。

路由分别对应

  • 以psr-2基础命名规则外,下面为实践中相关命名规则
  • 以user为例, 路由和路由名称为小写复数按功能以.分开
  • 控制器按功能,列表为复数,其它单条担任可为单数驼峰命名
  • 序列化模型名为单数驼峰命名
  • 数据库模型以单数驼峰命名
  • 表名以复数命名

下面为具体增删改查路由例子:

  1. $route->get('/users', 'users.index', ListUsersController::class);
  2. $route->get('/users/{id}', 'users.resource', ResourceUserController::class);
  3. $route->post('/users', 'users.create', CreateUserController::class);
  4. $route->patch('/users/{id}', 'users.update', UpdateUserController::class);
  5. $route->delete('/users/{id}', 'users.delete', DeleteUserController::class);

列表数据

继承 Discuz\Api\Controller\AbstractListController 需要指定 $serializer 要用于序列化模型的 data 方法,并实现返回模型集合的方法。该 data 方法接收 Request 对象和 tobscure/json-api Document

  1. class ListUsersController extends AbstractListController
  2. {
  3. public $serializer = UserSerializer::class;
  4. public function data(ServerRequestInterface $request, Document $document)
  5. {
  6. return User::all();
  7. }
  8. }

单条数据

同列表数据一样继承 Discuz\Api\Controller\AbstractResourceController 并实现相关属性和方法,接收参数一样

  1. class ResourceUserController extends AbstractResourceController
  2. {
  3. public $serializer = UserSerializer::class;
  4. public function data(ServerRequestInterface $request, Document $document)
  5. {
  6. $body = $request->getQueryParams();
  7. $id = $body->get('id');
  8. return User::findOrFail($id);
  9. }
  10. }

创建数据

继承 Discuz\Api\Controller\AbstractCreateController。这与单条数据相同,但响应状态代码将自动设置为 201 Created。您可以通过以下方式访问传入的 JSON:API POST传过来的数据可通过 $request->getParsedBody() 来取到 json 数组,并通过 Laravel Arr 类来获取数组相关信息

  1. class CreateUserController extends AbstractCreateController
  2. {
  3. public $serializer = UserSerializer::class;
  4. public function data(ServerRequestInterface $request, Document $document)
  5. {
  6. $attributes = Arr::get($request->getParsedBody(), 'data.attributes');
  7. return User::create([
  8. 'name' => Arr::get($attributes, 'name')
  9. ]);
  10. }
  11. }

更新数据

继承 Discuz\Api\Controller\AbstractResourceController 同单条数据一样,现相关属性和方法,接收参数一样,返回相关数据,获取同创建数据一样

  1. class UpdateUserController extends AbstractCreateController
  2. {
  3. public $serializer = UserSerializer::class;
  4. public function data(ServerRequestInterface $request, Document $document)
  5. {
  6. $attributes = Arr::get($request->getParsedBody(), 'data.attributes', []);
  7. $body = $request->getQueryParams();
  8. $attributes = $body->get('id');
  9. $user = User::findOrFail($id);
  10. $user->name = $attributes['name'];
  11. $user->save();
  12. return $user;
  13. }
  14. }

删除数据

继承 Discuz\Api\Controller\AbstractDeleteController 实现delete 方法,该方法接收 Request,默认返回空 204 No Content 响应。

  1. class DeleteUserController extends AbstractDeleteController
  2. public function delete(ServerRequestInterface $request)
  3. {
  4. $id = Arr::get($request->getQueryParams(), 'id');
  5. User::findOrFail($id)->delete();
  6. }
  7. }

序列化模型说明

继承 Discuz\Api\Serializer\AbstractSerializer 实现 getDefaultAttributes 方法,该方法接收数据库模型,写好 type 属性

  1. class UserSerializer extends AbstractSerializer
  2. {
  3. protected $type = 'user';
  4. public function getDefaultAttributes($model)
  5. {
  6. return [
  7. 'id' => $model->id,
  8. 'username' => $model->username,
  9. 'email' => $model->email
  10. ];
  11. }
  12. }

自定义返回类型

实现接口 Psr\Http\Server\RequestHandlerInterface

  1. <?php
  2. namespace Discuz\Api\Controller;
  3. use Psr\Http\Server\RequestHandlerInterface;
  4. class AbstractSerializeController implements RequestHandlerInterface
  5. {
  6. public function handle(ServerRequestInterface $request): ResponseInterface
  7. {
  8. return new HtmlResponse();
  9. }
  10. }