我经常使用response()
帮助器,我只是将数据返回给用户。现在我还必须包含http状态代码,但我不想更改每个响应(无论如何都可能是坏的)。
所以我试图通过在response()
中创建我自己的helpers.php
来覆盖app/Http/helpers.php
辅助函数。
当我将它添加到我的作曲家文件时,它首先从框架中自动加载当前的helpers.php,当我在bootstrap/global.php
中的autload include之前添加它时,我将无法使用app()
和其他Laravel函数。
我怎么能解决这个问题?我只想在响应数组中包含状态代码。
用这个逻辑编写的所有Laravel辅助函数
if ( ! function_exists('response'))
{
function response($content = '', $status = 200, array $headers = array())
{
// function body
}
}
对于第一个Laravel检查是否存在此函数,如果存在,Laravel将不会再次定义此函数(否则会抛出致命错误)。因此,如果您要在自动加载器包含vendor/laravel/framework/src/Illuminate/Foundation/helpers.php
文件之前定义您的函数,则可以定义自定义响应函数。
不幸的是,没有办法说作曲家首先加载你的autoload.files
部分,然后laravel autoload.files
。但你可以做小黑客......
打开bootstrap/autoload.php
文件并在自动加载器之前包含您的文件
// file with your custom helper functions
require __DIR__.'/../app/app/Http/helpers.php';
require __DIR__.'/../vendor/autoload.php';
我不会直接回答你的问题,因为我不知道是否有解决方案(不改变Laravels helpers.php
或重命名你的功能)
但是,这个常见用例的框架有一个解决方案。 Response Macros
您可以定义宏(这在服务提供商中完成)
Response::macro('foo', function($value){
// do some stuff
return Response::make($value);
});
你可以像这样使用它:
return response()->foo('bar');
我必须这样做才能覆盖now()
助手,这样我就可以控制运行测试时的表观时间。我跟着创建regular advice的app/Http/helpers.php
然后将其添加到bootstrap/autoload.php
,如下所示:
require __DIR__.'/../app/Http/helpers.php'; // added
require __DIR__.'/../vendor/autoload.php';
这通常有效,因为正如Marty所说,只有在没有具有该名称的现有函数时才会定义所有助手。因此,上面的两行加载您的自定义帮助程序,然后执行所有供应商自动加载,其中包括Laravel的帮助程序,并且您已定义的函数优先。
但不幸的是,在使用Behat进行测试时似乎没有使用autoload.php
,这正是我正在使用的。所以我需要一个替代解决方案。简而言之,通过使用https://github.com/funkjedi/composer-include-files软件包确保文件在供应商文件之前自动加载的唯一简单方法。引用其自述文件:
在过去,简单地修改
bootstrap/autoload.php
以包括帮助者就足够了。但是,新版本的PHPUnit在执行PHPUnit引导程序文件之前包含Composer Autoloader。因此,这种覆盖帮助程序的方法不再可行,因为它会在包含引导程序文件时触发致命错误。
所以我使用composer require funkjedi/composer-include-files
安装了这个包,然后将其添加到composer.json
:
"extra": {
"include_files": [
"app/Http/helpers.php"
]
},
完成后,运行composer dump-autoload
以重新生成自动加载文件。现在,覆盖在常规应用程序操作和运行测试期间都可以正常工作!