使用 cgo 构建时如何调试/转储 Go 变量?
Posted
技术标签:
【中文标题】使用 cgo 构建时如何调试/转储 Go 变量?【英文标题】:How to debug/dump Go variable while building with cgo? 【发布时间】:2018-12-28 23:24:12 【问题描述】:我正在尝试使用 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] (type uint32 does 不支持索引)
我不完全确定这意味着什么。这不应该是某种切片,而不是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>
)
【问题讨论】:
我认为您不能为初学者导出未导出的函数。这里没有什么可以转储的,因为你没有编译任何东西,编译器告诉你有一个类型错误:*args.arg_type
is a uint32
。
@JimB 是的,实际上我有一个可以工作的骨架 Go UDF gist.github.com/BrianLeishman/1720c3c3d47b799d5dd913cedea654c5,所以在这种情况下导出功能很好
@JimB 我明白了,我不能转储是有道理的,因为它没有被编译,但是有没有任何方法可以查看它的整个结构变量?
我不确定您的意思——args 的类型为C.UDF_ARGS
,这意味着某处有一个C 文件声明了您可以查看的UDF_ARGS
类型。 (而且 IMO 我实际上会导出这些导出的函数,因为它可能会让大多数 Go 程序员质疑那里到底发生了什么)
@JimB 这是有道理的,我可以更新它。实际上,我通过在 init 函数中调用 fmt.Fprintf(tmpfile, "%#v", args)
并删除我的类型强制使其编译得到了我正在寻找的 Go 语法,所以我们会看到我发现了什么
【参考方案1】:
好吧,尽管我显然不太擅长 Go(尤其是 CGO),但 @JimB 对我的帮助很大,但我有一个 UDF 的工作版本,它简单直接(并且fast) 函数,它从 URL 字符串中提取单个参数并正确解码它,而不是什么(例如 %20 作为空格返回,基本上是您期望它工作的方式)。
这对于纯 C UDF 来说似乎非常棘手,因为我并不真正了解 C(以及我了解其他语言),而且 URL 解析和 URL 参数解码以及本机 MySQL 函数可能会出错慢(也没有真正好的、干净的解码方式),所以对于这类问题,Go 似乎比完美的候选者更好,因为它具有强大的性能,易于编写,以及各种易于使用的内置插件和第三方库。
完整的 UDF 及其安装/使用说明在这里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 变量?的主要内容,如果未能解决你的问题,请参考以下文章