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

pkbketx9  于 2021-06-20  发布在  Mysql
关注(0)|答案(1)|浏览(360)

我正试图用go with cgo编写一个mysql自定义项,其中我有一个基本的功能,但是有一些细节我无法理解,因为我不知道go中的一些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\u 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>
})
wqlqzqxt

wqlqzqxt1#

好吧,@jimb给了我很大的帮助,尽管我对go(尤其是cgo)显然不太在行,但我有一个udf的工作版本,它是一个简单而直接(快速)的函数,可以从url字符串中提取一个参数,并正确地解码它,而不是什么(例如%20作为空格返回,基本上你会期望它如何工作)。
对于纯c udf来说,这似乎非常棘手,因为我不太懂c(也不太懂其他语言),而且url解析和url参数解码可能会出现很多问题,而且本机mysql函数速度很慢(而且也没有一种好的、干净的解码方法),因此,go似乎是解决这类问题的最佳选择,因为它具有强大的性能、易于编写以及各种易于使用的内置库和第三方库。
这里有完整的自定义项及其安装/使用说明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\u type是一个数组,就像在c中一样,但实际上是一个数字)之后,我需要将该数字引用的数据(指向c数组开头的指针)转换为一个go数组,以便设置它的值。

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

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

相关问题