source: trunk/athena/bin/desync/src/desync.c @ 25961

Revision 25961, 5.4 KB checked in by vasilvv, 11 years ago (diff)
In desync: * Use automake * Bump upstream version * Switch to debhelper7 * Switch to source format 3.0 (quilt)
RevLine 
[9727]1/* Copyright 1996, 1997 by the Massachusetts Institute of Technology.
2 *
3 * Permission to use, copy, modify, and distribute this
4 * software and its documentation for any purpose and without
5 * fee is hereby granted, provided that the above copyright
6 * notice appear in all copies and that both that copyright
7 * notice and this permission notice appear in supporting
8 * documentation, and that the name of M.I.T. not be used in
9 * advertising or publicity pertaining to distribution of the
10 * software without specific, written prior permission.
11 * M.I.T. makes no representations about the suitability of
12 * this software for any purpose.  It is provided "as is"
13 * without express or implied warranty.
14 */
15
[15156]16static const char rcsid[] = "$Id: desync.c,v 1.9 2000-09-30 21:08:32 rbasch Exp $";
[9727]17
[7973]18/*
19 * desync - desynchronize cron jobs on networks
20 *
21 * This program is a tool which sleeps an ip-address dependent period
[9727]22 * of time in order to skew over the course of a set time cron jobs
23 * that would otherwise be synchronized. It should reside on local disk
24 * so as not to cause a fileserver load at its own invocation.
[7973]25 */
26
27#include <sys/types.h>
28#include <sys/stat.h>
29#include <fcntl.h>
30#include <ctype.h>
31#include <string.h>
32#include <stdio.h>
[9727]33#include <stdlib.h>
[7973]34#include <errno.h>
[13769]35#include <unistd.h>
[13780]36#include <time.h>
[25696]37#include <limits.h>
[7973]38
[10934]39extern int optind;
40extern char *optarg;
41
[9727]42static char *progname;
[7973]43
[15156]44static unsigned long get_hash(const char *str);
[9727]45
[15156]46static void usage(void);
47
[9727]48int main(int argc, char **argv)
[7973]49{
[25696]50  const char *timefile = NULL, *hostname = NULL, *crontabhour = NULL;
[15156]51  char buf[128];
52  int range, interval, c, noop = 0;
53  unsigned long tval;
54  time_t timenow;
[9727]55  FILE *fp;
[7973]56
[9727]57  /* Save the program name. */
58  progname = argv[0];
[7973]59
[9727]60  /* Parse command-line flags. */
[25696]61  while ((c = getopt(argc, argv, "h:nc:t:")) != -1)
[7973]62    {
[9727]63      switch (c) {
[15156]64      case 'h':
65        hostname = optarg;
66        break;
67      case 'n':
68        noop = 1;
69        break;
[25696]70      case 'c':
71        crontabhour = optarg;
72        break;
[9727]73      case 't':
74        timefile = optarg;
75        break;
76      default:
[15156]77        usage();
[9727]78        return 2;
79      }
[7973]80    }
81
[25696]82  if (crontabhour && ((timefile != NULL) || (noop == 1))) {
83    usage();
84    return 2;
85  }
[9727]86  /* Get the time interval from the remaining argument, if there is one. */
87  argc -= optind;
88  argv += optind;
[25696]89  if ((argc > 1) && ! crontabhour) {
90    usage();
91    return 2;
92  }
93  if (argc >= 1) {
94    range = atoi(argv[0]);
95    argc -= 1;
96    argv += 1;
97  } else {
[25710]98    range = crontabhour == NULL ? 3600 : 60;
[25696]99  }
[25710]100  if ((range == 0) ||
101      (crontabhour && (range >= 1440)))
[15156]102    {
103      fprintf(stderr, "%s: Invalid range value\n", progname);
104      usage();
105      return 2;
106    }
[7973]107
[15156]108  /* Get a random number in the given range as the interval.  Seed the
109   * random number generator with a hash of the current host name, or
110   * the name given via the -h option.
111   */
112  if (hostname == NULL)
113    {
114      if (gethostname(buf, sizeof(buf)) != 0)
115        {
116          fprintf(stderr, "%s: Unable to obtain hostname: %s\n",
117                  progname, strerror(errno));
118          return 2;
119        }
120      buf[sizeof(buf) - 1] = '\0';
121      hostname = buf;
122    }
123  srand(get_hash(hostname));
[9727]124  interval = rand() % range;
[7973]125
[9727]126  if (timefile)
[7973]127    {
[15156]128      time(&timenow);
[9727]129      fp = fopen(timefile, "r");
130      if (fp)
[7973]131        {
[9727]132          if (fscanf(fp, "%lu", &tval) != 1)
[7973]133            {
[9727]134              fprintf(stderr, "%s: Invalid time file %s\n", progname,
135                      timefile);
136              return 2;
[7973]137            }
[9727]138          fclose(fp);
[15156]139          if (timenow >= tval)
[9727]140            {
[15156]141              if (noop)
142                puts("0");
143              else
144                unlink(timefile);
[9727]145              return 0;
146            }
147          else
[15156]148            {
149              if (noop)
150                printf("%lu\n", (unsigned long) (tval - timenow));
151              return 1;
152            }
[7973]153        }
[15156]154      else if (noop)
155        {
156          fprintf(stderr, "%s: Warning: Cannot open %s (%s)\n", progname,
157                  timefile, strerror(errno));
158          printf("%lu\n", (unsigned long) (interval));
159          return (interval == 0 ? 0 : 1);
160        }
[9727]161      else
162        {
163          if (interval == 0)
164            return 0;
165          fp = fopen(timefile, "w");
166          if (fp == NULL)
167            {
168              fprintf(stderr, "%s: Couldn't open %s for writing: %s\n",
169                      progname, timefile, strerror(errno));
170              return 2;
171            }
[15156]172          fprintf(fp, "%lu\n", (unsigned long)(timenow + interval));
[9727]173          fclose(fp);
174          return 1;
175        }
[7973]176    }
[25696]177  else if (crontabhour)
178    {
179      char *endptr;
180      int mins, hours = 0;
181      int j;
182      errno = 0;
183      hours = strtol(crontabhour, &endptr, 10);
184      if ((errno == ERANGE && (hours == LONG_MAX || hours == LONG_MIN))
185          || (errno != 0 && hours == 0)
186          || (endptr == crontabhour))
187        {
188          fprintf(stderr, "%s: Could not convert %s to integer\n", progname,
189                  crontabhour);
190        return 1;
191      }
192      if ((hours > 23) || (hours < 0 ))
193        {
194          fprintf(stderr,
195                  "%s: in crontab mode, hours must be between 0 and 23\n",
196                  progname);
197        return 1;
198      }
199      mins = interval % 60;
200      hours = (hours + (interval / 60)) % 24;
201      printf("%d %d * * *", mins, hours);
202      for (j = 0; j < argc; j++)
203        printf(" %s", argv[j]);
204      printf("\n");
205    }
[15156]206  else if (noop)
207    printf("%lu\n", (unsigned long) interval);
[9727]208  else
209    sleep(interval);
[7973]210
[9727]211  return 0;
[7973]212}
213
[15156]214static unsigned long get_hash(const char *str)
[7973]215{
[15156]216  const char *p;
[15092]217  unsigned long g, hashval = 0;
[7973]218
[15156]219  for (p = str; *p; p++)
[9727]220    {
[15092]221      hashval = (hashval << 4) + *p;
222      g = hashval & 0xf0000000;
223      if (g != 0) {
224        hashval ^= g >> 24;
225        hashval ^= g;
226      }
[7973]227    }
[9727]228
[15092]229  if (hashval == 0)
230    hashval = 1;
231
232  return hashval;
[7973]233}
[15156]234
235static void usage()
236{
237  fprintf(stderr,
[25710]238          "Usage: %s [-h name] [-n] [-t timefile] [range]\n       %s -c hour [range [crontab arguments]]\n",
[25696]239          progname, progname);
[15156]240}
Note: See TracBrowser for help on using the repository browser.