编辑
好的,由于似乎有很多混乱,我将把问题简化一点。您可以尝试回答下面的原始问题,或者您可以解决此版本并忽略该行下方的所有内容。
我的目标是采用任意表达式并在极其受限的环境中对其进行评估。此环境将仅包含具有以下类型值的变量:
- 数值向量
- 采用一个或多个数值向量并返回数值向量的纯函数(即算术运算符)
此外,表达式必须能够使用任何文字,例如数字和字符串常量(但不能使用数字或字符串向量,因为它们需要c
)。我想在这个环境中评估表达式,并确保表达式无法访问环境之外的任何内容,这样我就可以确定评估表达式不会有安全风险。那么,在下面的代码中,你能用一个在评估时会做一些淘气的字符串来填充空白吗?“一些顽皮的东西”被定义为在屏幕上打印一些东西,访问变量的值secret
,执行任何 shell 命令(最好是产生输出的命令),或者任何其他对你来说似乎很顽皮的东西(证明你的选择是正确的)。
a <- 1
b <- 2
x <- 5
y <- 1:10
z <- -1
## Give secret a random value so that you can't just compute it from
## the above variables
secret <- rnorm(5)
allowed.variables <- c(
## Numeric variables
"a", "b", "x", "y", "z",
## Arithmetic operators
"(", "+", "-", "/", "*", "^", "sqrt", "log", "log10", "log2", "exp", "log1p")
restricted.environment <- Map(get, allowed.variables)
## Example naughty expressions that my method successfully guards
## against
expr1 <- "secret"
expr2 <- "cat('Printing something with cat\n')"
expr3 <- "system('echo Printing something via shell command')"
arbitrary.expression <- "?????????" # Your naughty string constant here
eval(parse(text=arbitrary.expression), envir=restricted.environment, enclos=emptyenv())
原始问题
我正在编写一些代码来将算术表达式作为用户输入并对其进行评估。我有一组指定的可以使用的变量,以及算术函数的白名单(+
、-
、*
、/
、^
等)。有什么方法可以评估表达式,以便只有这些变量和运算符在范围内,以避免任意代码注入的任何可能性?我有一些我认为可行的东西,但我不想实际使用它,除非我确定它确实是防弹的:
## Shortcut for parse-then-eval pattern
evalparse <- function(expr, ...) eval(parse(text=expr), ...)
# I control these
arithmetic.operators <- Map(get, c("(", "+", "-", "/", "*", "^", "sqrt", "log", "log10", "log2", "exp", "log1p"))
vars <- list(a=1, b=2)
safe.envir <- c(vars, arithmetic.operators)
# Assume that these expressions are user input, e.g. from a web form.
nice.expr <- "a + b"
naughty.expr <- paste("cat('ARBITRARY R CODE INJECTION\n'); system('echo ARBITRARY SHELL COMMAND INJECTION');", nice.expr)
## NOT SAFE! Lookups outside env still possible.
evalparse(nice.expr, envir=safe.envir)
evalparse(naughty.expr, envir=safe.envir)
## Is this safe?
evalparse(nice.expr, envir=safe.envir, enclos=emptyenv())
evalparse(naughty.expr, envir=safe.envir, enclos=emptyenv())
如果你在 R 中运行上面的代码,你会看到我们第一次 evalnaughty.expr
时,它成功地执行了它的有效载荷。但是,第二次,使用enclose=emptyenv()
,评估只能访问变量a
,b
和指定的算术运算符,因此有效负载无法执行。
那么,这种方法(即eval(..., envir=safeenv, enclos=emptyenv())
)实际上可以在接受实际用户输入的生产中使用,还是我错过了一些在受限环境中仍然执行任意代码的偷偷摸摸的方法?