如何在使用cgo进行构建时调试/转储Go变量?

时间:2018-07-20 15:41:26

标签: mysql go cgo mysql-udf

我正在尝试使用cgo在Go中编写一个MySQL UDF,在其中我有一个基本的功能,但是由于我不知道其中的一些C变量,我无法弄清一些点点滴滴就围棋而言。

这是我用C语言编写的一个示例,该示例将MySQL参数之一的类型强制为int

my_bool unhex_sha3_init(UDF_INIT *initid, UDF_ARGS *args, char *message) {
    if (args->arg_count != 2) {
        strcpy(message, "`unhex_sha3`() requires 2 parameters: the message part, and the bits");
        return 1;
    }

    args->arg_type[1] = INT_RESULT;

    initid->maybe_null = 1; //can return null

    return 0;
}

那行得通,但是我尝试像这样用Go中的另一个功能做相同/相似的事情

//export get_url_param_init
func get_url_param_init(initid *C.UDF_INIT, args *C.UDF_ARGS, message *C.char) C.my_bool {
    if args.arg_count != 2 {
        message = C.CString("`get_url_param` require 2 parameters: the URL string and the param name")
        return 1
    }

    (*args.arg_type)[0] = C.STRING_RESULT
    (*args.arg_type)[1] = C.STRING_RESULT

    initid.maybe_null = 1

    return 0
}

出现此构建错误

  

./ main.go:24:无效操作:(* args.arg_type)[0](类型uint32可以   不支持索引)

我不确定那是什么意思。这不应该是某种切片,而不是uint32吗?

这是超级有用的方法,它可以某种方式将args结构以某种方式转储到某个地方(甚至在Go语法中也可以作为一个超级加号),以便我可以知道我正在使用什么。


好吧,我用spew将变量内容转储到init函数内部的tmp文件中(注释掉了使其无法编译的行),我明白了

(string) (len=3) "%#v"
(*main._Ctype_struct_st_udf_args)(0x7ff318006af8)({
 arg_count: (main._Ctype_uint) 2,
 _: ([4]uint8) (len=4 cap=4) {
  00000000  00 00 00 00                                       |....|
 },
 arg_type: (*uint32)(0x7ff318006d18)(0),
 args: (**main._Ctype_char)(0x7ff318006d20->0x7ff3180251b0)(0),
 lengths: (*main._Ctype_ulong)(0x7ff318006d30)(0),
 maybe_null: (*main._Ctype_char)(0x7ff318006d40)(0),
 attributes: (**main._Ctype_char)(0x7ff318006d58->0x7ff318006b88)(39),
 attribute_lengths: (*main._Ctype_ulong)(0x7ff318006d68)(2),
 extension: (unsafe.Pointer) <nil>
})

1 个答案:

答案 0 :(得分:0)

@JimB帮了我很大的忙,尽管我显然不擅长Go(尤其是CGO),但我仍然坚持使用我的UDF,这是一个简单而直接的工作(并且快速)功能,可以从URL字符串中提取单个参数并对其进行正确解码,而不进行解码(例如,%20返回为空格,基本上是您希望其工作的方式)。

对于纯C UDF来说,这似乎非常棘手,因为我不太了解C(以及其他语言),而且URL解析和URL参数解码以及本机MySQL函数可能会出错是 slow (而且也没有真正好的,干净的方法来进行解码),因此Go似乎是解决此类问题的理想之选,具有强大的性能,易于编写,以及各种易于使用的内置插件和第三方库。

完整的UDF及其安装/使用说明位于https://github.com/StirlingMarketingGroup/mysql-get-url-param/blob/master/main.go


第一个问题是调试输出。然后,我通过Fprintf而不是标准输出到tmp文件来做到这一点,以便可以检查文件以查看变量转储。

t, err := ioutil.TempFile(os.TempDir(), "get-url-param")

fmt.Fprintf(t, "%#v\n", args.arg_type)

然后,在获得输出之后(我期望args.arg_type是一个数组,就像在C中一样,但是是一个数字),我需要转换该数字所引用的数据(指向开头的指针) C数组)转到Go数组,以便我可以设置其值。

argsTypes := *(*[2]uint32)(unsafe.Pointer(args.arg_type))

argsTypes[0] = C.STRING_RESULT
argsTypes[1] = C.STRING_RESULT