我最近的一个项目是基于Laravel5.6开发的APP接口端+APP管理后台,接口中使用到了laravel-passport
组件,接口端的跨域问题是通过网络上非常通用的Laravel跨域中间件解决的,代码如下:
$response = $next($request);
$response->header('Access-Control-Allow-Origin', '*');
$response->header('Access-Control-Allow-Headers', 'Origin, Content-Type, Cookie, Accept, multipart/form-data, application/json, Authorization, X-CSRF-TOKEN');
$response->header('Access-Control-Allow-Methods', 'GET, POST, PATCH, PUT, OPTIONS, DELETE');
$response->header('Access-Control-Allow-Credentials', 'true');
return $response;
自APP上线以来,一直没有出现什么问题,直到项目中引入了laravel-excel
包,在使用laravel-excel
包导出表格时,出现了一个以前没有遇到过的错误:
Call to undefined method Symfony\Component\HttpFoundation\Response::header()
看样子是响应header的时候出了问题,随后查看了一下laravel-passport
的源码才发现,laravel-passport
实际上使用的是Symfony\Component\HttpFoundation\Response
类来响应消息的,而普通的http请求,则是通过Illuminate\Http\Response
类来响应消息的。
随后又发现这两个类设置header
的方式不太一样。
laravel-passport
是通过下面的方式来设置header的:
$response->headers->set($key, $value);
而普通的http请求则是通过另外一种方式来设置header:
$response->header($key, $value);
而中间件中我们通常获取到的Response
对象其实是Illuminate\Http\Response
类的一个实例。
解决方案其实也比较简单,我们只需要在中间件中判断Response
是哪一个类的实例,然后调用不同的设置header
的方法,我们将上面的代码替换为下面的就可以解决问题了。
$response = $next($request);
$IlluminateResponse = 'Illuminate\Http\Response';
$SymfonyResopnse = 'Symfony\Component\HttpFoundation\Response';
$headers = [
'Access-Control-Allow-Origin' => '*',
'Access-Control-Allow-Methods' => 'POST, GET, OPTIONS, PUT, PATCH, DELETE',
'Access-Control-Allow-Headers' => 'Access-Control-Allow-Headers, Origin,Accept, X-Requested-With, Content-Type, Access-Control-Request-Method, Authorization , Access-Control-Request-Headers, X-CSRF-TOKEN'
];
if ($response instanceof $IlluminateResponse) {
foreach ($headers as $key => $value) {
$response->header($key, $value);
}
return $response;
}
if ($response instanceof $SymfonyResopnse) {
foreach ($headers as $key => $value) {
$response->headers->set($key, $value);
}
return $response;
}
return $response;
最后不要忘记导入命名空间。