我经常使用response()
帮助程序,我只是向用户返回带有消息的数据。现在,我还必须包含http状态代码,但我不想更改每个响应(这可能很糟糕)。
因此,我试图通过在app/Http/helpers.php
中创建自己的helpers.php
来覆盖response()
助手函数。
当我将它添加到我的composer文件中时,它会首先从框架中自动加载当前的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
文件之前定义您的函数,您可以定义自定义响应函数。
不幸的是,没有办法说composer首先加载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';
为了覆盖now()
助手,我不得不这样做,这样我就可以控制运行测试时的明显时间。我遵循了创建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
重新生成自动加载文件。现在,覆盖既可以在常规应用程序操作期间工作,也可以在运行测试时工作!
我不会直接回答您的问题,因为我不知道是否有解决方案(不更改Laravels helpers.php
或重命名您的函数)
然而,对于这个常见的用例,框架中有一个解决方案。响应宏
您可以定义宏(这是在服务提供商中完成的)
Response::macro('foo', function($value){
// do some stuff
return Response::make($value);
});
你可以这样使用它:
return response()->foo('bar');
在Laravel 5.5+中,Laravel没有bootstrap/autoload.php
你必须使用这样的包装https://github.com/funkjedi/composer-include-files
如何操作:
首次运行命令
composer require funkjedi/composer-include-files
然后在composer.json 中添加您的助手函数文件路径
{
"extra": {
"include_files": [
"/path/to/file/you/want/to/include",
"/path/to/another/file/you/want/to/include"
]
},
}
然后只做
composer dump-autoload