如何在使用 cgo 构建时 debug/dump Go 变量?
How to debug/dump Go variable while building with cgo?
我正在尝试用 cgo 在 Go 中编写一个 MySQL UDF,其中我有一个基本的功能,但是有一些我无法弄清楚的点点滴滴,因为我不知道就 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 中的另一个函数做 same/similar 事情,就像这样
//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: invalid operation: (*args.arg_type)[0] (type uint32 does
not support indexing)
我不太确定那是什么意思。这不应该是某种切片,而不是 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>
})
好吧,@JimB 给了我很大的帮助,尽管我显然不太擅长 Go(尤其是 CGO),但我有一个 UDF 的工作版本,这是一个简单直接的方法(和快速)函数,从 URL 字符串中提取单个参数并正确解码它(例如 %20 作为 space 返回,基本上是您期望它如何工作)。
对于纯 C UDF,这似乎非常棘手,因为我并不真正了解 C(以及我对其他语言的了解),并且 URL 解析和 URL 参数解码,而原生 MySQL 函数是 slow(而且也没有真正好的、干净的解码方法),所以 Go 似乎更好-这类问题的完美候选者,具有强大的性能、易于编写以及各种易于使用的内置插件和第三方库。
完整的 UDF 及其 installation/usage 说明在这里 https://github.com/StirlingMarketingGroup/mysql-get-url-param/blob/master/main.go
第一个问题是调试输出。我通过 Fprintf
ing 到 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
我正在尝试用 cgo 在 Go 中编写一个 MySQL UDF,其中我有一个基本的功能,但是有一些我无法弄清楚的点点滴滴,因为我不知道就 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 中的另一个函数做 same/similar 事情,就像这样
//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: invalid operation: (*args.arg_type)[0] (type uint32 does not support indexing)
我不太确定那是什么意思。这不应该是某种切片,而不是 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>
})
好吧,@JimB 给了我很大的帮助,尽管我显然不太擅长 Go(尤其是 CGO),但我有一个 UDF 的工作版本,这是一个简单直接的方法(和快速)函数,从 URL 字符串中提取单个参数并正确解码它(例如 %20 作为 space 返回,基本上是您期望它如何工作)。
对于纯 C UDF,这似乎非常棘手,因为我并不真正了解 C(以及我对其他语言的了解),并且 URL 解析和 URL 参数解码,而原生 MySQL 函数是 slow(而且也没有真正好的、干净的解码方法),所以 Go 似乎更好-这类问题的完美候选者,具有强大的性能、易于编写以及各种易于使用的内置插件和第三方库。
完整的 UDF 及其 installation/usage 说明在这里 https://github.com/StirlingMarketingGroup/mysql-get-url-param/blob/master/main.go
第一个问题是调试输出。我通过 Fprintf
ing 到 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