全部例程

SMTP

W55MH32 其他标签

2025/02/17 更新

本篇文章,我们将详细介绍如何在W55MH32芯片上面实现SMTP协议。并通过实战例程,为大家讲解如何在W55MH32上使用SMTP协议给他人发送电子邮件。

该例程用到的其他网络协议,例如DHCPDNS, 请参考相关章节。有关 W55MH32 的初始化过程,请参考Network install,这里将不再赘述。

SMTP协议简介

SMTP(Simple Mail Transfer Protocol,简单邮件传输协议) 是一种用于电子邮件传输的通信协议。它是互联网标准协议之一,专门设计用于电子邮件的发送和路由。SMTP定义了邮件如何从发件人发送到收件人的电子邮件服务器,并规范了服务器之间的邮件中继操作。

SMTP协议特点

  • 面向文本:SMTP协议使用纯文本命令和响应(如HELO、MAIL FROM、RCPT TO等)来进行通信,邮件内容通常是以ASCII码表示。
  • 请求-响应模式:SMTP通信是基于请求-响应模型的,客户端发送请求,服务器根据请求返回响应。
  • 基于TCP:SMTP依赖于TCP协议来提供可靠的数据传输服务。SMTP会通过TCP建立连接并发送邮件。
  • 广泛兼容性:SMTP 是国际标准邮件传输协议,广泛应用于邮件系统,确保不同系统间高效互通。
  • 高效性与可靠性:SMTP 协议简单易用,支持错误处理和重试机制。邮件无法发送时,可暂存队列并重试,确保传输可靠。
  • 可扩展性:通过扩展SMTP协议的命令和响应码,可以支持更多的邮件传输特性和功能。
  • 安全性:SMTP 本身不处理加密,但可结合 SSL/TLS(SMTPS)提供加密通道,保障邮件传输的安全性。
  • 异步传输:SMTP 支持异步传输,使邮件发送和接收可在不同时间进行,提升效率并支持批量处理。
  • 灵活性:SMTP 设计灵活,可配置邮件路由、优先级、大小限制等,以满足不同需求。

SMTP应用场景

接下来,我们了解下在W55MH32上,可以使用SMTP协议完成哪些操作及应用呢?

  • 物联网(IoT)设备远程监控:使用W55MH32实现SMTP通信,收集传感器数据,并发送定期报告或警报邮件。
  • 环境监测系统:W55MH32通过SMTP协议将监测数据(如温湿度、CO2浓度等)发送到指定的邮箱,确保及时获取信息。
  • 设备状态报告与日志记录:通过定时任务触发W55MH32每隔一段时间自动发送设备的状态报告或运行日志到管理人员邮箱。
  • 远程故障报警与支持:结合传感器和W55MH32的邮件发送功能,可以实现自动报警,减少人工干预的需要。
  • 工业自动化和远程监控:使用W55MH32连接到互联网,并利用SMTP协议将设备状态、报警等信息发送到指定的邮箱。
  • 远程控制反馈:通过W55MH32接收邮件,解析邮件命令并执行相应操作,然后通过SMTP将执行结果反馈给发送者。

SMTP发送邮件流程

  1. 使用TCP协议连接SMTP服务器
  2. 发送握手消息
  3. 发送用户认证消息
  4. 设置邮件发送地址
  5. 传输邮件内容
  6. 完成邮件发送

STMP协议的主要命令

Blog Image

SMTP服务器响应状态码

Blog Image

实现过程

接下来,我们在W55MH32上实现SMTP邮件发送功能。

步骤1:SMTP发送内容初始化
发送指令定义:


char    hello[50]     = "HELO localhost";                   // Identity command
char    hello_reply[] = "250 OK";                           // Id successfully responded
char    AUTH[50]      = "AUTH LOGIN";                       // Authentication request
char    AUTH_reply[]  = "334 dXNlcm5hbWU6";                 // The authentication request was successfully 
sent
char    name_126[100] = "[email protected]";                 // 126 Login email address
char    base64name_126[200];                                // 126 base64 encoding of the login mailbox name
char    name_reply[]     = "334 UGFzc3dvcmQ6";              // The login name was sent successfully
char    password_126[50] = "ZPURADLGRUPQLVBK";              // 126 Email login password
char    base64password_126[100];                            // base64 123 Password for logging in to the 
mailbox
char    password_reply[] = "235 Authentication successful"; // Login successful response
char    from[]           = "[email protected]";              // Sender email
char    from_reply[]     = "250 Mail OK";
char    to[]             = "[email protected]";             // Recipient email address
char    to_reply[]       = "250 Mail OK";
char    data_init[10]    = "data";                          // Request data transfer
char    data_reply[]     = "354";                           // The request was successfully responded to 
HEAD
char    Cc[]             = "";                              // Cc to email
char    subject[]        = "Hello!WIZnet!";                 // subject
char    content[]        = "Hello!WIZnet!";                 // text part
char    mime_reply[]     = "250 Mail OK queued as";         // The email was sent successfully
char    mailfrom[50]     = "MAIL FROM:<>";
char    rcptto[50]       = "rcpt to:<>";
char    mime[200]        = "From:\r\n";
char    mime1[50]        = "To:\r\n";
char    mime2[50]        = "Cc:\r\n";
char    mime3[50]        = "Subject:\r\n";
char    mime4[50]        = "MIME-Version:1.0\r\nContent-Type:text/plain\r\n\r\n";
char    mime5[50]        = "\r\n.\r\n";

步骤2:发送邮件内容初始化:

mailmessage(); // Mail command information processing

mailmessage()函数内容如下:


void mailmessage(void)
{
    uint16_t len_from = strlen(from);
    uint16_t len_to   = strlen(to);
    uint16_t len_Cc   = strlen(Cc);
    uint16_t len_sub  = strlen(subject);
    strcat(hello, "\r\n");
    strcat(AUTH, "\r\n");
    base64encode(name_126, base64name_126);
    base64encode(password_126, base64password_126);
    strcat(base64name_126, "\r\n");
    strcat(base64password_126, "\r\n");
    str_insert(mailfrom, from, 11);
    strcat(mailfrom, "\r\n");
    str_insert(rcptto, to, 9);
    strcat(rcptto, "\r\n");
    strcat(data_init, "\r\n");
  
    str_insert(mime, from, 5);
    str_insert(mime1, to, 3);
    str_insert(mime2, Cc, 3);
    str_insert(mime3, subject, 8);
    str_insert(mime5, content, 0);
    strcat(mime, mime1);
    strcat(mime, mime2);
    strcat(mime, mime3);
    strcat(mime, mime4);
    strcat(mime, mime5);
}

步骤3:使用DNS协议解析SMTP服务器地址


if (do_dns(ethernet_buf, smtp_server_name, smtp_server_ip))
{
    while (1)
    {
    }
}

步骤4:SMTP发送邮件操作


while (1)
{
    do_smtp(SOCKET_ID, ethernet_buf, smtp_server_ip); // smtp run
}

do_smtp()函数内容如下:


void do_smtp(uint8_t sn, uint8_t *buf, uint8_t *smtp_server_ip)
{
  volatile uint8_t ret;
  uint32_t         len       = 0;
  uint16_t         anyport   = 5000;
  uint8_t          Smtp_PORT = 25;
  memset(buf, 0, ETHERNET_MAX_BUF_SIZE);
  switch (getSn_SR(sn))
  {
  case SOCK_INIT:
      ret = connect(sn, smtp_server_ip, Smtp_PORT);
      break;
  case SOCK_ESTABLISHED:
      if (getSn_IR(sn) & Sn_IR_CON)
      {
          setSn_IR(sn, Sn_IR_CON);
      }

      while (!Mail_Send_OK)
      {
          len = getSn_RX_RSR(sn);
          if (len > 0)
          {
              memset(buf, 0, ETHERNET_MAX_BUF_SIZE);
              len = recv(sn, (uint8_t *)buf, len);
              send_mail(sn, buf, smtp_server_ip);
          }
      }
      disconnect(sn);
      break;
  case SOCK_CLOSE_WAIT:
      if ((len = getSn_RX_RSR(sn)) > 0)
      {
          while (!Mail_Send_OK)
          {
              len = recv(sn, (uint8_t *)buf, len);
              send_mail(sn, buf, smtp_server_ip);
          }
      }
      disconnect(sn);
      break;
  case SOCK_CLOSED:
      socket(sn, Sn_MR_TCP, anyport++, 0x00);
      break;
  default:
      break;
  }
  if (Mail_Send_OK)
  {
      while (1)
      {
      }
  }
}

在该函数中,程序会执行一个 TCP Client 模式的状态机,详细讲解请参考TCP Client章节,这里不再赘述。当程序处于 SOCK_ESTABLISHED (即成功连接上SMTP服务器)状态时,SMTP服务器会主动发送一条消息给W55MH32,当接收到这条消息后,进入send_mail()函数进行SMTP发送邮件流程:
send_mail()函数内容如下:


void send_mail(uint8_t sn, uint8_t *buf, uint8_t *smtp_server_ip)
{
    volatile uint8_t ret;
    switch (SMTP_STATE)
    {
    case waitfor220:
        if (strstr((const char *)buf, "220") != NULL)
        {
            ret        = send(sn, (uint8_t *)hello, strlen(hello));
            SMTP_STATE = waitforHELO250;
        }
        else
        {
            printf("Connected failed!\r\n");
        }
        break;
    case waitforHELO250:
        if (strstr((const char *)buf, hello_reply) != NULL && strstr((const char *)buf, "Mail") == NULL)
        {
            ret        = send(sn, (uint8_t *)AUTH, strlen(AUTH));
            SMTP_STATE = waitforAUTH334;
        }
        else
        {
            printf("smtp handshake failed!\r\n");
        }
        break;
    case waitforAUTH334:
        if (strstr((const char *)buf, AUTH_reply) != NULL)
        {
            ret        = send(sn, (uint8_t *)base64name_126, strlen(base64name_126));
            SMTP_STATE = waitforuser334;
        }
        else
        {
            printf("AUTH authentication request failed!\r\n");
        }
        break;
    case waitforuser334:
        if (strstr((const char *)buf, name_reply) != NULL)
        {
            ret        = send(sn, (uint8_t *)base64password_126, strlen(base64password_126));
            SMTP_STATE = waitforpassword235;
        }
        else
        {
            printf("username send failed!\r\n");
        }
        break;
    case waitforpassword235:
        if (strstr((const char *)buf, password_reply) != NULL)
        {
            ret        = send(sn, (uint8_t *)mailfrom, strlen(mailfrom));
            SMTP_STATE = waitforsend250;
        }
        else
        {
            printf("password error!\r\n");
        }
        break;
    case waitforsend250:
        if (strstr((const char *)buf, from_reply) != NULL && strstr((const char *)buf, "queued as") == NULL)
        {
            ret        = send(sn, (uint8_t *)rcptto, strlen(rcptto));
            SMTP_STATE = waitforrcpt250;
        }
        else
        {
            printf("Send email failed to set up!\r\n");
        }
        break;
    case waitforrcpt250:
        if (strstr((const char *)buf, to_reply) != NULL)
        {
            ret        = send(sn, (uint8_t *)data_init, strlen(data_init));
            SMTP_STATE = waitfordate354;
        }
        else
        {
            printf("Failed to set the receiving mailbox!\r\n");
        }
        break;
    case waitfordate354:
        if (strstr((const char *)buf, data_reply) != NULL)
        {
            ret        = send(sn, (uint8_t *)mime, strlen(mime));
            SMTP_STATE = waitformime250;
        }
        else
        {
            printf("Failed to send content setup\r\n");
        }
        break;
    case waitformime250:
        if (strstr((const char *)buf, mime_reply) != NULL)
        {
            Mail_Send_OK = 1;
            printf("mail send OK\r\n");
        }
        break;
    default:
        break;
    }
}

send_mail()中,会执行一个发送邮件的状态机,流程如下图所示:

Blog Image

1.SMTP握手:发送 HELO命令,等待服务器返回 250 OK 响应码,确认握手成功。

2.用户认证:

(1) 请求认证:发送 AUTH LOGIN 命令,表明客户端需要认证,等待服务器返回 334 ,请求用户名。

(2) 提交用户名:发送经过Base64编码的用户名(如:base64name_126),等待服务器返回 334 ,请求密码。

(3) 提交密码:发送经过Base64编码的密码(如:base64password_126),等待服务器返回 235 ,确认认证成功。

3. 设置邮件发送地址:

(1) 设置发件人:发送 MAIL FROM:<发件人邮箱>,等待服务器返回 250 ,确认发件人地址成功设置。

(2) 设置收件人:发送 RCPT TO:<收件人邮箱>,等待服务器返回 250,确认收件人地址成功设置。

4. 传输邮件内容:

(1) 请求数据传输:发送 DATA 命令,表明开始传输邮件数据,等待服务器返回 354,表示已准备接收数据。

(2) 发送邮件数据:按以下格式发送邮件数据:

发件人信息:From:

收件人信息:To:

抄送信息:Cc:

邮件主题:Subject:

邮件内容:实际的文本部分

数据结束用 \r\n\r\n 表示

等待服务器返回 250 ,确认邮件内容已成功排队

5. 完成邮件发送:

如果服务器返回 250 ,标志邮件成功发送。

设置 Mail_Send_OK = 1,并打印提示信息 mail send OK

运行结果

请注意:

因为本示例需要访问互联网,请确保 W55MH32 的配置能够访问互联网。

烧录例程运行后,首先进行了PHY链路检测,接着是通过DHCP获取网络地址信息,然后是通过DNS解析SMTP服务器域名,最后是进行邮件发送,如下图所示:

Blog Image

可以在设置接收邮件的账号中查找接收到的邮件:

Blog Image

通过wireshark抓包查看,流程与send_mail()函数一致。

Blog Image

总结

本文讲解了如何在 W55MH32 芯片上实现 SMTP 协议,通过实例详细展示了在该芯片上使用 SMTP 协议发送电子邮件的实现流程,包括 SMTP 发送内容初始化、使用 DNS 协议解析 SMTP 服务器地址、SMTP 发送邮件操作等核心步骤。文章还对 SMTP 协议的简介、特点、应用场景,以及主要命令和服务器响应状态码进行了分析,帮助读者理解其在邮件传输中的实际应用价值。

下一篇文章将介绍NetBIOS的原理及在网络通信中的应用,同时讲解如何在W55MH32芯片上实现NetBIOS功能,敬请期待!

下载本章例程

我们提供完整的工程文件以及配套开发板,方便你随时测试,快速完成产品开发:

开发环境: Keil MDK5 配套开发板