【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

相关推荐
洛水水4 分钟前
数据库连接池详解
数据库·c++·mysql
WL_Aurora6 分钟前
MySQL慢查询分析与优化实战
mysql·性能优化·慢查询·查询优化
小江的记录本6 分钟前
【Java基础】Java 8-21新特性 :JDK17:密封类、模式匹配、Record类(附《思维导图》+《面试高频考点清单》)
java·数据结构·后端·python·mysql·面试·职场和发展
小江的记录本8 分钟前
【Java基础】集合框架: ArrayList vs LinkedList 核心区别、扩容机制(附《思维导图》+《面试高频考点清单》)
java·数据库·python·mysql·spring·面试·maven
石小千12 分钟前
mysql8全文检索
mysql·全文检索
NineData13 分钟前
NineData 将亮相 XCOPS 智能运维管理人年会 2026 广州站
运维·dba·开发工具·数据库管理工具·ninedata·数据库管理员·数据迁移工具
難釋懷14 分钟前
Nginx介绍与安装
运维·nginx
wujt888815 分钟前
mysql 比较数据库
数据库·mysql·oracle
windawdaysss17 分钟前
使用VMware Workstation Pro安装Ubuntu虚拟机教程
linux·运维·ubuntu
宋浮檀s20 分钟前
Linux后门持久化排查
linux·运维·服务器