libcurl smtp发送邮件附件大小限制问题

1 篇文章 0 订阅
1 篇文章 0 订阅

原文链接:http://codingfan.cn/index.php/archives/14/

  • 问题
    最近项目上有自动发送邮件的需求, 便用libcurl封装了个,发送hello world是ok的, 测试attachment的时候, 发现报错:read function returned funny value, 而且只要body或者attachment超过16k左右就报错

  • 分析过程
    翻看libcurl源码找到对应位置:

CURLcode Curl_fillreadbuffer(struct connectdata *conn, int bytes, int *nreadp)
{
  struct SessionHandle *data = conn->data;
  size_t buffersize = (size_t)bytes;

  ...

  //fread_func为用户实现的读取buffer的回调函数
  nread = (int)data->set.fread_func(data->req.upload_fromhere, 1,
                                    buffersize, data->set.in);
  ...
  else if((size_t)nread > buffersize) {
    /* the read function returned a too large value */
    *nreadp = 0;
    failf(data, "read function returned funny value");
    return CURLE_READ_ERROR;
  }

调用处:

static CURLcode readwrite_upload(struct SessionHandle *data,
                                 struct connectdata *conn,
                                 struct SingleRequest *k,
                                 int *didwhat)
{
    ...
    result = Curl_fillreadbuffer(conn, BUFSIZE, &fillcount);
    ...
}

可以看出buffersize 即BUFSIZE为固定大小16348, 当满足回调函数返回值大于该数值,就会报错。
再看看回调函数的实现:

struct WriteThis
    {
        int pos;
        int counter;
        std::vector<std::string> data;
    };

static size_t read_callback(void *ptr, size_t size, size_t nmemb, void *userp)
{
    struct WriteThis *pooh = (struct WriteThis *)userp;
    const char *data;

    if (size*nmemb < 1)
        return 0;

    data = text[pooh->counter];

    if (data)
    {
        size_t len = strlen(data);
        memcpy(ptr, data, len);
        pooh->counter++; /* advance pointer */
        return len;
    }
    return 0;                         /* no more data left to deliver */
}

很明显, 数组的任意一项大于该值,就会出错, 也就解释了为什么body或者attachment超出16k就会报错了

既然不支持大文件(不过16k居然能算大文件), 那我们就考虑chunk up分片读, 修改成如下, 解决问题:

#define CHUNCK_SIZE 1024 * 10
size_t CurlSmtp::read_callback(void *ptr, size_t size, size_t nmemb, void *userp)
{
  struct WriteThis *pooh = (struct WriteThis *)userp;
  const char *data;

  if(size * nmemb < 1)
    return 0;

  const std::string& str = pooh->data[pooh->counter];
  if(pooh->counter < pooh->data.size()) {
    size_t len = str.size();
    int size = len - pooh->pos;
    if (len < CHUNCK_SIZE || size <= CHUNCK_SIZE)
    {
        memcpy(ptr, str.c_str() + pooh->pos, size);
        pooh->counter++; /* advance pointer */
        pooh->pos = 0;
        return size;
    }
    else
    {
        memcpy(ptr, str.c_str() + pooh->pos, CHUNCK_SIZE);
        pooh->pos += CHUNCK_SIZE;
        return CHUNCK_SIZE;
    }

  }
  return 0;
}
  • 总结, libcurl的设计中, 默认只支持每次读取16k左右大小的buffer(包括读取smtp协议头, 消息体, 附件等), 一旦超过尺寸, 就需要考虑分片
  • 工具源码:curlsmtp
简单的控制台程序 主要是为了学习之用 本人花时间编写。为了和大家分享和学习用,如有什么错误或认为本人哪里处理不当 请和我联系~~ 内容: main.cpp 就一个cpp文件 //1.首先需要连接邮件服务器 这里用socket 邮件服务器端口 25 //2.现在就是和服务器对话了 //3.结束 #include #include #include #include #include #include #include #include #include #include #include #include #pragma comment(lib,"WS2_32.lib") using namespace std; /*加附件的版本*/ //base64编码 string Base64Encode(LPCTSTR lpszSrc); //base64解码 string Base64Decode(LPCTSTR lpszSrc); //读文件数据 bool ReadFromFile(const char* pszFilename,string &filename); unsigned char* m_pbText; int main() { //1.首先需要连接邮件服务器 这里用socket 邮件服务器端口 25 WSADATA Wsa; //进行WINSOCK的设置 WSAStartup(0x0101,&Wsa); SOCKET s = socket(AF_INET,SOCK_STREAM,IPPROTO_IP); SOCKADDR_IN sin; LPHOSTENT lphost = gethostbyname("smtp.163.com");//这里是用网易的邮件服务器 也可以修改 if(lphost) sin.sin_addr.s_addr = ((LPIN_ADDR)lphost->h_addr)->s_addr; else { printf("%s\n","获取地址失败"); return 1; } sin.sin_family = AF_INET; //注意邮件服务器的侦听端口 25 sin.sin_port = htons(IPPORT_SMTP); //连接服务器 if(connect(s,(LPSOCKADDR)&sin,sizeof(sin))==SOCKET_ERROR) { printf("%s\n","连接错误"); return 1; } printf("%s\n","连接成功"); //接收服务器初次回应 char buff[1024]; memset(buff,0,sizeof(buff)); recv(s,buff,sizeof(buff),0); printf("服务说:%s\n",buff); /////上面已经完成连接了///// string szLine="\r\n";//相当于你按下回车 //2.现在就是和服务器对话了 //问候服务器 string szHelo = "HELO smtp.163.com" + szLine; printf("我说:%s\n",szHelo.c_str()); send(s,szHelo.c_str(),szHelo.length(),0); memset(buff,0,sizeof(buff)); recv(s,buff,sizeof(buff),0); printf("服务说:%s\n",buff); //请求验证用户密码(需要编码) string szAL = "auth login" + szLine; //发送验证命令 printf("我说:%s\n",szAL.c_str()); send(s,szAL.c_str(),szAL.length(),0); memset(buff,0,sizeof(buff)); recv(s,buff,sizeof(buff),0); // printf("服务说:%s\n",buff); //服务器会回答说 可以输入帐号 //发送帐号 string szUser; .... 具体自己下载运行即可 装个VC6.0 即可
Libcurl是一个强大的开源库,用于从万维网传输数据,包括HTTP、HTTPS、FTP等协议。如果你想使用libcurl发送电子邮件(通常通过SMTP协议),这是一个简单的C语言示例: ```c #include <curl/curl.h> #include <stdio.h> // 定义回调函数处理邮件发送状态 static size_t write_function(void *ptr, size_t size, size_t nmemb, void *stream) { return fwrite(ptr, size, nmemb, stream); } int main() { CURL *curl; FILE *file; // 初始化libcurl curl_global_init(CURL_GLOBAL_DEFAULT); // 创建一个curl会话 curl = curl_easy_init(); if(curl) { // 设置SMTP服务器地址和端口 curl_easy_setopt(curl, CURLOPT_URL, "smtp://your-smtp-server.com:587"); // 设置用户名和密码(如果需要) curl_easy_setopt(curl, CURLOPT_USERNAME, "your-email@example.com"); curl_easy_setopt(curl, CURLOPT_PASSWORD, "your-password"); // 设置发件人和收件人的邮箱地址 curl_easy_setopt(curl, CURLOPT_MAIL_FROM, "your-email@example.com"); curl_slist_append(NULL, "recipient1@example.com"); // 添加多个收件人,用';'分隔 // 设置邮件正文内容 file = fopen("email.txt", "rb"); if(file) { curl_easy_setopt(curl, CURLOPT_READFUNCTION, write_function); curl_easy_setopt(curl, CURLOPT_INFILE, file); curl_easy_setopt(curl, CURLOPT_INFILESIZE, -1); // 使用文件长度作为上传大小 } else { perror("Error opening file"); return 1; } // 执行并捕获错误信息 int res = curl_easy_perform(curl); if(res != CURLE_OK) { fprintf(stderr, "curl_easy_perform() failed: %s\n", curl_easy_strerror(res)); curl_easy_cleanup(curl); return 1; } // 清理资源 fclose(file); curl_easy_cleanup(curl); curl_global_cleanup(); printf("Email sent successfully.\n"); } else { perror("curl_easy_init() failed"); return 1; } return 0; } ```
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值