【linux C】在mysql中增加自定义的C动态库

前言

可能这是在比较极端的需求场景下才会有的解决方案:有个QT应用需要激活码(需要进行DES加密运算)才可使用,应用部署到生成环境之后,销售希望能在web页面之间操作获取。公司已有对外开放的web管理端,并且限制优先用mysql方式解决,避免使用socket方式。

我已经用C编写DES的加解密相关功能,QT已经用此功能进行激活校验,既然不让用socket通信,那么就看看mysql是否支持自定义的C库(SO动态库)。

答案当然是支持,官方名称为MYSQL UDF(方便阅读者查找),本篇的操作步骤是支持国产系统(已在欧拉系统下验证OK)。

操作步骤

1、安装mysql开发环境

(1)centos系统

复制代码
# 安装 MySQL 客户端开发库和头文件
sudo yum install -y mysql-devel

(2)欧拉系统

复制代码
sudo dnf install -y mariadb-connector-c-devel

安装完成后,在/usr/include/下可以看到mysql目录,编译mysql udf时,需要引用mysql头文件

2、创建UDF文件

比如我的DES加解密代码(文件名称为mydes.c)提供了以下两个对外可调用的加解密函数

复制代码
void DesJiaMi(const char* mingwen, char* des_code);// DES 加密函数
void DesJieMi(const char* des_code, char* des_miwen);// DES 解密函数

需要手动创建一个mydes_udf.c文件(名称不限定必须用_udf后缀,只是方便用于编译),下面是mydes_udf.c代码

复制代码
#include <mysql/mysql.h>
#include <string.h>

// 声明来自 mydes.c 的函数
extern void DesJiaMi(const char* mingwen, char* des_code);

// ========================
// 加密函数:my_jiami
// ========================

my_bool my_jiami_init(UDF_INIT *initid, UDF_ARGS *args, char *message) {
    if (args->arg_count != 1 || args->arg_type[0] != STRING_RESULT) {
        strcpy(message, "ymore_jiami(plain_text): requires one string argument");
        return 1;
    }
    initid->maybe_null = 1;
    initid->max_length = 2048; // 支持长文本加密(每8字节→16 hex)
    return 0;
}

// 注意:无需 deinit,因为无动态内存

char* my_jiami(UDF_INIT *initid, UDF_ARGS *args, char *result,
                  unsigned long *length, char *is_null, char *error) {
    if (args->args[0] == NULL) {
        *is_null = 1;
        return NULL;
    }

    // 调用加密函数,直接写入 result
    DesJiaMi((const char*)args->args[0], result);

    // 计算实际长度(直到 \0)
    *length = strlen(result);
    
    // 确保以 \0 结尾(防御性)
    result[*length] = '\0';

    // 如果结果为空,设为 NULL
    if (*length == 0) {
        *is_null = 1;
        return NULL;
    }

    return result;
}

解密函数是一样的方法来造。

需要注意的是,在欧拉系统上编译时,会报以下错误

复制代码
error: unknown type name 'my_bool': did you mean 'Bool'?

只需要在mydes_udf.c文件的开头位置加上以下3行代码,再编译一次即可。

复制代码
#ifndef my_bool
#define my_bool char
#endif

3、编译UDF动态库

编译命令如下:

复制代码
gcc -shared -fPIC -o mydes.so mydes.c mydes_udf.c -I/usr/include/mysql -std=c99

会生成一个mydes.so动态库

4、动态库拷贝到mysql扩展目录中

把上面生成的mydes.so拷贝到l扩展目录中

复制代码
sudo cp mydes.so /usr/lib64/mysql/plugin/
sudo chmod 755 /usr/lib64/mysql/plugin/mydes.so

5、在数据库中创建加密函数

复制代码
CREATE FUNCTION my_jiami RETURNS STRING SONAME 'mydes.so';

测试看效果:

复制代码
SELECT my_jiami('helloworld');

这里要注意下,如果客户端版本比较低时,上面的测试命令返回的是BLOB,高版本是能拿到字符串的,当然也可以用以下强制方式获取到字符串

复制代码
SELECT CAST(my_jiami('helloworld') AS CHAR) AS forced_text;

如果是在调试,需要覆盖mydes.so,覆盖之前最好把自定义函数删除掉

复制代码
DROP FUNCTION IF EXISTS my_jiami;

覆盖之后,必须重启mysql服务

复制代码
sudo systemctl restart mysqld

然后再创建加密函数my_jiami

相关推荐
十日十行10 小时前
Linux和window共享文件夹
linux
木心月转码ing17 小时前
WSL+Cpp开发环境配置
linux
Turnip12022 天前
深度解析:为什么简单的数据库"写操作"会在 MySQL 中卡住?
后端·mysql
蝎子莱莱爱打怪2 天前
Centos7中一键安装K8s集群以及Rancher安装记录
运维·后端·kubernetes
崔小汤呀2 天前
最全的docker安装笔记,包含CentOS和Ubuntu
linux·后端
何中应2 天前
vi编辑器使用
linux·后端·操作系统
何中应2 天前
Linux进程无法被kill
linux·后端·操作系统
何中应2 天前
rm-rf /命令操作介绍
linux·后端·操作系统
何中应2 天前
Linux常用命令
linux·操作系统
葛立国2 天前
从 / 和 /dev 说起:Linux 文件系统与挂载点一文理清
linux