mirror of
				https://gitee.com/jiuyilian/embedded-framework.git
				synced 2025-10-24 18:20:15 -04:00 
			
		
		
		
	
		
			
				
	
	
		
			257 lines
		
	
	
		
			6.5 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			257 lines
		
	
	
		
			6.5 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| #include "smtp.h"
 | |
| 
 | |
| #include "hsocket.h"
 | |
| #include "herr.h"
 | |
| #include "base64.h"
 | |
| 
 | |
| const char* smtp_command_str(enum smtp_command cmd) {
 | |
|     switch (cmd) {
 | |
| #define XX(name, string) case SMTP_##name: return #string;
 | |
|     SMTP_COMMAND_MAP(XX)
 | |
| #undef  XX
 | |
|     default: return "<unknown>";
 | |
|     }
 | |
| }
 | |
| 
 | |
| const char* smtp_status_str(enum smtp_status status) {
 | |
|     switch (status) {
 | |
| #define XXX(code, name, string) case SMTP_STATUS_##name: return #string;
 | |
|     SMTP_STATUS_MAP(XXX)
 | |
| #undef  XXX
 | |
|     default: return "<unknown>";
 | |
|     }
 | |
| }
 | |
| 
 | |
| int smtp_build_command(enum smtp_command cmd, const char* param, char* buf, int buflen) {
 | |
|     switch (cmd) {
 | |
|     // unary
 | |
|     case SMTP_DATA:
 | |
|     case SMTP_QUIT:
 | |
|         return snprintf(buf, buflen, "%s\r\n", smtp_command_str(cmd));
 | |
|     // <address>
 | |
|     case SMTP_MAIL:
 | |
|     case SMTP_RCPT:
 | |
|         return snprintf(buf, buflen, "%s <%s>\r\n", smtp_command_str(cmd), param);
 | |
|     default:
 | |
|         return snprintf(buf, buflen, "%s %s\r\n", smtp_command_str(cmd), param);
 | |
|     }
 | |
| }
 | |
| 
 | |
| // EHLO => AUTH PLAIN => MAIL => RCPT => DATA => data => EOB => QUIT
 | |
| int sendmail(const char* smtp_server,
 | |
|              const char* username,
 | |
|              const char* password,
 | |
|              mail_t* mail) {
 | |
|     char buf[1024] = {0};
 | |
|     int  buflen = sizeof(buf);
 | |
|     int  cmdlen = 0;
 | |
|     int  status_code = 0;
 | |
|     char basic[256];
 | |
|     int  basiclen;
 | |
| 
 | |
|     int sockfd = ConnectTimeout(smtp_server, SMTP_PORT, DEFAULT_CONNECT_TIMEOUT);
 | |
|     if (sockfd < 0) {
 | |
|         return sockfd;
 | |
|     }
 | |
|     so_sndtimeo(sockfd, 5000);
 | |
|     so_rcvtimeo(sockfd, 5000);
 | |
| 
 | |
|     int ret, nsend, nrecv;
 | |
|     nrecv = recv(sockfd, buf, buflen, 0);
 | |
|     if (nrecv <= 0) {
 | |
|         ret = ERR_RECV;
 | |
|         goto error;
 | |
|     }
 | |
|     status_code = atoi(buf);
 | |
|     if (status_code != SMTP_STATUS_READY) {
 | |
|         ret = status_code;
 | |
|         goto error;
 | |
|     }
 | |
|     // EHLO smtp.xxx.com\r\n
 | |
|     cmdlen = smtp_build_command(SMTP_EHLO, smtp_server, buf, buflen);
 | |
|     nsend = send(sockfd, buf, cmdlen, 0);
 | |
|     if (nsend != cmdlen) {
 | |
|         ret = ERR_SEND;
 | |
|         goto error;
 | |
|     }
 | |
|     nrecv = recv(sockfd, buf, buflen, 0);
 | |
|     if (nrecv <= 0) {
 | |
|         ret = ERR_RECV;
 | |
|         goto error;
 | |
|     }
 | |
|     status_code = atoi(buf);
 | |
|     if (status_code != SMTP_STATUS_OK) {
 | |
|         ret = status_code;
 | |
|         goto error;
 | |
|     }
 | |
|     // AUTH PLAIN\r\n
 | |
|     cmdlen = smtp_build_command(SMTP_AUTH, "PLAIN", buf, buflen);
 | |
|     nsend = send(sockfd, buf, cmdlen, 0);
 | |
|     if (nsend != cmdlen) {
 | |
|         ret = ERR_SEND;
 | |
|         goto error;
 | |
|     }
 | |
|     nrecv = recv(sockfd, buf, buflen, 0);
 | |
|     if (nrecv <= 0) {
 | |
|         ret = ERR_RECV;
 | |
|         goto error;
 | |
|     }
 | |
|     status_code = atoi(buf);
 | |
|     if (status_code != SMTP_STATUS_AUTH) {
 | |
|         ret = status_code;
 | |
|         goto error;
 | |
|     }
 | |
|     {
 | |
|         // BASE64 \0username\0password
 | |
|         int usernamelen = strlen(username);
 | |
|         int passwordlen = strlen(password);
 | |
|         basic[0] = '\0';
 | |
|         memcpy(basic+1, username, usernamelen);
 | |
|         basic[1+usernamelen] = '\0';
 | |
|         memcpy(basic+1+usernamelen+1, password, passwordlen);
 | |
|         basiclen = 1 + usernamelen + 1 + passwordlen;
 | |
|     }
 | |
|     hv_base64_encode((unsigned char*)basic, basiclen, buf);
 | |
|     cmdlen = BASE64_ENCODE_OUT_SIZE(basiclen);
 | |
|     buf[cmdlen] = '\r';
 | |
|     buf[cmdlen+1] = '\n';
 | |
|     cmdlen += 2;
 | |
|     nsend = send(sockfd, buf, cmdlen, 0);
 | |
|     if (nsend != cmdlen) {
 | |
|         ret = ERR_SEND;
 | |
|         goto error;
 | |
|     }
 | |
|     nrecv = recv(sockfd, buf, buflen, 0);
 | |
|     if (nrecv <= 0) {
 | |
|         ret = ERR_RECV;
 | |
|         goto error;
 | |
|     }
 | |
|     status_code = atoi(buf);
 | |
|     if (status_code != SMTP_STATUS_AUTH_SUCCESS) {
 | |
|         ret = status_code;
 | |
|         goto error;
 | |
|     }
 | |
|     // MAIL FROM: <from>\r\n
 | |
|     cmdlen = smtp_build_command(SMTP_MAIL, mail->from, buf, buflen);
 | |
|     nsend = send(sockfd, buf, cmdlen, 0);
 | |
|     if (nsend != cmdlen) {
 | |
|         ret = ERR_SEND;
 | |
|         goto error;
 | |
|     }
 | |
|     nrecv = recv(sockfd, buf, buflen, 0);
 | |
|     if (nrecv <= 0) {
 | |
|         ret = ERR_RECV;
 | |
|         goto error;
 | |
|     }
 | |
|     status_code = atoi(buf);
 | |
|     if (status_code != SMTP_STATUS_OK) {
 | |
|         ret = status_code;
 | |
|         goto error;
 | |
|     }
 | |
|     // RCPT TO: <to>\r\n
 | |
|     cmdlen = smtp_build_command(SMTP_RCPT, mail->to, buf, buflen);
 | |
|     nsend = send(sockfd, buf, cmdlen, 0);
 | |
|     if (nsend != cmdlen) {
 | |
|         ret = ERR_SEND;
 | |
|         goto error;
 | |
|     }
 | |
|     nrecv = recv(sockfd, buf, buflen, 0);
 | |
|     if (nrecv <= 0) {
 | |
|         ret = ERR_RECV;
 | |
|         goto error;
 | |
|     }
 | |
|     status_code = atoi(buf);
 | |
|     if (status_code != SMTP_STATUS_OK) {
 | |
|         ret = status_code;
 | |
|         goto error;
 | |
|     }
 | |
|     // DATA\r\n
 | |
|     cmdlen = smtp_build_command(SMTP_DATA, NULL, buf, buflen);
 | |
|     nsend = send(sockfd, buf, cmdlen, 0);
 | |
|     if (nsend != cmdlen) {
 | |
|         ret = ERR_SEND;
 | |
|         goto error;
 | |
|     }
 | |
|     nrecv = recv(sockfd, buf, buflen, 0);
 | |
|     if (nrecv <= 0) {
 | |
|         ret = ERR_RECV;
 | |
|         goto error;
 | |
|     }
 | |
|     status_code = atoi(buf);
 | |
|     // SMTP_STATUS_DATA
 | |
|     if (status_code >= 400) {
 | |
|         ret = status_code;
 | |
|         goto error;
 | |
|     }
 | |
|     // From:
 | |
|     cmdlen = snprintf(buf, buflen, "From:%s\r\n", mail->from);
 | |
|     nsend = send(sockfd, buf, cmdlen, 0);
 | |
|     if (nsend != cmdlen) {
 | |
|         ret = ERR_SEND;
 | |
|         goto error;
 | |
|     }
 | |
|     // To:
 | |
|     cmdlen = snprintf(buf, buflen, "To:%s\r\n", mail->to);
 | |
|     nsend = send(sockfd, buf, cmdlen, 0);
 | |
|     if (nsend != cmdlen) {
 | |
|         ret = ERR_SEND;
 | |
|         goto error;
 | |
|     }
 | |
|     // Subject:
 | |
|     cmdlen = snprintf(buf, buflen, "Subject:%s\r\n\r\n", mail->subject);
 | |
|     nsend = send(sockfd, buf, cmdlen, 0);
 | |
|     if (nsend != cmdlen) {
 | |
|         ret = ERR_SEND;
 | |
|         goto error;
 | |
|     }
 | |
|     // body
 | |
|     cmdlen = strlen(mail->body);
 | |
|     nsend = send(sockfd, mail->body, cmdlen, 0);
 | |
|     if (nsend != cmdlen) {
 | |
|         ret = ERR_SEND;
 | |
|         goto error;
 | |
|     }
 | |
|     // EOB
 | |
|     nsend = send(sockfd, SMTP_EOB, SMTP_EOB_LEN, 0);
 | |
|     if (nsend != SMTP_EOB_LEN) {
 | |
|         ret = ERR_SEND;
 | |
|         goto error;
 | |
|     }
 | |
|     nrecv = recv(sockfd, buf, buflen, 0);
 | |
|     if (nrecv <= 0) {
 | |
|         ret = ERR_SEND;
 | |
|         goto error;
 | |
|     }
 | |
|     status_code = atoi(buf);
 | |
|     if (status_code != SMTP_STATUS_OK) {
 | |
|         ret = status_code;
 | |
|         goto error;
 | |
|     }
 | |
|     // QUIT\r\n
 | |
|     cmdlen = smtp_build_command(SMTP_QUIT, NULL, buf, buflen);
 | |
|     nsend = send(sockfd, buf, cmdlen, 0);
 | |
|     if (nsend != cmdlen) {
 | |
|         ret = ERR_SEND;
 | |
|         goto error;
 | |
|     }
 | |
|     nrecv = recv(sockfd, buf, buflen, 0);
 | |
|     if (nrecv <= 0) {
 | |
|         ret = ERR_RECV;
 | |
|         goto error;
 | |
|     }
 | |
|     /*
 | |
|     status_code = atoi(buf);
 | |
|     if (status_code != SMTP_STATUS_BYE) {
 | |
|         ret = status_code;
 | |
|         goto error;
 | |
|     }
 | |
|     */
 | |
|     ret = SMTP_STATUS_OK;
 | |
| 
 | |
| error:
 | |
|     if (sockfd != INVALID_SOCKET) {
 | |
|         closesocket(sockfd);
 | |
|     }
 | |
|     return ret;
 | |
| }
 | 
