python

关注公众号 jb51net

关闭
首页 > 脚本专栏 > python > Cron Pytest脚本无法发送邮件

解决Cron定时任务中Pytest脚本无法发送邮件的问题

作者:Melantha007

文章探讨解决在 Cron 定时任务中运行 Pytest 脚本时邮件发送失败的问题,先优化环境变量,再检查 Pytest 邮件配置,接着配置文件确保 SMTP 服务正常,包括编辑相关文件、配置认证信息等,还提及常见问题排查,如防火墙等,最终使邮件功能在定时任务中成功运行

引言

在使用Cron定时任务运行Pytest的manage.py脚本时,有时会遇到脚本能够成功执行,但邮件发送功能却无法正常工作的问题。这种问题通常是由于环境配置不足、路径设置不正确,或者邮件服务器配置不当等原因导致的。本文将探讨如何解决这一问题,并提供详细的步骤和解决方案。

1. 环境变量优化:确保Cron任务可以正确执行

首先,确保在Cron定时任务中运行的脚本环境和你平时在终端手动运行的环境一致。在开发环境中,Pytest脚本通常依赖于一些环境变量,这些环境变量需要在Cron任务中正确配置。

解决方案:

可以通过将较长的命令拆分成脚本来优化,并在cron中调用该脚本,避免直接在cron中写入过长的命令。这样不仅简洁,而且便于管理和维护。

1.1. 创建一个脚本

首先,在项目目录下创建一个新的脚本文件,比如 run_manage.sh,内容如下:

#!/bin/bash
PATH=/usr/local/sbin:/usr/local/bin:/usr/sbin:/usr/bin:/sbin:/bin
# 设置环境变量(替换成你的真实虚拟环境的地址)
export PATH=/root/myenv/bin:$PATH
export PYTHONPATH=/root/myenv/bin/packages:$PYTHONPATH

# 激活虚拟环境(替换成你的真实虚拟环境的地址)
source /root/myenv/bin/activate

# 执行脚本(/path/to/project/替换为你的脚本路径)
/root/myenv/bin/python /path/to/project/manage.py >> /var/log/manage.log 2>&1

确保脚本可执行(替换为你的而真实run_manage.sh路径):

chmod +x /path/to/run_manage.sh

1.2. 修改cron配置

然后,在cron中调用该脚本。修改cron的配置如下:

00 08 * * * cd /path/to/project && /path/to/run_manage.sh

这样就避免了在cron中写入过长的命令,提高了可读性和可维护性。

2. 脚本能够执行,但邮件发送失败

当确保Cron任务的环境配置正确后,脚本能够成功运行,但邮件发送依然不成功时,问题可能出在邮件服务器的配置或者Pytest的邮件设置上。

解决方案:

检查Pytest邮件配置确保Pytest项目的配置文件中正确配置了邮件相关的设置。QQ的邮件配置如下:

def send_email(total, passed, failed):
 # 邮件内容设置
 subject = "Pytest Test Report"
 body = f"""
 <html>
   <body>
     <p>Hi all:</p>

   
     <p>以下是本次接口测试的结果:</p>
<ul>
     <li ><strong>总测试数: {total}</strong></li>
     <li ><strong>通过数:{passed}</strong></li>
     <li style="color: red;"><strong>失败数:{failed}</strong></li>
</ul>
     <p>                               测试团队</p>
   </body>
 </html>
"""
 mail_host = "smtp.qq.com"  # 设置服务器
 sender_email = "ZZZZZZZZZ@qq.com"  # 用户名
 receiver_email = "ZZZZZZZZZZZZ@126.com"
 password = "############"  # 授权码

 # 设置MIME邮件对象
 msg = MIMEMultipart()
 msg['From'] = "ZZZZZZZZZ@qq.com"
 msg['To'] = receiver_email
 msg['Subject'] = subject

 # 使用'html'格式将报告嵌入到邮件正文
 msg.attach(MIMEText(body, 'html'))

 # 发送邮件
 try:
     with smtplib.SMTP(mail_host, 587) as server:
         server.ehlo()  # 再次发送 EHLO,更新安全会话
         server.starttls()  # 启动 TLS 加密
         server.ehlo()  # 发送 EHLO 命令
         server.login(sender_email, password)  # 使用授权码进行登录
         server.sendmail(sender_email, receiver_email, msg.as_string())
     print("Email sent successfully")
 except smtplib.SMTPException as e:
     print(f"SMTP error occurred: {e}")
 except Exception as e:
     print(f"Failed to send email: {e}")

3. 配置main.cf文件:确保SMTP服务正常工作

如果邮件发送还是不成功,问题可能出在邮件服务本身,特别是SMTP服务器配置。可以尝试配置main.cf文件,来调整邮件发送的相关参数。

3.1. 编辑main.cf文件

在使用Postfix等邮件服务器时,main.cf文件控制着邮件服务的配置。可以检查并确保以下几个配置项被正确设置:

# 编辑main.cf
sudo nano /etc/postfix/main.cf

确保以下设置项正确:

 # 启用 SASL 身份验证
 smtp_sasl_auth_enable = yes
 smtp_sasl_password_maps = hash:/etc/postfix/sasl_passwd
 smtp_sasl_security_options = noanonymous
 smtp_sasl_tls_security_options = noanonymous
 # TLS 配置
 smtp_use_tls = yes
 smtp_tls_security_level = encrypt
 smtp_tls_CAfile = /etc/ssl/certs/ca-certificates.crt
 alias_maps = hash:/etc/aliases
 alias_database = hash:/etc/aliases

 mydestination = localhost, localhost.localdomain,ajcloud,net
 myhostname = mail.example.com
 mynetworks = 127.0.0.0/8 [::ffff:127.0.0.0]/104 [::1]/128
 mailbox_size_limit = 0
 recipient_delimiter = +
 inet_interfaces = all
 inet_protocols = all
 relayhost = [smtp.qq.com]:587
 debug_peer_level = 3
 debug_peer_list = smtp.qq.com
 smtp_helo_name = mail.example.com
 smtpd_helo_required = yes
 myorigin = /etc/mailname
 smtp_generic_maps = hash:/etc/postfix/generic

这些设置项允许Postfix通过指定的SMTP服务器进行身份验证,并启用TLS加密来保证邮件的安全性。

3.2. 配置sasl_passwd文件

如果使用SMTP身份验证,还需要配置sasl_passwd文件,指定SMTP服务器的认证信息:

# 编辑sasl_passwd
sudo nano /etc/postfix/sasl_passwd

文件内容应为:

[smtp.your-email-provider.com]:587    your-email@example.com:your-email-password

然后运行以下命令生成密码哈希:

sudo postmap /etc/postfix/sasl_passwd

为了在Postfix的main.cf文件中增加这两个配置项,你可以按照以下步骤进行操作:

3.3. 配置generic 文件

在main.cf文件中添加以下配置项:

myorigin = /etc/mailname
smtp_generic_maps = hash:/etc/postfix/generic

这两个配置项的作用:

如果你还没有创建/etc/postfix/generic文件,需要创建该文件并配置映射规则。例如:

sudo nano /etc/postfix/generic

在文件中添加类似如下的映射规则:

# 格式: 本地用户 -> 外部邮件地址
root@mail.example.com   [填写脚本中的sender_email]
root@localhost          [填写脚本中的sender_email]
[填写脚本中的receiver_email] [填写脚本中的receiver_email]

3.4. 生成generic文件的哈希文件

配置好后,需要生成generic文件的哈希文件,命令如下:

sudo postmap /etc/postfix/generic

3.5. 重启Postfix服务

配置完成后,重启Postfix服务使配置生效:

sudo systemctl restart postfix

3.6. cronf中增加MAILTO

MAILTO=[填写脚本中的receiver_email]

4. 其他常见问题排查

总结

通过正确配置环境变量、检查Pytest邮件设置、调试脚本及服务器配置,可以有效解决通过Cron任务运行Pytest脚本时,邮件发送失败的问题。确保Cron任务的环境与手动运行时的一致,检查邮件服务器配置,并结合日志输出排查错误,最终可以使邮件功能在定时任务中成功运行。

以上就是解决Cron定时任务中Pytest脚本无法发送邮件的问题的详细内容,更多关于Cron Pytest脚本无法发送邮件的资料请关注脚本之家其它相关文章!

您可能感兴趣的文章:
阅读全文