CakePHP includes a PSR-18 compliant HTTP client which can be used for making requests. It is a great way to communicate with webservices, and remote APIs.
CakePHP には、リクエストの実行に使用できる基本的ながら強力な HTTP クライアントが含まれています。 これは、ウェブサービスや、リモート API と通信するための素晴らしい方法です。
リクエストの実行は、シンプルで簡単です。 GET リクエストは、次のようになります。
use Cake\Http\Client;
$http = new Client();
// 単純な GET
$response = $http->get('http://example.com/test.html');
// クエリー文字列を使用した単純な GET
$response = $http->get('http://example.com/search', ['q' => 'widget']);
// クエリー文字列と追加ヘッダーを使用した単純な GET
$response = $http->get('http://example.com/search', ['q' => 'widget'], [
'headers' => ['X-Requested-With' => 'XMLHttpRequest']
]);
POST や PUT のリクエストを実行することは、同様に簡単です。
// application/x-www-form-urlencoded エンコードデータを POST リクエストで送信
$http = new Client();
$response = $http->post('http://example.com/posts/add', [
'title' => 'testing',
'body' => 'content in the post'
]);
// application/x-www-form-urlencoded エンコードデータを PUT リクエストで送信
$response = $http->put('http://example.com/posts/add', [
'title' => 'testing',
'body' => 'content in the post'
]);
// 他のメソッドも同様に、
$http->delete(/* ... */);
$http->head(/* ... */);
$http->patch(/* ... */);
If you have created a PSR-7 request object you can send it using
sendRequest()
:
use Cake\Http\Client;
use Cake\Http\Client\Request as ClientRequest;
$request = new ClientRequest(
'http://example.com/search',
ClientRequest::METHOD_GET
);
$client = new Client();
$response = $client->sendRequest($request);
配列の中にファイルハンドルを含めることで、リクエストのボディー内にファイルを含めることができます。
$http = new Client();
$response = $http->post('http://example.com/api', [
'image' => fopen('/path/to/a/file', 'r'),
]);
ファイルハンドルは、最後まで読まれます。それが読まれる前に巻き戻されることはありません。
警告
互換性の理由から、 @
で始まる文字列は、ローカルまたはリモートのファイルパスとして評価されます。
この機能は、CakePHP 3.0.5 からは非推奨で、将来のバージョンで削除されます。 それまでは、HTTP クライアントに渡されるユーザーデータは、次のようにサニタイズする必要があります。
$response = $http->post('http://example.com/api', [
'search' => ltrim($this->request->getData('search'), '@'),
]);
クエリー文字列の先頭の @
文字を維持する必要がある場合は、
http_build_query()
で予めエンコードされたクエリー文字列を渡すことができます。
$response = $http->post('http://example.com/api', http_build_query([
'search' => $this->request->getData('search'),
]));
非常に特殊な方法でリクエストボディーを構築しなければならない場合もあるかもしれません。
このような状況では、多くの場合、あなたが望んだ特殊なマルチパートの HTTP リクエストを作るために
Cake\Http\Client\FormData
を使用することができます。
use Cake\Http\Client\FormData;
$data = new FormData();
// XML 部分を作成
$xml = $data->newPart('xml', $xmlString);
// コンテンツタイプを設定
$xml->type('application/xml');
$data->add($xml);
// addFile() でファイルアップロードの作成
// 同様にフォームデータにファイルを追加します。
$file = $data->addFile('upload', fopen('/some/file.txt', 'r'));
$file->contentId('abc123');
$file->disposition('attachment');
// リクエストの送信
$response = $http->post(
'http://example.com/api',
(string)$data,
['headers' => ['Content-Type' => $data->contentType()]]
);
REST API を扱うとき、多くの場合、フォームエンコードされていないリクエストボディーを送信する必要があります。 Http\Client は、type オプションを介してこれを公開します。
// JSON リクエストボディーの送信
$http = new Client();
$response = $http->post(
'http://example.com/tasks',
json_encode($data),
['type' => 'json']
);
type
キーは「json」、「xml」または完全な MIME タイプのいずれかになります。
type
オプションを使用するときは、文字列としてデータを提供してください。
クエリー文字列パラメーターとリクエストボディーの両方を必要とする GET リクエストを行う場合は、
次の操作で行うことができます。
// クエリー文字列パラメーター付きの GET リクエストで JSON ボディーを送信
$http = new Client();
$response = $http->get(
'http://example.com/tasks',
['q' => 'test', '_content' => json_encode($data)],
['type' => 'json']
);
各 HTTP メソッドは、追加のリクエスト情報を提供するための $options
パラメーターを受け取ります。
以下のキーが $options
で使用することができます。
headers
- 追加ヘッダーの配列。
cookie
- 使用するクッキーの配列。
proxy
- プロキシー情報の配列。
auth
- 認証データの配列、 type
キーが認証ストラテジーに委譲するために使用されます。
デフォルトでは、Basic 認証が使用されます。
ssl_verify_peer
- デフォルトは true
。SSL 証明書の検証を無効にするには
false
を設定します(推奨されません)。
ssl_verify_peer_name
- デフォルトは true
。SSL 証明書を検証する場合、
ホスト名検証を無効にするには false
を設定します(推奨されません)。
ssl_verify_depth
- デフォルトは 5 。CA チェーンを通過する深さ。
ssl_verify_host
- デフォルトは true
。ホスト名に対して SSL 証明書を検証します。
ssl_cafile
- デフォルトは組み込みの cafile 。カスタム CA バンドルを使用するためには
上書きしてください。
timeout
- 秒単位でタイムアウトするまで待つ時間。
type
- 独自のコンテンツタイプでリクエストボディーを送信します。 $data
を文字列にするか、
GET リクエストで _content
オプションを指定する必要があります。
redirect
- フォローするリダイレクトの数。デフォルトは false
です。
オプションのパラメーターは、いつも HTTP メソッドの 3 番目のパラメーターです。
スコープ指定クライアント を作成するために
Client
を構築する場合にも使用できます。
Cake\Http\Client
は、いくつかの異なる認証システムをサポートしています。
異なる認証ストラテジーを、開発者によって追加することができます。
認証ストラテジーは、リクエストが送信される前に呼び出され、
リクエストの文脈にヘッダーを追加することができます。
Basic 認証の例:
$http = new Client();
$response = $http->get('http://example.com/profile/1', [], [
'auth' => ['username' => 'mark', 'password' => 'secret']
]);
デフォルトでは、 auth オプションに 'type'
キーが存在しない場合、
Cake\Http\Client
は Basic 認証を使用します。
Basic 認証の例:
$http = new Client();
$response = $http->get('http://example.com/profile/1', [], [
'auth' => [
'type' => 'digest',
'username' => 'mark',
'password' => 'secret',
'realm' => 'myrealm',
'nonce' => 'onetimevalue',
'qop' => 1,
'opaque' => 'someval'
]
]);
'type' キーに 'digest' を設定することによって、 認証サブシステムにダイジェスト認証を使用することを伝えます。
多くのモダンなウェブサービスは、API にアクセスするために OAuth 認証を必要とします。 含まれる OAuth 認証は、すでにコンシューマキーとコンシューマシークレットがあることを前提としています。
$http = new Client();
$response = $http->get('http://example.com/profile/1', [], [
'auth' => [
'type' => 'oauth',
'consumerKey' => 'bigkey',
'consumerSecret' => 'secret',
'token' => '...',
'tokenSecret' => '...',
'realm' => 'tickets',
]
]);
OAuth2 は、多くの場合、単一のヘッダーであるため、特殊な認証アダプターがありません。 代わりに、アクセストークンを使用してクライアントを作成することができます。
$http = new Client([
'headers' => ['Authorization' => 'Bearer ' . $accessToken]
]);
$response = $http->get('https://example.com/api/profile/1');
いくつかのプロキシーは使用するために認証を必要とします。 一般に、この認証は Basic ですが、任意の認証アダプターによって実装することができます。 デフォルトでは、 type キーが設定されていない限り、 Http\Client は Basic 認証を前提としています。
$http = new Client();
$response = $http->get('http://example.com/test.php', [], [
'proxy' => [
'username' => 'mark',
'password' => 'testing',
'proxy' => '127.0.0.1:8080',
]
]);
2番目のプロキシーパラメーターは、プロトコルのない IP またはドメインの文字列でなければなりません。 ユーザー名とパスワードは、ヘッダー通じて渡されますが、プロキシー文字列は stream_context_create() を通じて渡されます。
ドメイン名を再入力すると、認証とプロキシーの設定が面倒になり、エラーが発生しやすくなります。 間違いの可能性を減らし、いくつかの退屈さを緩和するために、 スコープ指定クライアントを作成することができます。
// スコープ指定クライアントの作成
$http = new Client([
'host' => 'api.example.com',
'scheme' => 'https',
'auth' => ['username' => 'mark', 'password' => 'testing']
]);
// api.example.com にリクエストします
$response = $http->get('/test.php');
スコープ指定クライアントを作成する場合、以下の情報を使用することができます。
host
basePath
scheme
proxy
auth
port
cookies
timeout
ssl_verify_peer
ssl_verify_depth
ssl_verify_host
リクエスト実行時に、これらのオプションのいずれかを指定することで上書きすることができます。 リクエスト URL 中のホスト、スキーム、プロキシー、ポートが上書きされます。
// 先ほど作成したスコープ指定クライアントの使用
$response = $http->get('http://foo.com/test.php');
上記は、ドメインやスキーム、ポートが置き換えられます。ただし、このリクエストは、 スコープ指定クライアントの作成時に定義された他のすべてのオプションを使用して引き続き行われます。 対応するオプションの詳細については リクエストメソッドのオプション をご覧ください。
Http\Client はまた、リクエストを行うときクッキーを受け入れることができます。 クッキーを受け入れることに加えて、レスポンス中に自動的に設定された有効なクッキーを格納します。 クッキーを持つ任意のレスポンスが、元の Http\Client のインスタンスに格納されています。 Client インスタンスに格納されているクッキーは、それ以後のドメイン+パスの組み合わせが一致する リクエストに自動的に含まれます。
$http = new Client([
'host' => 'cakephp.org'
]);
// いくつかのクッキーをセットしたリクエストを実行
$response = $http->get('/');
// デフォルトで、初めのリクエストのクッキーが
// 含まれます。
$response2 = $http->get('/changelogs');
リクエストの $options
パラメーターの中に設定することにより、
自動で含まれるクッキーをいつでも上書きすることができます。
// 格納されたクッキーを独自の値に置き換えます。
$response = $http->get('/changelogs', [], [
'cookies' => ['sessionid' => '123abc']
]);
addCookie()
メソッドを使って、作成された後のクライアントにクッキーオブジェクトを
追加することができます。
use Cake\Http\Cookie\Cookie;
$http = new Client([
'host' => 'cakephp.org'
]);
$http->addCookie(new Cookie('session', 'abc123'));
Response オブジェクトは、レスポンスデータを検査するための多くのメソッドを持ちます。
文字列としてレスポンスボディー全体を読み込みます。
// 文字列としてレスポンス全体を読み込み
$response->getStringBody();
また、レスポンスのストリームオブジェクトにアクセスし、そのメソッドを使用することができます。
// レスポンスボディーを含む Psr\Http\Message\StreamInterface を取得
$stream = $response->getBody();
// ストリームを一度に 100 バイト読み込み
while (!$stream->eof()) {
echo $stream->read(100);
}
JSON や XML のレスポンスが一般的に使用されているので、レスポンスオブジェクトは、
デコードされたデータを読み取るためにアクセサーを簡単に使用することができます。
JSON は配列にデコードされ、XML データは、 SimpleXMLElement
ツリーにデコードされます。
// 何らかの XML を取得
$http = new Client();
$response = $http->get('http://example.com/test.xml');
$xml = $response->getXml();
// 何らかの JSON を取得
$http = new Client();
$response = $http->get('http://example.com/test.json');
$json = $response->getJson();
デコードされたレスポンスデータはそれを複数回アクセスし、レスポンスオブジェクトに格納されても、 何も追加コストはかかりません。
いくつかの異なるメソッドを介してヘッダーにアクセスすることができます。 メソッドを介してアクセスする際に、ヘッダー名は常に大文字と小文字を区別しない値として扱われます。
// 連想配列として全てのヘッダーを取得
$response->getHeaders();
// 配列として単一のヘッダーを取得
$response->getHeader('content-type');
// 文字列としてヘッダーを取得
$response->getHeaderLine('content-type');
// レスポンスのエンコーディングを取得
$response->getEncoding();
// 全てのヘッダーの key=>value の配列を取得
$response->headers;
クッキーについて必要なデータ量に応じて、いくつかの異なる方法でクッキーを読むことができます。
// 全てのクッキー (全データ) を取得
$response->getCookies();
// 単一のクッキーの値を取得
$response->getCookie('session_id');
// 単一のクッキーの完全なデータを取得
// value, expires, path, httponly, secure キーを含みます
$response->getCookieData('session_id');
// 全てのクッキーの完全なデータにアクセス
$response->cookies;
レスポンスオブジェクトは、ステータスコードを確認するためのいくつかのメソッドを提供します。
// レスポンスが 20x だった
$response->isOk();
// レスポンスが 30x だった
$response->isRedirect();
// ステータスコードの取得
$response->getStatusCode();
// __get() ヘルパー
$response->code;
By default Http\Client
will prefer using a curl
based transport adapter.
If the curl extension is not available a stream based adapter will be used
instead. You can force select a transport adapter using a constructor option:
use Cake\Http\Client\Adapter\Stream;
$client = new Client(['adapter' => Stream::class]);
In tests you will often want to create mock responses to external APIs. You can
use the HttpClientTrait
to define responses to the requests your application
is making:
use Cake\Http\TestSuite\HttpClientTrait;
use Cake\TestSuite\TestCase;
class CartControllerTests extends TestCase
{
use HttpClientTrait;
public function testCheckout()
{
// Mock a POST request that will be made.
$this->mockClientPost(
'https://example.com/process-payment',
$this->newClientResponse(200, [], json_encode(['ok' => true]))
);
$this->post("/cart/checkout");
// Do assertions.
}
}
There are methods to mock the most commonly used HTTP methods:
$this->mockClientGet(/* ... */);
$this->mockClientPatch(/* ... */);
$this->mockClientPost(/* ... */);
$this->mockClientPut(/* ... */);
$this->mockClientDelete(/* ... */);
As seen above you can use the newClientResponse()
method to create responses
for the requests your application will make. The headers need to be a list of
strings:
$headers = [
'Content-Type: application/json',
'Connection: close',
];
$response = $this->newClientResponse(200, $headers, $body)