Как отладить / сбросить переменную Go при сборке с помощью cgo?
Я пытаюсь написать MySQL UDF на Go с cgo, в котором у меня есть базовое функционирование, но есть маленькие кусочки, которые я не могу понять, потому что я понятия не имею, каковы некоторые из переменных 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, чтобы выгрузить содержимое переменной в файл tmp внутри функции init (закомментировав строки, которые сделали его не компилируемым), и я получил это
(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 ответ
Хорошо, такая огромная помощь с @JimB, который застрял со мной, хотя я явно менее искусен в Go (и особенно в CGO), но у меня есть рабочая версия моего UDF, которая является простой и понятной (и быстрой) функцией он извлекает один параметр из строки URL и правильно его декодирует, а что нет (например,%20 возвращается как пробел, в основном, как вы ожидаете, что он будет работать).
Это выглядело невероятно сложно с чистым UDF на языке C, потому что я не очень хорошо знаю C (как и другие языки), и есть много вещей, которые могут пойти не так с разбором URL и декодированием параметров URL, а собственные функции MySQL работают медленно (и нет действительно хорошего, чистого способа сделать декодирование либо), так что 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