• Lua


    【摘要】 在安全领域,lua编程语言因为其小巧在众多工具上都作为插件开发语言,常见的有openresty,nmap等。因此笔者将会开辟一个Lua相关的系列文章,主要记录工作过程中一些领悟或者是一些踩过的坑,希望能够借此平台帮助到读者们。

    0x00 背景

    最近在写一段nginx+redis的代码,主要基于openresty,其中使用到了lua-resty-redis库。我平时写代码都比较小心,针对外部输入的值一般都会进行异常判断,大概的代码如下:

    local redis = require "redis"
    local cjson = require "cjson"
    --[[省略部分代码]]
    local ok, err = redis:get("key")
    if not ok then
        ngx.log(ngx.ERR, '[ERROR]:', err)
        return
    end
    local data = cjson.decode(ok)

    在decode这里出现了错误提示,但是ok并没有为空或者nil,不然代码是走不到这里来。
    发现问题后,我们就在前面打印一下ok数据的类型吧,大概的代码如下:

    ngx.log(ngx.ERR, 'ok type: ', type(ok))
    if not ok then
        -- TODO
    end

    这个时候我们得到的结果是userdata,这个东西算是一种复杂结构体,一般都是跨语言产生的,比如ffi.C这些。当时我的思路大概也是这样,肯定redis存放的数据是二进制的,但是呀,存放什么数据都是我自己控制的,不可能有什么畸形数据,因此这一点也排除了。最后在自己查看中发现,其实就是这个key不存在。

    0x01 分析

    既然原因找到了,我们就去看看为什么会这样,主要通过阅读lua-resty-redis的源码:

    local function _read_reply(self, sock)
        local line, err = sock:receive()
        if not line then
            if err == "timeout" and not rawget(self, "_subscribed") then
                sock:close()
            end
            return nil, err
        end
    
        local prefix = byte(line)
    
        if prefix == 36 then    -- char '$'
            -- print("bulk reply")
    
            local size = tonumber(sub(line, 2))
            if size < 0 then
                return null
            end
    
            local data, err = sock:receive(size)
            if not data then
                if err == "timeout" then
                    sock:close()
                end
                return nil, err
            end
    
            local dummy, err = sock:receive(2) -- ignore CRLF
            if not dummy then
                return nil, err
            end
    
            return data
    
        elseif prefix == 43 then    -- char '+'
            -- print("status reply")
    
            return sub(line, 2)
    
        elseif prefix == 42 then -- char '*'
            local n = tonumber(sub(line, 2))
    
            -- print("multi-bulk reply: ", n)
            if n < 0 then
                return null
            end
    
            local vals = new_tab(n, 0)
            local nvals = 0
            for i = 1, n do
                local res, err = _read_reply(self, sock)
                if res then
                    nvals = nvals + 1
                    vals[nvals] = res
    
                elseif res == nil then
                    return nil, err
    
                else
                    -- be a valid redis error value
                    nvals = nvals + 1
                    vals[nvals] = {false, err}
                end
            end
    
            return vals
    
        elseif prefix == 58 then    -- char ':'
            -- print("integer reply")
            return tonumber(sub(line, 2))
    
        elseif prefix == 45 then    -- char '-'
            -- print("error reply: ", n)
    
            return false, sub(line, 2)
    
        else
            -- when `line` is an empty string, `prefix` will be equal to nil.
            return nil, "unknown prefix: "" .. tostring(prefix) .. """
        end
    end

    从上面的源码可以看到,在读取redis服务器返回数据的时候,如果某些格式不正确,比如数据长度的字节小于0这样的异常情况,函数就会返回null,注意是null不是nil
    这个null的定义来自ngx.null,这个东西可以追溯到其官方文档lua-nginx-module. 

    The ngx.null constant is a NULL light userdata usually used to represent nil values in Lua tables etc and is similar to the lua-cjson library’s cjson.null constant.

    从上面描述看,ngx.null就是一个代表null的userdata结构,类似一个自定义的类,但是没有什么具体含义,同时文档里面也提到了类似的值还有cjson.null,以后小心被坑。

    0x02 扩展

    同时文档中还提到了,使用ngx.log对几个空值进行字符串打印的时候

    • nil会显示成“nil”,

    • 逻辑值会显示成“true”或者“false”,

    • ngx.null会被显示成“null”。

    来源:华为云社区  作者:HuangJacky

  • 相关阅读:
    学习Mybatis中的一对多表关联
    学习Mybatis中的一对一表关联
    学习Mybatis中的动态sql
    学习Mybatis中的约定大于配置、数据库配置优化、定义别名、类型处理器、resultMap和parameterType
    第八周进度
    构建之法阅读笔记07
    正则表达式
    梦断代码之阅读笔记02
    顶会热词统计
    本周进度
  • 原文地址:https://www.cnblogs.com/2020-zhy-jzoj/p/13165783.html
Copyright © 2020-2023  润新知