不停服更新二进制文件
2018-12-04 07:13:51来源:博客园 阅读 ()
原文
http://www.zhaoch.top/操作系统/linux/不停服更新二进制文件.html
不停服更新二进制文件
www.zhaoch.top > 操作系统 > linux
虽然目前分布式架构和keepalived等工具的存在,对于某些特殊的程序,仍然需要不停服更新二进制文件。 这里参考nginx的实现介绍下如何实现这个功能的
痛点
- 已有的链接不中断,直至这个客户端完成业务处理
- 使用新的程序文件处理新的请求
- 新老进程同时存在还要监听相同的端口
思路
- 因为要监听相同端口所以一般都是父子进程
- 常用的fork不能更新二进制文件,需要再通过exec重新加载文件
- 监听相同端口可以通过把句柄作为参数或者环境变量传递给exec族函数继续使用
参考
ngx_exec_new_binary
过程
- 更新可执行程序
- 向主进程发送信号(例如:USR2),立即设置全局变量
- 事件处理循环中老进程检查到全局变量后,不再accept新的连接,已有的连接正常处理
- fork一个子进程,子进程通过exec族函数更新二进制,将listen的句柄通过参数或者环境变量
- 子进程开始accept,新接入的连接由新程序处理
- 老程序持续运行,直至所有的连接都完成业务(可设置超时时间),退出
- 此时只有子程序在运行
示例代码
编辑app_new.cpp 与 app_old.cpp两个文件,app_old.cpp内容如下
#include <stdlib.h> #include <iostream> #include <fcntl.h> #include <errno.h> #include <unistd.h> #include <signal.h> #include <sys/types.h> #include <sys/socket.h> #include <arpa/inet.h> #include <netinet/in.h> using namespace std; extern char **environ; // difference form old and new in binary file const char *TITLE = "APP-OLD "; bool g_update = false; bool g_stop = false; void signal_handler(int sig) { cout << "signal_handler" << sig << endl; if (sig == SIGUSR2) { g_update = true; } g_stop = true; } void update_binary(char* argv[], int fd) { if (fork() != 0) { return; } char *fdstr = new char[100]; snprintf(fdstr, 100, "FD=%d", fd); int n; for (n = 0; environ[n]; n++); // copy environ char **env = new char*[n + 1]; for (int i = 0; i < n - 1; i++) { env[i] = environ[i]; } // app fd into environ env[n - 1] = fdstr; env[n] = NULL; // importent exec app_new execvpe("./app_new", argv, env); } int main(int argc, char* argv[]) { int fd = -1; g_stop = false; g_update = false; cout << TITLE << getpid() << endl; char *oldfd = getenv("FD"); if (oldfd) { fd = atoi(oldfd); } else { struct sockaddr_in addr; addr.sin_family = AF_INET; addr.sin_port = htons(9999); addr.sin_addr.s_addr = inet_addr("127.0.0.1"); fd = socket(AF_INET, SOCK_STREAM|SOCK_NONBLOCK, 0); if (fd < 0) { return 1; } if (bind(fd, (struct sockaddr *)&addr, sizeof(addr)) < 0 || listen(fd, 4096) < 0) { close(fd); return 1; } if (signal(SIGUSR2, signal_handler) == SIG_ERR || signal(SIGTERM, signal_handler) == SIG_ERR || signal(SIGINT, signal_handler) == SIG_ERR) { close(fd); return 1; } } while (!g_stop) { cout << TITLE << getpid() << " fd=" << fd << endl; sleep(1); } if (g_update) { update_binary(argv, fd); } // semulate waiting for the all connection diedown for (int i = 0; i < 10; i++) { cout << TITLE << getpid() << " fd=" << fd << endl; sleep(1); } cout << TITLE << "end" << endl; return 0; }
app_new.cpp仅仅TITLE取名不同,用于区分新旧二进制文件
diff app_old.cpp app_new.cpp < const char *TITLE = "APP-OLD "; --- > const char *TITLE = "APP-NEW ";
编译两个文件用于对比测试
g++ app_old.cpp -o app_old g++ app_new.cpp -o app_new
执行结果
./app_old APP-OLD 8521 APP-OLD 8521 fd=3 APP-OLD 8521 fd=3 APP-OLD 8521 fd=3 signal_handler12 <- 执行命令 kill -USR2 8521 APP-OLD 8521 fd=3 <- 新旧并行 APP-NEW 8524 APP-NEW 8524 fd=3 APP-OLD 8521 fd=3 APP-NEW 8524 fd=3 APP-OLD 8521 fd=3 APP-NEW 8524 fd=3 APP-OLD 8521 fd=3 APP-NEW 8524 fd=3 APP-OLD 8521 fd=3 APP-NEW 8524 fd=3 APP-OLD 8521 fd=3 APP-NEW 8524 fd=3 APP-OLD 8521 fd=3 APP-NEW 8524 fd=3 APP-OLD 8521 fd=3 APP-NEW 8524 fd=3 APP-OLD 8521 fd=3 APP-NEW 8524 fd=3 APP-OLD 8521 fd=3 APP-NEW 8524 fd=3 APP-OLD end <- 旧进程退出,只剩新进程 APP-NEW 8524 fd=3 APP-NEW 8524 fd=3 APP-NEW 8524 fd=3 APP-NEW 8524 fd=3 APP-NEW 8524 fd=3 APP-NEW 8524 fd=3
The End
- My github location
- View Source of this website GhostZch.github.io
- Commit issues to discuss with me and others
标签:
版权申明:本站文章部分自网络,如有侵权,请联系:west999com@outlook.com
特别注意:本站所有转载文章言论不代表本站观点,本站所提供的摄影照片,插画,设计作品,如需使用,请与原作者联系,版权归原作者所有
下一篇:C语言--指针详解
- 二进制枚举之被RE完虐的我的一天 2020-01-03
- QT程序自启动 2019-12-03
- c++ 内存二进制表示 2019-10-25
- C++ 编程技巧笔记记录(持续更新) 2019-09-30
- 剑指offer11:输入一个整数,输出该数二进制表示中1的个数。 2019-08-26
IDC资讯: 主机资讯 注册资讯 托管资讯 vps资讯 网站建设
网站运营: 建站经验 策划盈利 搜索优化 网站推广 免费资源
网络编程: Asp.Net编程 Asp编程 Php编程 Xml编程 Access Mssql Mysql 其它
服务器技术: Web服务器 Ftp服务器 Mail服务器 Dns服务器 安全防护
软件技巧: 其它软件 Word Excel Powerpoint Ghost Vista QQ空间 QQ FlashGet 迅雷
网页制作: FrontPages Dreamweaver Javascript css photoshop fireworks Flash