Go 群发邮件Redis 实现邮件群发

一、安装

go get github.com/go-redis/redis/v8
go get gopkg.in/gomail.v2

 二、使用"gopkg.in/gomail.v2"群发 

package main

import (
	gomail "gopkg.in/gomail.v2"
)

func main() {
	// 邮件内容
	m := gomail.NewMessage()

	m.SetHeader("From", "demo@163.com")
	m.SetHeader("To", "demo@qq.com")
	m.SetHeader("Subject", "邮件标题")
	m.SetBody("text/html", "<h2>邮件内容</h2>")

	// 邮件服务器账号密码
	// func NewDialer(host string, port int, username, password string) *Dialer
	d := gomail.NewDialer(
		"smtp.163.com",
		25,
		"demo@163.com",
		"password",
	)

	// 发送邮件
	d.DialAndSend(m)
}

 三、邮箱服务配置,以QQ邮箱为例 

四、示例代码 

package main

import (
	"context"
	"fmt"
	redis "github.com/go-redis/redis/v8"
	"gopkg.in/gomail.v2"
	"gopkg.in/ini.v1"
	"log"
	"time"
)

func main() {
	// 1、连接redis,确保redis服务已开启
	// 创建一个Redis客户端实例
	rdb := redis.NewClient(&redis.Options{
		Addr:     "localhost:6379", // Redis 服务器地址
		Password: "",               // 没有密码就留空
		DB:       0,                // 使用默认DB
	})

	// 关闭 Redis 连接
	defer rdb.Close()

	ctx := context.Background()

	// 检查连接
	pong, err := rdb.Ping(context.Background()).Result()
	if err != nil {
		fmt.Println("连接Redis失败", err)
		return
	}
	fmt.Println(pong) // 输出 PING 命令的响应

	// 模拟添加邮件到待发送队列
	email1 := "123xxx@qq.com"
	email2 := "123xxx@qq.com"
	email3 := "123xxx@qq.com"
	// 使用LPush命令向名为'my_list'的列表中添加元素
	//values := []interface{}{"2576265911@qq.com", "2576265911@qq.com", "2576265911@qq.com"}
	//err := rdb.LPush(ctx, "email_queue", email1, email2, email3).Err()

	err = rdb.LPush(ctx, "email_queue", email1, email2, email3).Err()
	if err != nil {
		log.Fatalf("无法推送到列表: %v", err)
	}

	// 初始化邮件发送配置
	cfg, err := ini.Load("conf/email_config.ini")
	if err != nil {
		log.Fatalf("加载邮件配置失败: %v", err)
	}

	sender := cfg.Section("AUTHENTICATION").Key("username").String()
	password := cfg.Section("AUTHENTICATION").Key("password").String()
	smtpServer := cfg.Section("SMTP_SERVER").Key("smtp_server").String()
	smtpPort := cfg.Section("SMTP_SERVER").Key("smtp_port").MustInt(587)

	// 连接到 SMTP 服务器
	d := gomail.NewDialer(smtpServer, smtpPort, sender, password)

	// 循环处理邮件队列
	for {
		// 从 Redis 队列中获取邮件
		email, err := rdb.RPop(ctx, "email_queue").Result()
		if err == redis.Nil {
			// 队列为空,退出循环
			fmt.Println("电子邮件队列为空。")
			break
		} else if err != nil {
			log.Fatalf("无法从队列中获取电子邮件: %v", err)
		}

		// 创建邮件消息
		m := gomail.NewMessage()
		m.SetHeader("From", cfg.Section("EMAIL").Key("from").String())
		m.SetHeader("To", email)
		m.SetHeader("Subject", cfg.Section("EMAIL").Key("subject").String())

		// 设置邮件内容
		m.SetBody("text/plain", cfg.Section("EMAIL").Key("body").String())

		// 发送邮件
		if err := d.DialAndSend(m); err != nil {
			log.Fatalf("无法将电子邮件发送到:%s: %v", email, err)
		}

		// 模拟邮件发送后的一些操作
		fmt.Printf("发送电子邮件至:%s\n", email)
		time.Sleep(5 * time.Second) // 可以添加一些延迟以限制发送速率
	}
}

 五、邮件配置

# 以QQ邮箱为例
[SMTP_SERVER]
smtp_server = smtp.qq.com
#587或465,具体取决于SMTP服务器配置
smtp_port = 587

[AUTHENTICATION]
# 发送者邮箱
username = qwexxxx@qq.com
# 授权码,不是登录密码
password = aafrsdgsgsssfqw

[EMAIL]
# 发送者邮箱
from = qwexxxx@qq.com
subject = Test Email(邮件主题)
body = 这是我的应用程序发送的测试电子邮件。

[SECURITY]
use_tls = yes
tls_insecure_skip_verify = no


;这个配置文件包含以下部分:
;SMTP_SERVER: 存储 SMTP 服务器的主机名和端口号。
;AUTHENTICATION: 存储用于邮件认证的用户名和密码。
;EMAIL: 存储发送邮件时使用的发件人地址、邮件主题和邮件正文。
;SECURITY: 存储安全设置,比如是否使用 TLS 加密以及是否跳过 TLS 证书验证。
;请注意,你需要将示例中的占位符(如 smtp.example.com、your_email@example.com 和 your_password)替换为实际的值。此外,use_tls 和 tls_insecure_skip_verify 用于配置安全选项,具体值取决于你的邮件服务器配置。
;在实际使用中,出于安全考虑,不建议在配置文件中明文存储密码。可以使用环境变量或加密存储密码的方式来提高安全性。

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

实验11 OSPF协议配置

实验11 OSPF协议配置 一、OSPF单区域配置&#xff08;一&#xff09;原理描述&#xff08;二&#xff09;实验目的&#xff08;三&#xff09;实验内容&#xff08;四&#xff09;实验配置&#xff08;五&#xff09;实验步骤 二、OSPF多区域配置&#xff08;一&#xff09;原理…

44-5 waf绕过 - SQL注入绕WAF方法

环境准备: 43-5 waf绕过 - 安全狗简介及安装-CSDN博客然后安装sqlilabs靶场:构建完善的安全渗透测试环境:推荐工具、资源和下载链接_渗透测试靶机下载-CSDN博客 一、双写绕过 打开sql靶场的第一关:http://127.0.0.1/sqli-labs-master/Less-1/?id=1 验证一下waf是否开启防…

创新指南|2024企业如何开启生成式AI创新?从5大应用场景和6步抓手

想要了解如何采用生成式AI来提高企业效率和竞争力&#xff1f;本指南将介绍如何采用生成式AI来实现数字化转型&#xff0c;并打造智能化商业模式。从5大应用场景和6大步骤切入&#xff0c;让您了解如何开启生成式AI创新。立即连线创新专家咨询或观看创新战略方案视频进一步了解…

具有可编程电流限制的1.5A电源开关LPW5210用于5V或USB供电输出过流保护只要3毛

前言 适合要求反应时间较快的保护电路&#xff0c;保险丝或自恢复保险丝也能起到保护作用&#xff0c;但断开电流是额定电流的一倍&#xff0c;过流较小时&#xff0c;甚至需要数秒或更长的时间才能保护&#xff0c;因此半导体的过流保护开关更合适&#xff0c;相对成本要高一…

ABC318-E

挺有意思的一题&#xff0c;就当积累一下吧。 做法 枚举i和k会超时&#xff0c;那就只枚举j。 #include<bits/stdc.h> using namespace std; int n; int a[300010]; vector<int> v[300010]; int main(){ scanf("%d",&n); map<int,int&…

MQ之初识kafka

1. MQ简介 1.1 MQ的诞生背景 以前网络上的计算机&#xff08;或者说不同的进程&#xff09;传递数据&#xff0c;通信都是点对点的&#xff0c;而且要实现相同的协议&#xff08;HTTP、 TCP、WebService&#xff09;。1983 年的时候&#xff0c;有个在 MIT 工作的印度小伙突发…

Android AAudio——C API控制音频流(四)

上一篇文章我们介绍了 C API 中音频流的创建流程,以及打开音频流操作,这里我们再来看一下音频流的其他操作流程 一、音频流操作介绍 1、操作流程图 下图是状态变化流程图,虚线框表示瞬时状态,实线框表示稳定状态。 2、操作函数 上图中主要包含下面几个操作函数: aaudio…

AI-WEB-1 vulnhub靶场

AI-WEB-1 端口扫描 仅开放80端口 访问80端口 啥也没有 目录扫描 查看robots.txt 发现两个新目录 Disallow: /m3diNf0/ Disallow: /se3reTdir777/uploads/全都无权限访问 加入路径后再次扫描目录 发现/m3diNf0/目录下存在info.php&#xff0c;/se3reTdir777/目录下存在ind…

大文件上传处理:分卷压缩

大文件上传处理&#xff1a;分卷压缩 大文件上传处理&#xff1a;分卷压缩1、分卷压缩&#xff08;1&#xff09;Bandizip压缩工具&#xff1a;&#xff08;2&#xff09;分卷压缩后&#xff1a; 2、分卷压缩解压3、解压缩工具下载 大文件上传处理&#xff1a;分卷压缩 1、分卷…

将单列数据帧转换成多列数据帧

文章目录 1. 查看数据文件2. 读取数据文件得到单例数据帧3. 将单列数据帧转换成多列数据帧 在本次实战中&#xff0c;我们的目标是将存储在HDFS上的以逗号分隔的文本文件student.txt转换为结构化的Spark DataFrame。首先&#xff0c;使用spark.read.text读取文件&#xff0c;得…

Python | Leetcode Python题解之第132题分割回文串II

题目&#xff1a; 题解&#xff1a; class Solution:def minCut(self, s: str) -> int:n len(s)g [[True] * n for _ in range(n)]for i in range(n - 1, -1, -1):for j in range(i 1, n):g[i][j] (s[i] s[j]) and g[i 1][j - 1]f [float("inf")] * nfor …

TCP 建链(三次握手)和断链(四次握手)

TCP 建链&#xff08;三次握手&#xff09;和断链&#xff08;四次挥手&#xff09; 背景简介建链&#xff08;三次握手&#xff09;断链&#xff08;四次挥手&#xff09;序号及标志位延伸问题为什么建立连接需要握手三次&#xff0c;两次行不行&#xff1f;三次握手可以携带数…

【Vue】Vue生命周期

Vue生命周期&#xff1a;就是一个Vue实例从创建&#xff08;new一个Vue实例&#xff09; 到 销毁&#xff08;关闭网页&#xff09; 的整个过程。 生命周期四个阶段&#xff1a;① 创建 ② 挂载 ③ 更新 ④ 销毁 创建阶段&#xff1a;创建响应式数据 通过data给当前的Vue实例提…

【设计模式】结构型-适配器模式

前言 在软件开发中&#xff0c;经常会遇到需要将一个类的接口转换成另一个类的接口的情况。这可能是因为新旧系统之间的接口不兼容&#xff0c;或者是因为需要使用的第三方库的接口与当前系统的接口不匹配。为了解决这类问题&#xff0c;设计模式中的适配器模式应运而生。 一…

Leetcode3164. 优质数对的总数 II

Every day a Leetcode 题目来源&#xff1a;3164. 优质数对的总数 II 解法1&#xff1a;统计因子 遍历 nums1&#xff0c;统计所有元素的因子个数&#xff0c;记录到哈希表 cnt 中。 遍历 nums2&#xff0c;那么有 cnt[nums2[i]*k] 个数可以被 nums2[i]*k 整除&#xff0c;…

容器化部署Pig微服务快速开发框架

系统说明 基于 Spring Cloud 、Spring Boot、 OAuth2 的 RBAC 企业快速开发平台&#xff0c; 同时支持微服务架构和单体架构 提供对 Spring Authorization Server 生产级实践&#xff0c;支持多种安全授权模式 提供对常见容器化方案支持 Kubernetes、Rancher2 、Kubesphere、E…

南昌代理记账公司的收费标准及咨询服务

随着现代商业的快速发展&#xff0c;对于财务管理的需求也在不断增加&#xff0c;作为一家专业的会计代理公司&#xff0c;我们的目标是为各类企业提供全面、高效的财务管理服务&#xff0c;任何服务都应以公平合理的价格为基础&#xff0c;我们一直坚持这一原则。 关于常州代…

NDIS网络接口

在windows发行版本中&#xff0c;真的存在一个 ndis.sys 的驱动文件&#xff0c;和我们认知的不太一样&#xff0c;它真的是一个DLL&#xff0c;NDIS 库打包在 Ndis.sys&#xff08;一个内核模式导出库&#xff09;中&#xff0c;作为一组函数&#xff0c;强调宏以获得最佳性能…

0基础学习区块链技术——链之间数据同步样例

我们可以在https://blockchaindemo.io/体验这个过程。 创建区块 默认第一个链叫Satoshi(中本聪)。链上第一个区块叫“创世区块”——Genesis Block。后面我们会看到创建的第二条链第一个区块也是如此。 新增链 新创建的链叫Debby。默认上面有一个创世区块。 然后我们让这…

可视化数据科学平台在信贷领域应用系列一:数据探索

引言 信贷风险数据建模是金融机构在数据量日益庞杂的时代进行信贷业务风控的关键技术。它能够帮助机构更好地控制风险、减少违约损失&#xff0c;并提高业务效率。通过不断优化建模方法和利用建模工具&#xff0c;金融机构的风险控制能力得到了显著提升。 在本文中&#xff0c;…