POP3协议详解

2023-12-23 18:04
文章标签 详解 协议 pop3

本文主要是介绍POP3协议详解,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

基本介绍

POP3是一种用于从邮件服务器获取电子邮件的协议。它允许邮件客户端连接到邮件服务器,检索服务器上存储的邮件,并将邮件下载到客户端设备上。POP3的工作原理如下:

  1. 连接和身份验证: 邮件客户端通过TCP/IP连接到邮件服务器的POP3端口(通常是110)。客户端首先发送USER​命令提供用户名,然后发送PASS​命令提供密码以进行身份验证。
  2. 查看邮件状态: 客户端可以使用STAT​命令来查看邮箱中的邮件状态,包括邮件数量和总字节数。
  3. 查看邮件列表: 使用LIST​命令,客户端可以查看每封邮件的索引号和大小。如果提供邮件索引号,服务器将返回特定邮件的大小。
  4. 检索邮件: 使用RETR​命令,客户端可以检索特定邮件的内容,包括邮件头和正文。邮件被下载到客户端设备上。
  5. 删除邮件: 使用DELE​命令,客户端可以标记要删除的邮件。邮件在标记为删除后不会立即从服务器上删除,直到客户端执行QUIT​命令。
  6. 退出: 使用QUIT​命令,客户端结束POP3会话并断开连接。在断开连接之前,标记为删除的邮件将从服务器上删除。

相关概念

  • 邮件收件箱(Mailbox): 邮件服务器上存储用户接收到的电子邮件的地方。每个用户都有一个邮箱,其中包含他们的电子邮件。
  • 命令和响应: POP3协议使用一系列命令和响应进行通信。客户端发送命令(例如,USER​​,PASS​​,LIST​​等),服务器返回响应(例如,+OK​​表示成功,-ERR​​表示错误)。
  • 邮件索引号(Message Number): 每封邮件在邮箱中都有一个唯一的索引号。客户端可以使用索引号来指定要执行的操作,如检索或删除邮件。
  • 邮件大小(Message Size): 邮件的大小以字节为单位。LIST​​命令返回每封邮件的大小,以便客户端可以了解邮件的大小信息。
  • 标记为删除: 客户端可以使用DELE​​命令将邮件标记为删除。标记为删除的邮件在QUIT​​命令之前不会从服务器上真正删除。
  • 邮件头和正文: 邮件通常由邮件头和正文组成。邮件头包含元数据,如发件人、主题和日期。邮件正文包含邮件的实际内容。
  • MIME(Multipurpose Internet Mail Extensions): MIME是一种电子邮件标准,用于描述邮件内容的类型和格式,包括文本、HTML、图像和附件等。
  • 连接池(Connection Pool): 一些POP3客户端使用连接池来管理与邮件服务器的连接,以提高性能和效率。

POP3命令与响应

Transaction State

一旦客户机成功地向 POP3服务器标识了自己并且 POP3服务器已经锁定并打开了相应的邮件包, POP3会话现在处于 TRANSACTION 状态. 重复发出以下任何 POP3命令命令,POP3服务器发出响应 发出 QUIT 命令,POP3会话进入 UPDATE 状态。

STAT命令

  • STAT​:查询邮箱中邮件的统计信息,包括邮件数量和总字节数。

命令格式:

STAT

服务器应该响应一个包含两个值的响应:邮件数量和邮箱中所有邮件的总字节数。响应的格式如下:

+OK <邮件数量> <总字节数>

LIST命令

获取邮件服务器返回每封邮件的详细信息,包括邮件的索引号和大小。

命令格式:

LIST [<message_number>]
  • <message_number>​ 是可选的参数,用于指定要列出的特定邮件的索引号。如果省略了此参数,则会列出所有邮件的信息。

服务器会以以下格式响应LIST​命令:

如果未指定<message_number>​(列出所有邮件):

+OK <邮件总数> messages (<邮件总大小> octets)
<邮件1的索引号> <邮件1的大小>
<邮件2的索引号> <邮件2的大小>
...
<邮件n的索引号> <邮件n的大小>
.
  • +OK​ 表示命令执行成功。
  • <邮件总数>​ 是邮箱中邮件的数量。
  • <邮件总大小>​ 是所有邮件的总大小(以字节为单位)。
  • 每行中的 <邮件i的索引号>​ 表示第i封邮件的索引号。
  • 每行中的 <邮件i的大小>​ 表示第i封邮件的大小(以字节为单位)。

如果指定了<message_number>​(列出特定邮件):

+OK <邮件索引号> <邮件大小>
  • +OK​ 表示命令执行成功。
  • <邮件索引号>​ 是指定邮件的索引号。
  • <邮件大小>​ 是指定邮件的大小(以字节为单位)。

RETR

检索特定邮件的内容,其中邮件的索引号由客户端指定。RETR命令可用于下载邮件的正文和标头。

命令格式:

RETR <message_number>

服务器会以以下格式响应RETR​命令:

如果命令执行成功:

+OK
<邮件内容>
.
  • +OK​ 表示命令执行成功。
  • <邮件内容>​ 包含了邮件的完整内容,包括邮件头和正文。通常,邮件内容以文本格式返回。
  • 最后一行的.​(句号)表示邮件内容的结束。

如果命令执行失败(邮件不存在或索引号无效):

-ERR <错误消息>
  • -ERR​ 表示命令执行失败。
  • <错误消息>​ 包含有关失败原因的描述。

DELE

标记特定邮件以删除它们,但不会立即从服务器上删除它们。删除操作只有在客户端执行QUIT​命令时才会生效。

命令格式:

DELE <message_number>

**服务器响应: **

如果命令执行成功(邮件被成功标记为删除):

+OK Message <message_number> deleted
  • +OK​ 表示命令执行成功。
  • <message_number>​ 是已标记为删除的邮件的索引号。

如果命令执行失败(邮件不存在或索引号无效):

-ERR <错误消息>
  • -ERR​ 表示命令执行失败。

    • <错误消息>​ 包含有关失败原因的描述。

NOOP

它不执行任何实际的操作。它的主要目的是告诉邮件服务器,客户端仍然活跃,正在使用连接,以防止连接因长时间不活动而被服务器关闭。

REST

重置或取消标记为删除的邮件,以便这些邮件不会被删除。
RSET命令通常用于撤销之前使用DELE命令标记为删除的邮件。

命令格式:

RSET

命令响应:

如果命令执行成功:

+OK
  • +OK​ 表示命令执行成功。

如果命令执行失败(通常是因为RSET​命令不支持或出现其他问题):

-ERR <错误消息>
  • -ERR​ 表示命令执行失败。
  • <错误消息>​ 包含有关失败原因的描述。

UPDATE State

QUIT

告知邮件服务器客户端已经完成了邮件收取或管理任务,并请求结束与服务器的连接。

Optional命令

TOP

检索邮件的头部信息和指定行数的邮件正文。
TOP​命令允许客户端请求邮件服务器返回特定邮件的头部信息和部分邮件正文,以减少数据传输量和加快邮件检索速度。

命令格式:

TOP <message_number> <number_of_lines>
  • <message_number>​ 是要检索的邮件的索引号。客户端必须提供一个有效的索引号,以指示要检索的邮件。
  • <number_of_lines>​ 是要返回的邮件正文的行数。客户端可以指定要检索多少行邮件正文,通常用于查看邮件的摘要信息。

响应:

+OK
<邮件头部信息>
<邮件正文的前n行>
.
  • +OK​ 表示命令执行成功。
  • <邮件头部信息>​ 包含了邮件的头部信息,如发件人、主题、日期等。
  • <邮件正文的前n行>​ 包含了指定行数的邮件正文。
  • 最后一行的.​(句号)表示邮件内容的结束。

UIDL

检索每封邮件的唯一标识符(UID)。每封邮件都分配了一个唯一的UID,通常用于客户端在与服务器交互时标识邮件,而不仅仅依赖于邮件的序号。

命令格式:

UIDL [message_number]

响应

+OK
<message_number> <UID>
  • +OK​ 表示命令执行成功。
  • <message_number>​ 是邮件的索引号。
  • <UID>​ 是邮件的唯一标识符(UID)。

USER

USER​命令是POP3(Post Office Protocol Version 3)协议中的一条命令,用于向邮件服务器提供登录用户名。USER​命令通常是POP3会话的第一步,用于标识客户端要使用的邮箱账户。

PASS

PASS​命令是POP3(Post Office Protocol Version 3)协议中的一条命令,用于向邮件服务器提供登录密码以完成身份验证。一般情况下,在使用USER​命令提供用户名后,客户端必须使用PASS​命令提供相应的密码,以便访问邮箱账户。

APOP

APOP​(Authenticated Post Office Protocol)是POP3(Post Office Protocol Version 3)协议的一种认证机制。它是一种用于安全地认证邮件客户端与邮件服务器之间的身份的方法。与普通的POP3用户名和密码认证不同,APOP​使用一种基于哈希函数的挑战-响应机制来认证客户端的身份。

以下是APOP​认证的工作原理:

  1. 邮件服务器生成一个随机的挑战字符串,通常是一个长字符串,称为挑战值(challenge)。
  2. 邮件服务器将挑战值与密码的哈希值进行拼接,并计算拼接后的字符串的MD5哈希值。这个MD5哈希值称为响应值(response)。
  3. 邮件服务器将挑战值发送给客户端。
  4. 客户端收到挑战值后,将自己的密码和挑战值拼接,并计算拼接后的字符串的MD5哈希值,得到自己的响应值。
  5. 客户端将自己的用户名和响应值发送给邮件服务器。
  6. 邮件服务器收到客户端的响应后,会检查客户端的用户名是否存在,并将客户端的响应值与邮件服务器计算的响应值进行比较。
  7. 如果客户端的响应值与邮件服务器的响应值匹配,服务器将认为客户端通过了认证,允许客户端访问邮箱。

APOP​认证的优点是它通过使用挑战-响应机制,使得在网络上传输的密码不会以明文形式传输,从而提高了安全性。这种认证机制确保只有知道正确密码的客户端才能通过认证。

请注意,APOP​认证是一种POP3协议的扩展认证机制,因此并不是所有邮件服务器和邮件客户端都支持它。在使用APOP​认证之前,确保您的邮件服务器和客户端都支持此认证方式。

Go语言解析POP3

解析头部

  1. 通过mail.ReadMessage来解析邮件内容

    message, err := mail.ReadMessage(strings.NewReader(source))
    if err != nil {log.Error(err)return
    }
    
  2. 通过解析出来message内部的Header结构可以获取指定key的信息
    常见key有: From​、To​、Subject​、Date​、Content-Type​、Content-Transfer-Encoding

    from := message.Header.Get("From")
    to := message.Header.Get("To")
    subject := message.Header.Get("Subject")
    date := message.Header.Get("Date")
    message.Header.Get("Content-Type")
    message.Header.Get("Content-Transfer-Encoding")
    
  3. Header实际是个map, 所以可以通过for循环的方式遍历

    for k, v := range message.Header {blog.Info("key: ", k, "  value:", v)
    }
    

解析body

通常body会带有多行, 并且最后带有.结束符, 所以我们需要将这些过滤掉

	str := strings.ReplaceAll(string(body), "\r\n", "")str = strings.ReplaceAll(str, ".", "")

具体需要使用解析算法需要注根据Content-Transfer-Encoding来判断,比较常用的两种邮件编码方式为BASE64和Quoted-printable。后来的扩展 SMTP协议允许直接在邮件中传递二进制数据,而不用对它们进行邮件编码,人们将这种没有进行邮件编码的二进制数据的邮件内容称为8bit编码,为了与此 相区别,人们将没有进行邮件编码的纯ASCII码字符的邮件称为7bit编码。MIME消息体的邮件编码方式通过MIME消息头中的Content- Transfer- Encoding头字段指定,每种邮件编码方式的介绍如下:

  • 7Bit: 指消息体内容全部是没有经过编码的ASCII字符。
  • 8Bit: 指消息体内容是没有经过编码的原始数据,且其中包含有非ASCII字符的数据。现在的邮件服务器基本上都支持8Bit编码,使用支持8Bit编码的邮件服务器可以简化邮件的处理过程。
  • BASE64: Base64是将二进制数据转换成可打印的ASCII字符的一种最常见的编码方式,它的基本原理是将一组连续的字节数据按6个bit位进行分组,然后对每组数据用一个ASCII字符来表示。
  • Quoted-printable: 一种将二进制数据转换成可打印的ASCII字符的编码方式,它对ASCII字符不进行转换,只对非ASCII字符的数据进行编码转 化。

指消息体内容全部是没有经过编码的ASCII字符。

使用base64解析示例:

	decodedBody, err := base64.StdEncoding.DecodeString(str)if err != nil {log.Fatal("Error decoding Base64:", err)}log.Info(string(decodedBody))

参考资料

inbucket smtp,pop3,restful服务

pop3协议分析

docker使用Inbucket搭建邮件服务器用于接受邮件

rfc1939

Multipurpose Internet Mail Extensions(RFC2045)

这篇关于POP3协议详解的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/528929

相关文章

Mysql 中的多表连接和连接类型详解

《Mysql中的多表连接和连接类型详解》这篇文章详细介绍了MySQL中的多表连接及其各种类型,包括内连接、左连接、右连接、全外连接、自连接和交叉连接,通过这些连接方式,可以将分散在不同表中的相关数据... 目录什么是多表连接?1. 内连接(INNER JOIN)2. 左连接(LEFT JOIN 或 LEFT

Java中switch-case结构的使用方法举例详解

《Java中switch-case结构的使用方法举例详解》:本文主要介绍Java中switch-case结构使用的相关资料,switch-case结构是Java中处理多个分支条件的一种有效方式,它... 目录前言一、switch-case结构的基本语法二、使用示例三、注意事项四、总结前言对于Java初学者

Linux内核之内核裁剪详解

《Linux内核之内核裁剪详解》Linux内核裁剪是通过移除不必要的功能和模块,调整配置参数来优化内核,以满足特定需求,裁剪的方法包括使用配置选项、模块化设计和优化配置参数,图形裁剪工具如makeme... 目录简介一、 裁剪的原因二、裁剪的方法三、图形裁剪工具四、操作说明五、make menuconfig

详解Java中的敏感信息处理

《详解Java中的敏感信息处理》平时开发中常常会遇到像用户的手机号、姓名、身份证等敏感信息需要处理,这篇文章主要为大家整理了一些常用的方法,希望对大家有所帮助... 目录前后端传输AES 对称加密RSA 非对称加密混合加密数据库加密MD5 + Salt/SHA + SaltAES 加密平时开发中遇到像用户的

Springboot使用RabbitMQ实现关闭超时订单(示例详解)

《Springboot使用RabbitMQ实现关闭超时订单(示例详解)》介绍了如何在SpringBoot项目中使用RabbitMQ实现订单的延时处理和超时关闭,通过配置RabbitMQ的交换机、队列和... 目录1.maven中引入rabbitmq的依赖:2.application.yml中进行rabbit

C语言线程池的常见实现方式详解

《C语言线程池的常见实现方式详解》本文介绍了如何使用C语言实现一个基本的线程池,线程池的实现包括工作线程、任务队列、任务调度、线程池的初始化、任务添加、销毁等步骤,感兴趣的朋友跟随小编一起看看吧... 目录1. 线程池的基本结构2. 线程池的实现步骤3. 线程池的核心数据结构4. 线程池的详细实现4.1 初

Python绘制土地利用和土地覆盖类型图示例详解

《Python绘制土地利用和土地覆盖类型图示例详解》本文介绍了如何使用Python绘制土地利用和土地覆盖类型图,并提供了详细的代码示例,通过安装所需的库,准备地理数据,使用geopandas和matp... 目录一、所需库的安装二、数据准备三、绘制土地利用和土地覆盖类型图四、代码解释五、其他可视化形式1.

SpringBoot使用Apache POI库读取Excel文件的操作详解

《SpringBoot使用ApachePOI库读取Excel文件的操作详解》在日常开发中,我们经常需要处理Excel文件中的数据,无论是从数据库导入数据、处理数据报表,还是批量生成数据,都可能会遇到... 目录项目背景依赖导入读取Excel模板的实现代码实现代码解析ExcelDemoInfoDTO 数据传输

如何用Java结合经纬度位置计算目标点的日出日落时间详解

《如何用Java结合经纬度位置计算目标点的日出日落时间详解》这篇文章主详细讲解了如何基于目标点的经纬度计算日出日落时间,提供了在线API和Java库两种计算方法,并通过实际案例展示了其应用,需要的朋友... 目录前言一、应用示例1、天安门升旗时间2、湖南省日出日落信息二、Java日出日落计算1、在线API2

使用Spring Cache时设置缓存键的注意事项详解

《使用SpringCache时设置缓存键的注意事项详解》在现代的Web应用中,缓存是提高系统性能和响应速度的重要手段之一,Spring框架提供了强大的缓存支持,通过​​@Cacheable​​、​​... 目录引言1. 缓存键的基本概念2. 默认缓存键生成器3. 自定义缓存键3.1 使用​​@Cacheab