Guzzle7中文文档

目录
  1. 1. 发送请求
    1. 1.1. 创建客户端
    2. 1.2. 发送请求
    3. 1.3. 异步请求
    4. 1.4. 并发请求
  2. 2. 使用响应
  3. 3. 查询字符串参数
  4. 4. 上传数据
    1. 4.1. POST/表单请求
      1. 4.1.1. 发送表单字段
      2. 4.1.2. 发送表单文件
  5. 5. Cookies
  6. 6. 重定向
  7. 7. 异常
  8. 8. 环境变量
    1. 8.1. 相关ini设置

该页面提供了Guzzle的快速入门以及列子,如果你还没有安装Guzzle请前往 安装 页面。

发送请求

你可以使用Guzzle的 GuzzleHttp\ClientInterface 对象来发送请求。

创建客户端

1
2
3
4
5
6
7
8
use GuzzleHttp\Client;

$client = new Client([
// Base URI is used with relative requests
'base_uri' => 'http://httpbin.org',
// You can set any number of default request options.
'timeout' => 2.0,
]);

Client对象可以接收一个包含参数的数组:
base_uri
(string|UriInterface) 基URI用来合并到相关URI,可以是一个字符串或者UriInterface的实例,当提供了相关uri,将合并到基URI,遵循的规则请参考 RFC 3986, section 5.2 章节。

1
2
3
4
5
6
// Create a client with a base URI
$client = new GuzzleHttp\Client(['base_uri' => 'https://foo.com/api/']);
// Send a request to https://foo.com/api/test
$response = $client->request('GET', 'test');
// Send a request to https://foo.com/root
$response = $client->request('GET', '/root');

不想阅读RFC 3986?这里有一些关于 base_uri与其他URI处理器的快速例子:

base_uri URI Result
http://foo.com /bar http://foo.com/bar
http://foo.com/foo /bar http://foo.com/bar
http://foo.com/foo bar http://foo.com/bar
http://foo.com/foo/ bar http://foo.com/foo/bar
http://foo.com http://baz.com http://baz.com
http://foo.com/?bar bar http://foo.com/bar

handler
传输HTTP请求的(回调)函数。 该函数被调用的时候包含 Psr7\Http\Message\RequestInterface 以及参数数组,必须返回 GuzzleHttp\Promise\PromiseInterface,成功时满足 Psr7\Http\Message\ResponseInterfacehandler是一个构造方法,不能在请求参数里被重写。
...
(混合) 构造方法中传入的其他所有参数用来当作每次请求的默认参数。

发送请求

Client对象的方法可以很容易的发送请求:

1
2
3
4
5
6
7
$response = $client->get('http://httpbin.org/get');
$response = $client->delete('http://httpbin.org/delete');
$response = $client->head('http://httpbin.org/get');
$response = $client->options('http://httpbin.org/get');
$response = $client->patch('http://httpbin.org/patch');
$response = $client->post('http://httpbin.org/post');
$response = $client->put('http://httpbin.org/put');

你可以创建一个请求,一切就绪后将请求传送给client:

1
2
3
4
use GuzzleHttp\Psr7\Request;

$request = new Request('PUT', 'http://httpbin.org/put');
$response = $client->send($request, ['timeout' => 2]);

Client对象为传输请求提供了非常灵活的处理器方式,包括请求参数、每次请求使用的中间件以及传送多个相关请求的基URI。

你可以在 Handlers and Middleware 页面找到更多关于中间件的内容。

异步请求

你可以使用Client提供的方法来创建异步请求:

1
2
3
4
5
6
7
$promise = $client->getAsync('http://httpbin.org/get');
$promise = $client->deleteAsync('http://httpbin.org/delete');
$promise = $client->headAsync('http://httpbin.org/get');
$promise = $client->optionsAsync('http://httpbin.org/get');
$promise = $client->patchAsync('http://httpbin.org/patch');
$promise = $client->postAsync('http://httpbin.org/post');
$promise = $client->putAsync('http://httpbin.org/put');

你也可以使用Client的 sendAsync() and requestAsync() 方法:

1
2
3
4
5
6
7
8
9
10
use GuzzleHttp\Psr7\Request;

// Create a PSR-7 request object to send
$headers = ['X-Foo' => 'Bar'];
$body = 'Hello!';
$request = new Request('HEAD', 'http://httpbin.org/head', $headers, $body);
$promise = $client->sendAsync($request);

// Or, if you don't need to pass in a request instance:
$promise = $client->requestAsync('GET', 'http://httpbin.org/get');

这些方法返回了Promise对象,该对象实现了由 Guzzle promises library 提供的 Promises/A+ spec,这意味着你可以使用then()来调用返回值,成功使用Psr\Http\Message\ResponseInterface处理器,否则抛出一个异常。

1
2
3
4
5
6
7
8
9
10
11
12
13
use Psr\Http\Message\ResponseInterface;
use GuzzleHttp\Exception\RequestException;

$promise = $client->requestAsync('GET', 'http://httpbin.org/get');
$promise->then(
function (ResponseInterface $res) {
echo $res->getStatusCode() . "\n";
},
function (RequestException $e) {
echo $e->getMessage() . "\n";
echo $e->getRequest()->getMethod();
}
);

并发请求

你可以使用Promise和异步请求来同时发送多个请求:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
use GuzzleHttp\Client;
use GuzzleHttp\Promise;

$client = new Client(['base_uri' => 'http://httpbin.org/']);

// Initiate each request but do not block
$promises = [
'image' => $client->getAsync('/image'),
'png' => $client->getAsync('/image/png'),
'jpeg' => $client->getAsync('/image/jpeg'),
'webp' => $client->getAsync('/image/webp')
];

// Wait for the requests to complete; throws a ConnectException
// if any of the requests fail
$responses = Promise\Utils::unwrap($promises);

// You can access each response using the key of the promise
echo $responses['image']->getHeader('Content-Length')[0];
echo $responses['png']->getHeader('Content-Length')[0];

// Wait for the requests to complete, even if some of them fail
$responses = Promise\Utils::settle($promises)->wait();

// Values returned above are wrapped in an array with 2 keys: "state" (either fulfilled or rejected) and "value" (contains the response)
echo $responses['image']['state']; // returns "fulfilled"
echo $responses['image']['value']->getHeader('Content-Length')[0];
echo $responses['png']['value']->getHeader('Content-Length')[0];

当你想发送不确定数量的请求时,可以使用GuzzleHttp\Pool对象:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
use GuzzleHttp\Client;
use GuzzleHttp\Exception\RequestException;
use GuzzleHttp\Pool;
use GuzzleHttp\Psr7\Request;
use GuzzleHttp\Psr7\Response;

$client = new Client();

$requests = function ($total) {
$uri = 'http://127.0.0.1:8126/guzzle-server/perf';
for ($i = 0; $i < $total; $i++) {
yield new Request('GET', $uri);
}
};

$pool = new Pool($client, $requests(100), [
'concurrency' => 5,
'fulfilled' => function (Response $response, $index) {
// this is delivered each successful response
},
'rejected' => function (RequestException $reason, $index) {
// this is delivered each failed request
},
]);

// Initiate the transfers and create a promise
$promise = $pool->promise();

// Force the pool of requests to complete.
$promise->wait();

或者使用一个闭包,一旦池调用闭包,它将返回一个promise

1
2
3
4
5
6
7
8
9
10
11
12
$client = new Client();

$requests = function ($total) use ($client) {
$uri = 'http://127.0.0.1:8126/guzzle-server/perf';
for ($i = 0; $i < $total; $i++) {
yield function() use ($client, $uri) {
return $client->getAsync($uri);
};
}
};

$pool = new Pool($client, $requests(100));

使用响应

前面的例子里,我们取到了$response变量,或者从Promise得到了响应,Response对象实现了一个PSR-7接口Psr\Http\Message\ResponseInterface,包含了很多有用的信息。
你可以获取这个响应的状态码和和原因短语(reason phrase):

1
2
$code = $response->getStatusCode(); // 200
$reason = $response->getReasonPhrase(); // OK

你可以从响应获取头信息(header):

1
2
3
4
5
6
7
8
9
10
11
12
// Check if a header exists.
if ($response->hasHeader('Content-Length')) {
echo "It exists";
}

// Get a header from the response.
echo $response->getHeader('Content-Length');

// Get all of the response headers.
foreach ($response->getHeaders() as $name => $values) {
echo $name . ': ' . implode(', ', $values) . "\r\n";
}

使用getBody方法可以获取响应的主体部分(body),主体可以当成一个字符串或流对象使用

1
2
3
4
5
6
7
8
9
$body = $response->getBody();
// Implicitly cast the body to a string and echo it
echo $body;
// Explicitly cast the body to a string
$stringBody = (string) $body;
// Read 10 bytes from the body
$tenBytes = $body->read(10);
// Read the remaining contents of the body as a string
$remainingBytes = $body->getContents();

查询字符串参数

你可以有多种方式来提供请求的查询字符串 你可以在请求的URI中设置查询字符串:

1
$response = $client->request('GET', 'http://httpbin.org?foo=bar');

你可以使用query请求参数来声明查询字符串参数:

1
2
3
$client->request('GET', 'http://httpbin.org', [
'query' => ['foo' => 'bar']
]);

提供的数组参数将会使用PHP的http_build_query
最后,你可以提供一个字符串作为query请求参数:

1
$client->request('GET', 'http://httpbin.org', ['query' => 'foo=bar']);

上传数据

Guzzle为上传数据提供了一些方法。
你可以发送一个包含数据流的请求,将body请求参数设置成一个字符串、fopen返回的资源、或者一个Psr\Http\Message\StreamInterface的实例。

1
2
3
4
5
6
7
8
9
10
11
12
// Provide the body as a string.
$r = $client->request('POST', 'http://httpbin.org/post', [
'body' => 'raw data'
]);

// Provide an fopen resource.
$body = fopen('/path/to/file', 'r');
$r = $client->request('POST', 'http://httpbin.org/post', ['body' => $body]);

// Use the Utils::streamFor method to create a PSR-7 stream.
$body = \GuzzleHttp\Psr7\Utils::streamFor('hello!');
$r = $client->request('POST', 'http://httpbin.org/post', ['body' => $body]);

上传JSON数据以及设置合适的头信息可以使用json请求参数这个简单的方式:

1
2
3
$r = $client->request('PUT', 'http://httpbin.org/put', [
'json' => ['foo' => 'bar']
]);

POST/表单请求

除了使用body参数来指定请求数据外,Guzzle为发送POST数据提供了有用的方法。

发送表单字段

发送application/x-www-form-urlencodedPOST请求需要你传入form_params数组参数,数组内指定POST的字段。

1
2
3
4
5
6
7
8
9
$response = $client->request('POST', 'http://httpbin.org/post', [
'form_params' => [
'field_name' => 'abc',
'other_field' => '123',
'nested_field' => [
'nested' => 'hello'
]
]
]);

发送表单文件

你可以通过使用multipart请求参数来发送表单(表单enctype属性需要设置multipart/form-data)文件, 该参数接收一个包含多个关联数组的数组,每个关联数组包含一下键名:

  • name: (必须,字符串) 映射到表单字段的名称。
  • contents: (必须,混合) 提供一个字符串,可以是fopen返回的资源、或者一个Psr\Http\Message\StreamInterface以从PSR-7流中传输内容。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
$response = $client->request('POST', 'http://httpbin.org/post', [
'multipart' => [
[
'name' => 'field_name',
'contents' => 'abc'
],
[
'name' => 'file_name',
'contents' => fopen('/path/to/file', 'r')
],
[
'name' => 'other_file',
'contents' => 'hello',
'filename' => 'filename.txt',
'headers' => [
'X-Foo' => 'this is an extra header to include'
]
]
]
]);

Cookies

Guzzle可以使用cookies请求参数为你维护一个cookie会话,当发送一个请求时,cookies选项必须设置成GuzzleHttp\Cookie\CookieJarInterface的实例。

1
2
3
4
5
// Use a specific cookie jar
$jar = new \GuzzleHttp\Cookie\CookieJar;
$r = $client->request('GET', 'http://httpbin.org/cookies', [
'cookies' => $jar
]);

如果您想对所有请求使用共享的cookie jar,则可以在客户端构造函数中将cookie设置为true。

1
2
3
// Use a shared client cookie jar
$client = new \GuzzleHttp\Client(['cookies' => true]);
$r = $client->request('GET', 'http://httpbin.org/cookies');

GuzzleHttp\Cookie\CookieJarInterface存在不同的实现:

  • GuzzleHttp\Cookie\CookieJar类将cookie存储为数组。
  • GuzzleHttp\Cookie\FileCookieJar类使用JSON格式的文件保留非会话cookie。
  • GuzzleHttp\Cookie\SessionCookieJar类在客户端会话中保留cookie。
    您可以使用命名构造函数fromArray(array $cookies, $domain)将cookie手动设置到cookie jar中。
1
2
3
4
5
6
7
$jar = \GuzzleHttp\Cookie\CookieJar::fromArray(
[
'some_cookie' => 'foo',
'other_cookie' => 'barbaz1234'
],
'example.org'
);

您可以使用返回GuzzleHttp\Cookie\SetCookie实例的getCookieByName($name)方法获取其名称的cookie。

1
2
3
4
5
$cookie = $jar->getCookieByName('some_cookie');

$cookie->getValue(); // 'foo'
$cookie->getDomain(); // 'example.org'
$cookie->getExpires(); // expiration date as a Unix timestamp

借助toArray()方法,也可以将cookie提取到数组中。GuzzleHttp\Cookie\CookieJarInterface接口扩展了Traversable,因此可以在foreach循环中进行迭代。

重定向

树形视图
下面的树形视图描述了Guzzle异常如何相互依赖。

. \RuntimeException
└── TransferException (implements GuzzleException)
└── ConnectException (implements NetworkExceptionInterface)
└── RequestException
├── BadResponseException
│ ├── ServerException
│ └── ClientException
└── TooManyRedirectsException

如果你没有告诉Guzzle不要重定向,Guzzle会自动的进行重定向,你可以使用allow_redirects请求参数来自定义重定向行为。

  • 设置成true时将启用最大数量为5的重定向,这是默认设置。
  • 设置成false来禁用重定向。
  • 传入一个包含max键名的关联数组来声明最大重定向次数,提供可选的strict键名来声明是否使用严格的RFC标准重定向 (表示使用POST请求重定向POST请求 vs 大部分浏览器使用GET请求重定向POST请求)。
1
2
3
$response = $client->request('GET', 'http://github.com');
echo $response->getStatusCode();
// 200

下面的列子表示重定向被禁止:

1
2
3
4
5
$response = $client->request('GET', 'http://github.com', [
'allow_redirects' => false
]);
echo $response->getStatusCode();
// 301

异常

请求传输过程中出现的错误Guzzle将会抛出异常。

  • 在发送网络错误(连接超时、DNS错误等)时,将会抛出GuzzleHttp\Exception\RequestException异常。 该异常继承自GuzzleHttp\Exception\TransferException,捕获这个异常可以在传输请求过程中抛出异常。
1
2
3
4
5
6
7
8
9
10
11
use GuzzleHttp\Psr7;
use GuzzleHttp\Exception\RequestException;

try {
$client->request('GET', 'https://github.com/_abc_123_404');
} catch (RequestException $e) {
echo Psr7\Message::toString($e->getRequest());
if ($e->hasResponse()) {
echo Psr7\Message::toString($e->getResponse());
}
}
  • GuzzleHttp\Exception\ConnectException异常发生在网络错误时, 该异常继承自GuzzleHttp\Exception\RequestException

  • 如果http_errors请求参数设置成true,在400级别的错误的时候将会抛出GuzzleHttp\Exception\ClientException异常, 该异常继承自GuzzleHttp\Exception\BadResponseException``GuzzleHttp\Exception\BadResponseException继承自GuzzleHttp\Exception\RequestException

1
2
3
4
5
6
7
8
9
use GuzzleHttp\Psr7;
use GuzzleHttp\Exception\ClientException;

try {
$client->request('GET', 'https://github.com/_abc_123_404');
} catch (ClientException $e) {
echo Psr7\Message::toString($e->getRequest());
echo Psr7\Message::toString($e->getResponse());
}
  • 如果http_errors请求参数设置成true,在500级别的错误的时候将会抛出GuzzleHttp\Exception\ServerException异常。 该异常继承自GuzzleHttp\Exception\BadResponseException
  • GuzzleHttp\Exception\TooManyRedirectsException异常发生在重定向次数过多时, 该异常继承自GuzzleHttp\Exception\RequestException
    上述所有异常均继承自GuzzleHttp\Exception\TransferException

环境变量

Guzzle提供了一些可自定义的环境变量:
GUZZLE_CURL_SELECT_TIMEOUT
当在curl处理器时使用curl_multi_select()控制了 curl_multi_* 需要使用到的持续时间, 有些系统实现PHP的curl_multi_select()存在问题,调用该函数时总是等待超时的最大值。
HTTP_PROXY
定义了使用http协议发送请求时使用的代理。
HTTPS_PROXY
定义了使用https协议发送请求时使用的代理。

相关ini设置

Guzzle配置客户端时可以利用PHP的ini配置。
openssl.cafile
当发送到”https”协议的请求时需要用到指定磁盘上PEM格式的CA文件,参考:https://wiki.php.net/rfc/tls-peer-verification#phpini_defaults