mirror of
https://bitbucket.org/jsuto/piler.git
synced 2025-01-12 16:40:12 +01:00
292 lines
8.8 KiB
C
292 lines
8.8 KiB
C
|
/*
|
||
|
* smtp.c, SJ
|
||
|
*/
|
||
|
|
||
|
#include <stdio.h>
|
||
|
#include <stdlib.h>
|
||
|
#include <string.h>
|
||
|
#include <sys/time.h>
|
||
|
#include <sys/types.h>
|
||
|
#include <sys/stat.h>
|
||
|
#include <sys/socket.h>
|
||
|
#include <netdb.h>
|
||
|
#include <arpa/inet.h>
|
||
|
#include <netinet/in.h>
|
||
|
#include <fcntl.h>
|
||
|
#include <locale.h>
|
||
|
#include <getopt.h>
|
||
|
#include <stdbool.h>
|
||
|
#include <assert.h>
|
||
|
#include "../src/piler.h"
|
||
|
|
||
|
|
||
|
extern char *optarg;
|
||
|
extern int optind;
|
||
|
|
||
|
char *testmessage = "From: aaa@aaa.fu\nTo: bela@aaa.fu\nMessage-Id: ajajajaja\nSubject: this is a test\n\nAaaaaa.";
|
||
|
|
||
|
|
||
|
int connect_to_smtp_server(char *server, int port, int timeout, int use_ssl, struct __data *data);
|
||
|
|
||
|
|
||
|
void usage(){
|
||
|
printf("\nusage: smtp\n\n");
|
||
|
printf(" -s <smtp server> SMTP server\n");
|
||
|
printf(" -p <smtp port> SMTP port (25)\n");
|
||
|
printf(" -t <timeout> Timeout in sec (10)\n");
|
||
|
|
||
|
exit(0);
|
||
|
}
|
||
|
|
||
|
|
||
|
void send_smtp_command(int sd, char *cmd, char *buf, int buflen, int timeout, int use_ssl, struct __data *data){
|
||
|
|
||
|
printf("sent: %s", cmd);
|
||
|
write1(sd, cmd, strlen(cmd), use_ssl, data->ssl);
|
||
|
recvtimeoutssl(sd, buf, buflen, timeout, use_ssl, data->ssl);
|
||
|
printf("rcvd: %s", buf);
|
||
|
|
||
|
}
|
||
|
|
||
|
|
||
|
static void test_smtp_commands_one_at_a_time(char *server, int port, int timeout){
|
||
|
int sd, use_ssl = 0;
|
||
|
char recvbuf[MAXBUFSIZE], sendbuf[MAXBUFSIZE];
|
||
|
struct __data data;
|
||
|
|
||
|
sd = connect_to_smtp_server(server, port, timeout, use_ssl, &data);
|
||
|
|
||
|
send_smtp_command(sd, "HELO aaaa.fu\r\n", recvbuf, sizeof(recvbuf)-1, timeout, use_ssl, &data);
|
||
|
assert(strncmp(recvbuf, "250 ", 4) == 0 && "HELO");
|
||
|
|
||
|
send_smtp_command(sd, "MAIL FROM: <sender@aaa.fu>\r\n", recvbuf, sizeof(recvbuf)-1, timeout, use_ssl, &data);
|
||
|
assert(strncmp(recvbuf, "250 ", 4) == 0 && "MAIL");
|
||
|
|
||
|
send_smtp_command(sd, "RCPT TO: <archive@aaa.fu>\r\n", recvbuf, sizeof(recvbuf)-1, timeout, use_ssl, &data);
|
||
|
assert(strncmp(recvbuf, "250 ", 4) == 0 && "RCPT");
|
||
|
|
||
|
send_smtp_command(sd, "DATA\r\n", recvbuf, sizeof(recvbuf)-1, timeout, use_ssl, &data);
|
||
|
assert(strncmp(recvbuf, "354 ", 4) == 0 && "DATA");
|
||
|
|
||
|
snprintf(sendbuf, sizeof(sendbuf)-1, "%s\r\n.\r\n", testmessage);
|
||
|
|
||
|
send_smtp_command(sd, sendbuf, recvbuf, sizeof(recvbuf)-1, timeout, use_ssl, &data);
|
||
|
assert(strncmp(recvbuf, "250 ", 4) == 0 && "PERIOD");
|
||
|
|
||
|
send_smtp_command(sd, "QUIT\r\n", recvbuf, sizeof(recvbuf)-1, timeout, use_ssl, &data);
|
||
|
assert(strncmp(recvbuf, "221 ", 4) == 0 && "QUIT");
|
||
|
|
||
|
close(sd);
|
||
|
}
|
||
|
|
||
|
|
||
|
static void test_smtp_commands_pipelining(char *server, int port, int timeout){
|
||
|
int sd, use_ssl = 0;
|
||
|
char recvbuf[MAXBUFSIZE], sendbuf[MAXBUFSIZE];
|
||
|
struct __data data;
|
||
|
|
||
|
sd = connect_to_smtp_server(server, port, timeout, use_ssl, &data);
|
||
|
|
||
|
send_smtp_command(sd, "HELO aaaa.fu\r\n", recvbuf, sizeof(recvbuf)-1, timeout, use_ssl, &data);
|
||
|
assert(strncmp(recvbuf, "250 ", 4) == 0 && "HELO");
|
||
|
|
||
|
send_smtp_command(sd, "MAIL FROM: <sender@aaa.fu>\r\nRCPT TO: <archive@aaa.fu>\r\nDATA\r\n", recvbuf, sizeof(recvbuf)-1, timeout, use_ssl, &data);
|
||
|
assert(strncmp(recvbuf, "250 ", 4) == 0 && "MAIL");
|
||
|
|
||
|
snprintf(sendbuf, sizeof(sendbuf)-1, "%s\r\n.\r\nQUIT\r\n", testmessage);
|
||
|
|
||
|
send_smtp_command(sd, sendbuf, recvbuf, sizeof(recvbuf)-1, timeout, use_ssl, &data);
|
||
|
assert(strncmp(recvbuf, "250 ", 4) == 0 && "QUIT");
|
||
|
|
||
|
close(sd);
|
||
|
}
|
||
|
|
||
|
|
||
|
static void test_smtp_commands_with_reset_command(char *server, int port, int timeout){
|
||
|
int sd, use_ssl = 0;
|
||
|
char recvbuf[MAXBUFSIZE];
|
||
|
struct __data data;
|
||
|
|
||
|
sd = connect_to_smtp_server(server, port, timeout, use_ssl, &data);
|
||
|
|
||
|
send_smtp_command(sd, "HELO aaaa.fu\r\n", recvbuf, sizeof(recvbuf)-1, timeout, use_ssl, &data);
|
||
|
assert(strncmp(recvbuf, "250 ", 4) == 0 && "HELO");
|
||
|
|
||
|
send_smtp_command(sd, "MAIL FROM: <sender@aaa.fu>\r\n", recvbuf, sizeof(recvbuf)-1, timeout, use_ssl, &data);
|
||
|
assert(strncmp(recvbuf, "250 ", 4) == 0 && "MAIL");
|
||
|
|
||
|
send_smtp_command(sd, "RSET\r\n", recvbuf, sizeof(recvbuf)-1, timeout, use_ssl, &data);
|
||
|
assert(strncmp(recvbuf, "250 ", 4) == 0 && "RSET");
|
||
|
|
||
|
send_smtp_command(sd, "RCPT TO: <archive@aaa.fu>\r\n", recvbuf, sizeof(recvbuf)-1, timeout, use_ssl, &data);
|
||
|
assert(strncmp(recvbuf, "503 ", 4) == 0 && "RCPT");
|
||
|
|
||
|
send_smtp_command(sd, "QUIT\r\n", recvbuf, sizeof(recvbuf)-1, timeout, use_ssl, &data);
|
||
|
assert(strncmp(recvbuf, "221 ", 4) == 0 && "QUIT");
|
||
|
|
||
|
close(sd);
|
||
|
}
|
||
|
|
||
|
|
||
|
static void test_smtp_commands_partial_command(char *server, int port, int timeout){
|
||
|
int sd, use_ssl = 0;
|
||
|
char recvbuf[MAXBUFSIZE], sendbuf[MAXBUFSIZE];
|
||
|
struct __data data;
|
||
|
|
||
|
sd = connect_to_smtp_server(server, port, timeout, use_ssl, &data);
|
||
|
|
||
|
send_smtp_command(sd, "HELO aaaa.fu\r\n", recvbuf, sizeof(recvbuf)-1, timeout, use_ssl, &data);
|
||
|
assert(strncmp(recvbuf, "250 ", 4) == 0 && "HELO");
|
||
|
|
||
|
write1(sd, "M", 1, use_ssl, data.ssl);
|
||
|
printf("sent: M\n");
|
||
|
|
||
|
send_smtp_command(sd, "AIL FROM: <sender@aaa.fu>\r\n", recvbuf, sizeof(recvbuf)-1, timeout, use_ssl, &data);
|
||
|
assert(strncmp(recvbuf, "250 ", 4) == 0 && "MAIL");
|
||
|
|
||
|
send_smtp_command(sd, "RCPT TO: <archive@aaa.fu>\r\n", recvbuf, sizeof(recvbuf)-1, timeout, use_ssl, &data);
|
||
|
assert(strncmp(recvbuf, "250 ", 4) == 0 && "RCPT");
|
||
|
|
||
|
send_smtp_command(sd, "DATA\r\n", recvbuf, sizeof(recvbuf)-1, timeout, use_ssl, &data);
|
||
|
assert(strncmp(recvbuf, "354 ", 4) == 0 && "DATA");
|
||
|
|
||
|
snprintf(sendbuf, sizeof(sendbuf)-1, "%s\r\n.\r\n", testmessage);
|
||
|
|
||
|
send_smtp_command(sd, sendbuf, recvbuf, sizeof(recvbuf)-1, timeout, use_ssl, &data);
|
||
|
assert(strncmp(recvbuf, "250 ", 4) == 0 && "PERIOD");
|
||
|
|
||
|
send_smtp_command(sd, "QUIT\r\n", recvbuf, sizeof(recvbuf)-1, timeout, use_ssl, &data);
|
||
|
assert(strncmp(recvbuf, "221 ", 4) == 0 && "QUIT");
|
||
|
|
||
|
close(sd);
|
||
|
}
|
||
|
|
||
|
|
||
|
static void test_smtp_commands_partial_command_pipelining(char *server, int port, int timeout){
|
||
|
int sd, use_ssl = 0;
|
||
|
char recvbuf[MAXBUFSIZE], sendbuf[MAXBUFSIZE];
|
||
|
struct __data data;
|
||
|
|
||
|
sd = connect_to_smtp_server(server, port, timeout, use_ssl, &data);
|
||
|
|
||
|
send_smtp_command(sd, "HELO aaaa.fu\r\n", recvbuf, sizeof(recvbuf)-1, timeout, use_ssl, &data);
|
||
|
assert(strncmp(recvbuf, "250 ", 4) == 0 && "HELO");
|
||
|
|
||
|
write1(sd, "M", 1, use_ssl, data.ssl);
|
||
|
printf("sent: M\n");
|
||
|
|
||
|
send_smtp_command(sd, "AIL FROM: <sender@aaa.fu>\r\nRCPT TO: <archive@aaa.fu>\r\nDATA\r\n", recvbuf, sizeof(recvbuf)-1, timeout, use_ssl, &data);
|
||
|
assert(strncmp(recvbuf, "250 ", 4) == 0 && "MAIL");
|
||
|
|
||
|
snprintf(sendbuf, sizeof(sendbuf)-1, "%s\r\n.\r\nQUIT\r\n", testmessage);
|
||
|
|
||
|
send_smtp_command(sd, sendbuf, recvbuf, sizeof(recvbuf)-1, timeout, use_ssl, &data);
|
||
|
assert(strncmp(recvbuf, "250 ", 4) == 0 && "QUIT");
|
||
|
|
||
|
close(sd);
|
||
|
}
|
||
|
|
||
|
|
||
|
int connect_to_smtp_server(char *server, int port, int timeout, int use_ssl, struct __data *data){
|
||
|
int rc, sd = -1;
|
||
|
char port_string[8], buf[MAXBUFSIZE];
|
||
|
struct addrinfo hints, *res;
|
||
|
|
||
|
snprintf(port_string, sizeof(port_string)-1, "%d", port);
|
||
|
|
||
|
memset(&hints, 0, sizeof(hints));
|
||
|
hints.ai_family = AF_UNSPEC;
|
||
|
hints.ai_socktype = SOCK_STREAM;
|
||
|
|
||
|
if((rc = getaddrinfo(server, port_string, &hints, &res)) != 0){
|
||
|
printf("getaddrinfo for '%s': %s\n", server, gai_strerror(rc));
|
||
|
return sd;
|
||
|
}
|
||
|
|
||
|
if((sd = socket(res->ai_family, res->ai_socktype, res->ai_protocol)) == -1){
|
||
|
printf("cannot create socket\n");
|
||
|
goto ENDE;
|
||
|
}
|
||
|
|
||
|
if(connect(sd, res->ai_addr, res->ai_addrlen) == -1){
|
||
|
printf("connect()\n");
|
||
|
goto ENDE;
|
||
|
}
|
||
|
|
||
|
recvtimeoutssl(sd, buf, sizeof(buf), timeout, use_ssl, data->ssl);
|
||
|
printf("rcvd: %s", buf);
|
||
|
|
||
|
ENDE:
|
||
|
freeaddrinfo(res);
|
||
|
|
||
|
return sd;
|
||
|
}
|
||
|
|
||
|
|
||
|
int main(int argc, char **argv){
|
||
|
int c, port=25, timeout = 10;
|
||
|
char *server=NULL;
|
||
|
|
||
|
while(1){
|
||
|
|
||
|
#ifdef _GNU_SOURCE
|
||
|
static struct option long_options[] =
|
||
|
{
|
||
|
{"server", required_argument, 0, 's' },
|
||
|
{"port", required_argument, 0, 'p' },
|
||
|
{"timeout", required_argument, 0, 't' },
|
||
|
{"help", no_argument, 0, 'h' },
|
||
|
{0,0,0,0}
|
||
|
};
|
||
|
|
||
|
int option_index = 0;
|
||
|
|
||
|
c = getopt_long(argc, argv, "c:s:p:t:h?", long_options, &option_index);
|
||
|
#else
|
||
|
c = getopt(argc, argv, "c:s:p:t:h?");
|
||
|
#endif
|
||
|
|
||
|
|
||
|
if(c == -1) break;
|
||
|
|
||
|
switch(c){
|
||
|
|
||
|
case 's' :
|
||
|
server = optarg;
|
||
|
break;
|
||
|
|
||
|
case 'p' :
|
||
|
port = atoi(optarg);
|
||
|
break;
|
||
|
|
||
|
case 't' :
|
||
|
timeout = atoi(optarg);
|
||
|
break;
|
||
|
|
||
|
case 'h' :
|
||
|
case '?' :
|
||
|
usage();
|
||
|
break;
|
||
|
|
||
|
|
||
|
default :
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if(!server) usage();
|
||
|
|
||
|
|
||
|
test_smtp_commands_one_at_a_time(server, port, timeout);
|
||
|
test_smtp_commands_pipelining(server, port, timeout);
|
||
|
test_smtp_commands_with_reset_command(server, port, timeout);
|
||
|
test_smtp_commands_partial_command(server, port, timeout);
|
||
|
test_smtp_commands_partial_command_pipelining(server, port, timeout);
|
||
|
|
||
|
|
||
|
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
|