/* * pilerimport.c, SJ */ #define _FILE_OFFSET_BITS 64 #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #define SKIPLIST "junk,trash,spam,draft," #define MBOX_ARGS 1024 extern char *optarg; extern int optind; int dryrun=0; int import_from_gui=0; char *folder_imap=NULL; int connect_to_imap_server(int sd, int *seq, char *username, char *password, int port, struct __data *data, int use_ssl); int list_folders(int sd, int *seq, int use_ssl, struct __data *data); int process_imap_folder(int sd, int *seq, char *folder, struct session_data *sdata, struct __data *data, int use_ssl, int dryrun, struct __config *cfg); int connect_to_pop3_server(int sd, char *username, char *password, int port, struct __data *data, int use_ssl); int process_pop3_emails(int sd, struct session_data *sdata, struct __data *data, int use_ssl, int dryrun, struct __config *cfg); void send_imap_close(int sd, int *seq, struct __data *data, int use_ssl); void close_connection(int sd, struct __data *data, int use_ssl); void update_import_job_stat(struct session_data *sdata, struct __data *data); int import_from_mailbox(char *mailbox, struct session_data *sdata, struct __data *data, struct __config *cfg){ FILE *F, *f=NULL; int rc=ERR, tot_msgs=0, ret=OK; char buf[MAXBUFSIZE], fname[SMALLBUFSIZE]; time_t t; F = fopen(mailbox, "r"); if(!F){ printf("cannot open mailbox: %s\n", mailbox); return rc; } t = time(NULL); while(fgets(buf, sizeof(buf)-1, F)){ if(buf[0] == 'F' && buf[1] == 'r' && buf[2] == 'o' && buf[3] == 'm' && buf[4] == ' '){ tot_msgs++; if(f){ fclose(f); f = NULL; rc = import_message(fname, sdata, data, cfg); if(rc == ERR){ printf("error importing: '%s'\n", fname); ret = ERR; } else unlink(fname); if(data->quiet == 0) printf("processed: %7d\r", tot_msgs); fflush(stdout); } snprintf(fname, sizeof(fname)-1, "%ld-%d", t, tot_msgs); f = fopen(fname, "w+"); continue; } if(f) fprintf(f, "%s", buf); } if(f){ fclose(f); rc = import_message(fname, sdata, data, cfg); if(rc == ERR){ printf("error importing: '%s'\n", fname); ret = ERR; } else unlink(fname); if(data->quiet == 0) printf("processed: %7d\r", tot_msgs); fflush(stdout); } fclose(F); return ret; } int import_mbox_from_dir(char *directory, struct session_data *sdata, struct __data *data, int *tot_msgs, struct __config *cfg){ DIR *dir; struct dirent *de; int rc=ERR, ret=OK, i=0; int folder; char fname[SMALLBUFSIZE]; struct stat st; dir = opendir(directory); if(!dir){ printf("cannot open directory: %s\n", directory); return ERR; } while((de = readdir(dir))){ if(strcmp(de->d_name, ".") == 0 || strcmp(de->d_name, "..") == 0) continue; snprintf(fname, sizeof(fname)-1, "%s/%s", directory, de->d_name); if(stat(fname, &st) == 0){ if(S_ISDIR(st.st_mode)){ folder = data->folder; rc = import_mbox_from_dir(fname, sdata, data, tot_msgs, cfg); data->folder = folder; if(rc == ERR) ret = ERR; } else { if(S_ISREG(st.st_mode)){ if(i == 0 && data->recursive_folder_names == 1){ folder = get_folder_id(sdata, data, fname, data->folder, cfg); if(folder == ERR_FOLDER){ folder = add_new_folder(sdata, data, fname, data->folder, cfg); if(folder == ERR_FOLDER){ printf("error: cannot get/add folder '%s' to parent id: %d\n", fname, data->folder); return ERR; } else { data->folder = folder; } } } rc = import_from_mailbox(fname, sdata, data, cfg); if(rc == OK) (*tot_msgs)++; else ret = ERR; i++; } else { printf("%s is not a file\n", fname); } } } else { printf("cannot stat() %s\n", fname); } } closedir(dir); return ret; } int import_from_maildir(char *directory, struct session_data *sdata, struct __data *data, int *tot_msgs, struct __config *cfg){ DIR *dir; struct dirent *de; int rc=ERR, ret=OK, i=0; int folder; char *p, fname[SMALLBUFSIZE]; struct stat st; dir = opendir(directory); if(!dir){ printf("cannot open directory: %s\n", directory); return ERR; } while((de = readdir(dir))){ if(strcmp(de->d_name, ".") == 0 || strcmp(de->d_name, "..") == 0) continue; snprintf(fname, sizeof(fname)-1, "%s/%s", directory, de->d_name); if(stat(fname, &st) == 0){ if(S_ISDIR(st.st_mode)){ folder = data->folder; rc = import_from_maildir(fname, sdata, data, tot_msgs, cfg); data->folder = folder; if(rc == ERR) ret = ERR; } else { if(S_ISREG(st.st_mode)){ if(i == 0 && data->recursive_folder_names == 1){ p = strrchr(directory, '/'); if(p) p++; else { printf("invalid directory name: '%s'\n", directory); return ERR; } folder = get_folder_id(sdata, data, p, data->folder, cfg); if(folder == ERR_FOLDER){ folder = add_new_folder(sdata, data, p, data->folder, cfg); if(folder == ERR_FOLDER){ printf("error: cannot get/add folder '%s' to parent id: %d\n", p, data->folder); return ERR; } else { data->folder = folder; } } } rc = import_message(fname, sdata, data, cfg); if(rc == OK) (*tot_msgs)++; else { printf("error importing: '%s'\n", fname); ret = ERR; } if(data->import->remove_after_import == 1 && ret != ERR) unlink(fname); i++; if(data->quiet == 0) printf("processed: %7d\r", *tot_msgs); fflush(stdout); } else { printf("%s is not a file\n", fname); } } } else { printf("cannot stat() %s\n", fname); } } closedir(dir); return ret; } int import_from_imap_server(char *server, char *username, char *password, int port, struct session_data *sdata, struct __data *data, char *skiplist, int dryrun, struct __config *cfg){ int i, rc=ERR, ret=OK, sd, seq=1, skipmatch, use_ssl=0; char port_string[8], puf[SMALLBUFSIZE]; struct addrinfo hints, *res; struct node *q; inithash(data->imapfolders); 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 ERR; } if(port == 993) use_ssl = 1; if((sd = socket(res->ai_family, res->ai_socktype, res->ai_protocol)) == -1){ printf("cannot create socket\n"); ret = ERR; goto ENDE_IMAP; } if(connect(sd, res->ai_addr, res->ai_addrlen) == -1){ printf("connect()\n"); ret = ERR; goto ENDE_IMAP; } if(connect_to_imap_server(sd, &seq, username, password, port, data, use_ssl) == ERR){ close(sd); ret = ERR; goto ENDE_IMAP; } /* * if the user gives -f , then don't iterate through the folder list * rather build the folderlist based on the option, 2014.10.14, SJ */ if(folder_imap){ addnode(data->imapfolders, folder_imap); } else { rc = list_folders(sd, &seq, use_ssl, data); if(rc == ERR) goto ENDE_IMAP; } for(i=0;iimapfolders[i]; while(q != NULL){ if(q && q->str && strlen(q->str) > 1){ skipmatch = 0; if(skiplist && strlen(skiplist) > 0){ snprintf(puf, sizeof(puf)-1, "%s,", (char *)q->str); if(strstr(skiplist, puf)) skipmatch = 1; } if(folder_imap && strstr(q->str, folder_imap) == NULL){ skipmatch = 1; } if(skipmatch == 1){ if(data->quiet == 0) printf("SKIPPING FOLDER: %s\n", (char *)q->str); } else { if(data->quiet == 0) printf("processing folder: %s... ", (char *)q->str); if(process_imap_folder(sd, &seq, q->str, sdata, data, use_ssl, dryrun, cfg) == ERR) ret = ERR; } } q = q->r; } } send_imap_close(sd, &seq, data, use_ssl); close_connection(sd, data, use_ssl); ENDE_IMAP: freeaddrinfo(res); clearhash(data->imapfolders); data->import->status = 2; return ret; } int import_from_pop3_server(char *server, char *username, char *password, int port, struct session_data *sdata, struct __data *data, int dryrun, struct __config *cfg){ int rc, ret=OK, sd, use_ssl=0; char port_string[8]; 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 ERR; } if(port == 995) use_ssl = 1; if((sd = socket(res->ai_family, res->ai_socktype, res->ai_protocol)) == -1){ printf("cannot create socket\n"); ret = ERR; goto ENDE_POP3; } if(connect(sd, res->ai_addr, res->ai_addrlen) == -1){ printf("connect()\n"); ret = ERR; goto ENDE_POP3; } if(connect_to_pop3_server(sd, username, password, port, data, use_ssl) == ERR){ close(sd); ret = ERR; goto ENDE_POP3; } if(process_pop3_emails(sd, sdata, data, use_ssl, dryrun, cfg) == ERR) ret = ERR; close_connection(sd, data, use_ssl); ENDE_POP3: freeaddrinfo(res); return ret; } int read_gui_import_data(struct session_data *sdata, struct __data *data, char *skiplist, int dryrun, struct __config *cfg){ int rc=ERR; char s_type[SMALLBUFSIZE], s_username[SMALLBUFSIZE], s_password[SMALLBUFSIZE], s_server[SMALLBUFSIZE]; memset(s_type, 0, sizeof(s_type)); memset(s_username, 0, sizeof(s_username)); memset(s_password, 0, sizeof(s_password)); memset(s_server, 0, sizeof(s_server)); if(prepare_sql_statement(sdata, &(data->stmt_generic), SQL_PREPARED_STMT_GET_GUI_IMPORT_JOBS, cfg) == ERR) return ERR; p_bind_init(data); if(p_exec_query(sdata, data->stmt_generic, data) == OK){ p_bind_init(data); data->sql[data->pos] = (char *)&(data->import->import_job_id); data->type[data->pos] = TYPE_LONG; data->len[data->pos] = sizeof(int); data->pos++; data->sql[data->pos] = &s_type[0]; data->type[data->pos] = TYPE_STRING; data->len[data->pos] = sizeof(s_type)-2; data->pos++; data->sql[data->pos] = &s_username[0]; data->type[data->pos] = TYPE_STRING; data->len[data->pos] = sizeof(s_username)-2; data->pos++; data->sql[data->pos] = &s_password[0]; data->type[data->pos] = TYPE_STRING; data->len[data->pos] = sizeof(s_password)-2; data->pos++; data->sql[data->pos] = &s_server[0]; data->type[data->pos] = TYPE_STRING; data->len[data->pos] = sizeof(s_server)-2; data->pos++; p_store_results(sdata, data->stmt_generic, data); if(p_fetch_results(data->stmt_generic) == OK) rc = OK; p_free_results(data->stmt_generic); } close_prepared_statement(data->stmt_generic); data->import->processed_messages = 0; data->import->total_messages = 0; time(&(data->import->started)); data->import->status = 1; update_import_job_stat(sdata, data); if(strcmp(s_type, "pop3") == 0){ rc = import_from_pop3_server(s_server, s_username, s_password, 110, sdata, data, dryrun, cfg); } if(strcmp(s_type, "imap") == 0){ rc = import_from_imap_server(s_server, s_username, s_password, 143, sdata, data, skiplist, dryrun, cfg); } update_import_job_stat(sdata, data); // don't set error in case of a problem, because it // will scare users looking at the gui progressbar /*if(rc == ERR){ data->import->status = 3; update_import_job_stat(sdata, data); }*/ return rc; } void usage(){ printf("\nusage: pilerimport\n\n"); printf(" [-c ] Config file to use if not the default\n"); printf(" -e EML file to import\n"); printf(" -m Mbox file to import\n"); printf(" -d Directory with EML files to import\n"); printf(" -i IMAP server to connect\n"); printf(" -K POP3 server to connect\n"); printf(" -u Username for imap/pop3 import\n"); printf(" -p Password for imap/pop3 import\n"); printf(" -P Port for imap/pop3 import (default: 143/110\n"); printf(" -t Timeout in sec for imap/pop3 import\n"); printf(" -x Comma separated list of imap folders to skip. Add the trailing comma!\n"); printf(" -f IMAP folder name to import\n"); printf(" -g Move email after import to this IMAP folder\n"); printf(" -F Piler folder name to assign to this import\n"); printf(" -R Assign IMAP folder names as Piler folder names\n"); printf(" -b Import only this many emails\n"); printf(" -s Start importing POP3 emails from this position\n"); printf(" -D Dry-run, do not import anything\n"); printf(" -o Only download emails for POP3/IMAP import\n"); printf(" -r Remove imported emails\n"); printf(" -q Quiet mode\n"); exit(0); } int main(int argc, char **argv){ int i, c, rc=0, n_mbox=0, tot_msgs=0, port=143; char *configfile=CONFIG_FILE, *emlfile=NULL, *mboxdir=NULL, *mbox[MBOX_ARGS], *directory=NULL; char *imapserver=NULL, *pop3server=NULL, *username=NULL, *password=NULL, *skiplist=SKIPLIST, *folder=NULL; struct session_data sdata; struct __config cfg; struct __data data; struct import import; for(i=0; i arguments: %s\n", optarg); } break; case 'M' : mboxdir = optarg; break; case 'i' : imapserver = optarg; break; case 'K' : pop3server = optarg; if(port == 143) port = 110; break; case 'u' : username = optarg; break; case 'p' : password = optarg; break; case 'P' : port = atoi(optarg); break; case 'x' : skiplist = optarg; break; case 'F' : folder = optarg; break; case 'f' : folder_imap = optarg; break; case 'R' : data.recursive_folder_names = 1; break; case 'r' : data.import->remove_after_import = 1; break; case 'g' : data.import->move_folder = optarg; break; case 'o' : data.import->download_only = 1; dryrun = 1; break; case 'b' : data.import->batch_processing_limit = atoi(optarg); break; case 't' : data.import->timeout = atoi(optarg); break; case 's' : if(atoi(optarg) < 1){ printf("invalid start position: %s\n", optarg); return -1; } data.import->start_position = atoi(optarg); break; case 'a' : data.import->extra_recipient = optarg; break; case 'G' : import_from_gui = 1; break; case 'D' : dryrun = 1; break; case 'q' : data.quiet = 1; break; case 'h' : case '?' : usage(); break; default : break; } } if(!mbox[0] && !mboxdir && !emlfile && !directory && !imapserver && !pop3server && import_from_gui == 0) usage(); if(!can_i_write_current_directory()) __fatal("cannot write current directory!"); cfg = read_config(configfile); if((data.recursive_folder_names == 1 || folder) && cfg.enable_folders == 0){ printf("please set enable_folders=1 in piler.conf to use the folder options\n"); return ERR; } /* make sure we don't discard messages without a valid Message-Id when importing manually */ cfg.archive_emails_not_having_message_id = 1; if(read_key(&cfg)){ printf("%s\n", ERR_READING_KEY); return ERR; } if(open_database(&sdata, &cfg) == ERR) return 0; setlocale(LC_CTYPE, cfg.locale); (void) openlog("pilerimport", LOG_PID, LOG_MAIL); #ifdef HAVE_MEMCACHED memcached_init(&(data.memc), cfg.memcached_servers, 11211); #endif if(folder){ data.folder = get_folder_id(&sdata, &data, folder, 0, &cfg); if(data.folder == ERR_FOLDER){ data.folder = add_new_folder(&sdata, &data, folder, 0, &cfg); } if(data.folder == ERR_FOLDER){ printf("error: cannot get/add folder '%s'\n", folder); close_database(&sdata); return 0; } } load_rules(&sdata, &data, data.archiving_rules, SQL_ARCHIVING_RULE_TABLE, &cfg); load_rules(&sdata, &data, data.retention_rules, SQL_RETENTION_RULE_TABLE, &cfg); load_rules(&sdata, &data, data.folder_rules, SQL_FOLDER_RULE_TABLE, &cfg); load_mydomains(&sdata, &data, &cfg); if(emlfile) rc = import_message(emlfile, &sdata, &data, &cfg); if(mbox[0]){ for(i=0; i