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

相关文章

Linux线程同步/互斥过程详解

《Linux线程同步/互斥过程详解》文章讲解多线程并发访问导致竞态条件,需通过互斥锁、原子操作和条件变量实现线程安全与同步,分析死锁条件及避免方法,并介绍RAII封装技术提升资源管理效率... 目录01. 资源共享问题1.1 多线程并发访问1.2 临界区与临界资源1.3 锁的引入02. 多线程案例2.1 为

Python使用Tenacity一行代码实现自动重试详解

《Python使用Tenacity一行代码实现自动重试详解》tenacity是一个专为Python设计的通用重试库,它的核心理念就是用简单、清晰的方式,为任何可能失败的操作添加重试能力,下面我们就来看... 目录一切始于一个简单的 API 调用Tenacity 入门:一行代码实现优雅重试精细控制:让重试按我

Python标准库之数据压缩和存档的应用详解

《Python标准库之数据压缩和存档的应用详解》在数据处理与存储领域,压缩和存档是提升效率的关键技术,Python标准库提供了一套完整的工具链,下面小编就来和大家简单介绍一下吧... 目录一、核心模块架构与设计哲学二、关键模块深度解析1.tarfile:专业级归档工具2.zipfile:跨平台归档首选3.

idea的终端(Terminal)cmd的命令换成linux的命令详解

《idea的终端(Terminal)cmd的命令换成linux的命令详解》本文介绍IDEA配置Git的步骤:安装Git、修改终端设置并重启IDEA,强调顺序,作为个人经验分享,希望提供参考并支持脚本之... 目录一编程、设置前二、前置条件三、android设置四、设置后总结一、php设置前二、前置条件

python中列表应用和扩展性实用详解

《python中列表应用和扩展性实用详解》文章介绍了Python列表的核心特性:有序数据集合,用[]定义,元素类型可不同,支持迭代、循环、切片,可执行增删改查、排序、推导式及嵌套操作,是常用的数据处理... 目录1、列表定义2、格式3、列表是可迭代对象4、列表的常见操作总结1、列表定义是处理一组有序项目的

python使用try函数详解

《python使用try函数详解》Pythontry语句用于异常处理,支持捕获特定/多种异常、else/final子句确保资源释放,结合with语句自动清理,可自定义异常及嵌套结构,灵活应对错误场景... 目录try 函数的基本语法捕获特定异常捕获多个异常使用 else 子句使用 finally 子句捕获所

C++11范围for初始化列表auto decltype详解

《C++11范围for初始化列表autodecltype详解》C++11引入auto类型推导、decltype类型推断、统一列表初始化、范围for循环及智能指针,提升代码简洁性、类型安全与资源管理效... 目录C++11新特性1. 自动类型推导auto1.1 基本语法2. decltype3. 列表初始化3

SQL Server 中的 WITH (NOLOCK) 示例详解

《SQLServer中的WITH(NOLOCK)示例详解》SQLServer中的WITH(NOLOCK)是一种表提示,等同于READUNCOMMITTED隔离级别,允许查询在不获取共享锁的情... 目录SQL Server 中的 WITH (NOLOCK) 详解一、WITH (NOLOCK) 的本质二、工作

springboot自定义注解RateLimiter限流注解技术文档详解

《springboot自定义注解RateLimiter限流注解技术文档详解》文章介绍了限流技术的概念、作用及实现方式,通过SpringAOP拦截方法、缓存存储计数器,结合注解、枚举、异常类等核心组件,... 目录什么是限流系统架构核心组件详解1. 限流注解 (@RateLimiter)2. 限流类型枚举 (

Java Thread中join方法使用举例详解

《JavaThread中join方法使用举例详解》JavaThread中join()方法主要是让调用改方法的thread完成run方法里面的东西后,在执行join()方法后面的代码,这篇文章主要介绍... 目录前言1.join()方法的定义和作用2.join()方法的三个重载版本3.join()方法的工作原