【Python系列】Python自动发邮件脚本-html邮件内容

缘起

这段时间给朋友搞了个群发邮件的脚本,为了防止进入垃圾邮件,作了不少工做,刚搞完,垃圾邮件进入率50%,以为还不错,若是要将垃圾邮件的进入率再调低,估计就要花钱买主机了,想一想也就算了,先发一个月,看看效果再拓展吧。html

脚本主要是经过Python写的,调的smtplib库,这些是基础,你们在网上一搜一大堆,今天主要给你们讲解下如何避免进入垃圾邮件系统,以及整个系统搭建时的一些思想。可能刚搞Python不久,有不少多是错误的写法望你们提出来哈~python

配置

CentOS7.0系统c++

Python 3.4git

CentOS7.0下面默认的是Python2.7.5,咱们先来将Python的版本提高上去vim

#wget https://www.python.org/ftp/python/3.4.3/Python-3.4.3.tgz

下载Python3.4版本数组

#tar -xf Python-3.4.3.tgz
#cd Python-3.4.3/
#./configure

这边configure的时候可能会遇到你的环境没有安装gcc编译环境,执行下面的语句再configure便可安全

#yum -y install gcc
#yum -y install gdb
#yum -y install gcc-c++

编译安装服务器

#make
#make install

由于替换了python版本以后yum可能不能正常使用,需改两个文件架构

#vim /usr/bin/yum
#vim /usr/libexec/urlgrabber-ext-down

将这两个文件的头部的#!/usr/bin/python改为#!/usr/bin/python2.7便可,保存退出,yum满状态复活app

编译完了以后,将python3.4设置为默认python解析。

#ln -s /usr/local/bin/python3.4 /usr/bin/python

连接完成以后检查python版本

#python -V

出现Python3.4就标识版本切换完成

系统架构

Account:用于存放发送者邮箱帐号的目录,个人163邮箱,sina邮箱,sohu邮箱和tom邮箱均在淘宝上购买了30个能够发送smtp服务的帐号,花费一顿饭的钱不到就能够搞到啦~帐号密码用【:】分割,每一个帐号之间使用【,】分割。

Common:引用类文件夹,里面是日至系统配置文件和日志系统源代码

Conf:全局配置文件,目前还木有用到

Image:邮件发送过程当中须要使用到的图片资源

Log:日志文件,按日期区分

Logbackups:日志备份文件,用于备份过时日志

Sendmail:用于存储收件人的邮箱信息,帐号之间用【,】分割

mail_html.py:主要执行脚本

README.md:git版本控制用户须知,我是经过码云来管理个人代码的

日志系统

提起脚本系统,日志是至关关键的一个角色,尤为是当你的脚本出错,你要查错的时候,就很是重要了,我也是从网上搞来的一段Log日志系统的代码,以为挺好用,供大伙参考~

主要思想是,打印log到指定文件,打印log到屏幕,啥也不说了,先上代码,由于是网上的代码,我就放上来啦~

# coding: utf-8

#from lxml import etree
import logging.handlers
import logging
import os
import sys
import time
import datetime

try: 
  import xml.etree.cElementTree as ET 
except ImportError: 
  import xml.etree.ElementTree as ET 

# 提供日志功能
class logger:
    # 先读取XML文件中的配置数据
    # 因为config.xml放置在与当前文件相同的目录下,所以经过 __file__ 来获取XML文件的目录,而后再拼接成绝对路径
    # 这里利用了lxml库来解析XML
    root = ET.parse(os.path.join(os.path.dirname(__file__), 'config.xml')).getroot()
    # 读取日志文件保存路径
    logpath = root.find('logpath').text
    # 读取日志文件容量,转换为字节
    logsize = 1024*1024*int(root.find('logsize').text)
    # 读取日志文件保存个数
    lognum = int(root.find('lognum').text)
    
    # 添加分天日志名
    now = datetime.datetime.now()
    now_time = now.strftime('%Y%m%d')
    log_file_name = sys.argv[0].split('/')[-1].split('.')[0] + '_' + now_time
    # 日志文件名:由用例脚本的名称,结合日志保存路径,获得日志文件的绝对路径
    logname = os.path.join(logpath, log_file_name)

    # 初始化logger
    log = logging.getLogger()
    # 日志格式,能够根据须要设置
    fmt = logging.Formatter('[%(asctime)s][%(filename)s][line:%(lineno)d][%(levelname)s] %(message)s', '%Y-%m-%d %H:%M:%S')

    # 日志输出到文件,这里用到了上面获取的日志名称,大小,保存个数
    handle1 = logging.handlers.RotatingFileHandler(logname, maxBytes=logsize, backupCount=lognum)
    handle1.setFormatter(fmt)
    # 同时输出到屏幕,便于实施观察
    handle2 = logging.StreamHandler(stream=sys.stdout)
    handle2.setFormatter(fmt)
    log.addHandler(handle1)
    log.addHandler(handle2)

    # 设置日志基本,这里设置为INFO,表示只有INFO级别及以上的会打印
    log.setLevel(logging.INFO)

    # 日志接口,用户只需调用这里的接口便可,这里只定位了INFO, WARNING, ERROR三个级别的日志,可根据须要定义更多接口
    @classmethod
    def info(cls, msg):
        cls.log.info(msg)
        return

    @classmethod
    def warning(cls, msg):
        cls.log.warning(msg)
        return

    @classmethod
    def error(cls, msg):
        cls.log.error(msg)
        return

日志系统的配置文件

<?xml version="1.0" encoding="utf-8"?>

<config>
    <!--  日志保存路径  -->
    <logpath>/Users/litao/Desktop/mail_html/Log</logpath>

    <!-- 每一个脚本对应的日志文件大小,单位MB -->
    <logsize>8</logsize>

    <!-- 每一个脚本保存的日志文件个数 -->
    <lognum>100</lognum>
</config>

保存的路径各位随意哈。

如何使用

logger.info('邮件总数量【'+str(len(recivers))+'')                                                                                          
logger.info('总计发送邮件数量【'+str(send_num)+'')                                                                                         
logger.info('总计发送错误数量【'+str(error_num)+'')                                                                                        
logger.info('成功邮箱帐号集合:'+','.join(send_success_account))                                                                             
logger.info('失败邮箱帐号集合:'+','.join(send_failure_account))                                                                             
logger.info('脚本结束------------------------------------------------------------------')                                                    
logger.info('')                                                                        

error的话将info换成error便可

执行主文件

提起垃圾邮件,你们首先想到的就是那个使人讨厌的垃圾箱里面的营销邮件,可是,若是你的邮件内容写的很棒,是否是就能够避免被封杀,答案是否认的,邮件被封杀是机器干的,若是是机器干的事,那就好办了,首先,咱们得先知道机器的工做原理。

大部分被列为垃圾邮件的邮件均有两个特征:内容不变,IP不变,其实作到内容一直变,IP一直变理论上就能够作到不进垃圾邮箱,可是哪有那么多的人力物力作这事,因此,咱们要作的是解决几率性的问题。

内容混淆

内容不变咱们可使用多套模板,嵌套着发,这个问题好解决,可是IP不变,这个就难一点了,其实我也没解决,主要是怕花钱,我能作的就是经过多套模板来实现内容几率性的不被封杀。

好了,咱们先准备30个邮件的subject,30套邮件的内容模板,下面就是个人全局subject配置

这样作的好处就是能够防止邮件的内容被封杀,假设咱们30秒发一封邮件,那么在20分钟内的邮件,没有一封是重复的。咱们是从接受邮箱域名的角度考虑的,也就是若是咱们的营销对象全是QQ邮箱,那么QQ邮箱的邮箱服务器在20分钟内收到同一IP的邮件内容是不同的,这很大程度上就能避免被封杀。

帐户混淆

设置这么多帐号是干吗用的呢,主要仍是想混淆机器,让垃圾邮件进率更低。

下面我我的通过测试,发现邮箱服务器具备的一些特性。

163邮箱

163邮箱设置了天天每一个帐号邮件发送的上限位50封,帐号554出错重发的时间是3小时。

tom邮箱

tom邮箱天天邮件发送数量不作限制,咱们也假设是50封,可是每封邮件之间的发送间隔必定要超过30秒,要否则会被短期链接数过大报错。

sohu邮箱

业界良心,基本上没出过啥错误,一直保持着良好的发送成功率。咱们也将其定位发送间隔30秒,每日上线50封。

sina邮箱

恶心的玩意儿,每次发送邮箱前须要先登陆,认证手机号,每一个手机号5个邮箱哈,可是效果显著,认证完毕,和sohu同样,基本没出错过。

时间混淆

有了这些基础,咱们就能够知道了,咱们有120个帐号,30个邮件模板,天天一刻不停的发送,每封邮件之间的间隔为30秒,一天的邮件发送量在2800封左右。

我以为一天2800封,若是有钱的话,一台ESC的费用是3元天天,独立ip哈,若是找第三方发送,一封邮件是3分钱,量大2分钱,他们是EDM的,我测试过1500封,达到率不足千分之一。也就是说,咱们发送1500封,只须要1块多钱,找第三方发送,1500封怎么也得40块钱。成本是否是很低。

好的,那就来看看邮件是如何发送的吧。

邮件发送

下面咱们来看下个人主文件是如何搞的

#coding=utf-8
import smtplib
import mimetypes
import time
import datetime

from email.mime.text import MIMEText
from email.mime.multipart import MIMEMultipart
from email.mime.image import MIMEImage

#引入外部文件
from Common.log import *

导入模块,以来的外部库和内部的文件

#目录主位置
_root_dir = '/Users/litao/Desktop/mail_html/'
_title_common = '愚人节'

愚人节主题礼物,也是为了之后省事,subject和内容中设计到title的均会被改成愚人节,立刻愚人节了吗,营销方式,代码实现,异常方便修改。

#邮箱内容设置
_content = """\
<html>
    <style> .title{font-weight:bold;font-size:18px;}</style>
    <body>
        <p>
            <img src="cid:image1">
            <br><br>
            <span
            class="title">【愚人节】将至,您还没准备礼物?那你必定会过个开心的愚人节的</span>
            <br>
            愚人节就要来啦,礼朵朵给大伙准备了大批量的礼物伴你度过愚人节,具体百度一下【礼朵朵】,赶忙进站选礼物吧~
            <br>
            选礼物前别忘了先去心愿墙许愿哟,你的愿望可能被礼朵朵看到,能够帮你实现哟,实现的时候别忘了来礼朵朵还愿哈~
            <br><br>
            <span class="title">【礼朵朵】介绍</span>
            <br>
            国人从古至今都有送礼的习俗,送礼做为传统之一,一直流传至今,礼尚往来成为人生必修课。
            <br>
            【礼朵朵】集合商业送礼和现代送礼搭建礼物导购分享平台【朵朵礼物】,带给老小皆宜的送礼分享体验新体验。
            <br>
            与此同时,礼朵朵还给大伙准备了礼物攻略【礼物说】,让你们能够对礼物有个更全面的了解~
            <br><br>
            <span class="title">百度搜索【礼朵朵】,开启你的礼物新旅程吧~</span>
            <br><br>
        </p>
    </body>
</html> 
""" 

营销内容模板,html模式实现邮件的发送,少不了有模板~

#发送邮箱smtp地址
_smtp_address = ['smtp.163.com','smtp.sina.cn','smtp.tom.com','smtp.sohu.com']

smtp地址数组,用于在不一样的邮件服务器间切换。

def sendMail(sender,reciver,subject,content,passwd,smtpadd):
    username=sender
    password=passwd
    msg=MIMEMultipart('related')
    msg['Subject']=subject
    #html格式
    html=content
    htm = MIMEText(html,'html','utf-8')
    msg.attach(htm)
    #构造图片
    fp=open(_root_dir+'Image/logo_small.png','rb')
    msgImage=MIMEImage(fp.read())
    fp.close()
    msgImage.add_header("Content-ID", "<image1>")
    msg.attach(msgImage)
    fp2=open(_root_dir+'Image/yurenjie.png','rb')
    msgImage2=MIMEImage(fp2.read())
    fp2.close()
    msgImage2.add_header('Content-Disposition', 'attachment',
                         filename="愚人节活动海报.jpg")
    msg.attach(msgImage2)
    msg['From']=sender
    msg['To']=reciver

    #发送邮件
    smtp=smtplib.SMTP()
    smtp.connect(smtpadd)
    smtp.login(username, password)
    smtp.sendmail(sender, reciver, msg.as_string())
    smtp.quit()

发邮件方法,里面有两个地方须要注意,一个是

msgImage.add_header("Content-ID", "<image1>")
msg.attach(msgImage)

将邮件模板中的image1的img标签内容替换成咱们想要的图片

第二个

    fp2=open(_root_dir+'Image/yurenjie.png','rb')
    msgImage2=MIMEImage(fp2.read())
    fp2.close()
    msgImage2.add_header('Content-Disposition', 'attachment',
                         filename="愚人节活动海报.jpg")

插入附件,图片是一个海报,提及海报,强烈建议你们使用创客贴这个平台,很是好用。

下面就是发送邮件啦!!!

#发送邮件
    smtp=smtplib.SMTP()
    smtp.connect(smtpadd)
    smtp.login(username, password)
    smtp.sendmail(sender, reciver, msg.as_string())
    smtp.quit()

通用方法,将文件中的以,分割的内容以数组形式返回

#读取文件中的数据,并将使用,分割的数据变为数组
def readFileToSplit(filepath):
    file_stream = open(filepath)
    try:
        data = file_stream.read()
    finally:
        file_stream.close()
    data_split = data.split(',')
    return data_split

主方法

一、切割帐号

二、切换邮件服务器

三、每发送一封邮件,休息25秒,切换帐号,继续发送

四、日志记录

五、错误处理

if __name__=="__main__":
    content=_content
    # 接收人的邮箱按照天天2000封来,天天的邮箱都须要更换,文件名最后以日期为准,邮件发送量以日志为准
    recivers=readFileToSplit(_root_dir+'Sendmail/mail_test.txt')
    # 把4个邮箱的帐号都获取到,方便下面for循环中使用
    account_163=readFileToSplit(_root_dir+'Account/account163')
    account_sina=readFileToSplit(_root_dir+'Account/accountsina')
    account_tom = readFileToSplit(_root_dir+'Account/accounttom')
    account_sohu = readFileToSplit(_root_dir+'Account/accountsohu')

    # 获取邮件发送模板
    # 注意模板之间的切换

    #log_file_stream = open(_root_dir+'log', 'w+')
    logger.info('')
    logger.info('脚本开始------------------------------------------------------------------')

    # 统计邮件发送量
    send_num = 0
    # 统计发送出错量
    error_num = 0
    # 统计发送失败的邮箱发送帐号
    send_success_account = []
    # 统计发送成功的邮箱发送帐号
    send_failure_account = []

    subject_num = len(_subject)

    # 最后统计没有发出去的邮箱号,放到下日,继续发送
    for i in range(0, len(recivers)):
        try:
            sendindex = i - error_num
            num = i % 30
            account = account_163[num].split(':')
            addindex=i%4
            subjectindex = sendindex%subject_num
            if addindex == 1:
                account=account_sina[num].split(':')
            elif addindex == 2:
                account=account_tom[num].split(':')
            elif addindex == 3:
                account=account_sohu[num].split(':')
            sender=account[0]
            passwd=account[1]
            smtpadd = _smtp_address[addindex]
            #smtpstr=str('163')
            sendMail(sender, recivers[sendindex], _subject[subjectindex], content, passwd, smtpadd)
            #print('发送帐号', sender, '正在发送')
            str_success_1 = '发送帐号【'+sender+'】正在发送'
            logger.info(str_success_1)
            #writeLog(log_file_stream,str_success_1)
            #print('接收序号', i, recivers[i],'发送成功')
            str_success_2 = '接受序号【'+str(i)+'】【'+recivers[sendindex]+'】发送成功'
            #writeLog(log_file_stream,str_success_2)
            logger.info(str_success_2)
            logger.info('')
            #print('')
            send_num+=1
            send_success_account.append(sender)
            time.sleep(25)
        except Exception as e:
            #print('中止于:', i, recivers[i],',发送失败')
            str_failure_1 = '产生错误于:【'+sender+'】发送失败'
            #writeLog(log_file_stream,str_failure_1)
            logger.error(str_failure_1)
            #print(e)
            str_failure_2 = str(e)
            #writeLog(log_file_stream,str_failure_2)
            logger.error(str_failure_2)
            logger.info('')
            error_num+=1
            send_failure_account.append(sender)
            #print('')
            #break
    #print('安全抵达底部')
    #writeLog(log_file_stream,'脚本结束')
    set(send_success_account)
    set(send_failure_account)
    logger.info('邮件总数量【'+str(len(recivers))+'')
    logger.info('总计发送邮件数量【'+str(send_num)+'')
    logger.info('总计发送错误数量【'+str(error_num)+'')
    logger.info('成功邮箱帐号集合:'+','.join(send_success_account))
    logger.info('失败邮箱帐号集合:'+','.join(send_failure_account))
    logger.info('脚本结束------------------------------------------------------------------')
    logger.info('')
    #log_file_stream.close()

代码就这么多,至于subject邮件主题和模板怎么搞,能够自由发挥哈,能够放在主执行文件中,也能够放到配置文件中,实现能够配置,这里就再也不赘述啦

开工

下面就能够开工啦,直接到项目主目录

#python mail_html.py

看到屏幕上有输出就OK啦,下面就是等待收获的季节,好了,今天给你们讲解了我这段时间一直在搞的两个小玩意儿,下面开始又要回到正题了,PHP之Yii框架的详解。但愿你们都沟通交流哈~~~

相关文章
相关标签/搜索