常见原因是SMTP服务端拒绝未认证或未加密连接,需用PlainAuth显式认证、选587/465端口、QQ邮箱用授权码、Gmail用应用专用密码。
net/smtp 发不出邮件却没报错?常见现象是调用 smtp.SendMail 返回 nil 错误,但收件人根本没收到——这通常不是代码问题,而是 SMTP 服务端拒绝了未认证或未加密的连接。Gmail、Outlook、QQ 邮箱等主流服务商默认禁用不带身份验证的明文 SMTP(端口 25),也拒绝未启用 TLS 的连接。
auth := smtp.PlainAuth("", user, password, host) 显式传入认证凭据587(STARTTLS)或 465(SMTPS),避免用 25
user 必须是完整邮箱地址(如 "xxx@qq.com"),密码需填「SMTP 授权码」而非登录密码535 5.7.8 Username and Password not accepted
Go 标准库没有开箱即用的 MIME 构建器,得手动拼装 multipart/mixed 和 multipart/alternative 结构。关键在于边界(boundary)一致、头部顺序正确、正文编码合规。
mime/multipart 创建 Writer,调用 w.WriteField("To", "...") 设置基础头字段multipart/alternative 子部分(含纯文本 text/plain 和 HTML text/html),再写附件部分base64 编码,并声明 Content-Transfer-Encoding: base64
mime.BEncoding.Encode("UTF-8", "简历.pdf") 编码,否则 Outlook 会显示乱码body := &bytes.Buffer{}
writer := multipart.NewWriter(body)
writer.SetBoundary("myboundary123")
// HTML 部分
part, _ := writer.CreatePart(map[string][]string{
"Content-Type": {"text/html; charset=UTF-8"},
"Content-Transfer-Encoding": {"base64"},
})
base64.NewEncoder(base64.StdEncoding, part).Write([]byte(`你好
`))
// 附件
part, _ = writer.CreatePart(map[string][]string{
"Content-Type":
{"application/pdf"},
"Content-Disposition": {`attachment; filename*=UTF-8''%E7%AE%80%E5%8E%86.pdf`},
})
io.Copy(part, file)
writer.Close()
错误信息常被忽略,但 smtp.SendMail 返回的 error 值里包含原始 SMTP 状态码和服务器提示,是第一手线索。
*textproto.Error 类型:if e, ok := err.(*textproto.Error); ok { fmt.Println(e.Code, e.Msg) }
421 表示服务不可用(DNS 解析失败或目标主机拒连);454 是 TLS 启动失败;535 是认证失败;554 多为内容被判定为垃圾邮件telnet smtp.qq.com 587 手动走一遍 EHLO → AUTH → MAIL FROM 流程,确认基础连通性550 Mail from ... not allowed,说明发信域名未配置 SPF 或 DKIM,需在 DNS 添加记录直接调用第三方 SMTP(如 SendGrid、Mailgun、腾讯云 SES)比自建更可靠,尤其对通知类高频小邮件。
os.Getenv("MAILGUN_API_KEY") + .env 文件管理