node-imap-sync-client, imap 客户端, 例子

news2025/4/2 4:36:57

说明

本文是 node-imap-sync-client imap客户端库的使用例子

https://blog.csdn.net/eli960/article/details/146049717

例子


import { imapSyncClient, imapUtf7ToUtf8, utf8ToImapUtf7 } from "imap-sync-client"

const sleep = async (t) => {
    return new Promise((resolv) => {
        setTimeout(() => {
            resolv("")
        }, t)
    })
}

/**
 * 执行 IMAP 客户端测试的异步函数
 * @param {Object} attrs - 包含 IMAP 客户端连接所需参数的对象
 * @param {string} attrs.host - IMAP 服务器地址
 * @param {number} attrs.port - IMAP 服务器端口
 * @param {boolean} [attrs.ssl] - 是否使用 SSL 连接
 * @param {string} attrs.user - 用户名
 * @param {string} attrs.pass - 密码
 * @param {boolean} [attrs.startTLS] - 是否使用 STARTTLS 加密
 * @param {string} attrs.appendMsg - 用于上传测试的邮件内容
 */
async function do_test(attrs) {
    // 打印传入的参数
    console.log(attrs)

    // 创建对象
    // 实例化 imapSyncClient 类,传入连接所需的参数
    let ic = new imapSyncClient({
        host: attrs.host,
        port: attrs.port,
        ssl: attrs.ssl,
        user: attrs.user,
        pass: attrs.pass,
        startTLS: attrs.startTLS
    })
    // 开启调试模式
    ic.setDebugMode()

    // open,包括 连接,STARTTLS,认证,ID,CAPABILITY 等
    // 尝试打开与 IMAP 服务器的连接
    if (!await ic.open()) {
        // 若连接失败,打印错误信息
        console.error("imap connect, error: ", ic.getLastReadedBuffer().toString())
        return
    }
    // 若连接成功,打印成功信息
    console.log("connect, success")

    // 在操作中, 试着检查是否有新的信件
    // 设置新邮件检查的处理函数
    ic.setMaybeHaveNewMailHandler(() => {
        console.log("maybe have new mail")
    })

    // 通过命令 LIST,获取文件夹列表
    // 调用方法获取邮箱文件夹列表
    let mboxs = await ic.getMboxList()
    if (mboxs === null) {
        // 若获取失败,打印错误信息
        console.error("cmdList, error: ", ic.getLastReadedBuffer().toString())
        return
    }
    // 若获取成功,打印文件夹列表
    console.log("mboxs", mboxs)

    // 遍历文件夹列表,执行 STATUS 命令
    for (let i = 0; i < mboxs.length; i++) {
        let fo = mboxs[i]
        let mboxName = fo.mboxName
        // 若文件夹不是不可选择状态
        if (!fo.attrs.noselect) {
            // STATUS 名
            // 调用方法获取文件夹状态
            let status = await ic.getMboxStatus(mboxName)
            if (status === null) {
                // 若获取失败,打印错误信息
                console.error("cmdStatus " + mboxName.toString() + ", error: ", ic.getLastReadedBuffer().toString())
                return
            }
            // 若获取成功,打印文件夹状态
            console.log("status " + mboxName.toString() + ", success:", status)
        }
    }
    // 遍历文件夹列表,打印文件夹名称及其编码转换后的结果
    for (let i = 0; i < mboxs.length; i++) {
        let fo = mboxs[i]
        let mboxName = fo.mboxName
        console.log("mboxName:", mboxName.toString(), imapUtf7ToUtf8(mboxName), utf8ToImapUtf7(imapUtf7ToUtf8(mboxName)))
    }

    // 命令 SELECT, 选择 INBOX
    // 调用方法选择 INBOX 文件夹
    const selector = await ic.selectMbox("INBOX")
    if (selector === null) {
        // 若选择失败,打印错误信息
        console.error("cmdSelect INBOX, error: ", ic.getLastReadedBuffer().toString())
        return
    }
    // 若选择成功,打印成功信息
    console.log("select INBOX, success: ", selector)

    // 通过命令 FETCH, 获取当前文件夹(INBOX)下所有邮件的UID和其标记列表
    // 调用方法获取 INBOX 文件夹下所有邮件的 UID 和标记列表
    let uidWithFlags = await ic.fetchUidListWithFlags()
    if (uidWithFlags === null) {
        // 若获取失败,打印错误信息
        console.error("fetchUidListWithFlags INBOX, error: ", ic.getLastReadedBuffer().toString())
        return
    }
    if (uidWithFlags.length < 2) {
        // 若邮件数量少于 2,打印所有邮件信息
        console.log("fetch uid list, success: ", uidWithFlags)
    } else {
        // 若邮件数量多于 2,打印前 2 条信息和剩余数量
        console.log("fetch uid list, success: ", uidWithFlags.slice(0, 2), uidWithFlags.length - 2, + " MORE")
    }

    // 通过命令 UID SEARCH,获取星标邮件UID列表
    // 调用方法获取星标邮件的 UID 列表
    let flaggedUids = await ic.searchFlaggedUids()

    if (flaggedUids === null) {
        // 若获取失败,打印错误信息
        console.error("searchAllUids INBOX, error: ", ic.getLastReadedBuffer().toString())
        return
    }
    // 若获取成功,打印星标邮件 UID 列表
    console.log("searchFlaggedUids, success: ", flaggedUids)

    // 通过命令 UID SEARCH,获取所有邮件UID列表
    // 调用方法获取所有邮件的 UID 列表
    let allUids = await ic.searchAllUids()
    if (allUids === null) {
        // 若获取失败,打印错误信息
        console.error("searchAllUids INBOX, error: ", ic.getLastReadedBuffer().toString())
        return
    }
    // 若获取成功,打印所有邮件 UID 列表
    console.log("searchAllUids, success: ", allUids)

    // 下面几个是文件夹相关操作,顾名思义即可
    // 创建名为 "你好啦啦啦" 的文件夹
    await ic.createMbox(utf8ToImapUtf7("你好啦啦啦"))
    // 订阅名为 "你好啦啦啦" 的文件夹
    await ic.subscribeMbox(utf8ToImapUtf7("你好啦啦啦"))
    // 创建名为 "abc" 的文件夹
    await ic.createMbox("abc")
    // 订阅名为 "abc" 的文件夹
    await ic.subscribeMbox("abc")
    // 删除名为 "abc" 的文件夹
    await ic.deleteMbox("abc")
    // 尝试订阅已删除的 "abc" 文件夹
    await ic.subscribeMbox("abc")
    // 尝试取消订阅 "abc" 文件夹
    await ic.unSubscribeMbox("abc")
    // 再次尝试删除 "abc" 文件夹
    await ic.deleteMbox("abc")
    // 删除名为 "ddd" 的文件夹
    await ic.deleteMbox("ddd")
    // 删除名为 "eee" 的文件夹
    await ic.deleteMbox("eee")
    // 将 "abc" 文件夹重命名为 "ddd"
    await ic.renameMbox("abc", "ddd")
    // 再次创建名为 "abc" 的文件夹
    await ic.createMbox("abc")
    // 将 "abc" 文件夹重命名为 "ddd"
    await ic.renameMbox("abc", "ddd")
    // 将 "abc" 文件夹重命名为 "eee"
    await ic.renameMbox("abc", "eee")

    // 如果想测试新的信件检查:
    if (0) {
        // 等 10 秒
        console.log("等十秒, 期间, 请上传信件到收件箱")
        // 调用 sleep 函数延迟 10 秒
        await sleep(10 * 1000)
        // 发送 NOOP 命令
        await ic.cmdNoop()
    }

    // 再次创建名为 "abc" 的文件夹
    await ic.createMbox("abc")
    // 选择 INBOX 文件夹
    await ic.selectMbox("INBOX")

    // 通过命令 APPEND, 上传信件到 文件夹 abc
    // 将邮件内容转换为 Buffer 对象
    let msgbf = Buffer.from(attrs.appendMsg)
    // 调用方法将邮件上传到 "abc" 文件夹
    await ic.appendMail("abc", msgbf.length, () => {
        return msgbf
    }, {
        // 上传成功后,打印邮件的 UID 信息
        callbackForAppendUid(r) {
            console.log("append mail, appenduid: ", r.uidvalidity, r.uid)
        }
    })

    // 获取 inbox 文件夹下, 一封信件的信封等信息
    // 选择 "inbox" 文件夹
    await ic.selectMbox("inbox")
    // 调用方法获取 "inbox" 文件夹下所有邮件的 UID 列表
    let uids = await ic.searchAllUids()
    if (uids && uids.length) {
        // 获取第一封邮件的 UID
        let uid = uids[0]
        // 信封
        // 调用方法获取邮件的信封信息
        let envelope = await ic.fetchMailEnvelope(uid)
        console.log("envelope", envelope)
        // 结构
        // 调用方法获取邮件的结构信息
        let structure = await ic.fetchMailStructure(uid)
        console.log("structure", structure)
        // 信息(信封 + 结构)
        // 调用方法获取邮件的综合信息
        let mi = await ic.fetchMailInfo(uid)
        console.log("mailinfo", mi)

        // 获取第一个可读mime的数据
        if (mi.textMimes.length) {
            // 调用方法获取邮件第一个可读 MIME 部分的数据
            await ic.fetchMimeDataBySection(uid, mi.textMimes[0].section,
                async (pieceData) => {
                    // 打印获取到的数据
                    console.log("part", pieceData.toString())
                },
                {
                    partial: {
                        offset: 0,
                        length: 128
                    }
                })
        }
        // 获取 信件
        // 调用方法获取邮件的完整数据
        await ic.fetchMailData(uid,
            async (pieceData) => {
                // 打印获取到的邮件数据
                console.log("eml", pieceData.toString())
            },
            {
                partial: {
                    offset: 0,
                    length: 256
                },
                // 打印邮件的标记信息
                callbackForFlags: (flags) => {
                    console.log("flags", flags)
                }
            })
    }

    // 断开与 IMAP 服务器的连接
    await ic.logout()
}

// 程序开始

// 获取命令行参数
let argv = process.argv
if (argv.length < 6) {
    // 若参数数量不足,打印使用说明和示例
    console.log("USAGE: " + argv[0] + " " + argv[1] + "host port user pass [ SSL / STARTTLS ]")
    console.log("examples");
    console.log(argv[0], argv[1], "127.0.0.1 143 x1@a.com password123")
    console.log(argv[0], argv[1], "127.0.0.1 143 x1@a.com password123 STARTTLS")
    console.log(argv[0], argv[1], "127.0.0.1 993 x1@a.com password123 SSL")
    // 退出程序
    process.exit(1)
}

// 准备对象的参数
let attrs = {}
// 从命令行参数中获取 IMAP 服务器地址
attrs.host = argv[2]
// 从命令行参数中获取 IMAP 服务器端口,并转换为数字类型
attrs.port = parseInt(argv[3])
// 从命令行参数中获取用户名
attrs.user = argv[4]
// 从命令行参数中获取密码
attrs.pass = argv[5]

// 根据命令行参数设置连接方式
if (argv[6] === "SSL") {
    attrs.ssl = true
} else if (argv[6] === "STARTTLS") {
    attrs.startTLS = true
}

// 准备一封信,用于上传测试 
let msgs = []
// 添加邮件主题
msgs.push("Subject: imap-sync-client append cmd, " + (new Date()).getTime() + "\r\n")
// 添加发件人信息
msgs.push("From: nihao@linuxmail.cn\r\n")
// 添加收件人信息
msgs.push("to: thanks@linuxmail.cn\r\n")
// 添加邮件 ID
msgs.push("Message-Id: message_id_random_string_" + (new Date()).getTime() + "\r\n")
// 添加邮件内容类型
msgs.push("Content-Type: text/plain; charset=utf-8\r\n")
// 添加邮件编码方式
msgs.push("Content-Transfer-Encoding: 8bit\r\n")
msgs.push("\r\n")
msgs.push("\r\n")
// 添加邮件正文
msgs.push("imap-sync-client 邮件上传测试\r\n")
msgs.push("\r\n")
// 将邮件内容拼接成字符串
attrs.appendMsg = msgs.join("")

// 测试开始
// 调用 do_test 函数开始测试,并在测试结束后打印测试结束信息
do_test(attrs).then(() => {
    console.log("\ntest over\n")
})

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2323897.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

boost.asio

as&#xff08;async&#xff09;:异步 同步io&#xff1a; reactor (非阻塞)&#xff08;需要注册一次&#xff0c;在等待消息时可以干别的事&#xff09; 阻塞io网络模型 接口&#xff1a;read\accept\connect\write 接口返回时&#xff0c;io完成 异步…

当贝AI知识库评测 AI如何让知识检索快人一步

近日,国内领先的人工智能服务商当贝AI正式推出“个人知识库”功能,这一创新性工具迅速引发行业关注。在信息爆炸的时代,如何高效管理个人知识资产、快速获取精准答案成为用户的核心需求。当贝AI通过将“闭卷考试”变为“开卷考试”的独特设计,为用户打造了一个高度个性化的智能…

深度解读:智能体2.0 AI Agent多推演进

AI Agent即AI 代理&#xff0c;长期以来&#xff0c;研究人员一直在追求更完美的AI&#xff0c;可以与人类相当、甚至是超越人类。在1950年代&#xff0c;AIan Turing就将“智能”的概念扩展到了人工实体&#xff0c;并提出了著名的图灵测试。这些人工智能实体就被称为——Agen…

Golang 的 GMP 调度机制常见问题及解答

文章目录 Golang GMP 调度模型详解常见问题基础概念1. GMP 各组件的作用是什么&#xff1f;2. 为什么 Go 需要自己的调度器&#xff1f;3. GOMAXPROCS 的作用是什么&#xff1f; 调度流程4. Goroutine 如何被调度到 M 上执行&#xff1f;5. 系统调用会阻塞整个线程吗&#xff1…

项目-苍穹外卖(十五) Apache ECharts+数据统计

一、介绍 二、营业额统计 需求分析和设计&#xff1a; Controller: Service: /*** 营业额统计* param begindate* param enddate* return* */Overridepublic TurnoverReportVO turnoverStatistics(LocalDate begindate, LocalDate enddate) {//创建时间集合List<LocalDate&…

Spring Data审计利器:@LastModifiedDate详解(依赖关系补充篇)!!!

&#x1f552; Spring Data审计利器&#xff1a;LastModifiedDate详解&#x1f525;&#xff08;依赖关系补充篇&#xff09; &#x1f50c; 核心依赖解析 使用LastModifiedDate必须知道的依赖关系 #mermaid-svg-qm1OUa9Era9ktbeK {font-family:"trebuchet ms",verd…

Tweak Power:全方位电脑系统优化的高效工具

Tweak Power&#xff08;系统&#xff09; Tweak Power是一款功能强大的系统优化工具&#xff0c;专为提升Windows电脑的性能和稳定性而设计。它提供了全面的清理、优化和调整选项&#xff0c;帮助用户轻松管理系统资源、提高运行速度、延长设备寿命。 快速扫描并清理系统垃圾…

CLion下载安装(Windows11)

目录 CLion工具下载安装其他 CLion CLion-2024.1.4.exe 工具 系统&#xff1a;Windows 11 下载 1.通过百度网盘分享的文件&#xff1a;CLion-2024.1.4.exe 链接&#xff1a;https://pan.baidu.com/s/1-zH0rZPCZtQ60IqdHA7Cew?pwdux5a 提取码&#xff1a;ux5a 安装 打开…

如何用 Postman 进行高效的 Mock 测试?

Postman 是一个强大的 API 开发和测试工具&#xff0c;它可以让你轻松地创建和发送各种 HTTP 请求&#xff0c;查看响应结果&#xff0c;并进行调试和优化。但是有时候&#xff0c;你可能还没有开发好后端服务&#xff0c;或者想要模拟不同的响应场景&#xff0c;这时候就可以使…

DeepSeek API集成开发指南——Flask示例实践

DeepSeek API集成开发指南——Flask示例实践 序言&#xff1a;智能化开发新范式 DeepSeek API提供了覆盖自然语言处理、代码生成等多领域的先进AI能力。本文将以一个功能完备的Flask示例系统为载体&#xff0c;详解API的集成方法与最佳实践。通过本案例&#xff0c;开发者可快…

【天梯赛】L2-004 这是二叉搜索树吗(经典问题C++)

解题反思 //镜像树满足&#xff1a;左子树>根节点>右子树 //特殊&#xff1a;独腿二叉树&#xff0c;如pre {2&#xff0c;3&#xff0c;4}&#xff0c;递归函数用if(root tail) return&#xff1b;无法识别这种二叉树 // 用ismirror来将一般二叉树和镜像二叉搜索树的…

Postman 全局 Header 如何设置?全局设置了解一下

在使用 Postman 设置全局请求头信息的关键步骤包括&#xff1a;在集合设置页面中添加所需的头部信息&#xff0c;并确保选择适当的类型和值&#xff1b;如果需要&#xff0c;可通过 JavaScript 脚本添加其他请求头&#xff1b;最后&#xff0c;验证设置是否成功生效。 Postman…

科技赋能建筑业变革:中建海龙创新引领高质量发展新路径

在建筑工业化浪潮中&#xff0c;中建海龙科技有限公司&#xff08;以下简称“中建海龙”&#xff09;凭借深厚的技术积累与持续创新&#xff0c;成为推动行业转型升级的标杆企业。作为中国建筑国际集团旗下核心科技力量&#xff0c;中建海龙深耕模块化集成建筑&#xff08;MiC&…

QT计算器开发

1.项目架构 1.图形化界面 ​ 2.widget.h​ #ifndef WIDGET_H #define WIDGET_H#include <QWidget> #include <QString> #include <QStack>QT_BEGIN_NAMESPACE namespace Ui { class Widget; } QT_END_NAMESPACEclass Widget : public QWidget {Q_OBJECTp…

R语言对偏态换数据进行转换(对数、平方根、立方根)

我们进行研究的时候经常会遇见偏态数据&#xff0c;数据转换是统计分析和数据预处理中的一项基本技术。使用 R 时&#xff0c;了解如何正确转换数据有助于满足统计假设、标准化分布并提高分析的准确性。在 R 中实现和可视化最常见的数据转换&#xff1a;对数、平方根和立方根转…

【云服务器】在 Linux(Ubuntu / CentOS 7)上快速搭建我的世界 Minecraft 服务器,并实现远程联机,详细教程

【云服务器】在 Linux&#xff08;Ubuntu / CentOS 7&#xff09;上快速搭建我的世界 Minecraft 服务器&#xff0c;并实现远程联机&#xff0c;详细教程 一、 服务器介绍二、下载 Minecraft 服务端二、安装 JRE 21三、安装 MCS manager 面板四、搭建服务器五、本地测试连接六、…

docker torcherve打包mar包并部署模型

使用Docker打包深度网络模型mar包到服务端 参考链接&#xff1a;Docker torchserve 部署模型流程——以WSL部署YOLO-FaceV2为例_class myhandler(basehandler): def initialize(self,-CSDN博客 1、docker拉取环境镜像命令 docker images出现此提示为没有权限取执行命令&…

【安当产品应用案例100集】042-基于安当KADP实现机密文件安全流转

一、客户需求 某集团公司客户&#xff0c;在系统业务流中&#xff0c;存在大量的内部文件流转的需求。内部业务文件有不同的安全密级&#xff0c;最初在文件流转时&#xff0c;公司内部规定点对点的文件传输&#xff0c;要使用加密工具加密后再发给需要的一方。这种方式虽然能…

附录C SLAC匹配过程命令定义与实际抓包

附录C SLAC匹配过程命令定义与实际抓包 ISO15118-3 附录A中规定了SLAC匹配过程中的请求命令及应答&#xff0c; 本文将会对比协议中的定义和实际抓包内容&#xff0c;以便读者获得直观的认识。 1 CM_SET_KEY.REQ 定义内容&#xff1a; 实际数据&#xff1a; 注意报文中的 08…

【QT】新建QT工程(详细步骤)

新建QT工程 1.方法(1)点击new project按钮&#xff0c;弹出对话框&#xff0c;新建即可&#xff0c;步骤如下&#xff1a;(2) 点击文件菜单&#xff0c;选择新建文件或者工程&#xff0c;后续步骤如上 2.QT工程文件介绍(1).pro文件 --》QT工程配置文件(2)main.cpp --》QT工程主…