Redis:将值中的键引用展开为它们的值

时间:2014-11-11 08:39:21

标签: lua redis

我对redis相当新鲜。我有一个CMS,在这样的结构中将JSON输出到redis:

partial:1 => {id: 1, title: 'Foo1', ...}
partial:2 => {id: 2, title: 'Foo2', ...}
partial:3 => {id: 3, title: 'Foo3', ...}

page:home => {
   id: 'Homepage',
   teaser1: 'partial:1',
   teaser2: 'partial:2',
   teaser3: {type: Slider, content: 'partial:3'}
}

因此JSON可以包含符合其结构中某个命名方案的其他redis-key。我想要的是一种查询redis的方法,这样当我得到page:home - 键时,对json中其他键的引用得到了扩展'各自的价值观,如下:

{
   id: 'Homepage',
   teaser1: {id: 1, title: 'Foo1', ...},
   teaser2: {id: 2, title: 'Foo2', ...},
   teaser3: {type: Slider, content: {id: 3, title: 'Foo3', ...}
}

这可能吗?如何实现?

2 个答案:

答案 0 :(得分:2)

是。这是很有可能。这是一种有效的方法:

  1. 创建一个函数将redis哈希值(例如:partial:*,page:home)转换为lua表: hgetall

  2. 创建一个函数来检查给定的redis hashname是否符合您的命名方案,如果是,请使用 hgetall 将其转换为lua表;否则返回相同的值: evaluate_hash

  3. 创建一个函数来评估和转换给定哈希名称的所有属性并将其作为json返回: expand
  4. 这是一个简单的实现:

    - 脚本:redis_expand_as_json.lua

    --Function to Convert a given hash name (e.g: partial:1..n, page:home) to a lua table
    local function hgetall(a)local b=redis.call('HGETALL',a)local c={}local d;for e,f in ipairs(b)do if e%2==1 then d=f else c[d]=f end end;return c end
    
    --Function to check if the given value conforms with a naming scheme, 
    -- if so convert it to a lua table; otherwise return the values as is.
    local function evaluate_hash(value)
      local pattern = "partial:%d+"
      if string.match(value, pattern) then          
        return hgetall(value)
      else
        return value
      end 
    end
    
    --Function to convert a given hash_name to a lua table, 
    -- iterate all elements and convert them to lua table if necessary
    -- returns the table as a json object
    local function expand(hash_name)
      local obj_table = hgetall(hash_name)
      for k, val in pairs(obj_table) do
        obj_table[k] = evaluate_hash(val)
      end 
      return cjson.encode(obj_table)
    end 
    
    local page = KEYS[1]
    local json_result = expand(page) or {}
    
    redis.log(redis.LOG_NOTICE, tostring(json_result))
    return json_result
    

    在控制台中测试:

      

    redis-cli -c hmset“partial:1”id 1 title foo1

      

    redis-cli -c hmset“partial:2”id 2 title foo2

      

    redis-cli -c hmset'page:home'id'主页'teaser1'部分:1'teaser2'partial:2'

      

    redis-cli EVAL“$(cat redis_expand_as_json.lua)”1'页:主页'

    {"teaser1":{"id":"1","title":"foo1"},"teaser2":{"id":"2","title":"foo2"},"id":"Homepage"}
    

答案 1 :(得分:0)

我不认识Redis,但也许这适合你:

local T=[[
partial:1 => {id: 1, title: 'Foo1', ...}
partial:2 => {id: 2, title: 'Foo2', ...}
partial:3 => {id: 3, title: 'Foo3', ...}

page:home => {
   id: 'Homepage',
   teaser1: 'partial:1',
   teaser2: 'partial:2',
   teaser3: {type: Slider, content: 'partial:3'}
}
]]

local D={}
for k,v in T:gmatch("(%w+:%w+)%s*=>%s*(%b{})") do
    D[k]=v
end

print((T:gsub("'(.-)'",D)))