openldap/servers/slurpd/st.c

332 lines
7.0 KiB
C
Raw Normal View History

/* $OpenLDAP$ */
2003-11-27 02:19:00 +08:00
/* This work is part of OpenLDAP Software <http://www.openldap.org/>.
*
2005-01-02 04:49:32 +08:00
* Copyright 1998-2005 The OpenLDAP Foundation.
2003-11-27 02:19:00 +08:00
* All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted only as authorized by the OpenLDAP
* Public License.
*
* A copy of this license is available in file LICENSE in the
* top-level directory of the distribution or, alternatively, at
* <http://www.OpenLDAP.org/license.html>.
2001-01-21 01:49:05 +08:00
*/
2003-11-27 02:19:00 +08:00
/* Portions Copyright (c) 1996 Regents of the University of Michigan.
1998-08-09 08:43:13 +08:00
* All rights reserved.
*
* Redistribution and use in source and binary forms are permitted
* provided that this notice is preserved and that due credit is given
* to the University of Michigan at Ann Arbor. The name of the University
* may not be used to endorse or promote products derived from this
* software without specific prior written permission. This software
* is provided ``as is'' without express or implied warranty.
*/
2003-11-27 02:19:00 +08:00
/* ACKNOWLEDGEMENTS:
* This work was originally developed by the University of Michigan
* (as part of U-MICH LDAP).
*/
1998-08-09 08:43:13 +08:00
/*
* st.c - routines for managing the status structure, and for reading and
* writing status information to disk.
*/
1998-10-25 09:41:42 +08:00
#include "portable.h"
1998-08-09 08:43:13 +08:00
#include <stdio.h>
1999-06-03 08:37:44 +08:00
#include <ac/stdlib.h>
1998-10-25 09:41:42 +08:00
#include <ac/string.h>
#include <ac/unistd.h>
1998-08-09 08:43:13 +08:00
#include "slurp.h"
#include "globals.h"
2005-11-24 09:10:05 +08:00
#include "lutil.h"
1998-08-09 08:43:13 +08:00
/*
* Add information about replica host specified by Ri to list
* of hosts.
*/
static Stel *
St_add(
St *st,
Ri *ri
)
{
int ind;
if ( st == NULL || ri == NULL ) {
return NULL;
}
/* Serialize access to the St struct */
ldap_pvt_thread_mutex_lock( &(st->st_mutex ));
1998-08-09 08:43:13 +08:00
st->st_nreplicas++;
ind = st->st_nreplicas - 1;
st->st_data = ( Stel ** ) ch_realloc( st->st_data,
( st->st_nreplicas * sizeof( Stel * )));
if ( st->st_data == NULL ) {
ldap_pvt_thread_mutex_unlock( &(st->st_mutex ));
1998-08-09 08:43:13 +08:00
return NULL;
}
1998-08-09 12:52:36 +08:00
st->st_data[ ind ] = ( Stel * ) ch_malloc( sizeof( Stel ) );
1998-08-09 08:43:13 +08:00
if ( st->st_data[ ind ] == NULL ) {
ldap_pvt_thread_mutex_unlock( &(st->st_mutex ));
1998-08-09 08:43:13 +08:00
return NULL;
}
st->st_data[ ind ]->hostname = strdup( ri->ri_hostname );
st->st_data[ ind ]->port = ri->ri_port;
st->st_data[ ind ]->last = 0;
1998-08-09 08:43:13 +08:00
st->st_data[ ind ]->seq = 0;
ldap_pvt_thread_mutex_unlock( &(st->st_mutex ));
1998-08-09 08:43:13 +08:00
return st->st_data[ ind ];
}
/*
* Write the contents of an St to disk.
*/
static int
St_write (
St *st
)
{
int rc;
Stel *stel;
int i;
if ( st == NULL ) {
return -1;
}
ldap_pvt_thread_mutex_lock( &(st->st_mutex ));
1998-08-09 08:43:13 +08:00
if ( st->st_fp == NULL ) {
/* Open file */
if (( rc = acquire_lock( sglob->slurpd_status_file, &(st->st_fp),
&(st->st_lfp))) < 0 ) {
if ( !st->st_err_logged ) {
Debug( LDAP_DEBUG_ANY,
"Error: cannot open status file \"%s\": %s\n",
sglob->slurpd_status_file, sys_errlist[ errno ], 0 );
st->st_err_logged = 1;
ldap_pvt_thread_mutex_unlock( &(st->st_mutex ));
1998-08-09 08:43:13 +08:00
return -1;
}
} else {
st->st_err_logged = 0;
}
}
/* Write data to the file */
truncate( sglob->slurpd_status_file, 0L );
fseek( st->st_fp, 0L, 0 );
for ( i = 0; i < st->st_nreplicas; i++ ) {
stel = st->st_data[ i ];
fprintf( st->st_fp, "%s:%d:%ld:%d\n",
stel->hostname, stel->port,
(long) stel->last, stel->seq );
1998-08-09 08:43:13 +08:00
}
fflush( st->st_fp );
ldap_pvt_thread_mutex_unlock( &(st->st_mutex ));
1998-08-09 08:43:13 +08:00
return 0;
}
/*
* Update the entry for a given host.
*/
static int
St_update(
St *st,
Stel *stel,
Re *re
)
{
if ( stel == NULL || re == NULL ) {
return -1;
}
ldap_pvt_thread_mutex_lock( &(st->st_mutex ));
stel->last = re->re_timestamp;
1998-08-09 08:43:13 +08:00
stel->seq = re->re_seq;
ldap_pvt_thread_mutex_unlock( &(st->st_mutex ));
1998-08-09 08:43:13 +08:00
return 0;
}
/*
* Read status information from disk file.
*/
static int
St_read(
St *st
)
{
FILE *fp;
FILE *lfp;
char buf[ 255 ];
int i;
int rc;
char *hostname, *port, *timestamp, *seq, *p, *t;
int found;
if ( st == NULL ) {
return -1;
}
ldap_pvt_thread_mutex_lock( &(st->st_mutex ));
1998-08-09 08:43:13 +08:00
if ( access( sglob->slurpd_status_file, F_OK ) < 0 ) {
/*
* File doesn't exist, so create it and return.
*/
if (( fp = fopen( sglob->slurpd_status_file, "w" )) == NULL ) {
Debug( LDAP_DEBUG_ANY, "Error: cannot create status file \"%s\"\n",
sglob->slurpd_status_file, 0, 0 );
ldap_pvt_thread_mutex_unlock( &(st->st_mutex ));
1998-08-09 08:43:13 +08:00
return -1;
}
(void) fclose( fp );
ldap_pvt_thread_mutex_unlock( &(st->st_mutex ));
1998-08-09 08:43:13 +08:00
Debug( LDAP_DEBUG_ARGS, "No status file found, defaulting values\n",
0, 0, 0 );
return 0;
}
if (( rc = acquire_lock( sglob->slurpd_status_file, &fp, &lfp)) < 0 ) {
ldap_pvt_thread_mutex_unlock( &(st->st_mutex ));
1998-08-09 08:43:13 +08:00
return 0;
}
while ( fgets( buf, sizeof( buf ), fp ) != NULL ) {
p = buf;
hostname = p;
if (( t = strchr( p, ':' )) == NULL ) {
goto bad;
}
*t++ = '\0';
p = t;
port = p;
if (( t = strchr( p, ':' )) == NULL ) {
goto bad;
}
*t++ = '\0';
p = t;
timestamp = p;
if (( t = strchr( p, ':' )) == NULL ) {
goto bad;
}
*t++ = '\0';
seq = t;
if (( t = strchr( seq, '\n' )) != NULL ) {
*t = '\0';
}
found = 0;
for ( i = 0; i < sglob->st->st_nreplicas; i++ ) {
2005-11-24 09:10:05 +08:00
int p;
1998-08-09 08:43:13 +08:00
if ( !strcmp( hostname, sglob->st->st_data[ i ]->hostname ) &&
2005-11-24 09:10:05 +08:00
lutil_atoi( &p, port ) == 0 && p == sglob->st->st_data[ i ]->port )
{
1998-08-09 08:43:13 +08:00
found = 1;
2005-11-24 09:10:05 +08:00
if ( lutil_atol( &sglob->st->st_data[ i ]->last, timestamp ) != 0
|| lutil_atoi( &sglob->st->st_data[ i ]->seq, seq ) != 0 )
{
found = 0;
}
1998-08-09 08:43:13 +08:00
break;
}
}
if ( found ) {
char tbuf[ 255 ];
sprintf( tbuf, "%s:%s (timestamp %s.%s)", hostname, port,
timestamp, seq );
Debug( LDAP_DEBUG_ARGS,
"Retrieved state information for %s\n", tbuf, 0, 0 );
} else {
Debug( LDAP_DEBUG_ANY,
"Warning: saved state for %s:%s, not a known replica\n",
hostname, port, 0 );
}
}
(void) relinquish_lock( sglob->slurpd_status_file, fp, lfp);
ldap_pvt_thread_mutex_unlock( &(st->st_mutex ));
1998-08-09 08:43:13 +08:00
return 0;
bad:
(void) relinquish_lock( sglob->slurpd_status_file, fp, lfp);
ldap_pvt_thread_mutex_unlock( &(st->st_mutex ));
1998-08-09 08:43:13 +08:00
return -1;
}
/*
* Lock an St struct.
*/
static int
St_lock(
St *st
)
{
return( ldap_pvt_thread_mutex_lock( &st->st_mutex ));
1998-08-09 08:43:13 +08:00
}
/*
* Lock an St struct.
*/
static int
St_unlock(
St *st
)
{
return( ldap_pvt_thread_mutex_unlock( &st->st_mutex ));
1998-08-09 08:43:13 +08:00
}
/*
* Allocate and initialize an St struct.
*/
int
St_init(
St **st
)
{
if ( st == NULL ) {
return -1;
}
(*st) = (St *) malloc( sizeof( St ));
if ( *st == NULL ) {
return -1;
}
ldap_pvt_thread_mutex_init( &((*st)->st_mutex) );
1998-08-09 08:43:13 +08:00
(*st)->st_data = NULL;
(*st)->st_fp = NULL;
(*st)->st_lfp = NULL;
(*st)->st_nreplicas = 0;
(*st)->st_err_logged = 0;
(*st)->st_update = St_update;
(*st)->st_add = St_add;
(*st)->st_write = St_write;
(*st)->st_read = St_read;
(*st)->st_lock = St_lock;
(*st)->st_unlock = St_unlock;
return 0;
}