【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

相关推荐
Ujimatsu19 小时前
虚拟机安装Debian 13.x及其常用软件(2026.4)
linux·运维·ubuntu
千百元19 小时前
zookeeper启不来了
linux·zookeeper·debian
志栋智能19 小时前
超自动化安全:构建智能安全运营的核心引擎
大数据·运维·服务器·数据库·安全·自动化·产品运营
WinterKay20 小时前
【开源】我写了一个轻量级本地数据库浏览工具,支持 MySQL/Redis 只读查询
数据库·mysql·开源
AnalogElectronic20 小时前
linux 测试网络和端口是否连通的命令详解
linux·网络·php
Edward1111111121 小时前
4月28日防火墙问题
linux·运维·服务器
想学后端的前端工程师21 小时前
【补充内外网突然不通的情况】
运维·服务器
面汤放盐21 小时前
何时使用以及何时不应使用微服务:没有银弹
java·运维·云计算
子琦啊1 天前
【算法复习】字符串 | 两个底层直觉,吃透高频题
linux·运维·算法