在阿里云和树莓派上编写一个守护进程程序

news2025/4/21 14:59:04

目录

一、阿里云邮件守护进程

1. 安装必要库

2. 创建邮件发送脚本 mail_daemon.py

3. 设置后台运行

二、树莓派串口守护进程

1. 启用树莓派串口

2. 安装依赖库

3. 创建串口输出脚本 serial_daemon.py

4. 设置开机自启

5. 使用串口助手接收


一、阿里云邮件守护进程

1. 安装必要库
pip3 install smtplib email schedule
2. 创建邮件发送脚本 mail_daemon.py
import smtplib
import schedule
import time
from email.mime.text import MIMEText
from email.header import Header

# 配置信息(替换为你的信息)
SMTP_SERVER = 'smtp.163.com'        # 例如163邮箱
SMTP_PORT = 465                     # SSL端口
SENDER_EMAIL = 'your_email@163.com'
SENDER_PASSWORD = 'your_password'   # 或授权码
RECEIVER_EMAIL = 'receiver@example.com'

def send_email():
    message = MIMEText('守护进程正在运行中....', 'plain', 'utf-8')
    message['From'] = Header(SENDER_EMAIL)
    message['To'] = Header(RECEIVER_EMAIL)
    message['Subject'] = Header('阿里云守护进程状态')

    try:
        smtp = smtplib.SMTP_SSL(SMTP_SERVER, SMTP_PORT)
        smtp.login(SENDER_EMAIL, SENDER_PASSWORD)
        smtp.sendmail(SENDER_EMAIL, [RECEIVER_EMAIL], message.as_string())
        smtp.quit()
        print("邮件发送成功")
    except Exception as e:
        print(f"邮件发送失败: {str(e)}")

def job():
    send_email()

if __name__ == '__main__':
    schedule.every(1).minutes.do(job)
    while True:
        schedule.run_pending()
        time.sleep(1)
3. 设置后台运行
nohup python3 mail_daemon.py > /dev/null 2>&1 &

二、树莓派串口守护进程

1. 启用树莓派串口
# 禁用蓝牙占用串口
sudo raspi-config
# 选择 Interfacing Options -> Serial -> No 关闭登录shell -> Yes 启用硬件串口
sudo reboot

# 验证串口
ls -l /dev/serial*  # 应显示 /dev/serial0 -> ttyAMA0
2. 安装依赖库
sudo apt-get install python3-serial
3. 创建串口输出脚本 serial_daemon.py
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <sys/socket.h>
#include <netdb.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <string.h>
#include <fcntl.h>
#include <signal.h>
#include <time.h>
#include <openssl/ssl.h>
#include <openssl/err.h>

#define SMTP_SERVER "smtp.qq.com"
#define SMTP_PORT 465
#define SENDER_EMAIL "your_email@qq.com"
#define SENDER_PASSWORD "your_auth_code"
#define RECEIVER_EMAIL "receiver@example.com"
#define LOG_FILE "/tmp/daemon_email.log"

// 守护进程初始化
void daemonize() {
    pid_t pid = fork();
    if (pid < 0) {
        perror("First fork failed");
        exit(EXIT_FAILURE);
    }
    if (pid > 0) exit(EXIT_SUCCESS); // 父进程退出

    // 创建新会话
    if (setsid() < 0) {
        perror("setsid failed");
        exit(EXIT_FAILURE);
    }

    // 第二次fork
    pid = fork();
    if (pid < 0) {
        perror("Second fork failed");
        exit(EXIT_FAILURE);
    }
    if (pid > 0) exit(EXIT_SUCCESS);

    // 设置文件权限掩码
    umask(0);

    // 改变工作目录
    chdir("/");

    // 关闭标准文件描述符
    close(STDIN_FILENO);
    close(STDOUT_FILENO);
    close(STDERR_FILENO);
}

// 日志记录函数
void log_message(const char* message) {
    time_t now;
    time(&now);
    char* dt = ctime(&now);
    dt[strlen(dt)-1] = '\0'; // 去除换行符

    FILE* log_file = fopen(LOG_FILE, "a");
    if (log_file) {
        fprintf(log_file, "%s - %s\n", dt, message);
        fclose(log_file);
    }
}

// Base64编码(简化版,实际应使用完整实现)
char* base64_encode(const char* input) {
    BIO *bio, *b64;
    BUF_MEM *bufferPtr;

    b64 = BIO_new(BIO_f_base64());
    bio = BIO_new(BIO_s_mem());
    bio = BIO_push(b64, bio);

    BIO_write(bio, input, strlen(input));
    BIO_flush(bio);
    BIO_get_mem_ptr(bio, &bufferPtr);

    char* output = (char*)malloc(bufferPtr->length + 1);
    memcpy(output, bufferPtr->data, bufferPtr->length);
    output[bufferPtr->length] = '\0';

    BIO_free_all(bio);
    return output;
}

// 发送SMTP命令并检查响应
int smtp_command(SSL* ssl, const char* cmd, const char* expect_code, int log) {
    char buffer[1024];
    
    if (cmd) SSL_write(ssl, cmd, strlen(cmd));
    SSL_read(ssl, buffer, sizeof(buffer));

    if (log) {
        log_message(buffer);
    }

    return (strstr(buffer, expect_code) != NULL);
}

// 发送邮件函数
int send_email() {
    struct sockaddr_in addr;
    int sock;
    SSL_CTX* ctx;
    SSL* ssl;

    // 初始化OpenSSL
    SSL_library_init();
    OpenSSL_add_all_algorithms();
    SSL_load_error_strings();
    ctx = SSL_CTX_new(SSLv23_client_method());

    // 创建socket
    sock = socket(AF_INET, SOCK_STREAM, 0);
    addr.sin_family = AF_INET;
    addr.sin_port = htons(SMTP_PORT);
    inet_pton(AF_INET, SMTP_SERVER, &addr.sin_addr);

    // 连接服务器
    if (connect(sock, (struct sockaddr*)&addr, sizeof(addr)) {
        log_message("Connection failed");
        return -1;
    }

    // 创建SSL连接
    ssl = SSL_new(ctx);
    SSL_set_fd(ssl, sock);
    if (SSL_connect(ssl) <= 0) {
        log_message("SSL connection failed");
        return -1;
    }

    // SMTP协议交互
    if (!smtp_command(ssl, NULL, "220", 1)) goto error; // 读取欢迎消息
    if (!smtp_command(ssl, "EHLO localhost\r\n", "250", 1)) goto error;
    
    // 认证
    char auth_cmd[256];
    sprintf(auth_cmd, "AUTH LOGIN\r\n");
    if (!smtp_command(ssl, auth_cmd, "334", 1)) goto error;

    char* encoded_user = base64_encode(SENDER_EMAIL);
    sprintf(auth_cmd, "%s\r\n", encoded_user);
    if (!smtp_command(ssl, auth_cmd, "334", 1)) goto error;
    free(encoded_user);

    char* encoded_pass = base64_encode(SENDER_PASSWORD);
    sprintf(auth_cmd, "%s\r\n", encoded_pass);
    if (!smtp_command(ssl, auth_cmd, "235", 1)) goto error;
    free(encoded_pass);

    // 构造邮件
    sprintf(auth_cmd, "MAIL FROM:<%s>\r\n", SENDER_EMAIL);
    if (!smtp_command(ssl, auth_cmd, "250", 1)) goto error;

    sprintf(auth_cmd, "RCPT TO:<%s>\r\n", RECEIVER_EMAIL);
    if (!smtp_command(ssl, auth_cmd, "250", 1)) goto error;

    if (!smtp_command(ssl, "DATA\r\n", "354", 1)) goto error;

    const char* email_body = "From: \"Daemon\" <" SENDER_EMAIL ">\r\n"
                             "To: " RECEIVER_EMAIL "\r\n"
                             "Subject: Status Report\r\n\r\n"
                             "Daemon is running...\r\n.\r\n";
    
    SSL_write(ssl, email_body, strlen(email_body));
    if (!smtp_command(ssl, NULL, "250", 1)) goto error;

    // 退出
    smtp_command(ssl, "QUIT\r\n", "221", 1);

    SSL_free(ssl);
    close(sock);
    SSL_CTX_free(ctx);
    return 0;

error:
    SSL_free(ssl);
    close(sock);
    SSL_CTX_free(ctx);
    return -1;
}

int main() {
    daemonize();
    log_message("Daemon started");

    while(1) {
        if (send_email() == 0) {
            log_message("Email sent successfully");
        } else {
            log_message("Failed to send email");
        }
        sleep(60); // 每分钟发送一次
    }

    return 0;
}

4. 设置开机自启

创建 /etc/systemd/system/serial_daemon.service

[Unit]
Description=Serial Daemon Service

[Service]
ExecStart=/usr/bin/python3 /home/pi/serial_daemon.py
Restart=always
User=pi

[Install]
WantedBy=multi-user.target
5. 使用串口助手接收
sudo apt-get install minicom
minicom -b 115200 -o -D /dev/serial0

实验结果:

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2339500.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

基于前端技术的QR码API开发实战:从原理到部署

前言 QR码&#xff08;Quick Response Code&#xff09;是一种二维码&#xff0c;于1994年开发。它能快速存储和识别数据&#xff0c;包含黑白方块图案&#xff0c;常用于扫描获取信息。QR码具有高容错性和快速读取的优点&#xff0c;广泛应用于广告、支付、物流等领域。通过扫…

RenderStage::drawInner

文章目录 RenderStage::drawInnerOSG渲染后台关系图OSG的渲染流程RenderBin::draw(renderInfo,previous)RenderBin::drawImplementationRenderLeaf::renderosg::State::apply(const StateSet*)Drawable::draw(RenderInfo& renderInfo)Drawable::drawInner(RenderInfo& …

C++初阶-类和对象(中)

目录 1.类的默认成员函数 2.构造函数&#xff08;难度较高&#xff09; ​编辑 ​编辑 ​编辑 3.析构函数 4.拷贝构造函数 5.赋值运算符重载 5.1运算符重载 5.2赋值运算符重载 6.取地址运算符重载 6.1const成员函数 6.2取地址运算符重载 7.总结 1.类的默认成员函数…

智谱开源新一代GLM模型,全面布局AI智能体生态

2024年4月15日&#xff0c;智谱在中关村论坛上正式发布了全球首个集深度研究与实际操作能力于一体的AI智能体——AutoGLM沉思。这一革命性技术的发布标志着智谱在AGI&#xff08;通用人工智能&#xff09;领域的又一次重要突破。智谱的最新模型不仅推动了AI智能体技术的升级&am…

分治-快排-75.颜色分类-力扣(LeetCode)

一、题目解析 给定一个数组将其元素按照0&#xff0c;1,&#xff0c;2三段式排序&#xff0c;并且在原地进行排序。 二、算法原理 解法&#xff1a;三指针 用cur遍历数组&#xff0c;left记录0的最左侧&#xff0c;right记录2的最右侧。 left初始值为-1&#xff0c;right的初…

铅酸电池充电器方案EG1253+EG4321

参考&#xff1a; 基于EG1253EG4321铅酸电池(48V20AH)三段式充电器 屹晶微高性价比的电瓶车充电器方案——EG1253 电瓶电压 48V电瓶锂电池&#xff0c;其充满约为55V~56V&#xff0c;因此充电器输出电压为55V~56V&#xff1b; 若是48V铅酸电池&#xff0c;标称电压为48V&…

vue 中formatter

:formatter 是前端表格组件&#xff08;如 Element UI、Vxe-Table 等&#xff09;中用于 ​​自定义单元格内容显示格式​​ 的属性。它的核心作用是&#xff1a;将后端返回的原始数据&#xff08;如编码、状态值等&#xff09;转换为更友好、更易读的文本。 这段代码 :forma…

协程?协程与线程的区别?Java是否支持协程?

一、前言 协程&#xff08;Coroutine&#xff09; 是一种轻量级的并发编程模型&#xff0c;允许在单线程内通过协作式多任务调度实现并发。由用户代码显式控制&#xff08;用户态调度而非操作系统内核调度&#xff09;&#xff0c;避免了线程上下文切换的开销&#xff0c;适合…

Muduo网络库实现 [十六] - HttpServer模块

目录 设计思路 类的设计 模块的实现 公有接口 私有接口 疑问点 设计思路 本模块就是设计一个HttpServer模块&#xff0c;提供便携的搭建http协议的服务器的方法。那么这个模块需要如何设计呢&#xff1f; 这还需要从Http请求说起。 首先从http请求的请求行开始分析&…

关于进程状态

目录 进程的各种状态 运行状态 阻塞状态 挂起状态 linux中的进程状态、 进程状态查看 S状态&#xff08;浅睡眠&#xff09; t 状态&#xff08;追踪状态&#xff09; T状态&#xff08;暂停状态&#xff09; ​编辑 kill命令手册 D状态&#xff08;深度睡眠&#…

SQL注入 01

0x01 用户、脚本、数据库之间的关系 首先客户端发出了ID36的请求&#xff0c;脚本引擎收到后将ID36的请求先代入脚本的sql查询语句Select * from A where id 36 &#xff0c; 然后将此代入到数据库中进行查询&#xff0c;查到后将返回查询到的所有记录给脚本引擎&#xff0c;接…

学习笔记:黑马程序员JavaWeb开发教程(2025.3.24)

11.2 案例-文件上传-简介 火狐浏览器可以看到文件上传传递的底层数据&#xff0c;而chrome对这一块数据进行了包装 在输出日志代码处打了一个断点&#xff0c;看服务端接收到的数据&#xff0c;在上传文件的保存地址中&#xff0c;可以看到&#xff0c;有三个临时文件&…

计算机视觉cv2入门之视频处理

在我们进行计算机视觉任务时&#xff0c;经常会对视频中的图像进行操作&#xff0c;这里我来给大家分享一下&#xff0c;cv2对视频文件的操作方法。这里我们主要介绍cv2.VideoCapture函数的基本使用方法。 cv2.VideoCapture函数 当我们在使用cv2.VideoCapture函数时&#xff…

【Linux】Rhcsa复习5

一、Linux文件系统权限 1、文件的一般权限 文件权限针对三类对象进行定义&#xff1a; owner 属主&#xff0c;缩写u group 属组&#xff0c; 缩写g other 其他&#xff0c;缩写o 每个文件针对每类访问者定义了三种主要权限&#xff1a; r&#xff1a;read 读 w&…

FFmpeg:M3U8的AES加密

1、加密用的key&#xff0c;命令&#xff1a; openssl rand 16>enc.key 2、目的是生成一个enc.key文件 生成iv openssl rand -hex 16 生成后记录下来这个字符串 3、新建一个enc.keyinfo文件&#xff0c;内容有如下三行&#xff1a; key URIenc.key的路径&#xff0c;…

VMware虚拟机走主机代理上网

&#x1f310; VMware虚拟机走主机代理上网&#x1f511; 你是否也遇到过这样的困境&#xff1f;&#x1f4a1; 在虚拟机中测试某个项目&#xff0c;却因为网络限制而寸步难行。今天&#xff0c;就让我们一起探索如何让VMware虚拟机轻松调用本机的代理上网工具&#xff0c;开启…

Cursor入门教程-JetBrains过度向

Cursor使用笔记 **前置&#xff1a;**之前博主使用的是JetBrains的IDE&#xff0c;VSCode使用比较少&#xff0c;所以会尽量朝着JetBrains的使用习惯及样式去调整。 一、设置语言为中文 如果刚上手Cursor&#xff0c;那么肯定对Cursor中的众多选项配置项不熟悉&#xff0c;这…

Mac OS系统下kernel_task占用大量CPU资源导致系统卡顿

CPU负载突然飙升&#xff0c;如截图&#xff1a; 根本原因&#xff0c;大家从各种博主上已知晓&#xff0c;现在提供自己的解决办法&#xff0c;亲测有效 一、设置开机自动禁用温度管理守护进程 1.创建脚本文件 mkdir -p ~/Scripts touch ~/Scripts/disable_thermald.sh …

宝塔面板部署 Dify-latest 最新版本

一、本地部署Windows 版本宝塔面板 宝塔面板是一款简单容易上手使用的服务器管理软件&#xff0c;它可以帮助用户方便地管理服务器以及部署网站等。 &#xff08;1&#xff09;在宝塔面板官网的下载界面&#xff0c;选择 windows 版本下载。点此进入下载 &#xff08;2&#x…

《TCP/IP网络编程》学习笔记 | Chapter 24:制作 HTTP 服务器端

《TCP/IP网络编程》学习笔记 | Chapter 24&#xff1a;制作 HTTP 服务器端 《TCP/IP网络编程》学习笔记 | Chapter 24&#xff1a;制作 HTTP 服务器端HTTP 概要理解 Web 服务器端无状态的 Stateless 协议请求消息&#xff08;Request Message&#xff09;的结构响应消息&#x…