1. SMTP简介
SMTP协议是邮件发送的应用层协议,被称为简单邮件传输协议,采用请求应答方式进行通讯,SMTP命令和响应是基于ASCII文本的,并且以回车换行符(CR和LF)结束,SMTP服务端的底层一般使用TCP协议,在掌握了TCP通讯的方法后,就可以自己实现一个简单的邮件发送客户端,关于TCP通讯的常用方法可以参见我上一篇文章鸿蒙网络编程系列3-TCP客户端通讯示例
SMTP的标准命令有14个,后来通过扩展命令又增加了一些,针对本文的示例,需要了解ehlo(问候信息)、auth login(登录)、mail from(发件人)、rcpt to(收件人)、data(邮件内容)、quit(登出)等命令,关于这些命令的详细解释可以查找相关文档,这些命令比较简单,不深入研究的话,光看看本文的具体示例也大概能明白用法。
2.邮件发送客户端示例
本示例演示登录腾讯邮箱SMTP服务器并发送邮件的过程,不同的邮件服务器对密码的定义可能不一样,在腾讯的邮件服务器里,密码是指授权码,可以登录官方网站了解生成方式。本示例成功登录并发送邮件后的截图如下所示:
对于应用界面,上部是邮箱服务器地址和用户名密码,腾讯邮箱服务器地址为smtp.qq.com,对应的ip地址为157.148.54.34;中部是要发送的邮件信息,包括收件人邮箱、标题、内容等部分;最下面是发送日志,把客户端和服务端交互的过程记录下来,方便调试分析。
下面详细介绍创建该应用的步骤。
步骤1:创建Empty Ability项目。
步骤2:在module.json5配置文件加上对权限的声明:
"requestPermissions": [
{
"name": "ohos.permission.INTERNET"
}
]
这里添加了访问互联网的权限。
步骤3:在Index.ets文件里添加如下的代码:
import socket from '@ohos.net.socket';
import util from '@ohos.util';
//执行TCP通讯的对象
let tcpSocket = socket.constructTCPSocketInstance();
@Entry
@Component
struct Index {
//连接、通讯历史记录
@State msgHistory: string = ''
//套接字是否已绑定本地地址
bindLocal: boolean = false
//服务器是否响应(发送数据到客户端)
isServerResponse: boolean = false
//服务端地址,smtp.qq.com的ip地址为157.148.54.34
@State serverAddr: string = "157.148.54.34"
//服务端端口,smtp.qq.com的端口为587
@State serverPort: number = 587
//用户名
@State userName: string = "用户名,一般是你的邮箱地址"
//密码,对于腾讯邮箱,这里是授权码
@State passwd: string = "you auth code or password"
//收件人邮箱列表(如果多个使用逗号分隔)
@State rcptList: string = "8888@qq.com,9999@gmail.com"
//邮件标题
@State mailTitle: string = "测试邮件标题"
//发件人邮箱
@State mailFrom: string = "你的邮箱"
//邮件内容
@State mailContent: string = "This is greeting from Harmony OS"
//是否可以登录
@State canLogin: boolean = false
//是否可以发送邮件
@State canSend: boolean = false
scroller: Scroller = new Scroller()
build() {
Row() {
Column() {
Text("邮件发送客户端")
.fontSize(14)
.fontWeight(FontWeight.Bold)
.width('100%')
.textAlign(TextAlign.Center)
.padding(10)
Flex({ justifyContent: FlexAlign.Start, alignItems: ItemAlign.Center }) {
Text("邮箱服务器地址:")
.width(120)
.fontSize(14)
.flexGrow(0)
TextInput({ text: this.serverAddr.toString() })
.onChange((value) => {
this.serverAddr = value
})
.width(110)
.fontSize(12)
.flexGrow(1)
}.width('100%')
.padding(5)
Flex({ justifyContent: FlexAlign.Start, alignItems: ItemAlign.Center }) {
Text("邮箱服务器端口:")
.width(120)
.fontSize(14)
.flexGrow(0)
TextInput({ text: this.serverPort.toString() })
.type(InputType.Number)
.onChange((value) => {
this.serverPort = parseInt(value)
})
.width(110)
.fontSize(12)
.flexGrow(1)
}.width('100%')
.padding(5)
Flex({ justifyContent: FlexAlign.Start, alignItems: ItemAlign.Center }) {
Text("邮箱用户名:")
.width(90)
.fontSize(14)
.flexGrow(0)
TextInput({ text: this.userName.toString() })
.onChange((value) => {
this.userName = value
})
.width(110)
.fontSize(12)
.flexGrow(1)
}.width('100%')
.padding(5)
Flex({ justifyContent: FlexAlign.Start, alignItems: ItemAlign.Center }) {
Text("登录密码(授权码):")
.width(130)
.fontSize(14)
.flexGrow(0)
TextInput({ text: this.passwd.toString() })
.onChange((value) => {
this.passwd = value
})
.width(100)
.fontSize(12)
.flexGrow(1)
Button("登录")
.onClick(() => {
this.login()
})
.width(70)
.fontSize(14)
.flexGrow(0)
}.width('100%')
.padding(5)
Flex({ justifyContent: FlexAlign.Start, alignItems: ItemAlign.Center }) {
Text("收件人邮箱:")
.width(90)
.fontSize(14)
.flexGrow(0)
TextInput({ placeholder: "多个使用逗号分隔",text:this.rcptList })
.onChange((value) => {
this.rcptList = value
})
.width(110)
.fontSize(12)
.flexGrow(1)
}.width('100%')
.padding(5)
Flex({ justifyContent: FlexAlign.Start, alignItems: ItemAlign.Center }) {
Text("标题:")
.width(50)
.fontSize(14)
.flexGrow(0)
TextInput({text:this.mailTitle})
.onChange((value) => {
this.mailTitle = value
})
.width(110)
.fontSize(12)
.flexGrow(1)
}.width('100%')
.padding(5)
Flex({ justifyContent: FlexAlign.Start, alignItems: ItemAlign.Center }) {
Text("发件人邮箱:")
.width(90)
.fontSize(14)
.flexGrow(0)
TextInput({text:this.mailFrom})
.onChange((value) => {
this.mailFrom = value
})
.width(110)
.fontSize(12)
.flexGrow(1)
}.width('100%')
.padding(5)
Flex({ justifyContent: FlexAlign.Start, direction: FlexDirection.Column, alignItems: ItemAlign.Center }) {
Text("邮件内容:")
.width('100%')
.fontSize(14)
TextArea({text:this.mailContent})
.onChange((value) => {
this.mailContent = value
})
.width('100%')
.height(80)
.fontSize(12)
Button("发送")
.enabled(this.canSend)
.onClick(() => {
this.sendMail()
})
.width(70)
.fontSize(14)
Scroll(this.scroller) {
Text(this.msgHistory)
.textAlign(TextAlign.Start)
.padding(10)
.width('100%')
.backgroundColor(0xeeeeee)
.fontSize(10)
}
.align(Alignment.Top)
.backgroundColor(0xeeeeee)
.height(200)
.flexGrow(1)
.scrollable(ScrollDirection.Vertical)
.scrollBar(BarState.On)
.scrollBarWidth(20)
}
.flexGrow(1)
.width('100%')
.padding(5)
.height(300)
}
.width('100%')
.justifyContent(FlexAlign.Start)
.height('100%')
.padding(10)
}
.height('100%')
}
//发送邮件
async sendMail() {
//发送发件人信箱
await this.exeCmdAndWait4Response(`mail from:<${this.mailFrom}>`)
let rcptMails = this.rcptList.split(',')
for(let i=0;i<rcptMails.length;i++){
//发送收件人信箱
let rcpt = rcptMails[i]
await this.exeCmdAndWait4Response(`rcpt to:<${rcpt}>`)
}
//准备发送邮件内容
await this.exeCmdAndWait4Response("data")
let mailBody =`Subject: ${this.mailTitle} \r\nFrom: ${this.mailFrom}\r\n\r\n${this.mailContent}\r\n.`
//发送邮件内容
await this.exeCmdAndWait4Response(mailBody)
//登出
await this.exeCmdAndWait4Response("quit")
}
async bindSocket() {
//本地地址
let localAddress = { address: "0.0.0.0", family: 1 }
await tcpSocket.bind(localAddress)
.then(() => {
this.msgHistory += 'C:bind success' + "\r\n";
})
.catch((e) => {
this.msgHistory += 'C:bind fail ' + e.message + "\r\n";
})
//收到消息时的处理
tcpSocket.on("message", async (value) => {
this.isServerResponse = true
let msg = buf2String(value.message)
this.msgHistory += "S:" + msg + "\r\n"
this.scroller.scrollEdge(Edge.Bottom)
})
this.bindLocal = true
}
//登录服务器
async login() {
//首先判断套接字是否绑定到本地地址,如果没有绑定就绑定一下
if (!this.bindLocal) {
this.bindSocket()
}
//服务器地址
let serverAddress = { address: this.serverAddr, port: this.serverPort, family: 1 }
//连接smtp服务器
await tcpSocket.connect({ address: serverAddress })
.then(() => {
this.msgHistory += 'C:connect success ' + "\r\n";
})
.catch((e) => {
this.msgHistory += 'C:connect fail ' + e.message + "\r\n";
return
})
//等待服务器响应
await this.wait4ServerResponse()
//服务端发送ehlo,anyname为发送服务器名称,可以随便写,但不能没有
await this.exeCmdAndWait4Response("ehlo anyname")
//告诉服务器,我要登录了
await this.exeCmdAndWait4Response("auth login")
//发送用户名,用户名需要base64编码
let loginName = string2Base64(this.userName)
await this.exeCmdAndWait4Response(loginName)
//发送密码,密码需要base64编码,对于腾讯邮箱,这里是授权码,具体的可以参考腾讯邮箱文档
//你说为什么要编码?自欺欺人罢了,起不到加密作用,还给新手带来一堆bug
//正常情况下,登录后服务器就认可你了,下面就可以正式发送邮件了
let passWd = string2Base64(this.passwd)
await this.exeCmdAndWait4Response(passWd)
//设置发送按钮可用,当然,这里还要一堆逻辑需要判断,比如判断服务端的返回信息是否表明登录成功了
//毕竟本例只是演示smtp的使用,实际中可以添加上这些逻辑
this.canSend = true
}
//给服务器发送命令并等待响应
async exeCmdAndWait4Response(cmd:string){
this.isServerResponse = false
let result = await this.sendCmd2ServerWithCRLF(cmd)
if (result != true) {
return
}
//等待服务器响应
await this.wait4ServerResponse()
}
//等待服务器响应
async wait4ServerResponse() {
while (!this.isServerResponse) {
await sleep(100)
}
}
//发送命令到服务端,自动在命令后加上回车换行
async sendCmd2ServerWithCRLF(cmd: string): Promise<boolean> {
cmd = cmd + "\r\n"
let result = false
await tcpSocket.send({ data: cmd })
.then(() => {
this.msgHistory += "C:" + cmd;
result = true
})
.catch((e) => {
this.msgHistory += 'C:send fail ' + e.message + "\r\n";
result = false
})
return result
}
}
//对字符串base64编码
function string2Base64(src: string) {
let textEncoder = new util.TextEncoder();
let encodeValue = textEncoder.encodeInto(src)
let tool = new util.Base64Helper()
return tool.encodeToStringSync(encodeValue)
}
//ArrayBuffer转utf8字符串
function buf2String(buf: ArrayBuffer) {
let msgArray = new Uint8Array(buf);
let textDecoder = util.TextDecoder.create("utf-8");
return textDecoder.decodeWithStream(msgArray)
}
//休眠指定的毫秒数
function sleep(time) {
return new Promise((resolve) => setTimeout(resolve, time));
}
步骤4:编译运行,可以使用模拟器或者真机。
步骤5:配置服务器信息、填写用户名密码以及要发送的邮件信息,然后单击“登录”按钮进行登录,成功后单击“发送”按钮发送邮件。
步骤6:查看收件人信箱看看是否发送成功,首先是第一个腾讯邮箱,发送成功了:
然后看看第二个,第二个是谷歌的gmail信箱,也成功了:
这样,我们就创建了鸿蒙版本的邮件发送客户端。
3.注意事项
在SMTP协议里,对于用户名和密码需要转换为base64格式,所以代码里包括了一个string2Base64()函数,执行这个转换。
(本文作者原创,除非明确授权禁止转载)
本文源码地址:
https://gitee.com/zl3624/harmonyos_network_samples/tree/maste...
本系列源码地址:
**粗体** _斜体_ [链接](http://example.com) `代码` - 列表 > 引用
。你还可以使用@
来通知其他用户。