Lua: передать контекст в loadstring? - PullRequest
7 голосов
/ 14 февраля 2012

Я пытаюсь передать контекст в динамическое выражение, которое я оцениваю на каждой итерации цикла for. Я понимаю, что строка загрузки оценивается только в глобальном контексте, что означает, что локальные переменные недоступны. В моем случае я обхожу это ограничение, преобразовывая локальное в глобальное для оценки строки. Вот что у меня есть:

require 'cosmo'

model = { { player = "Cliff", age = 35, gender = "male" }, { player = "Ally", age = 36, gender = "female" }, { player = "Jasmine", age = 13, gender = "female" }, { player = "Lauren", age = 6.5, gender = "female" } }

values = { eval = function(args)
    output = ''
    condition = assert(loadstring('return ' .. args.condition))
    for _, it in ipairs(model) do
        each = it
        if condition() then
            output = output .. each.player .. ' age: ' .. each.age .. ' ' .. '\n'
        end
    end
    return output
end }
template = "$eval{ condition = 'each.age < 30' }"

result = cosmo.fill(template, values)
print (result)

Моя конечная цель (кроме освоения Lua) состоит в том, чтобы создать XSLT-подобный соблазнительный движок, где я мог бы сделать что-то вроде:

apply_templates{ match = each.age > 30}[[<parent-player>$each.player</parent-player>]]

apply_templates{ match = each.age > 30}[[<child-player>$each.player</child-player>]]

... И генерировать разные выходы. В настоящее время я застрял в своих вышеупомянутых ястребиных способах делиться локальным контекстом через глобальный. У кого-нибудь здесь есть лучшее понимание того, как я буду делать то, что пытаюсь сделать?

Ответы [ 2 ]

10 голосов
/ 14 февраля 2012

Стоит отметить, что setfenv был удален из Lua 5.2 и loadstring устарел . 5.2 довольно нов, поэтому вам не придется беспокоиться об этом какое-то время, но можно написать процедуру загрузки, которая работает для обеих версий:

local function load_code(code, environment)
    if setfenv and loadstring then
        local f = assert(loadstring(code))
        setfenv(f,environment)
        return f
    else
        return assert(load(code, nil,"t",environment))
    end
end

local context = {}
context.string = string
context.table = table
-- etc. add libraries/functions that are safe for your application.
-- see: http://lua-users.org/wiki/SandBoxes
local condition = load_code("return " .. args.condition, context)

Версия 5.2 load обрабатывает как старое поведение loadstring, так и задает среду (контекст, в вашем примере). Версия 5.2 также меняет концепцию окружений , поэтому loadstring может меньше всего беспокоить вас. Тем не менее, стоит подумать, чтобы спасти себя от работы в будущем.

7 голосов
/ 14 февраля 2012

Вы можете изменить контекст функции с помощью setfenv(). Это позволяет вам в основном загружать изолированную программную среду в свою собственную среду. Должно работать что-то вроде следующего:

local context = {}
local condition = assert(loadstring('return ' .. args.condition))
setfenv(condition, context)
for _, it in ipairs(model) do
    context['each'] = it
    if condition() then
        -- ...

Это также не позволит значению условия получить доступ к любым данным, которые вам не нужны, или, что более важно, к изменению данных, которые вам не нужны. Тем не менее, обратите внимание, что вам нужно будет выставить любые привязки верхнего уровня в таблицу context, к которой вы хотите иметь доступ condition (например, если вы хотите, чтобы у нее был доступ к пакету math, вы нужно вставить это в context). В качестве альтернативы, если у вас нет проблем с condition, имеющим глобальный доступ, и вы просто хотите не делать свой локальный объект глобальным, вы можете использовать метатаблицу на context, чтобы она передавала неизвестные индексы на _G:

setmetatable(context, { __index = _G })
...