PHP:laravel中间件和控制器的请求参数传递与获取

本文涉及的产品
服务治理 MSE Sentinel/OpenSergo,Agent数量 不受限
简介: PHP:laravel中间件和控制器的请求参数传递与获取

image.png

目录

接口开发中,通常我们需要在中间件里边做一些全局性的前置判断,获取请求中的公共参数,然后传递给控制器

参考网络上的大部分文章,整理出以下传递参数的方式,并深入研究,作出一些思考

1、中间件和控制器测试

中间件

<?php
namespace App\Http\Middleware;
use Illuminate\Http\Request;
/**
 * 测试中间件
 */
class TestMiddleware
{
    public function handle(Request $request, \Closure $next)
    {
        // 设置参数
        $request->attributes->set('name', 'Tom');
        return $next($request);
    }
}

控制器中取值

<?php
namespace App\Http\Controllers;
use Illuminate\Http\Request;
class TestController extends Controller
{
    public function index(Request $request): array
    {
        return [
            'name:input' => $request->input('name'),
            'name:get'   => $request->get('name')
        ];
    }
}

请求提交参数

{
    "name":  "Jack"
}

返回参数

{
  "data": {
    "name:input": "Jack",
    "name:get": "Tom"
  },
  "msg": "success",
  "code": 0
}

2、安全隐患

目前看,都挺好的

如果中间件中没有设置参数,就会取到用户传递过来的参数

public function handle(Request $request, \Closure $next)
    {
        // $request->attributes->set('name', 'Tom');
        return $next($request);
    }

如果用于安全性较高的场景,有点不适用

{
  "data": {
    "name:input": "Jack",
    "name:get": "Jack"
  },
  "msg": "success",
  "code": 0
}

假设,需要一个当前登录用户的id,假设是变量 current_login_user_id

如果我们没有加中间件,前端中接口中传递了这个参数,那么我们的代码将会出现安全隐患。

3、支持的传参方式

我们看下支持的传参方式

中间件

class TestMiddleware
{
    public function handle(Request $request, \Closure $next)
    {
        $request->name = 'prototype_name';
        $request->attributes->set('name', 'attr_name');
        return $next($request);
    }
}

控制器

public function index(Request $request): array
    {
        return [
            'name:prototype'  => $request->name,
            'name:input' => $request->input('name'),
            'name:query' => $request->query('name'),
            'name:attr'   => $request->get('name')
        ];
    }

请求

POST {{api_url}}/test?name=query_name
Content-Type: application/json; charset=utf-8
{
    "name":  "post_name"
}
{
    "name:prototype": "prototype_name",
    "name:input": "post_name",
    "name:query": "query_name",
    "name:attr": "attr_name"
}

如果取消中间件,发现数据并不是所期待的那样,取到null值

{
    "name:prototype": "post_name",
    "name:input": "post_name",
    "name:query": "query_name",
    "name:attr": "query_name"
}

4、总结

综上,可以采用如下方式传递中间件参数到控制器是可行的

// 中间件设置值 设置为null,就算请求参数中携带了name, 也能取到null值
$request->attributes->set('name', null);
// 控制器取值
$request->get('name')

看下get的实现代码

public function get(string $key, mixed $default = null): mixed
    {
        if ($this !== $result = $this->attributes->get($key, $this)) {
            return $result;
        }
        if ($this->query->has($key)) {
            return $this->query->all()[$key];
        }
        if ($this->request->has($key)) {
            return $this->request->all()[$key];
        }
        return $default;
}
public function get(string $key, mixed $default = null): mixed
{
        return \array_key_exists($key, $this->parameters) ? $this->parameters[$key] : $default;
}

array_key_exists 意味着,如果设置了值,就会直接返回,不再往下查找其他值

5、一种更为安全的做法

一种更为安全的做法,是建立一个枚举变量,来替代字符串变量

const key = '192b9bdd22ab9ed4d12e236c78afcb9a393ec15f71bbf5dc987d54727823bcbf'
// 就算没有设置值,也不会错误的从请求参数中获取值
$value = input->get(key)

利用Python,可以很容易生成这种很长的数字

Python编程:使用os.urandom生成Flask的秘钥SECRET_KEY


相关文章
|
8月前
|
PHP
PHP - Laravel 表单验证错误切换为中文
PHP - Laravel 表单验证错误切换为中文
112 0
|
8月前
|
JavaScript 前端开发 PHP
PHP - Laravel 视图模板(blade.php) @ 原始形态输出(Vue 与 PHP 混编)
PHP - Laravel 视图模板(blade.php) @ 原始形态输出(Vue 与 PHP 混编)
95 0
|
8月前
|
安全 PHP
PHP - Laravel 表单验证(验证规则与使用 $this->validate()、Validator::make()、Requests)
PHP - Laravel 表单验证(验证规则与使用 $this->validate()、Validator::make()、Requests)
151 0
|
8月前
|
PHP
PHP - Laravel Blade模板注释 {{-- 注释 --}} 与 <!-- 注释 --> 的区别
PHP - Laravel Blade模板注释 {{-- 注释 --}} 与 <!-- 注释 --> 的区别
62 0
|
8月前
|
PHP
PHP - Laravel 三元运算
PHP - Laravel 三元运算
48 0
|
4月前
|
前端开发 PHP
【PHP学习】—get请求传递参数(五)
【PHP学习】—get请求传递参数(五)
|
6月前
|
监控 安全 BI
一套医疗安全不良事件管理系统源码(PHP+ vue2+element+ laravel)
不良事件报告管理系统按照不良事件的管理部门不同,分为医疗不良事件、护理不良事件、药品不良反应事件、院内感染事件、输血不良反应事件、医疗器械不良事件、医技相关不良事件、安保后勤不良事件、信息不良事件、费用窗口服务不良事件共10大类事件。
一套医疗安全不良事件管理系统源码(PHP+ vue2+element+ laravel)
|
7月前
|
Ubuntu PHP Windows
在安装PHP时出现了冲突请求的问题
在安装PHP时出现了冲突请求的问题
83 1