2
2

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?

【Laravel】Swagger-phpライブラリを使ったOPEN APIドキュメント生成

Last updated at Posted at 2023-01-13

Swagger-phpとは

ソースコードに記載したAnnotationsもしくはAttributesの内容を元に、RESTful APIのOPEN APIドキュメントを生成するライブラリです。

※Attributeを使用するにはPHP8.1以上であること。

Swagger-PHP公式

環境

Laravel 9.35.0
PHP 8.1.2

インストール

$ composer require zircote/swagger-php

準備

ベースとなるControllerに以下のAttributesを記述し、コントローラで使用できるように設定します。

#[OA\Info(title: "My First API", version: "0.1")]
#[OA\Server(url: "http://localhost:8080")]
class Controller extends BaseController
{
    use AuthorizesRequests, DispatchesJobs, ValidatesRequests;
}

#[OAから始まるのがAttributes記法です。

サンプル実装

例として、入力した名前とsubクレームでユーザテーブルにデータを登録する、という割とシンプルなユーザ追加APIに対してAttributesで記述してみます。

/**
 * Store a newly created resource in storage.
 *
 * @param  UserCreateRequest  $request
 * @return \Illuminate\Http\Response
 */
#[OA\Post(path: '/api/users', tags: ['User'])]
#[OA\RequestBody(content: [new OA\JsonContent(ref: Constant::SCHEMA_URL. 'UserCreateInput')])]
#[OA\Response(response: Response::HTTP_CREATED, description: 'The data', content: [new OA\JsonContent(ref: Constant::SCHEMA_URL. 'UserCreateOutput')])]
public function store(UserCreateRequest $request)
{
    $validated = $request->validated();

    $input = new UserCreateInput(
        $validated['cognito_sub'],
        $validated['name']
    );

    $res = $this->user_service->create($input);

    return response()->json($res->toArray());
}

1つずつ読み解いていきます。

まず以下の記述で、どのメソッドでどのURLかを定義します。

#[OA\Post(path: '/api/users', tags: ['User'])]

pathは該当APIのリクエストルート、tagsはOPEN APIドキュメントにおいて分類分けに使用します。

続いて、リクエストボディ用にスキーマを定義します。

#[OA\RequestBody(content: [new OA\JsonContent(ref: Constant::SCHEMA_URL. 'UserCreateInput')])]

contentにはリクエストボディの具体的な内容を定義します。

上記ではJsonContent内でrefを使用していますが、使用しない場合の記述は以下の通りになります。

#[OA\RequestBody(content: [new OA\JsonContent(required: ['cognito_sub', 'name'], properties: [new OA\Property(property: 'cognito_sub', type:'string', description:'cognito_sub'), new OA\Property(property: 'name', type:'string', description:'ユーザ名')])])]

refを使用することで、別途作成したスキーマを参照することができます。

そのスキーマの内容は以下の通りです。

#[OA\Schema(required: ['cognito_sub', 'name'])]
final class UserCreateInput extends CreateInput
{
    #[OA\Property(description: 'cognito_sub')]
    protected string $cognito_sub;

    #[OA\Property(description: 'ユーザ名')]
    protected string $name;

    /**
     * @param string $cognito_sub
     * @param string $name
     */
    public function __construct(string $cognito_sub, string $name)
    {
        $this->cognito_sub    = $cognito_sub;
        $this->name           = $name;
    }
}

スキーマ名は明示的に定義しない場合、クラス名と同じになります。cognito_sub、nameプロパティに対して、requiredの設定をしています。

型については、明示的に定義しない場合、各プロパティに定義してある(ここではstring)の型がスキーマの各プロパティに反映されます。

最後に、レスポンス内容に関してもスキーマ定義します。

#[OA\Response(response: Response::HTTP_CREATED, description: 'The data', content: [new OA\JsonContent(ref: Constant::SCHEMA_URL. 'UserCreateOutput')])]

contentにはレスポンスの具体的な値を定義します。

ここでも、refを使用することで、別途作成したスキーマを参照することができます。

スキーマの内容は以下の通りです。

#[OA\Schema(required: ['id', 'cognito_sub', 'name', 'profile', 'icon'])]
final class UserCreateOutput
{
    #[OA\Property(description: 'ユーザID')]
    private int $id;

    #[OA\Property(description: 'cognito_sub')]
    private string $cognito_sub;

    #[OA\Property(description: 'ユーザ名')]
    private string $name;

    #[OA\Property(description: '自己紹介')]
    private string $profile;

    #[OA\Property(description: 'ユーザアイコン')]
    private string $icon;


    /**
     * @param int $id
     * @param string $cognito_sub
     * @param string $name
     * @param string $profile
     * @param string $icon
     */
    public function __construct(int $id, string $cognito_sub, string $name, string $profile, string $icon)
    {
        $this->id             = $id;
        $this->cognito_sub    = $cognito_sub;
        $this->name           = $name;
        $this->profile        = $profile;
        $this->icon           = $icon;
    }

    /**
     * @return array
     */
    public function toArray(): array
    {
        return [
            'id'             => $this->id,
            'cognito_sub'    => $this->cognito_sub,
            'name'           => $this->name,
            'profile'        => $this->profile,
            'icon'           => $this->icon,
        ];
    }
}

以下のコマンドでOPEN APIドキュメントを生成します。

$ ./vendor/bin/openapi app -o storage/app/openapi.yaml

-o 以降は出力先のファイルパスをファイル名および拡張子込みで記載します。

コマンドを実行して生成されたopenapi.yamlの内容は以下の通りになります。

info:
  title: "My First API"
  version: "0.1"
servers:
  - url: "http://localhost:8080"

... 省略 ...

paths:
	/api/users:
	    post:
	      tags:
	        - User
	      summary: 'Store a newly created resource in storage.'
	      operationId: be551c1d694a01c164966f58bfa77013
	      requestBody:
	        content:
	          application/json:
	            schema:
	              $ref: '#/components/schemas/UserCreateInput'
	      responses:
	        '201':
	          description: 'The data'
	          content:
	            application/json:
	              schema:
	                $ref: '#/components/schemas/UserCreateOutput'

... 省略 ...

components:
  schemas:
    UserCreateInput:
      required:
        - cognito_sub
        - name
      properties:
        cognito_sub:
          description: cognito_sub
          type: string
        name:
          description: ユーザ名
          type: string
      type: object

... 省略 ...

  UserCreateOutput:
      required:
        - id
        - cognito_sub
        - name
        - profile
        - icon
      properties:
        id:
          description: ユーザID
          type: integer
        cognito_sub:
          description: cognito_sub
          type: string
        name:
          description: ユーザ名
          type: string
        profile:
          description: 自己紹介
          type: string
        icon:
          description: ユーザアイコン
          type: string
      type: object

Swagger UIで見たときに以下のように表示されています。
localhost_8002_ (1).png

Swagger UIでもちゃんと表示されていることがわかります。

Annotationsで記述すると

上記のユーザ追加APIをAnnotationsで置き換えた場合以下の通りになります。

/**
 * Store a newly created resource in storage.
 *
 * @param  UserCreateRequest  $request
 * @return \Illuminate\Http\Response
 *
 * @OA\Post(
 *     path="/api/users",
 *     tags={"User"},
 *     @OA\RequestBody(
 *         required=true,
 *         @OA\JsonContent(type="object",ref="#/components/schemas/UserCreateInput")
 *     ),
 *     @OA\Response(
 *         response=\Symfony\Component\HttpFoundation\Response::HTTP_CREATED,
 *         description="The data",
 *         @OA\JsonContent(type="object",ref="#/components/schemas/UserCreateOutput")
 *      )
 *   )
 * )
 */
public function store(UserCreateRequest $request)
{

... 省略 ...

}

リクエストボディのスキーマ

/**
 * @OA\Schema(
 *     required={"cognito_sub", "name"}
 * )
 */
final class UserCreateInput extends CreateInput
{
    /**
     * @OA\Property(
     *      property="cognito_sub",
     *      description="cognito_sub",
     * )
     */
    protected string $cognito_sub;

    /**
     * @OA\Property(
     *      property="name",
     *      description="ユーザ名",
     * )
     */
    protected string $name;

    /**
     * @param string $cognito_sub
     * @param string $name
     */
    public function __construct(string $cognito_sub, string $name)
    {
        $this->cognito_sub    = $cognito_sub;
        $this->name           = $name;
    }

... 省略 ..

}

レスポンス用のスキーマ

/**
 * @OA\Schema(
 *     required={"id", "cognito_sub", "name", "profile", "icon"}
 * )
 */
final class UserCreateOutput
{
    /**
     * @OA\Property(
     *      description="ユーザID",
     * )
     */
    private int $id;

    /**
     * @OA\Property(
     *      description="cognito_sub",
     * )
     */
    private string $cognito_sub;

    /**
     * @OA\Property(
     *      description="ユーザ名",
     * )
     */
    private string $name;

    /**
     * @OA\Property(
     *      description="自己紹介",
     * )
     */
    private string $profile;

    /**
     * @OA\Property(
     *      description="ユーザアイコン",
     * )
     */
    private string $icon;


    /**
     * @param int $id
     * @param string $cognito_sub
     * @param string $name
     * @param string $profile
     * @param string $icon
     */
    public function __construct(int $id, string $cognito_sub, string $name, string $profile, string $icon)
    {
        $this->id             = $id;
        $this->cognito_sub    = $cognito_sub;
        $this->name           = $name;
        $this->profile        = $profile;
        $this->icon           = $icon;
    }

... 省略 ...

}

どちらも生成されるOPEN APIドキュメントに差異は無いですが、Attributesで記述する方が補完を効かせることができるので、個人的にはおすすめです。

その他

パスパラメータ、クエリパラメータを持つAPIの場合はRequestBodyではなく以下の通りで定義します。

パスパラメータ(api/users/{id}のようなケース)

#[OA\PathParameter(required:true, name:'id', schema: new OA\Schema(type:'integer'))]

クエリパラメータ(api/users?page=1のようなケース)

#[OA\Parameter(in:"query", required:true, name:'page', description:'ページ番号', schema: new OA\Schema(type:'integer'))]
2
2
0

Register as a new user and use Qiita more conveniently

  1. You get articles that match your needs
  2. You can efficiently read back useful information
  3. You can use dark theme
What you can do with signing up
2
2

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?