#include <sendmail.h>
SM_RCSID("@(#)$Id: stats.c,v 8.57 2006/08/15 23:24:58 ca Exp $")
#include <sendmail/mailstats.h>
static struct statistics Stat;
static bool GotStats = false;
#define ONE_K 1000
#define KBYTES(x) (((x) + (ONE_K - 1)) / ONE_K)
void
markstats(e, to, type)
register ENVELOPE *e;
register ADDRESS *to;
int type;
{
switch (type)
{
case STATS_QUARANTINE:
if (e->e_from.q_mailer != NULL)
Stat.stat_nq[e->e_from.q_mailer->m_mno]++;
break;
case STATS_REJECT:
if (e->e_from.q_mailer != NULL)
{
if (bitset(EF_DISCARD, e->e_flags))
Stat.stat_nd[e->e_from.q_mailer->m_mno]++;
else
Stat.stat_nr[e->e_from.q_mailer->m_mno]++;
}
Stat.stat_cr++;
break;
case STATS_CONNECT:
if (to == NULL)
Stat.stat_cf++;
else
Stat.stat_ct++;
break;
case STATS_NORMAL:
if (to == NULL)
{
if (e->e_from.q_mailer != NULL)
{
Stat.stat_nf[e->e_from.q_mailer->m_mno]++;
Stat.stat_bf[e->e_from.q_mailer->m_mno] +=
KBYTES(e->e_msgsize);
}
}
else
{
Stat.stat_nt[to->q_mailer->m_mno]++;
Stat.stat_bt[to->q_mailer->m_mno] += KBYTES(e->e_msgsize);
}
break;
default:
return;
}
GotStats = true;
}
void
clearstats()
{
memset(&Stat, '\0', sizeof(Stat));
GotStats = false;
}
void
poststats(sfile)
char *sfile;
{
int fd;
static bool entered = false;
long sff = SFF_REGONLY|SFF_OPENASROOT;
struct statistics stats;
extern off_t lseek();
if (sfile == NULL || *sfile == '\0' || !GotStats || entered)
return;
entered = true;
(void) time(&Stat.stat_itime);
Stat.stat_size = sizeof(Stat);
Stat.stat_magic = STAT_MAGIC;
Stat.stat_version = STAT_VERSION;
if (!bitnset(DBS_WRITESTATSTOSYMLINK, DontBlameSendmail))
sff |= SFF_NOSLINK;
if (!bitnset(DBS_WRITESTATSTOHARDLINK, DontBlameSendmail))
sff |= SFF_NOHLINK;
fd = safeopen(sfile, O_RDWR, 0600, sff);
if (fd < 0)
{
if (LogLevel > 12)
sm_syslog(LOG_INFO, NOQID, "poststats: %s: %s",
sfile, sm_errstring(errno));
errno = 0;
entered = false;
return;
}
if (read(fd, (char *) &stats, sizeof(stats)) == sizeof(stats) &&
stats.stat_size == sizeof(stats) &&
stats.stat_magic == Stat.stat_magic &&
stats.stat_version == Stat.stat_version)
{
register int i;
for (i = 0; i < MAXMAILERS; i++)
{
stats.stat_nf[i] += Stat.stat_nf[i];
stats.stat_bf[i] += Stat.stat_bf[i];
stats.stat_nt[i] += Stat.stat_nt[i];
stats.stat_bt[i] += Stat.stat_bt[i];
stats.stat_nr[i] += Stat.stat_nr[i];
stats.stat_nd[i] += Stat.stat_nd[i];
stats.stat_nq[i] += Stat.stat_nq[i];
}
stats.stat_cr += Stat.stat_cr;
stats.stat_ct += Stat.stat_ct;
stats.stat_cf += Stat.stat_cf;
}
else
memmove((char *) &stats, (char *) &Stat, sizeof(stats));
(void) lseek(fd, (off_t) 0, 0);
(void) write(fd, (char *) &stats, sizeof(stats));
(void) close(fd);
clearstats();
entered = false;
}