1 /*
   2  * CDDL HEADER START
   3  *
   4  * The contents of this file are subject to the terms of the
   5  * Common Development and Distribution License (the "License").
   6  * You may not use this file except in compliance with the License.
   7  *
   8  * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
   9  * or http://www.opensolaris.org/os/licensing.
  10  * See the License for the specific language governing permissions
  11  * and limitations under the License.
  12  *
  13  * When distributing Covered Code, include this CDDL HEADER in each
  14  * file and include the License file at usr/src/OPENSOLARIS.LICENSE.
  15  * If applicable, add the following below this CDDL HEADER, with the
  16  * fields enclosed by brackets "[]" replaced with your own identifying
  17  * information: Portions Copyright [yyyy] [name of copyright owner]
  18  *
  19  * CDDL HEADER END
  20  */
  21 /*
  22  * Copyright (c) 2001, 2010, Oracle and/or its affiliates. All rights reserved.
  23  *
  24  * logadm/main.c -- main routines for logadm
  25  *
  26  * this program is 90% argument processing, 10% actions...
  27  */
  28 
  29 #include <stdio.h>
  30 #include <stdlib.h>
  31 #include <unistd.h>
  32 #include <strings.h>
  33 #include <libintl.h>
  34 #include <locale.h>
  35 #include <fcntl.h>
  36 #include <sys/types.h>
  37 #include <sys/stat.h>
  38 #include <sys/wait.h>
  39 #include <sys/filio.h>
  40 #include <time.h>
  41 #include <utime.h>
  42 #include "err.h"
  43 #include "lut.h"
  44 #include "fn.h"
  45 #include "opts.h"
  46 #include "conf.h"
  47 #include "glob.h"
  48 #include "kw.h"
  49 
  50 /* forward declarations for functions in this file */
  51 static void usage(const char *msg);
  52 static void commajoin(const char *lhs, void *rhs, void *arg);
  53 static void doaftercmd(const char *lhs, void *rhs, void *arg);
  54 static void dologname(struct fn *fnp, struct opts *clopts);
  55 static boolean_t rotatelog(struct fn *fnp, struct opts *opts);
  56 static void rotateto(struct fn *fnp, struct opts *opts, int n,
  57     struct fn *recentlog, boolean_t isgz);
  58 static void do_delayed_gzip(const char *lhs, void *rhs, void *arg);
  59 static void expirefiles(struct fn *fnp, struct opts *opts);
  60 static void dorm(struct opts *opts, const char *msg, struct fn *fnp);
  61 static void docmd(struct opts *opts, const char *msg, const char *cmd,
  62     const char *arg1, const char *arg2, const char *arg3);
  63 static void docopytruncate(struct opts *opts, const char *file,
  64     const char *file_copy);
  65 
  66 /* our configuration file, unless otherwise specified by -f */
  67 static char *Default_conffile = "/etc/logadm.conf";
  68 /* our timestamps file, unless otherwise specified by -F */
  69 static char *Default_timestamps = "/var/logadm/timestamps";
  70 
  71 /* default pathnames to the commands we invoke */
  72 static char *Sh = "/bin/sh";
  73 static char *Mv = "/bin/mv";
  74 static char *Rm = "/bin/rm";
  75 static char *Touch = "/bin/touch";
  76 static char *Chmod = "/bin/chmod";
  77 static char *Chown = "/bin/chown";
  78 static char *Gzip = "/bin/gzip";
  79 static char *Mkdir = "/bin/mkdir";
  80 
  81 /* return from time(0), gathered early on to avoid slewed timestamps */
  82 time_t Now;
  83 
  84 /* list of before commands that have been executed */
  85 static struct lut *Beforecmds;
  86 
  87 /* list of after commands to execute before exiting */
  88 static struct lut *Aftercmds;
  89 
  90 /* list of conffile entry names that are considered "done" */
  91 static struct lut *Donenames;
  92 
  93 /* A list of names of files to be gzipped */
  94 static struct lut *Gzipnames = NULL;
  95 
  96 /*
  97  * only the "FfhnVv" options are allowed in the first form of this command,
  98  * so this defines the list of options that are an error in they appear
  99  * in the first form.  In other words, it is not allowed to run logadm
 100  * with any of these options unless at least one logname is also provided.
 101  */
 102 #define OPTIONS_NOT_FIRST_FORM  "eNrwpPsabcglmoRtzACEST"
 103 
 104 /* text that we spew with the -h flag */
 105 #define HELP1 \
 106 "Usage: logadm [options]\n"\
 107 "       (processes all entries in /etc/logadm.conf or conffile given by -f)\n"\
 108 "   or: logadm [options] logname...\n"\
 109 "       (processes the given lognames)\n"\
 110 "\n"\
 111 "General options:\n"\
 112 "        -e mailaddr     mail errors to given address\n"\
 113 "        -F timestamps   use timestamps instead of /var/logadm/timestamps\n"\
 114 "        -f conffile     use conffile instead of /etc/logadm.conf\n"\
 115 "        -h              display help\n"\
 116 "        -N              not an error if log file nonexistent\n"\
 117 "        -n              show actions, don't perform them\n"\
 118 "        -r              remove logname entry from conffile\n"\
 119 "        -V              ensure conffile entries exist, correct\n"\
 120 "        -v              print info about actions happening\n"\
 121 "        -w entryname    write entry to config file\n"\
 122 "\n"\
 123 "Options which control when a logfile is rotated:\n"\
 124 "(default is: -s1b -p1w if no -s or -p)\n"\
 125 "        -p period       only rotate if period passed since last rotate\n"\
 126 "        -P timestamp    used to store rotation date in conffile\n"\
 127 "        -s size         only rotate if given size or greater\n"\
 128 "\n"
 129 #define HELP2 \
 130 "Options which control how a logfile is rotated:\n"\
 131 "(default is: -t '$file.$n', owner/group/mode taken from log file)\n"\
 132 "        -a cmd          execute cmd after taking actions\n"\
 133 "        -b cmd          execute cmd before taking actions\n"\
 134 "        -c              copy & truncate logfile, don't rename\n"\
 135 "        -g group        new empty log file group\n"\
 136 "        -l              rotate log file with local time rather than UTC\n"\
 137 "        -m mode         new empty log file mode\n"\
 138 "        -M cmd          execute cmd to rotate the log file\n"\
 139 "        -o owner        new empty log file owner\n"\
 140 "        -R cmd          run cmd on file after rotate\n"\
 141 "        -t template     template for naming old logs\n"\
 142 "        -z count        gzip old logs except most recent count\n"\
 143 "\n"\
 144 "Options which control the expiration of old logfiles:\n"\
 145 "(default is: -C10 if no -A, -C, or -S)\n"\
 146 "        -A age          expire logs older than age\n"\
 147 "        -C count        expire old logs until count remain\n"\
 148 "        -E cmd          run cmd on file to expire\n"\
 149 "        -S size         expire until space used is below size \n"\
 150 "        -T pattern      pattern for finding old logs\n"
 151 
 152 /*
 153  * main -- where it all begins
 154  */
 155 /*ARGSUSED*/
 156 int
 157 main(int argc, char *argv[])
 158 {
 159         struct opts *clopts;            /* from parsing command line */
 160         const char *conffile;           /* our configuration file */
 161         const char *timestamps;         /* our timestamps file */
 162         struct fn_list *lognames;       /* list of lognames we're processing */
 163         struct fn *fnp;
 164         char *val;
 165         char *buf;
 166         int status;
 167 
 168         (void) setlocale(LC_ALL, "");
 169 
 170 #if !defined(TEXT_DOMAIN)
 171 #define TEXT_DOMAIN "SYS_TEST"  /* only used if Makefiles don't define it */
 172 #endif
 173 
 174         (void) textdomain(TEXT_DOMAIN);
 175 
 176         /* we only print times into the timestamps file, so make them uniform */
 177         (void) setlocale(LC_TIME, "C");
 178 
 179         /* give our name to error routines & skip it for arg parsing */
 180         err_init(*argv++);
 181         (void) setlinebuf(stdout);
 182 
 183         if (putenv("PATH=/bin"))
 184                 err(EF_SYS, "putenv PATH");
 185         if (putenv("TZ=UTC"))
 186                 err(EF_SYS, "putenv TZ");
 187         tzset();
 188 
 189         (void) umask(0);
 190 
 191         Now = time(0);
 192 
 193         /* check for (undocumented) debugging environment variables */
 194         if (val = getenv("_LOGADM_DEFAULT_CONFFILE"))
 195                 Default_conffile = val;
 196         if (val = getenv("_LOGADM_DEFAULT_TIMESTAMPS"))
 197                 Default_timestamps = val;
 198         if (val = getenv("_LOGADM_DEBUG"))
 199                 Debug = atoi(val);
 200         if (val = getenv("_LOGADM_SH"))
 201                 Sh = val;
 202         if (val = getenv("_LOGADM_MV"))
 203                 Mv = val;
 204         if (val = getenv("_LOGADM_RM"))
 205                 Rm = val;
 206         if (val = getenv("_LOGADM_TOUCH"))
 207                 Touch = val;
 208         if (val = getenv("_LOGADM_CHMOD"))
 209                 Chmod = val;
 210         if (val = getenv("_LOGADM_CHOWN"))
 211                 Chown = val;
 212         if (val = getenv("_LOGADM_GZIP"))
 213                 Gzip = val;
 214         if (val = getenv("_LOGADM_MKDIR"))
 215                 Mkdir = val;
 216 
 217         opts_init(Opttable, Opttable_cnt);
 218 
 219         /* parse command line arguments */
 220         if (SETJMP)
 221                 usage("bailing out due to command line errors");
 222         else
 223                 clopts = opts_parse(NULL, argv, OPTF_CLI);
 224 
 225         if (Debug) {
 226                 (void) fprintf(stderr, "command line opts:");
 227                 opts_print(clopts, stderr, NULL);
 228                 (void) fprintf(stderr, "\n");
 229         }
 230 
 231         /*
 232          * There are many moods of logadm:
 233          *
 234          *      1. "-h" for help was given.  We spew a canned help
 235          *         message and exit, regardless of any other options given.
 236          *
 237          *      2. "-r" or "-w" asking us to write to the conffile.  Lots
 238          *         of argument checking, then we make the change to conffile
 239          *         and exit.  (-r processing actually happens in dologname().)
 240          *
 241          *      3. "-V" to search/verify the conffile was given.  We do
 242          *         the appropriate run through the conffile and exit.
 243          *         (-V processing actually happens in dologname().)
 244          *
 245          *      4. No lognames were given, so we're being asked to go through
 246          *         every entry in conffile.  We verify that only the options
 247          *         that make sense for this form of the command are present
 248          *         and fall into the main processing loop below.
 249          *
 250          *      5. lognames were given, so we fall into the main processing
 251          *         loop below to work our way through them.
 252          *
 253          * The last two cases are where the option processing gets more
 254          * complex.  Each time around the main processing loop, we're
 255          * in one of these cases:
 256          *
 257          *      A. No cmdargs were found (we're in case 4), the entry
 258          *         in conffile supplies no log file names, so the entry
 259          *         name itself is the logfile name (or names, if it globs
 260          *         to multiple file names).
 261          *
 262          *      B. No cmdargs were found (we're in case 4), the entry
 263          *         in conffile gives log file names that we then loop
 264          *         through and rotate/expire.  In this case, the entry
 265          *         name is specifically NOT one of the log file names.
 266          *
 267          *      C. We're going through the cmdargs (we're in case 5),
 268          *         the entry in conffile either doesn't exist or it exists
 269          *         but supplies no log file names, so the cmdarg itself
 270          *         is the log file name.
 271          *
 272          *      D. We're going through the cmdargs (we're in case 5),
 273          *         a matching entry in conffile supplies log file names
 274          *         that we then loop through and rotate/expire.  In this
 275          *         case the entry name is specifically NOT one of the log
 276          *         file names.
 277          *
 278          * As we're doing all this, any options given on the command line
 279          * override any found in the conffile, and we apply the defaults
 280          * for rotation conditions and expiration conditions, etc. at the
 281          * last opportunity, when we're sure they haven't been overridden
 282          * by an option somewhere along the way.
 283          *
 284          */
 285 
 286         /* help option overrides anything else */
 287         if (opts_count(clopts, "h")) {
 288                 (void) fputs(HELP1, stderr);
 289                 (void) fputs(HELP2, stderr);
 290                 err_done(0);
 291                 /*NOTREACHED*/
 292         }
 293 
 294         /* detect illegal option combinations */
 295         if (opts_count(clopts, "rwV") > 1)
 296                 usage("Only one of -r, -w, or -V may be used at a time.");
 297         if (opts_count(clopts, "cM") > 1)
 298                 usage("Only one of -c or -M may be used at a time.");
 299 
 300         /* arrange for error output to be mailed if clopts includes -e */
 301         if (opts_count(clopts, "e"))
 302                 err_mailto(opts_optarg(clopts, "e"));
 303 
 304         /* this implements the default conffile and timestamps */
 305         if ((conffile = opts_optarg(clopts, "f")) == NULL)
 306                 conffile = Default_conffile;
 307         if ((timestamps = opts_optarg(clopts, "F")) == NULL)
 308                 timestamps = Default_timestamps;
 309         if (opts_count(clopts, "v"))
 310                 (void) out("# loading %s\n", conffile);
 311         status = conf_open(conffile, timestamps, clopts);
 312         if (!status && opts_count(clopts, "V"))
 313                 err_done(0);
 314 
 315         /* handle conffile write option */
 316         if (opts_count(clopts, "w")) {
 317                 if (Debug)
 318                         (void) fprintf(stderr,
 319                             "main: add/replace conffile entry: <%s>\n",
 320                             opts_optarg(clopts, "w"));
 321                 conf_replace(opts_optarg(clopts, "w"), clopts);
 322                 conf_close(clopts);
 323                 err_done(0);
 324                 /*NOTREACHED*/
 325         }
 326 
 327         /*
 328          * lognames is either a list supplied on the command line,
 329          * or every entry in the conffile if none were supplied.
 330          */
 331         lognames = opts_cmdargs(clopts);
 332         if (fn_list_empty(lognames)) {
 333                 /*
 334                  * being asked to do all entries in conffile
 335                  *
 336                  * check to see if any options were given that only
 337                  * make sense when lognames are given specifically
 338                  * on the command line.
 339                  */
 340                 if (opts_count(clopts, OPTIONS_NOT_FIRST_FORM))
 341                         usage("some options require logname argument");
 342                 if (Debug)
 343                         (void) fprintf(stderr,
 344                             "main: run all entries in conffile\n");
 345                 lognames = conf_entries();
 346         }
 347 
 348         /* foreach logname... */
 349         fn_list_rewind(lognames);
 350         while ((fnp = fn_list_next(lognames)) != NULL) {
 351                 buf = fn_s(fnp);
 352                 if (buf != NULL && lut_lookup(Donenames, buf) != NULL) {
 353                         if (Debug)
 354                                 (void) fprintf(stderr,
 355                                     "main: logname already done: <%s>\n",
 356                                     buf);
 357                         continue;
 358                 }
 359                 if (buf != NULL && SETJMP)
 360                         err(EF_FILE, "bailing out on logname \"%s\" "
 361                             "due to errors", buf);
 362                 else
 363                         dologname(fnp, clopts);
 364         }
 365 
 366         /* execute any after commands */
 367         lut_walk(Aftercmds, doaftercmd, clopts);
 368 
 369         /* execute any gzip commands */
 370         lut_walk(Gzipnames, do_delayed_gzip, clopts);
 371 
 372         /* write out any conffile changes */
 373         conf_close(clopts);
 374 
 375         err_done(0);
 376         /*NOTREACHED*/
 377         return (0);     /* for lint's little mind */
 378 }
 379 
 380 /* spew a message, then a usage message, then exit */
 381 static void
 382 usage(const char *msg)
 383 {
 384         if (msg)
 385                 err(0, "%s\nUse \"logadm -h\" for help.", msg);
 386         else
 387                 err(EF_RAW, "Use \"logadm -h\" for help.\n");
 388 }
 389 
 390 /* helper function used by doaftercmd() to join mail addrs with commas */
 391 /*ARGSUSED1*/
 392 static void
 393 commajoin(const char *lhs, void *rhs, void *arg)
 394 {
 395         struct fn *fnp = (struct fn *)arg;
 396         char *buf;
 397 
 398         buf = fn_s(fnp);
 399         if (buf != NULL && *buf)
 400                 fn_putc(fnp, ',');
 401         fn_puts(fnp, lhs);
 402 }
 403 
 404 /* helper function used by main() to run "after" commands */
 405 static void
 406 doaftercmd(const char *lhs, void *rhs, void *arg)
 407 {
 408         struct opts *opts = (struct opts *)arg;
 409         struct lut *addrs = (struct lut *)rhs;
 410 
 411         if (addrs) {
 412                 struct fn *fnp = fn_new(NULL);
 413 
 414                 /*
 415                  * addrs contains list of email addrs that should get
 416                  * the error output when this after command is executed.
 417                  */
 418                 lut_walk(addrs, commajoin, fnp);
 419                 err_mailto(fn_s(fnp));
 420         }
 421 
 422         docmd(opts, "-a cmd", Sh, "-c", lhs, NULL);
 423 }
 424 
 425 /* perform delayed gzip */
 426 
 427 static void
 428 do_delayed_gzip(const char *lhs, void *rhs, void *arg)
 429 {
 430         struct opts *opts = (struct opts *)arg;
 431 
 432         if (rhs == NULL) {
 433                 if (Debug) {
 434                         (void) fprintf(stderr, "do_delayed_gzip: not gzipping "
 435                             "expired file <%s>\n", lhs);
 436                 }
 437                 return;
 438         }
 439         docmd(opts, "compress old log (-z flag)", Gzip, "-f", lhs, NULL);
 440 }
 441 
 442 
 443 /* main logname processing */
 444 static void
 445 dologname(struct fn *fnp, struct opts *clopts)
 446 {
 447         const char *logname = fn_s(fnp);
 448         struct opts *cfopts;
 449         struct opts *allopts;
 450         struct fn_list *logfiles;
 451         struct fn_list *globbedfiles;
 452         struct fn *nextfnp;
 453 
 454         /* look up options set by config file */
 455         cfopts = conf_opts(logname);
 456 
 457         if (opts_count(clopts, "v"))
 458                 (void) out("# processing logname: %s\n", logname);
 459 
 460         if (Debug) {
 461                 if (logname != NULL)
 462                         (void) fprintf(stderr, "dologname: logname <%s>\n",
 463                             logname);
 464                 (void) fprintf(stderr, "conffile opts:");
 465                 opts_print(cfopts, stderr, NULL);
 466                 (void) fprintf(stderr, "\n");
 467         }
 468 
 469         /* handle conffile lookup option */
 470         if (opts_count(clopts, "V")) {
 471                 /* lookup an entry in conffile */
 472                 if (Debug)
 473                         (void) fprintf(stderr,
 474                             "dologname: lookup conffile entry\n");
 475                 if (conf_lookup(logname)) {
 476                         opts_printword(logname, stdout);
 477                         opts_print(cfopts, stdout, NULL);
 478                         (void) out("\n");
 479                 } else
 480                         err_exitcode(1);
 481                 return;
 482         }
 483 
 484         /* handle conffile removal option */
 485         if (opts_count(clopts, "r")) {
 486                 if (Debug)
 487                         (void) fprintf(stderr,
 488                             "dologname: remove conffile entry\n");
 489                 if (conf_lookup(logname))
 490                         conf_replace(logname, NULL);
 491                 else
 492                         err_exitcode(1);
 493                 return;
 494         }
 495 
 496         /* generate combined options */
 497         allopts = opts_merge(cfopts, clopts);
 498 
 499         /* arrange for error output to be mailed if allopts includes -e */
 500         if (opts_count(allopts, "e"))
 501                 err_mailto(opts_optarg(allopts, "e"));
 502         else
 503                 err_mailto(NULL);
 504 
 505         /* this implements the default rotation rules */
 506         if (opts_count(allopts, "sp") == 0) {
 507                 if (opts_count(clopts, "v"))
 508                         (void) out(
 509                             "#     using default rotate rules: -s1b -p1w\n");
 510                 (void) opts_set(allopts, "s", "1b");
 511                 (void) opts_set(allopts, "p", "1w");
 512         }
 513 
 514         /* this implements the default expiration rules */
 515         if (opts_count(allopts, "ACS") == 0) {
 516                 if (opts_count(clopts, "v"))
 517                         (void) out("#     using default expire rule: -C10\n");
 518                 (void) opts_set(allopts, "C", "10");
 519         }
 520 
 521         /* this implements the default template */
 522         if (opts_count(allopts, "t") == 0) {
 523                 if (opts_count(clopts, "v"))
 524                         (void) out("#     using default template: $file.$n\n");
 525                 (void) opts_set(allopts, "t", "$file.$n");
 526         }
 527 
 528         if (Debug) {
 529                 (void) fprintf(stderr, "merged opts:");
 530                 opts_print(allopts, stderr, NULL);
 531                 (void) fprintf(stderr, "\n");
 532         }
 533 
 534         /*
 535          * if the conffile entry supplied log file names, then
 536          * logname is NOT one of the log file names (it was just
 537          * the entry name in conffile).
 538          */
 539         logfiles = opts_cmdargs(cfopts);
 540         if (Debug) {
 541                 char *buf;
 542                 (void) fprintf(stderr, "dologname: logfiles from cfopts:\n");
 543                 fn_list_rewind(logfiles);
 544                 while ((nextfnp = fn_list_next(logfiles)) != NULL)
 545                         buf = fn_s(nextfnp);
 546                         if (buf != NULL)
 547                                 (void) fprintf(stderr, "    <%s>\n", buf);
 548         }
 549         if (fn_list_empty(logfiles))
 550                 globbedfiles = glob_glob(fnp);
 551         else
 552                 globbedfiles = glob_glob_list(logfiles);
 553 
 554         /* go through the list produced by glob expansion */
 555         fn_list_rewind(globbedfiles);
 556         while ((nextfnp = fn_list_next(globbedfiles)) != NULL)
 557                 if (rotatelog(nextfnp, allopts))
 558                         expirefiles(nextfnp, allopts);
 559 
 560         fn_list_free(globbedfiles);
 561         opts_free(allopts);
 562 }
 563 
 564 
 565 /* absurdly long buffer lengths for holding user/group/mode strings */
 566 #define TIMESTRMAX      100
 567 #define MAXATTR         100
 568 
 569 /* rotate a log file if necessary, returns true if ok to go on to expire step */
 570 static boolean_t
 571 rotatelog(struct fn *fnp, struct opts *opts)
 572 {
 573         char *fname = fn_s(fnp);
 574         struct stat stbuf;
 575         char nowstr[TIMESTRMAX];
 576         struct fn *recentlog = fn_new(NULL);    /* for -R cmd */
 577         char ownerbuf[MAXATTR];
 578         char groupbuf[MAXATTR];
 579         char modebuf[MAXATTR];
 580         const char *owner;
 581         const char *group;
 582         const char *mode;
 583 
 584         if (Debug && fname != NULL)
 585                 (void) fprintf(stderr, "rotatelog: fname <%s>\n", fname);
 586 
 587         if (opts_count(opts, "p") && opts_optarg_int(opts, "p") == OPTP_NEVER)
 588                 return (B_TRUE);        /* "-p never" forced no rotate */
 589 
 590         /* prepare the keywords */
 591         kw_init(fnp, NULL);
 592         if (Debug > 1) {
 593                 (void) fprintf(stderr, "rotatelog keywords:\n");
 594                 kw_print(stderr);
 595         }
 596 
 597         if (lstat(fname, &stbuf) < 0) {
 598                 if (opts_count(opts, "N"))
 599                         return (1);
 600                 err(EF_WARN|EF_SYS, "%s", fname);
 601                 return (B_FALSE);
 602         }
 603 
 604         if ((stbuf.st_mode & S_IFMT) == S_IFLNK) {
 605                 err(EF_WARN, "%s is a symlink", fname);
 606                 return (B_FALSE);
 607         }
 608 
 609         if ((stbuf.st_mode & S_IFMT) != S_IFREG) {
 610                 err(EF_WARN, "%s is not a regular file", fname);
 611                 return (B_FALSE);
 612         }
 613 
 614         /* even if size condition is not met, this entry is "done" */
 615         if (opts_count(opts, "s") &&
 616             stbuf.st_size < opts_optarg_int(opts, "s")) {
 617                 Donenames = lut_add(Donenames, fname, "1");
 618                 return (B_TRUE);
 619         }
 620 
 621         /* see if age condition is present, and return if not met */
 622         if (opts_count(opts, "p")) {
 623                 off_t when = opts_optarg_int(opts, "p");
 624                 struct opts *cfopts;
 625 
 626                 /* unless rotate forced by "-p now", see if period has passed */
 627                 if (when != OPTP_NOW) {
 628                         /*
 629                          * "when" holds the number of seconds that must have
 630                          * passed since the last time this log was rotated.
 631                          * of course, running logadm can take a little time
 632                          * (typically a second or two, but longer if the
 633                          * conffile has lots of stuff in it) and that amount
 634                          * of time is variable, depending on system load, etc.
 635                          * so we want to allow a little "slop" in the value of
 636                          * "when".  this way, if a log should be rotated every
 637                          * week, and the number of seconds passed is really a
 638                          * few seconds short of a week, we'll go ahead and
 639                          * rotate the log as expected.
 640                          *
 641                          */
 642                         if (when >= 60 * 60)
 643                                 when -= 59;
 644 
 645                         /*
 646                          * last rotation is recorded as argument to -P,
 647                          * but if logname isn't the same as log file name
 648                          * then the timestamp would be recorded on a
 649                          * separate line in the conf file.  so if we
 650                          * haven't seen a -P already, we check to see if
 651                          * it is part of a specific entry for the log
 652                          * file name.  this handles the case where the
 653                          * logname is "apache", it supplies a log file
 654                          * name like "/var/apache2/2.2/logs/[a-z]*_log",
 655                          * which expands to multiple file names.  if one of
 656                          * the file names is "/var/apache2/2.2/logs/access_log"
 657                          * then the -P will be attached to a line with that
 658                          * logname in the conf file.
 659                          */
 660                         if (opts_count(opts, "P")) {
 661                                 off_t last = opts_optarg_int(opts, "P");
 662 
 663                                 /* return if not enough time has passed */
 664                                 if (Now - last < when)
 665                                         return (B_TRUE);
 666                         } else if ((cfopts = conf_opts(fname)) != NULL &&
 667                             opts_count(cfopts, "P")) {
 668                                 off_t last = opts_optarg_int(cfopts, "P");
 669 
 670                                 /*
 671                                  * just checking this means this entry
 672                                  * is now "done" if we're going through
 673                                  * the entire conffile
 674                                  */
 675                                 Donenames = lut_add(Donenames, fname, "1");
 676 
 677                                 /* return if not enough time has passed */
 678                                 if (Now - last < when)
 679                                         return (B_TRUE);
 680                         }
 681                 }
 682         }
 683 
 684         if (Debug)
 685                 (void) fprintf(stderr, "rotatelog: conditions met\n");
 686         if (opts_count(opts, "l")) {
 687                 /* Change the time zone to local time zone */
 688                 if (putenv("TZ="))
 689                         err(EF_SYS, "putenv TZ");
 690                 tzset();
 691                 Now = time(0);
 692 
 693                 /* rename the log file */
 694                 rotateto(fnp, opts, 0, recentlog, B_FALSE);
 695 
 696                 /* Change the time zone to UTC */
 697                 if (putenv("TZ=UTC"))
 698                         err(EF_SYS, "putenv TZ");
 699                 tzset();
 700                 Now = time(0);
 701         } else {
 702                 /* rename the log file */
 703                 rotateto(fnp, opts, 0, recentlog, B_FALSE);
 704         }
 705 
 706         /* determine owner, group, mode for empty log file */
 707         if (opts_count(opts, "o"))
 708                 (void) strlcpy(ownerbuf, opts_optarg(opts, "o"), MAXATTR);
 709         else {
 710                 (void) snprintf(ownerbuf, MAXATTR, "%ld", stbuf.st_uid);
 711         }
 712         owner = ownerbuf;
 713         if (opts_count(opts, "g"))
 714                 group = opts_optarg(opts, "g");
 715         else {
 716                 (void) snprintf(groupbuf, MAXATTR, "%ld", stbuf.st_gid);
 717                 group = groupbuf;
 718         }
 719         (void) strlcat(ownerbuf, ":", MAXATTR - strlen(ownerbuf));
 720         (void) strlcat(ownerbuf, group, MAXATTR - strlen(ownerbuf));
 721         if (opts_count(opts, "m"))
 722                 mode = opts_optarg(opts, "m");
 723         else {
 724                 (void) snprintf(modebuf, MAXATTR,
 725                     "%03lo", stbuf.st_mode & 0777);
 726                 mode = modebuf;
 727         }
 728 
 729         /* create the empty log file */
 730         docmd(opts, NULL, Touch, fname, NULL, NULL);
 731         docmd(opts, NULL, Chown, owner, fname, NULL);
 732         docmd(opts, NULL, Chmod, mode, fname, NULL);
 733 
 734         /* execute post-rotation command */
 735         if (opts_count(opts, "R")) {
 736                 struct fn *rawcmd = fn_new(opts_optarg(opts, "R"));
 737                 struct fn *cmd = fn_new(NULL);
 738 
 739                 kw_init(recentlog, NULL);
 740                 (void) kw_expand(rawcmd, cmd, 0, B_FALSE);
 741                 docmd(opts, "-R cmd", Sh, "-c", fn_s(cmd), NULL);
 742                 fn_free(rawcmd);
 743                 fn_free(cmd);
 744         }
 745         fn_free(recentlog);
 746 
 747         /*
 748          * add "after" command to list of after commands.  we also record
 749          * the email address, if any, where the error output of the after
 750          * command should be sent.  if the after command is already on
 751          * our list, add the email addr to the list the email addrs for
 752          * that command (the after command will only be executed once,
 753          * so the error output gets mailed to every address we've come
 754          * across associated with this command).
 755          */
 756         if (opts_count(opts, "a")) {
 757                 const char *cmd = opts_optarg(opts, "a");
 758                 struct lut *addrs = (struct lut *)lut_lookup(Aftercmds, cmd);
 759                 if (opts_count(opts, "e"))
 760                         addrs = lut_add(addrs, opts_optarg(opts, "e"), NULL);
 761                 Aftercmds = lut_add(Aftercmds, opts_optarg(opts, "a"), addrs);
 762         }
 763 
 764         /* record the rotation date */
 765         (void) strftime(nowstr, sizeof (nowstr),
 766             "%a %b %e %T %Y", gmtime(&Now));
 767         if (opts_count(opts, "v") && fname != NULL)
 768                 (void) out("#     recording rotation date %s for %s\n",
 769                     nowstr, fname);
 770         conf_set(fname, "P", STRDUP(nowstr));
 771         Donenames = lut_add(Donenames, fname, "1");
 772         return (B_TRUE);
 773 }
 774 
 775 /* rotate files "up" according to current template */
 776 static void
 777 rotateto(struct fn *fnp, struct opts *opts, int n, struct fn *recentlog,
 778     boolean_t isgz)
 779 {
 780         struct fn *template = fn_new(opts_optarg(opts, "t"));
 781         struct fn *newfile = fn_new(NULL);
 782         struct fn *dirname;
 783         int hasn;
 784         struct stat stbuf;
 785         char *buf1;
 786         char *buf2;
 787 
 788         /* expand template to figure out new filename */
 789         hasn = kw_expand(template, newfile, n, isgz);
 790 
 791         buf1 = fn_s(fnp);
 792         buf2 = fn_s(newfile);
 793 
 794         if (Debug)
 795                 if (buf1 != NULL && buf2 != NULL) {
 796                         (void) fprintf(stderr, "rotateto: %s -> %s (%d)\n",
 797                             buf1, buf2, n);
 798                 }
 799         /* if filename is there already, rotate "up" */
 800         if (hasn && lstat(buf2, &stbuf) != -1)
 801                 rotateto(newfile, opts, n + 1, recentlog, isgz);
 802         else if (hasn && opts_count(opts, "z")) {
 803                 struct fn *gzfnp = fn_dup(newfile);
 804                 /*
 805                  * since we're compressing old files, see if we
 806                  * about to rotate into one.
 807                  */
 808                 fn_puts(gzfnp, ".gz");
 809                 if (lstat(fn_s(gzfnp), &stbuf) != -1)
 810                         rotateto(gzfnp, opts, n + 1, recentlog, B_TRUE);
 811                 fn_free(gzfnp);
 812         }
 813 
 814         /* first time through run "before" cmd if not run already */
 815         if (n == 0 && opts_count(opts, "b")) {
 816                 const char *cmd = opts_optarg(opts, "b");
 817 
 818                 if (lut_lookup(Beforecmds, cmd) == NULL) {
 819                         docmd(opts, "-b cmd", Sh, "-c", cmd, NULL);
 820                         Beforecmds = lut_add(Beforecmds, cmd, "1");
 821                 }
 822         }
 823 
 824         /* ensure destination directory exists */
 825         dirname = fn_dirname(newfile);
 826         docmd(opts, "verify directory exists", Mkdir, "-p",
 827             fn_s(dirname), NULL);
 828         fn_free(dirname);
 829 
 830         /* do the rename */
 831         if (opts_count(opts, "c") != NULL) {
 832                 docopytruncate(opts, fn_s(fnp), fn_s(newfile));
 833         } else if (n == 0 && opts_count(opts, "M")) {
 834                 struct fn *rawcmd = fn_new(opts_optarg(opts, "M"));
 835                 struct fn *cmd = fn_new(NULL);
 836 
 837                 /* use specified command to mv the log file */
 838                 kw_init(fnp, newfile);
 839                 (void) kw_expand(rawcmd, cmd, 0, B_FALSE);
 840                 docmd(opts, "-M cmd", Sh, "-c", fn_s(cmd), NULL);
 841                 fn_free(rawcmd);
 842                 fn_free(cmd);
 843         } else
 844                 /* common case: we call "mv" to handle the actual rename */
 845                 docmd(opts, "rotate log file", Mv, "-f",
 846                     fn_s(fnp), fn_s(newfile));
 847 
 848         /* first time through, gather interesting info for caller */
 849         if (n == 0)
 850                 fn_renew(recentlog, fn_s(newfile));
 851 }
 852 
 853 /* expire phase of logname processing */
 854 static void
 855 expirefiles(struct fn *fnp, struct opts *opts)
 856 {
 857         char *fname = fn_s(fnp);
 858         struct fn *template;
 859         struct fn *pattern;
 860         struct fn_list *files;
 861         struct fn *nextfnp;
 862         off_t count;
 863         off_t size;
 864 
 865         if (Debug && fname != NULL)
 866                 (void) fprintf(stderr, "expirefiles: fname <%s>\n", fname);
 867 
 868         /* return if no potential expire conditions */
 869         if (opts_count(opts, "zAS") == 0 && opts_optarg_int(opts, "C") == 0)
 870                 return;
 871 
 872         kw_init(fnp, NULL);
 873         if (Debug > 1) {
 874                 (void) fprintf(stderr, "expirefiles keywords:\n");
 875                 kw_print(stderr);
 876         }
 877 
 878         /* see if pattern was supplied by user */
 879         if (opts_count(opts, "T")) {
 880                 template = fn_new(opts_optarg(opts, "T"));
 881                 pattern = glob_to_reglob(template);
 882         } else {
 883                 /* nope, generate pattern based on rotation template */
 884                 template = fn_new(opts_optarg(opts, "t"));
 885                 pattern = fn_new(NULL);
 886                 (void) kw_expand(template, pattern, -1,
 887                     opts_count(opts, "z") != 0);
 888         }
 889 
 890         /* match all old log files (hopefully not any others as well!) */
 891         files = glob_reglob(pattern);
 892 
 893         if (Debug) {
 894                 char *buf;
 895 
 896                 buf = fn_s(pattern);
 897                 if (buf != NULL) {
 898                         (void) fprintf(stderr, "expirefiles: pattern <%s>\n",
 899                             buf);
 900                 }
 901                 fn_list_rewind(files);
 902                 while ((nextfnp = fn_list_next(files)) != NULL)
 903                         buf = fn_s(nextfnp);
 904                         if (buf != NULL)
 905                                 (void) fprintf(stderr, "    <%s>\n", buf);
 906         }
 907 
 908         /* see if count causes expiration */
 909         if ((count = opts_optarg_int(opts, "C")) > 0) {
 910                 int needexpire = fn_list_count(files) - count;
 911 
 912                 if (Debug)
 913                         (void) fprintf(stderr, "expirefiles: needexpire %d\n",
 914                             needexpire);
 915 
 916                 while (needexpire > 0 &&
 917                     ((nextfnp = fn_list_popoldest(files)) != NULL)) {
 918                         dorm(opts, "expire by count rule", nextfnp);
 919                         fn_free(nextfnp);
 920                         needexpire--;
 921                 }
 922         }
 923 
 924         /* see if total size causes expiration */
 925         if (opts_count(opts, "S") && (size = opts_optarg_int(opts, "S")) > 0) {
 926                 while (fn_list_totalsize(files) > size &&
 927                     ((nextfnp = fn_list_popoldest(files)) != NULL)) {
 928                         dorm(opts, "expire by size rule", nextfnp);
 929                         fn_free(nextfnp);
 930                 }
 931         }
 932 
 933         /* see if age causes expiration */
 934         if (opts_count(opts, "A")) {
 935                 int mtime = (int)time(0) - (int)opts_optarg_int(opts, "A");
 936 
 937                 while ((nextfnp = fn_list_popoldest(files)) != NULL) {
 938                         if (fn_getstat(nextfnp)->st_mtime < mtime) {
 939                                 dorm(opts, "expire by age rule", nextfnp);
 940                                 fn_free(nextfnp);
 941                         } else {
 942                                 fn_list_addfn(files, nextfnp);
 943                                 break;
 944                         }
 945                 }
 946         }
 947 
 948         /* record old log files to be gzip'ed according to -z count */
 949         if (opts_count(opts, "z")) {
 950                 int zcount = (int)opts_optarg_int(opts, "z");
 951                 int fcount = fn_list_count(files);
 952 
 953                 while (fcount > zcount &&
 954                     (nextfnp = fn_list_popoldest(files)) != NULL) {
 955                         if (!fn_isgz(nextfnp)) {
 956                                 /*
 957                                  * Don't gzip the old log file yet -
 958                                  * it takes too long. Just remember that we
 959                                  * need to gzip.
 960                                  */
 961                                 if (Debug) {
 962                                         (void) fprintf(stderr,
 963                                             "will compress %s count %d\n",
 964                                             fn_s(nextfnp), fcount);
 965                                 }
 966                                 Gzipnames = lut_add(Gzipnames,
 967                                     fn_s(nextfnp), "1");
 968                         }
 969                         fn_free(nextfnp);
 970                         fcount--;
 971                 }
 972         }
 973 
 974         fn_free(template);
 975         fn_list_free(files);
 976 }
 977 
 978 /* execute a command to remove an expired log file */
 979 static void
 980 dorm(struct opts *opts, const char *msg, struct fn *fnp)
 981 {
 982         if (opts_count(opts, "E")) {
 983                 struct fn *rawcmd = fn_new(opts_optarg(opts, "E"));
 984                 struct fn *cmd = fn_new(NULL);
 985 
 986                 /* user supplied cmd, expand $file */
 987                 kw_init(fnp, NULL);
 988                 (void) kw_expand(rawcmd, cmd, 0, B_FALSE);
 989                 docmd(opts, msg, Sh, "-c", fn_s(cmd), NULL);
 990                 fn_free(rawcmd);
 991                 fn_free(cmd);
 992         } else
 993                 docmd(opts, msg, Rm, "-f", fn_s(fnp), NULL);
 994         Gzipnames = lut_add(Gzipnames, fn_s(fnp), NULL);
 995 }
 996 
 997 /* execute a command, producing -n and -v output as necessary */
 998 static void
 999 docmd(struct opts *opts, const char *msg, const char *cmd,
1000     const char *arg1, const char *arg2, const char *arg3)
1001 {
1002         int pid;
1003         int errpipe[2];
1004 
1005         /* print info about command if necessary */
1006         if (opts_count(opts, "vn")) {
1007                 const char *simplecmd;
1008 
1009                 if ((simplecmd = strrchr(cmd, '/')) == NULL)
1010                         simplecmd = cmd;
1011                 else
1012                         simplecmd++;
1013                 (void) out("%s", simplecmd);
1014                 if (arg1)
1015                         (void) out(" %s", arg1);
1016                 if (arg2)
1017                         (void) out(" %s", arg2);
1018                 if (arg3)
1019                         (void) out(" %s", arg3);
1020                 if (msg)
1021                         (void) out(" # %s", msg);
1022                 (void) out("\n");
1023         }
1024 
1025         if (opts_count(opts, "n"))
1026                 return;         /* -n means don't really do it */
1027 
1028         /*
1029          * run the cmd and see if it failed.  this function is *not* a
1030          * generic command runner -- we depend on some knowledge we
1031          * have about the commands we run.  first of all, we expect
1032          * errors to spew something to stderr, and that something is
1033          * typically short enough to fit into a pipe so we can wait()
1034          * for the command to complete and then fetch the error text
1035          * from the pipe.  we also expect the exit codes to make sense.
1036          * notice also that we only allow a command name which is an
1037          * absolute pathname, and two args must be supplied (the
1038          * second may be NULL, or they may both be NULL).
1039          */
1040         if (pipe(errpipe) < 0)
1041                 err(EF_SYS, "pipe");
1042 
1043         if ((pid = fork()) < 0)
1044                 err(EF_SYS, "fork");
1045         else if (pid) {
1046                 int wstat;
1047                 int count;
1048 
1049                 /* parent */
1050                 (void) close(errpipe[1]);
1051                 if (waitpid(pid, &wstat, 0) < 0)
1052                         err(EF_SYS, "waitpid");
1053 
1054                 /* check for stderr output */
1055                 if (ioctl(errpipe[0], FIONREAD, &count) >= 0 && count) {
1056                         err(EF_WARN, "command failed: %s%s%s%s%s%s%s",
1057                             cmd,
1058                             (arg1) ? " " : "",
1059                             (arg1) ? arg1 : "",
1060                             (arg2) ? " " : "",
1061                             (arg2) ? arg2 : "",
1062                             (arg3) ? " " : "",
1063                             (arg3) ? arg3 : "");
1064                         err_fromfd(errpipe[0]);
1065                 } else if (WIFSIGNALED(wstat))
1066                         err(EF_WARN,
1067                             "command died, signal %d: %s%s%s%s%s%s%s",
1068                             WTERMSIG(wstat),
1069                             cmd,
1070                             (arg1) ? " " : "",
1071                             (arg1) ? arg1 : "",
1072                             (arg2) ? " " : "",
1073                             (arg2) ? arg2 : "",
1074                             (arg3) ? " " : "",
1075                             (arg3) ? arg3 : "");
1076                 else if (WIFEXITED(wstat) && WEXITSTATUS(wstat))
1077                         err(EF_WARN,
1078                             "command error, exit %d: %s%s%s%s%s%s%s",
1079                             WEXITSTATUS(wstat),
1080                             cmd,
1081                             (arg1) ? " " : "",
1082                             (arg1) ? arg1 : "",
1083                             (arg2) ? " " : "",
1084                             (arg2) ? arg2 : "",
1085                             (arg3) ? " " : "",
1086                             (arg3) ? arg3 : "");
1087 
1088                 (void) close(errpipe[0]);
1089         } else {
1090                 /* child */
1091                 (void) dup2(errpipe[1], fileno(stderr));
1092                 (void) close(errpipe[0]);
1093                 (void) execl(cmd, cmd, arg1, arg2, arg3, 0);
1094                 perror(cmd);
1095                 _exit(1);
1096         }
1097 }
1098 
1099 /* do internal atomic file copy and truncation */
1100 static void
1101 docopytruncate(struct opts *opts, const char *file, const char *file_copy)
1102 {
1103         int fi, fo, len;
1104         char buf[4096];
1105         struct stat s;
1106         struct utimbuf times;
1107 
1108         /* print info if necessary */
1109         if (opts_count(opts, "vn") != NULL) {
1110                 (void) out("# log rotation via atomic copy and truncation"
1111                     " (-c flag):\n");
1112                 (void) out("# copy %s to %s\n", file, file_copy);
1113                 (void) out("# truncate %s\n", file);
1114         }
1115 
1116         if (opts_count(opts, "n"))
1117                 return;         /* -n means don't really do it */
1118 
1119         /* open log file to be rotated and remember its chmod mask */
1120         if ((fi = open(file, O_RDWR)) < 0) {
1121                 err(EF_SYS, "cannot open file %s", file);
1122                 return;
1123         }
1124 
1125         if (fstat(fi, &s) < 0) {
1126                 err(EF_SYS, "cannot access: %s", file);
1127                 (void) close(fi);
1128                 return;
1129         }
1130 
1131         /* create new file for copy destination with correct attributes */
1132         if ((fo = open(file_copy, O_CREAT|O_APPEND|O_WRONLY, s.st_mode)) < 0) {
1133                 err(EF_SYS, "cannot create file: %s", file_copy);
1134                 (void) close(fi);
1135                 return;
1136         }
1137 
1138         (void) fchown(fo, s.st_uid, s.st_gid);
1139 
1140         /* lock log file so that nobody can write into it before we are done */
1141         if (fchmod(fi, s.st_mode|S_ISGID) < 0)
1142                 err(EF_SYS, "cannot set mandatory lock bit for: %s", file);
1143 
1144         if (lockf(fi, F_LOCK, 0) == -1)
1145                 err(EF_SYS, "cannot lock file %s", file);
1146 
1147         /* do atomic copy and truncation */
1148         while ((len = read(fi, buf, sizeof (buf))) > 0)
1149                 if (write(fo, buf, len) != len) {
1150                         err(EF_SYS, "cannot write into file %s", file_copy);
1151                         (void) lockf(fi, F_ULOCK, 0);
1152                         (void) fchmod(fi, s.st_mode);
1153                         (void) close(fi);
1154                         (void) close(fo);
1155                         (void) remove(file_copy);
1156                         return;
1157                 }
1158 
1159         (void) ftruncate(fi, 0);
1160 
1161         /* unlock log file */
1162         if (lockf(fi, F_ULOCK, 0) == -1)
1163                 err(EF_SYS, "cannot unlock file %s", file);
1164 
1165         if (fchmod(fi, s.st_mode) < 0)
1166                 err(EF_SYS, "cannot reset mandatory lock bit for: %s", file);
1167 
1168         (void) close(fi);
1169         (void) close(fo);
1170 
1171         /* keep times from original file */
1172         times.actime = s.st_atime;
1173         times.modtime = s.st_mtime;
1174         (void) utime(file_copy, &times);
1175 }