使用 JSON 请求正文测试 laravel 控制器
Posted
技术标签:
【中文标题】使用 JSON 请求正文测试 laravel 控制器【英文标题】:Testing laravel controllers with JSON request body 【发布时间】:2012-12-19 18:11:09 【问题描述】:我正在尝试为 Laravel 控制器编写一个 phpunit 测试,该控制器需要带有 JSON 格式正文的 post 请求。
控制器的简化版本:
class Account_Controller extends Base_Controller
public $restful = true;
public function post_login()
$credentials = Input::json();
return json_encode(array(
'email' => $credentials->email,
'session' => 'random_session_key'
));
目前我有一个测试方法可以正确地将数据作为 urlencoded 表单数据发送,但我不知道如何将数据作为 JSON 发送。
我的测试方法(我在写测试时使用了github gisthere)
class AccountControllerTest extends PHPUnit_Framework_TestCase
public function testLogin()
$post_data = array(
'email' => 'user@example.com',
'password' => 'example_password'
);
Request::foundation()->server->set('REQUEST_METHOD', 'POST');
Request::foundation()->request->add($post_data);
$response = Controller::call('account@login', $post_data);
//check the $response
我在前端使用 angularjs,默认情况下,发送到服务器的请求是 JSON 格式。我不希望将其更改为发送 urlencoded 表单。
有谁知道我如何编写一个为控制器提供 JSON 编码主体的测试方法?
【问题讨论】:
【参考方案1】:至少从 Laravel 5.2 开始,Illuminate\Foundation\Testing\Concerns\MakesHttpRequests
中有一个 json()
方法,因此您可以执行以下操作:
$data = [
"name" => "Foobar"
];
$response = $this->json('POST', '/endpoint', $data);
此外,从 Laravel 5.3 开始,还有 putJson()
、postJson()
等方便的方法。因此它甚至可以进一步缩短为:
$data = [
"name" => "Foobar"
];
$response = $this->postJson('/endpooint', $data);
然后你可以像$response->assertJson(...)
一样:
$response->assertJson(fn (AssertableJson $json) => $json->hasAll(['id', 'name']));
【讨论】:
【参考方案2】:从 Laravel 5.1 开始,有一种更简单的方法可以通过 PHPunit 测试 JSON 控制器。只需传递一个包含数据的数组,它就会自动编码。
public function testBasicExample()
$this->post('/user', ['name' => 'Sally'])
->seeJson([
'created' => true,
]);
来自文档:http://laravel.com/docs/5.1/testing#testing-json-apis
【讨论】:
我认为这不会使请求本身具有 application/json 的内容类型标头。据我所知,它仍会将 application/x-www-form-urlencoded 发送到 /users 端点。我相信用 json 强制请求的方法是做@eoinoc 所做的事情。 使用$this->json('POST', ...
而不是 $this->post(...
来使用适当的内容类型。【参考方案3】:
在 Laravel 5 中,call()
方法发生了变化:
$this->call(
'PUT',
$url,
[],
[],
[],
['CONTENT_TYPE' => 'application/json'],
json_encode($data_array)
);
我认为 Symphony 的 request()
方法正在被调用:
http://symfony.com/doc/current/book/testing.html
【讨论】:
【参考方案4】:这对我有用。
$postData = array('foo' => 'bar');
$postRequest = $this->action('POST', 'MyController@myaction', array(), array(), array(), array(), json_encode($postData));
$this->assertTrue($this->client->getResponse()->isOk());
$this->action
的第七个参数是content
。请参阅http://laravel.com/api/source-class-Illuminate.Foundation.Testing.TestCase.html#_action 的文档
【讨论】:
我也在寻找解决方案,在the docs找到相同的答案。 好答案,但请确保将 json 传递给第 6 个参数,而不是第 7 个。【参考方案5】:这就是我在 Laravel4 中执行此操作的方式
// Now Up-vote something with id 53
$this->client->request('POST', '/api/1.0/something/53/rating', array('rating' => 1) );
// I hope we always get a 200 OK
$this->assertTrue($this->client->getResponse()->isOk());
// Get the response and decode it
$jsonResponse = $this->client->getResponse()->getContent();
$responseData = json_decode($jsonResponse);
$responseData
将是一个等于 json 响应的 PHP 对象,并允许您随后测试响应:)
【讨论】:
【参考方案6】:有很多更简单的方法可以做到这一点。您可以简单地将 Input::$json 属性设置为要作为 post 参数发送的对象。请参阅下面的示例代码
$data = array(
'name' => 'sample name',
'email' => 'abc@yahoo.com',
);
Input::$json = (object)$data;
Request::setMethod('POST');
$response = Controller::call('client@create');
$this->assertNotNull($response);
$this->assertEquals(200, $response->status());
我希望这对您的测试用例有所帮助
更新:原始文章可在此处获得http://forums.laravel.io/viewtopic.php?id=2521
【讨论】:
我有机会使用这种方法重写我的一些测试。我发现它更有用,因为它允许我检查单个代码是否在工作,而不是检查用于 http 请求的所有代码。当出现问题时,此方法还提供回溯,这在调试时比 http 状态代码更有用。 当我尝试这个时,我得到Fatal error: Access to undeclared static property: Illuminate\Support\Facades\Input::$json
- 我错过了一些上下文吗?
@AaronPollock 因为 json() 是一种方法,而不是属性。将其用作 Input::json()【参考方案7】:
一个简单的解决方案是使用 CURL - 这样您还可以从服务器捕获“响应”。
class AccountControllerTest extends PHPUnit_Framework_TestCase
public function testLogin()
$url = "account/login";
$post_data = array(
'email' => 'user@example.com',
'password' => 'example_password'
);
$content = json_encode($post_data);
$curl = curl_init($url);
curl_setopt($curl, CURLOPT_HEADER, false);
curl_setopt($curl, CURLOPT_RETURNTRANSFER, true);
curl_setopt($curl, CURLOPT_HTTPHEADER, array("Content-type: application/json"));
curl_setopt($curl, CURLOPT_POST, true);
curl_setopt($curl, CURLOPT_POSTFIELDS, $content);
$json_response = curl_exec($curl);
$status = curl_getinfo($curl, CURLINFO_HTTP_CODE);
curl_close($curl);
$response = json_decode($json_response, true);
// Do some $this->Assert() stuff here on the $status
CURL 实际上会使用 JSON 模拟原始 HTTP 帖子 - 所以您知道您正在真正测试您的功能;
【讨论】:
好方法。但我建议使用Httpful bundle 来保持测试代码更简洁。 我最终创建了一个类,它将所有 curl 功能包装成一些更易于使用的函数并更新我的测试以从新类继承。 ApiTestCase.php以上是关于使用 JSON 请求正文测试 laravel 控制器的主要内容,如果未能解决你的问题,请参考以下文章
Laravel 5.2 PHPUnit JSON Api 请求正文未设置
在 Symfony 2.3 控制器中访问 POST 请求的 JSON 正文
带有 JSON 请求的 Laravel/phpunit 测试丢失了重要的请求细节