php - 在代码中使用 eval(),您将如何避免它?

标签 php eval unset variable-variables

在我的代码中,我想尝试获取 $_POST 或 $_GET 变量,并在请求一次后立即取消设置。

这个函数返回使用的方法,够简单了。不幸的是,我不能简单地直接或通过引用返回 $_POST 或 $_GET 变量,因为我无法取消设置它们(通过引用取消设置不起作用)。所以,我返回变量名:

class Input {
  protected function interface_get($method): string {
    switch ($method) {
      case 'post':
        return '_POST';
        break;
      case 'get':
        return '_GET';
        break;
      case 'ajax':
        // Not supported yet.
        return null;
        break;
      default:
        return null; // returns '_GET' or '_POST'
    }
  }
  public function fetch_eval(string $method, ?string $request = null) { // ...fetch('post', 'username')
    if ($request !== null) {
      $request = '[\'' . $request . '\']'; // "['username']"
    }
    $request = $this->interface_get($method) . $request; #"$_POST['username']"
    eval('$return = $' . $request . ';'); #$return = $_POST['username'];
    eval('unset($' . $request . ');'); #unset($_POST['username']);
    return $return;
  }
  public function fetch_varvar(string $method, ?string $request = null) {
    $interface = $this->interface_get($method); #$interface = '_POST';
    if ($request === null) {
      $return = (${$interface});
      unset(${$interface});
    } else {
      $return = ${$interface}; #"Notice:  Undefined variable: _POST in [...]"
      $return = ${$interface}[$request]; #"Warning:  Illegal string offset 'email' in [...]"
      unset($interface[$request]);
    }
    return $result;
}
}
// set $_POST = ['email'=>'spam@me'];
$in = new Input();
echo $in->fetch_eval('post', 'email'); #'spam@me'
// $_POST = [];

// set $_POST = ['email'=>'spam@me']; again
echo $in->fetch_varvar('post', 'email'); #'Notice:  Undefined variable: _POST [...]'

有趣的部分是处理输出。这是我的 fetch 函数,我认为这是最简单但最脏的方法:

我尝试使用可变变量,它在测试脚本中有效:

// $_POST = [0 => 5, 'bob' => 5];
$e = '_POST';
$var = 'bob';
echo ${$e}[$var]; #5
unset(${$e}[$var]);
echo ${$e}[$var ]; #NOTICE Undefined index: bob on line number 22
// Works as expected.

但它在我的脚本中不起作用(Undefined variable: _POST [...]。)。我想可能是在一个类(class)的原因,但无论如何我无法解决它。更重要的是,如果您将这些函数放在类之外并删除 $this->-es,它们就可以工作了!但不是在类里面。

如果有人能告诉我为什么我后面的代码不起作用,我将不胜感激。但无论如何,你会说使用 eval() 是合理的吗?我知道有些人无论如何都会避免它。显然,在这种情况下,它打开了一个相当大的漏洞,所以与其净化和担心,我宁愿完全避免它。

我想将 interface_get() 保留为一个单独的函数,但如果需要,我想我也可以在 fetch() 中复制它。

非常感谢您。

最佳答案

您可以使用 $GLOBALS[$interface] 获取数据。

假设 $_POST$_GET 始终被定义,但是,您可以检查之前是否定义了键以获取它并取消设置它,以避免警告。

public function fetch(string $method, ?string $request = null) {
    $interface = $this->interface_get($method); #$interface = '_POST';
    if ($request === null) {
        $result = $GLOBALS[$interface];
        unset($GLOBALS[$interface]);
        return $result;
    }
    if (isset($GLOBALS[$interface][$request])) {
        $result = $GLOBALS[$interface][$request];
        unset($GLOBALS[$interface][$request]);
        return $result;
    }
    return null;
}

关于php - 在代码中使用 eval(),您将如何避免它?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/49267080/

相关文章:

javascript - 评估后的返回值

php - 使用 verifySignature PHP api 返回 "InvalidSignature"的 Amazon FPS 返回 URL 验证

vba - 如何在 VBA 中使用方括号进行字符串计算?

php - 如何使用sqlsrv和 "?"样式参数在php中执行存储过程

javascript - 在 javascript 中替换 eval() 的正确方法是什么?

mongodb - 如何从子集合中删除列

php - 通过引用取消设置数组的元素

php - 如果我们想在 __destruct 中使用复杂的代码,我们是否应该尽快取消设置一个类?

php - 获取错误外键违规

php - 如何跟踪 php-fpm 使用过多 CPU 的原因