WordPress 上で REST API 開発をするためのプラグインです。
アイデアやインターフェースは BEAR.Sunday から影響を受けています。
前提: WordPress 管理画面からパーマリンク設定を「投稿名」などにしておいてください。
$ cd /path/to/theme
$ composer require tenjuu99/wp-resta
functions.php
で初期化
以下は、サンプルのディレクトリにあるAPIを読みこむ設定です。
<?php
require_once __DIR__ . '/vendor/autoload.php';
(new Wp\Resta\Resta)->init([
'routeDirectory' => [
[__DIR__ . '/vendor/wp/resta/src/REST/Example/Routes', 'Wp\\Resta\\REST\\Example\\Routes\\', 'example']
],
'use-swagger' => true,
'schemaDirectory' => [
[__DIR__ . '/vendor/wp/resta/src/REST/Example/Schemas', 'Wp\\Resta\\REST\\Example\\Schemas\\'],
],
]);
WordPress のプラグインとしても利用できます。
次の例は composer/installers
を利用して WordPress プラグインを wp-content/plugins/
以下に配置しています。
$ composer config "extra.installer-paths.wp-content/plugins/{\$name}/" "['type:wordpress-plugin']"
$ composer require composer/installers tenjuu99/wp-resta
無事 wp-content/plugins 以下に展開できたら、管理画面からプラグインを有効化してください。
インストールしたら、管理画面に REST API doc
というメニューが追加されます。
このページでは、 Swagger UI を使ってAPI定義をドキュメント化しています。
このサンプル実装は src/REST/Example/Routes/
以下にあります。
自分のルーティング定義を追加するためには、 functions.php
での初期化時のコードにルーティング用ディレクトリの設定を記述してください。
routeDirectory
に渡す配列は、 ['ディレクトリ名', 'php namespace', 'api namespace']
となっています。
schemaDirectory
は ['ディレクトリ名', 'php namespace']
です。
(new Wp\Resta\Resta)->init([
'routeDirectory' => [
['wp-content/themes/mytheme/vendor/wp/resta/src/REST/Example/Routes', 'Wp\\Resta\\REST\\Example\\Routes\\', 'example'],
+ ['src/Routes', 'MyREST\\Routes\\', 'myroute']
],
'schemaDirectory' => [
['wp-content/themes/mytheme/vendor/wp/resta/src/REST/Example/Schemas', 'Wp\\Resta\\REST\\Example\\Schemas\\'],
+ ['src/Schemas', 'MyREST\\Schemas\\']
],
]);
必要に応じて composer.json
にも autoload 設定を追加してください。
"autoload": {
+ "psr-4": {
+ "MyREST\\": "src/"
+ }
}
src/Routes/HelloWorld.php
を作ります。
<?php
namespace MyREST\Routes;
use Wp\Resta\REST\AbstractRoute;
class HelloWorld extends AbstractRoute
{
public $body = 'Hello, world!';
}
次のURLが生成されます。
$ curl http://example.com/wp-json/myroute/helloworld
"Hello, world!" と値が返ってきていることを確かめてください。
myroute
が functions.php
で定義した namespace ですが、 helloworld
にはクラス名がそのまま利用されています。
URL 定義は ROUTE
定数を定義することで変更できます。ついでにURL変数も定義してみます。
<?php
namespace MyREST\Routes;
use Wp\Resta\REST\AbstractRoute;
class HelloWorld extends AbstractRoute
{
protected const ROUTE = 'hello/[name]';
protected const URL_PARAMS = [
'name' => 'string',
];
public function callback(string $name) : string
{
return "Hello, ${name}!";
}
}
次のURLが生成されるとおもいます。
$ curl http://example.com/wp-json/myroute/hello/amashige
"Hello, amashige!"
と返ってくるとおもいます。
ROUTE定数のなかに [var]
と []
で囲えばパスパラメータとして扱うことができます。
変数は次のようなパターンを許容します。
protected const URL_PARAMS = [
'id' => 'integer',
'id_not_required' => '!integer',
'name' => 'string',
'name_not_required' => '!string',
'ok_or_ng' => '(ok|ng)',
'first_name' => [
'type' => 'string',
'required' => false,
'regex' => '[a-z]+'
],
];
これらは、ROUTE定数に user/[id]
のような形でURL埋め込み変数として定義されていればパスパラメータとして機能します。
パスパラメータとして利用されていないが URL_PARAMS
に定義されている変数はクエリパラメータとして利用されます。
次のような例は、 http://example.com/wp-json/myroute/user/2?name=tenjuu99
として展開されます。
protected const ROUTE = 'user/[id]';
protected const URL_PARAMS = [
'id' => 'integer',
'name' => '?string',
];
これらの変数は callback メソッドで受け取ることができます。
AbstractRoute
を継承したクラスが callback
という名のメソッドを持っている場合、このメソッドを呼びだしてレスポンスの body にします。body として返してよいのは、 WP_REST_Response
が body として解釈できるものになります。また、Psr\Http\Message\ResponseInterface
を返した場合にはそのまま利用されます。
callback
メソッドの引数は、URL変数を受けとることができます。 URL変数
に id
を定義していれば callback(int $id)
と定義して問題ありません。
また、簡易な DI があるため、解決可能なクラスを引数に定義すると受け取ることができます。ランタイムに値が決まるもの(例えば WP_REST_Response
)などはコンストラクタインジェクションでは値が決まっていませんが、コールバックが呼び出される時点では確定しているので、利用できます。
簡易なDIを用意しています。
基本的には autorwiring で、ほとんどの場合に設定なしで利用できます。
また、基本的にはコンストラクタインジェクションのみ対応しています(AbstractRoute::callback
は例外)。
// src/Lib/Foo.php
namespace MyREST\Lib;
class Foo
{
private Bar $bar;
public function __construct(Bar $bar)
{
$this->bar = $bar;
}
public function getBarString(): string
{
return $this->bar->get();
}
}
// src/Lib/Bar.php
namespace MyREST\Lib;
class Bar
{
public function get(): string
{
return 'bar';
}
}
// src/Routes/Sample.php
namespace MyREST\Routes;
use MyREST\Lib\Foo;
class Sample extends AbstractRoute
{
private Foo $foo;
public function __construct(Foo $foo)
{
$this->foo = $foo;
}
public function callback()
{
return $this->foo->getBarString();
}
}
Bar::get
の返り値が解決されます。
上記は DI の autowiring で解決可能ですが、interface を注入する場合は自動で解決できません。この場合は設定が必要になります。
設定は、初期化コードに dependencies
を渡すことができます。
(new Wp\Resta\Resta)->init([
'routeDirectory' => [
...
],
'schemaDirectory' => [
...
],
+ 'dependencies' => [
+ PSR\Log\LoggerInterface::class => MonoLog\Logger::class,
+ ],
]);
また、autowiring 機構はクラスやインターフェースの依存しか解決しないため、コンストラクタがクラスやインターフェース以外の値を受けている場合は解決できません。 この場合は関数を使ってください。
(new Wp\Resta\Resta)->init([
'routeDirectory' => [
...
],
'schemaDirectory' => [
...
],
'dependencies' => [
PSR\Log\LoggerInterface::class => MonoLog\Logger::class,
+ WP_Query::class => function () {
+ return new WP_Query(['post_type' => 'post']);
+ }
],
]);