2011-11-14 15:57:52 +01:00
/*
* message . c , SJ
*/
# include <stdio.h>
# include <stdlib.h>
# include <string.h>
# include <time.h>
# include <sys/types.h>
# include <sys/mman.h>
# include <sys/socket.h>
# include <sys/stat.h>
# include <fcntl.h>
# include <unistd.h>
# include <syslog.h>
# include <piler.h>
# include <zlib.h>
2012-08-23 10:23:58 +02:00
int store_index_data ( struct session_data * sdata , struct _state * state , struct __data * data , uint64 id , struct __config * cfg ) {
2012-01-29 09:02:21 +01:00
int rc = ERR ;
2013-01-27 23:42:45 +01:00
char * subj ;
2011-12-03 23:05:00 +01:00
2011-11-28 14:21:14 +01:00
subj = state - > b_subject ;
if ( * subj = = ' ' ) subj + + ;
2011-11-14 15:57:52 +01:00
2013-05-05 11:57:14 +02:00
if ( prepare_sql_statement ( sdata , & ( data - > stmt_insert_into_sphinx_table ) , SQL_PREPARED_STMT_INSERT_INTO_SPHINX_TABLE ) = = ERR ) return rc ;
2011-11-14 15:57:52 +01:00
2011-12-03 23:05:00 +01:00
2011-12-05 17:18:03 +01:00
fix_email_address_for_sphinx ( state - > b_from ) ;
fix_email_address_for_sphinx ( state - > b_to ) ;
2012-01-14 09:53:26 +01:00
fix_email_address_for_sphinx ( state - > b_from_domain ) ;
fix_email_address_for_sphinx ( state - > b_to_domain ) ;
2011-12-05 17:18:03 +01:00
2011-12-03 23:05:00 +01:00
2013-05-05 11:57:14 +02:00
p_bind_init ( data ) ;
2011-12-03 23:05:00 +01:00
2013-05-05 11:57:14 +02:00
data - > sql [ data - > pos ] = ( char * ) & id ; data - > type [ data - > pos ] = TYPE_LONGLONG ; data - > pos + + ;
data - > sql [ data - > pos ] = state - > b_from ; data - > type [ data - > pos ] = TYPE_STRING ; data - > pos + + ;
data - > sql [ data - > pos ] = state - > b_to ; data - > type [ data - > pos ] = TYPE_STRING ; data - > pos + + ;
data - > sql [ data - > pos ] = state - > b_from_domain ; data - > type [ data - > pos ] = TYPE_STRING ; data - > pos + + ;
data - > sql [ data - > pos ] = state - > b_to_domain ; data - > type [ data - > pos ] = TYPE_STRING ; data - > pos + + ;
data - > sql [ data - > pos ] = subj ; data - > type [ data - > pos ] = TYPE_STRING ; data - > pos + + ;
data - > sql [ data - > pos ] = state - > b_body ; data - > type [ data - > pos ] = TYPE_STRING ; data - > pos + + ;
data - > sql [ data - > pos ] = ( char * ) & sdata - > now ; data - > type [ data - > pos ] = TYPE_LONG ; data - > pos + + ;
data - > sql [ data - > pos ] = ( char * ) & sdata - > sent ; data - > type [ data - > pos ] = TYPE_LONG ; data - > pos + + ;
data - > sql [ data - > pos ] = ( char * ) & sdata - > tot_len ; data - > type [ data - > pos ] = TYPE_LONG ; data - > pos + + ;
data - > sql [ data - > pos ] = ( char * ) & sdata - > direction ; data - > type [ data - > pos ] = TYPE_LONG ; data - > pos + + ;
data - > sql [ data - > pos ] = ( char * ) & data - > folder ; data - > type [ data - > pos ] = TYPE_LONG ; data - > pos + + ;
data - > sql [ data - > pos ] = ( char * ) & state - > n_attachments ; data - > type [ data - > pos ] = TYPE_LONG ; data - > pos + + ;
data - > sql [ data - > pos ] = sdata - > attachments ; data - > type [ data - > pos ] = TYPE_STRING ; data - > pos + + ;
2013-05-15 11:46:08 +02:00
# ifdef HAVE_MULTITENANCY
data - > sql [ data - > pos ] = ( char * ) & sdata - > customer_id ; data - > type [ data - > pos ] = TYPE_SHORT ; data - > pos + + ;
# endif
2011-12-03 23:05:00 +01:00
2013-05-05 11:57:14 +02:00
if ( p_exec_query ( sdata , data - > stmt_insert_into_sphinx_table , data ) = = OK ) rc = OK ;
2011-12-03 23:05:00 +01:00
2013-05-05 11:57:14 +02:00
close_prepared_statement ( data - > stmt_insert_into_sphinx_table ) ;
2012-01-29 09:02:21 +01:00
return rc ;
2011-12-03 23:05:00 +01:00
}
2013-01-27 23:42:45 +01:00
uint64 get_metaid_by_messageid ( struct session_data * sdata , struct __data * data , char * message_id , struct __config * cfg ) {
2012-10-05 14:27:03 +02:00
uint64 id = 0 ;
2013-06-29 15:43:16 +02:00
if ( prepare_sql_statement ( sdata , & ( data - > stmt_get_meta_id_by_message_id ) , SQL_PREPARED_STMT_GET_META_ID_BY_MESSAGE_ID ) = = ERR ) return id ;
2012-10-05 14:27:03 +02:00
2013-05-05 11:57:14 +02:00
p_bind_init ( data ) ;
data - > sql [ data - > pos ] = message_id ; data - > type [ data - > pos ] = TYPE_STRING ; data - > pos + + ;
2012-10-05 14:27:03 +02:00
2013-05-05 11:57:14 +02:00
if ( p_exec_query ( sdata , data - > stmt_get_meta_id_by_message_id , data ) = = ERR ) goto CLOSE ;
2012-10-05 14:27:03 +02:00
2013-05-05 11:57:14 +02:00
p_bind_init ( data ) ;
data - > sql [ data - > pos ] = ( char * ) & id ; data - > type [ data - > pos ] = TYPE_LONGLONG ; data - > len [ data - > pos ] = sizeof ( uint64 ) ; data - > pos + + ;
2012-10-05 14:27:03 +02:00
2013-05-05 11:57:14 +02:00
p_store_results ( sdata , data - > stmt_get_meta_id_by_message_id , data ) ;
2012-10-05 14:27:03 +02:00
2013-05-05 11:57:14 +02:00
p_fetch_results ( data - > stmt_get_meta_id_by_message_id ) ;
2012-10-05 14:27:03 +02:00
2013-05-05 11:57:14 +02:00
p_free_results ( data - > stmt_get_meta_id_by_message_id ) ;
2012-10-05 14:27:03 +02:00
CLOSE :
2013-06-29 15:43:16 +02:00
mysql_stmt_close ( data - > stmt_get_meta_id_by_message_id ) ;
2012-10-05 14:27:03 +02:00
return id ;
}
2013-01-27 23:42:45 +01:00
int store_recipients ( struct session_data * sdata , struct __data * data , char * to , uint64 id , int log_errors , struct __config * cfg ) {
2012-09-04 14:49:56 +02:00
int ret = OK , n = 0 ;
2013-01-27 23:42:45 +01:00
char * p , * q , puf [ SMALLBUFSIZE ] ;
2011-12-03 23:05:00 +01:00
2013-06-29 15:43:16 +02:00
if ( prepare_sql_statement ( sdata , & ( data - > stmt_insert_into_rcpt_table ) , SQL_PREPARED_STMT_INSERT_INTO_RCPT_TABLE ) = = ERR ) return ret ;
2011-12-03 23:05:00 +01:00
p = to ;
do {
p = split_str ( p , " " , puf , sizeof ( puf ) - 1 ) ;
2012-01-09 23:15:39 +01:00
q = strchr ( puf , ' @ ' ) ;
if ( q & & strlen ( q ) > 3 & & does_it_seem_like_an_email_address ( puf ) = = 1 ) {
q + + ;
2011-12-03 23:05:00 +01:00
2013-05-05 11:57:14 +02:00
p_bind_init ( data ) ;
2011-12-03 23:05:00 +01:00
2013-05-05 11:57:14 +02:00
data - > sql [ data - > pos ] = ( char * ) & id ; data - > type [ data - > pos ] = TYPE_LONGLONG ; data - > pos + + ;
data - > sql [ data - > pos ] = & puf [ 0 ] ; data - > type [ data - > pos ] = TYPE_STRING ; data - > pos + + ;
data - > sql [ data - > pos ] = q ; data - > type [ data - > pos ] = TYPE_STRING ; data - > pos + + ;
2012-01-09 23:15:39 +01:00
2013-01-27 23:42:45 +01:00
2013-05-05 11:57:14 +02:00
if ( p_exec_query ( sdata , data - > stmt_insert_into_rcpt_table , data ) = = ERR ) {
2011-12-03 23:05:00 +01:00
ret = ERR ;
}
2012-09-04 14:49:56 +02:00
else n + + ;
2011-12-03 23:05:00 +01:00
}
} while ( p ) ;
2013-06-29 15:43:16 +02:00
mysql_stmt_close ( data - > stmt_insert_into_rcpt_table ) ;
2012-01-29 09:02:21 +01:00
2012-09-04 14:49:56 +02:00
if ( cfg - > verbosity > = _LOG_DEBUG ) syslog ( LOG_PRIORITY , " %s: added %d recipients " , sdata - > ttmpfile , n ) ;
2011-12-03 23:05:00 +01:00
return ret ;
2011-11-14 15:57:52 +01:00
}
2013-03-18 10:13:26 +01:00
int update_metadata_reference ( struct session_data * sdata , struct _state * state , struct __data * data , char * ref , struct __config * cfg ) {
2013-05-05 11:57:14 +02:00
int ret = ERR ;
2013-03-18 10:13:26 +01:00
2013-06-29 15:43:16 +02:00
if ( prepare_sql_statement ( sdata , & ( data - > stmt_update_metadata_reference ) , SQL_PREPARED_STMT_UPDATE_METADATA_REFERENCE ) = = ERR ) return ret ;
2013-03-18 10:13:26 +01:00
2013-05-05 11:57:14 +02:00
p_bind_init ( data ) ;
2013-03-18 10:13:26 +01:00
2013-05-05 11:57:14 +02:00
data - > sql [ data - > pos ] = ref ; data - > type [ data - > pos ] = TYPE_STRING ; data - > pos + + ;
data - > sql [ data - > pos ] = state - > reference ; data - > type [ data - > pos ] = TYPE_STRING ; data - > pos + + ;
2013-03-18 10:13:26 +01:00
2013-05-05 11:57:14 +02:00
if ( p_exec_query ( sdata , data - > stmt_update_metadata_reference , data ) = = OK ) ret = OK ;
2013-03-18 10:13:26 +01:00
2013-06-29 15:43:16 +02:00
mysql_stmt_close ( data - > stmt_update_metadata_reference ) ;
2013-05-05 11:57:14 +02:00
if ( cfg - > verbosity > = _LOG_DEBUG ) syslog ( LOG_PRIORITY , " %s: updated meta reference for '%s', rc=%d " , sdata - > ttmpfile , state - > reference , ret ) ;
2013-03-18 10:13:26 +01:00
return ret ;
}
2012-08-23 10:23:58 +02:00
int store_meta_data ( struct session_data * sdata , struct _state * state , struct __data * data , struct __config * cfg ) {
2011-12-03 23:05:00 +01:00
int rc , ret = ERR ;
2012-02-10 14:06:00 +01:00
char * subj , * p , s [ MAXBUFSIZE ] , s2 [ SMALLBUFSIZE ] , vcode [ 2 * DIGEST_LENGTH + 1 ] , ref [ 2 * DIGEST_LENGTH + 1 ] ;
2013-05-05 11:57:14 +02:00
uint64 id = 0 ;
2011-12-03 23:05:00 +01:00
2011-11-28 14:21:14 +01:00
subj = state - > b_subject ;
if ( * subj = = ' ' ) subj + + ;
2012-02-19 22:59:47 +01:00
snprintf ( s , sizeof ( s ) - 1 , " %llu+%s%s%s%ld%ld%ld%d%d%d%d%s%s%s " , id , subj , state - > b_from , state - > message_id , sdata - > now , sdata - > sent , sdata - > retained , sdata - > tot_len , sdata - > hdr_len , sdata - > direction , state - > n_attachments , sdata - > ttmpfile , sdata - > digest , sdata - > bodydigest ) ;
2012-02-10 14:06:00 +01:00
2012-01-14 09:53:26 +01:00
digest_string ( s , & vcode [ 0 ] ) ;
2012-02-10 14:06:00 +01:00
memset ( ref , 0 , sizeof ( ref ) ) ;
2013-03-18 10:13:26 +01:00
if ( strlen ( state - > reference ) > 10 ) {
digest_string ( state - > reference , & ref [ 0 ] ) ;
update_metadata_reference ( sdata , state , data , & ref [ 0 ] , cfg ) ;
}
2012-02-10 14:06:00 +01:00
2012-01-14 09:53:26 +01:00
2013-05-05 11:57:14 +02:00
if ( prepare_sql_statement ( sdata , & ( data - > stmt_insert_into_meta_table ) , SQL_PREPARED_STMT_INSERT_INTO_META_TABLE ) = = ERR ) return ERR ;
2012-02-07 17:21:23 +01:00
memset ( s2 , 0 , sizeof ( s2 ) ) ;
2011-11-14 15:57:52 +01:00
2012-02-07 17:21:23 +01:00
p = state - > b_from ;
do {
memset ( s2 , 0 , sizeof ( s2 ) ) ;
p = split ( p , ' ' , s2 , sizeof ( s2 ) - 1 ) ;
if ( s2 [ 0 ] = = ' \0 ' ) continue ;
if ( does_it_seem_like_an_email_address ( s2 ) = = 1 ) { break ; }
} while ( p ) ;
2011-11-14 15:57:52 +01:00
if ( strlen ( state - > b_to ) < 5 ) {
2012-01-09 23:15:39 +01:00
snprintf ( state - > b_to , SMALLBUFSIZE - 1 , " undisclosed-recipients@no.domain " ) ;
2011-11-14 15:57:52 +01:00
}
2013-05-05 11:57:14 +02:00
p_bind_init ( data ) ;
data - > sql [ data - > pos ] = & s2 [ 0 ] ; data - > type [ data - > pos ] = TYPE_STRING ; data - > pos + + ;
data - > sql [ data - > pos ] = state - > b_from_domain ; data - > type [ data - > pos ] = TYPE_STRING ; data - > pos + + ;
data - > sql [ data - > pos ] = subj ; data - > type [ data - > pos ] = TYPE_STRING ; data - > pos + + ;
data - > sql [ data - > pos ] = ( char * ) & sdata - > spam_message ; data - > type [ data - > pos ] = TYPE_LONG ; data - > pos + + ;
data - > sql [ data - > pos ] = ( char * ) & sdata - > now ; data - > type [ data - > pos ] = TYPE_LONG ; data - > pos + + ;
data - > sql [ data - > pos ] = ( char * ) & sdata - > sent ; data - > type [ data - > pos ] = TYPE_LONG ; data - > pos + + ;
data - > sql [ data - > pos ] = ( char * ) & sdata - > retained ; data - > type [ data - > pos ] = TYPE_LONG ; data - > pos + + ;
data - > sql [ data - > pos ] = ( char * ) & sdata - > tot_len ; data - > type [ data - > pos ] = TYPE_LONG ; data - > pos + + ;
data - > sql [ data - > pos ] = ( char * ) & sdata - > hdr_len ; data - > type [ data - > pos ] = TYPE_LONG ; data - > pos + + ;
data - > sql [ data - > pos ] = ( char * ) & sdata - > direction ; data - > type [ data - > pos ] = TYPE_LONG ; data - > pos + + ;
data - > sql [ data - > pos ] = ( char * ) & state - > n_attachments ; data - > type [ data - > pos ] = TYPE_LONG ; data - > pos + + ;
data - > sql [ data - > pos ] = sdata - > ttmpfile ; data - > type [ data - > pos ] = TYPE_STRING ; data - > pos + + ;
data - > sql [ data - > pos ] = state - > message_id ; data - > type [ data - > pos ] = TYPE_STRING ; data - > pos + + ;
data - > sql [ data - > pos ] = & ref [ 0 ] ; data - > type [ data - > pos ] = TYPE_STRING ; data - > pos + + ;
data - > sql [ data - > pos ] = sdata - > digest ; data - > type [ data - > pos ] = TYPE_STRING ; data - > pos + + ;
data - > sql [ data - > pos ] = sdata - > bodydigest ; data - > type [ data - > pos ] = TYPE_STRING ; data - > pos + + ;
data - > sql [ data - > pos ] = & vcode [ 0 ] ; data - > type [ data - > pos ] = TYPE_STRING ; data - > pos + + ;
2013-05-15 11:46:08 +02:00
# ifdef HAVE_MULTITENANCY
data - > sql [ data - > pos ] = ( char * ) & sdata - > customer_id ; data - > type [ data - > pos ] = TYPE_SHORT ; data - > pos + + ;
# endif
2013-05-05 11:57:14 +02:00
if ( p_exec_query ( sdata , data - > stmt_insert_into_meta_table , data ) = = ERR ) {
2011-12-03 23:05:00 +01:00
ret = ERR_EXISTS ;
}
else {
2013-05-05 11:57:14 +02:00
id = p_get_insert_id ( data - > stmt_insert_into_meta_table ) ;
2012-01-09 23:15:39 +01:00
2013-01-27 23:42:45 +01:00
rc = store_recipients ( sdata , data , state - > b_to , id , 1 , cfg ) ;
2011-11-14 15:57:52 +01:00
2011-12-03 23:05:00 +01:00
if ( cfg - > verbosity > = _LOG_DEBUG ) syslog ( LOG_PRIORITY , " %s: stored recipients, rc=%d " , sdata - > ttmpfile , rc ) ;
2011-11-14 15:57:52 +01:00
2011-12-03 23:05:00 +01:00
if ( rc = = OK ) {
2012-01-09 23:15:39 +01:00
2012-08-23 10:23:58 +02:00
rc = store_index_data ( sdata , state , data , id , cfg ) ;
2011-11-14 15:57:52 +01:00
2011-12-03 23:05:00 +01:00
if ( cfg - > verbosity > = _LOG_DEBUG ) syslog ( LOG_PRIORITY , " %s: stored indexdata, rc=%d " , sdata - > ttmpfile , rc ) ;
2011-11-14 15:57:52 +01:00
2011-12-03 23:05:00 +01:00
if ( rc = = OK )
ret = OK ;
2011-11-14 15:57:52 +01:00
}
2011-12-03 23:05:00 +01:00
}
2011-11-14 15:57:52 +01:00
2013-05-05 11:57:14 +02:00
close_prepared_statement ( data - > stmt_insert_into_meta_table ) ;
2012-01-29 09:02:21 +01:00
2011-11-14 15:57:52 +01:00
return ret ;
}
2012-12-15 21:25:36 +01:00
void remove_stripped_attachments ( struct _state * state ) {
int i ;
for ( i = 1 ; i < = state - > n_attachments ; i + + ) unlink ( state - > attachments [ i ] . internalname ) ;
}
2012-02-07 17:21:23 +01:00
int process_message ( struct session_data * sdata , struct _state * state , struct __data * data , struct __config * cfg ) {
2013-07-12 23:01:12 +02:00
int rc ;
2012-10-05 14:27:03 +02:00
uint64 id = 0 ;
2011-11-14 15:57:52 +01:00
/* discard if existing message_id */
2013-01-27 23:42:45 +01:00
id = get_metaid_by_messageid ( sdata , data , state - > message_id , cfg ) ;
if ( id > 0 ) {
2012-12-15 21:25:36 +01:00
remove_stripped_attachments ( state ) ;
2012-10-05 14:27:03 +02:00
if ( strlen ( state - > b_journal_to ) > 0 ) {
if ( cfg - > verbosity > = _LOG_DEBUG ) syslog ( LOG_PRIORITY , " %s: trying to add journal rcpt (%s) to id=%llu for message-id: '%s' " , sdata - > ttmpfile , state - > b_journal_to , id , state - > message_id ) ;
2013-01-27 23:42:45 +01:00
store_recipients ( sdata , data , state - > b_journal_to , id , 0 , cfg ) ;
2012-10-05 14:27:03 +02:00
}
2011-11-14 15:57:52 +01:00
return ERR_EXISTS ;
}
2011-11-23 12:24:21 +01:00
/* store base64 encoded file attachments */
2011-11-19 21:25:44 +01:00
2011-11-23 12:24:21 +01:00
if ( state - > n_attachments > 0 ) {
2013-01-28 12:10:39 +01:00
rc = store_attachments ( sdata , state , data , cfg ) ;
2011-11-19 21:25:44 +01:00
2013-07-12 23:01:12 +02:00
remove_stripped_attachments ( state ) ;
2011-11-23 12:24:21 +01:00
if ( rc ) return ERR ;
2011-11-19 21:25:44 +01:00
}
rc = store_file ( sdata , sdata - > tmpframe , 0 , 0 , cfg ) ;
if ( rc = = 0 ) {
syslog ( LOG_PRIORITY , " %s: error storing message: %s " , sdata - > ttmpfile , sdata - > tmpframe ) ;
return ERR ;
}
2011-11-14 15:57:52 +01:00
2013-07-31 09:10:26 +02:00
sdata - > retained + = query_retain_period ( data , state , sdata - > tot_len , sdata - > spam_message , cfg ) ;
2012-02-19 22:59:47 +01:00
2012-08-23 10:23:58 +02:00
rc = store_meta_data ( sdata , state , data , cfg ) ;
2011-11-14 15:57:52 +01:00
if ( cfg - > verbosity > = _LOG_DEBUG ) syslog ( LOG_PRIORITY , " %s: stored metadata, rc=%d " , sdata - > ttmpfile , rc ) ;
2012-05-29 14:08:56 +02:00
if ( rc = = ERR_EXISTS ) {
remove_stored_message_files ( sdata , state , cfg ) ;
return ERR_EXISTS ;
}
2011-11-14 15:57:52 +01:00
return OK ;
}