2012-06-01 14:25:49 +02:00
/*
* reindex . c , SJ
*/
# include <stdio.h>
# include <stdlib.h>
# include <string.h>
# include <sys/time.h>
# include <sys/types.h>
# include <sys/stat.h>
# include <fcntl.h>
# include <unistd.h>
# include <time.h>
# include <locale.h>
# include <syslog.h>
# include <getopt.h>
# include <piler.h>
extern char * optarg ;
extern int optind ;
int progressbar = 0 ;
void usage ( ) {
printf ( " \n usage: reindex \n \n " ) ;
printf ( " [-c|--config <config file>] \n " ) ;
printf ( " -f <from id> \n " ) ;
printf ( " -t <to id> \n " ) ;
2012-11-30 12:27:18 +01:00
printf ( " -a \n " ) ;
2012-06-01 14:25:49 +02:00
printf ( " [-p] \n " ) ;
exit ( 0 ) ;
}
2012-10-29 10:22:31 +01:00
void p_clean_exit ( char * msg , int rc ) {
2012-06-01 14:25:49 +02:00
if ( msg ) printf ( " error: %s \n " , msg ) ;
exit ( rc ) ;
}
2012-11-30 12:27:18 +01:00
uint64 get_max_meta_id ( struct session_data * sdata ) {
MYSQL_RES * res ;
MYSQL_ROW row ;
char s [ SMALLBUFSIZE ] ;
int rc ;
uint64 id = 0 ;
snprintf ( s , sizeof ( s ) - 1 , " SELECT MAX(`id`) FROM %s " , SQL_METADATA_TABLE ) ;
rc = mysql_real_query ( & ( sdata - > mysql ) , s , strlen ( s ) ) ;
if ( rc = = 0 ) {
res = mysql_store_result ( & ( sdata - > mysql ) ) ;
if ( res ) {
row = mysql_fetch_row ( res ) ;
if ( row ) {
id = strtoull ( row [ 0 ] , NULL , 10 ) ;
}
mysql_free_result ( res ) ;
}
}
return id ;
}
2012-08-23 10:23:58 +02:00
uint64 retrieve_email_by_metadata_id ( struct session_data * sdata , struct __data * data , uint64 from_id , uint64 to_id , struct __config * cfg ) {
2012-06-01 14:25:49 +02:00
MYSQL_RES * res ;
MYSQL_ROW row ;
FILE * f ;
char filename [ SMALLBUFSIZE ] ;
char s [ SMALLBUFSIZE ] ;
int rc = 0 ;
uint64 stored_id = 0 , reindexed = 0 ;
struct _state state ;
2012-06-01 14:54:13 +02:00
snprintf ( s , sizeof ( s ) - 1 , " SELECT `id`, `piler_id`, `arrived`, `sent` FROM %s WHERE (id BETWEEN %llu AND %llu) AND `deleted`=0 " , SQL_METADATA_TABLE , from_id , to_id ) ;
2012-06-01 14:25:49 +02:00
rc = mysql_real_query ( & ( sdata - > mysql ) , s , strlen ( s ) ) ;
if ( rc = = 0 ) {
res = mysql_store_result ( & ( sdata - > mysql ) ) ;
if ( res ) {
while ( ( row = mysql_fetch_row ( res ) ) ) {
stored_id = strtoull ( row [ 0 ] , NULL , 10 ) ;
if ( stored_id > 0 ) {
snprintf ( sdata - > ttmpfile , SMALLBUFSIZE - 1 , " %s " , ( char * ) row [ 1 ] ) ;
snprintf ( filename , sizeof ( filename ) - 1 , " %llu.eml " , stored_id ) ;
f = fopen ( filename , " w " ) ;
if ( f ) {
2013-01-28 12:10:39 +01:00
rc = retrieve_email_from_archive ( sdata , data , f , cfg ) ;
2012-06-01 14:25:49 +02:00
fclose ( f ) ;
if ( rc ) {
printf ( " cannot retrieve: %s \n " , filename ) ;
unlink ( filename ) ;
continue ;
}
snprintf ( sdata - > filename , SMALLBUFSIZE - 1 , " %s " , filename ) ;
2013-01-06 22:16:21 +01:00
state = parse_message ( sdata , 0 , data , cfg ) ;
2012-06-01 14:25:49 +02:00
post_parse ( sdata , & state , cfg ) ;
sdata - > now = strtoul ( row [ 2 ] , NULL , 10 ) ;
sdata - > sent = strtoul ( row [ 3 ] , NULL , 10 ) ;
2012-08-23 10:23:58 +02:00
rc = store_index_data ( sdata , & state , data , stored_id , cfg ) ;
2012-06-01 14:25:49 +02:00
if ( rc = = OK ) reindexed + + ;
else printf ( " failed to add to %s table: %s \n " , SQL_SPHINX_TABLE , filename ) ;
unlink ( filename ) ;
2012-06-01 14:54:13 +02:00
if ( progressbar & & reindexed % 100 = = 0 ) printf ( " . " ) ;
2012-06-01 14:25:49 +02:00
}
else printf ( " cannot open: %s \n " , filename ) ;
}
}
mysql_free_result ( res ) ;
}
else rc = 1 ;
}
if ( progressbar ) printf ( " \n " ) ;
return reindexed ;
}
int main ( int argc , char * * argv ) {
2012-11-30 12:27:18 +01:00
int c , all = 0 ;
2012-06-01 14:25:49 +02:00
uint64 from_id = 0 , to_id = 0 , n = 0 ;
2012-08-23 10:23:58 +02:00
char * configfile = CONFIG_FILE , * folder = NULL ;
2012-06-01 14:25:49 +02:00
struct session_data sdata ;
2012-08-23 10:23:58 +02:00
struct __data data ;
2012-06-01 14:25:49 +02:00
struct __config cfg ;
while ( 1 ) {
2012-11-30 12:27:18 +01:00
c = getopt ( argc , argv , " c:f:t:F:pahv? " ) ;
2012-06-01 14:25:49 +02:00
if ( c = = - 1 ) break ;
switch ( c ) {
case ' c ' :
configfile = optarg ;
break ;
case ' f ' :
from_id = strtoull ( optarg , NULL , 10 ) ;
break ;
case ' t ' :
to_id = strtoull ( optarg , NULL , 10 ) ;
break ;
2012-11-30 12:27:18 +01:00
case ' a ' :
all = 1 ;
break ;
2012-08-23 10:23:58 +02:00
case ' F ' :
folder = optarg ;
break ;
2012-06-01 14:25:49 +02:00
case ' p ' :
progressbar = 1 ;
break ;
default :
usage ( ) ;
break ;
}
}
2012-11-30 12:27:18 +01:00
if ( all = = 0 & & ( from_id < = 0 | | to_id < = 0 ) ) usage ( ) ;
2012-06-01 14:25:49 +02:00
( void ) openlog ( " reindex " , LOG_PID , LOG_MAIL ) ;
cfg = read_config ( configfile ) ;
if ( read_key ( & cfg ) ) {
printf ( " %s \n " , ERR_READING_KEY ) ;
return 1 ;
}
2012-08-23 10:23:58 +02:00
data . folder = 0 ;
2012-09-03 17:12:02 +02:00
data . recursive_folder_names = 0 ;
2012-08-23 10:23:58 +02:00
data . archiving_rules = NULL ;
data . retention_rules = NULL ;
if ( folder ) {
2013-01-28 13:40:44 +01:00
data . folder = get_folder_id ( & sdata , & data , folder , 0 ) ;
2012-08-23 10:23:58 +02:00
if ( data . folder = = 0 ) {
printf ( " error: could not get folder id for '%s' \n " , folder ) ;
return 0 ;
}
}
2013-02-22 15:01:21 +01:00
init_session_data ( & sdata , & cfg ) ;
2012-06-01 14:25:49 +02:00
mysql_init ( & ( sdata . mysql ) ) ;
mysql_options ( & ( sdata . mysql ) , MYSQL_OPT_CONNECT_TIMEOUT , ( const char * ) & cfg . mysql_connect_timeout ) ;
if ( mysql_real_connect ( & ( sdata . mysql ) , cfg . mysqlhost , cfg . mysqluser , cfg . mysqlpwd , cfg . mysqldb , cfg . mysqlport , cfg . mysqlsocket , 0 ) = = 0 ) {
2012-10-29 10:22:31 +01:00
p_clean_exit ( " cannot connect to mysql server " , 1 ) ;
2012-06-01 14:25:49 +02:00
}
mysql_real_query ( & ( sdata . mysql ) , " SET NAMES utf8 " , strlen ( " SET NAMES utf8 " ) ) ;
mysql_real_query ( & ( sdata . mysql ) , " SET CHARACTER SET utf8 " , strlen ( " SET CHARACTER SET utf8 " ) ) ;
2013-01-06 22:16:21 +01:00
load_mydomains ( & sdata , & data , & cfg ) ;
2012-06-01 14:25:49 +02:00
2012-11-30 12:27:18 +01:00
if ( all = = 1 ) {
from_id = 1 ;
to_id = get_max_meta_id ( & sdata ) ;
}
2012-08-23 10:23:58 +02:00
n = retrieve_email_by_metadata_id ( & sdata , & data , from_id , to_id , & cfg ) ;
2012-06-01 14:25:49 +02:00
printf ( " put %llu messages to %s table for reindexing \n " , n , SQL_SPHINX_TABLE ) ;
mysql_close ( & ( sdata . mysql ) ) ;
return 0 ;
}