home *** CD-ROM | disk | FTP | other *** search
/ Beijing Paradise BBS Backup / PARADISE.ISO / software / BBSDOORW / UUPC11XS.ZIP / UUCP / UUSTAT.C < prev    next >
C/C++ Source or Header  |  1992-11-15  |  38KB  |  1,088 lines

  1. /*--------------------------------------------------------------------*/
  2. /*    u u s t a t . c                                                 */
  3. /*                                                                    */
  4. /*    Job status report for UUPC/extended                             */
  5. /*                                                                    */
  6. /*                                                                    */
  7. /*    Copyright 1988 (C), Dewey Coffman                               */
  8. /*    Changes Copyright 1991 (C), Andrew H. Derbyshire                */
  9. /*--------------------------------------------------------------------*/
  10.  
  11. /*--------------------------------------------------------------------*/
  12. /*         System include files                                       */
  13. /*--------------------------------------------------------------------*/
  14.  
  15. #include <stdio.h>
  16. #include <io.h>
  17. #include <sys/types.h>
  18. #include <sys/stat.h>
  19. #include <string.h>
  20. #include <time.h>
  21. #include <fcntl.h>
  22. #include <stdlib.h>
  23.  
  24. /*--------------------------------------------------------------------*/
  25. /*         Local include files                                        */
  26. /*--------------------------------------------------------------------*/
  27.  
  28. #include "lib.h"
  29. #include "dater.h"
  30. #include "export.h"
  31. #include "getopt.h"
  32. #include "getseq.h"
  33. #include "hlib.h"
  34. #include "hostable.h"
  35. #include "hostatus.h"
  36. #include "import.h"
  37. #include "pushpop.h"
  38. #include "readnext.h"
  39. #include "security.h"
  40. #include "stater.h"
  41. #include "timestmp.h"
  42.  
  43. #define DAY (60l * 60l * 24l)
  44. #define ALL    "all"
  45. #define MAXL      128      /* MAX LINE LENGTH */
  46.  
  47. #define STRCREAT(s, s2, s3)\
  48.    strcpy(s, s2);\
  49.    strcat(s, s3);\
  50.  
  51. /*--------------------------------------------------------------------*/
  52. /*                          Local data types                          */
  53. /*--------------------------------------------------------------------*/
  54.  
  55. typedef enum {
  56.       POLL_CALL = 'P',
  57.       RECEIVE_CALL = 'R',
  58.       SEND_CALL = 'S'
  59.       } CALLTYPE;
  60.  
  61.  
  62. struct data_queue {
  63.    char name[FILENAME_MAX];
  64.    struct data_queue *next_link;
  65.    time_t created;
  66.    long size;
  67.    boolean execute;
  68.    char type;
  69. } ;
  70.  
  71. /*--------------------------------------------------------------------*/
  72. /*                             Verb list                              */
  73. /*--------------------------------------------------------------------*/
  74.  
  75. typedef enum {
  76.    LIST_DEFAULT = 1,
  77.    LIST_ALL,
  78.    LIST_ACCESS,
  79.    LIST_QUEUE,
  80.    KILL_JOB,
  81.    REFRESH_JOB,
  82.    FORCE_POLL
  83.    } COMMAND_CLASS;
  84.  
  85. /*--------------------------------------------------------------------*/
  86. /*                Processing to be taken by open_call                 */
  87. /*--------------------------------------------------------------------*/
  88.  
  89. typedef enum {
  90.    JOB_STATUS = 1,
  91.    JOB_KILL,
  92.    JOB_REFRESH
  93.    } CALL_ACTION;
  94.  
  95. /*--------------------------------------------------------------------*/
  96. /*                          Global variables                          */
  97. /*--------------------------------------------------------------------*/
  98.  
  99. currentfile();
  100.  
  101. static const char *host_status[] = {
  102.       "(invalid - entry not properly initialized)",
  103.       "(local host system)",
  104.       "(host for gateway purposes only)",
  105.       "Never called",
  106.       "Dialing now",
  107.       "Invalid device or speed in SYSTEMS file",
  108.       "Device not available",
  109.       "Conversation start-up failed",
  110.       "Talking",
  111.       "Callback required",
  112.       "Modem initialization script failed",
  113.       "Dial failed",
  114.       "Script failed",
  115.       "Max retry reached",
  116.       "Retry time not reached",
  117.       "Call succeeded",
  118.       "Wrong machine name",
  119.       "Unknown host",
  120.       "Failed",
  121.       "Wrong time to call",
  122.       "(call successed, entry not reset)",
  123.    } ;
  124.  
  125. /*--------------------------------------------------------------------*/
  126. /*                        Internal prototypes                         */
  127. /*--------------------------------------------------------------------*/
  128.  
  129. static void all( const char *system, const char *userid );
  130.  
  131. static char *is_job(const char *jobid );
  132.  
  133. static void kill_job(const char *s);
  134.  
  135. static void long_stats( const char *system );
  136.  
  137. static void short_stats( const char *system );
  138.  
  139. static CALLTYPE open_call( const char *callname,
  140.                            const char *remote,
  141.                                  struct data_queue **current,
  142.                                  char *user,
  143.                                  char *sys,
  144.                            const CALL_ACTION action);
  145.  
  146. static void open_data(const char *file,
  147.                             char *user,
  148.                             char *sys,
  149.                             char *command);
  150.  
  151. static void poll(const char *callee);
  152.  
  153. static void print_all(       char *job,
  154.                        struct data_queue *current,
  155.                        const char *user,
  156.                        const char *sys);
  157.  
  158. static void refresh_job(const char *s);
  159.  
  160. static void touch( const char *fname );
  161.  
  162. static void usage( void );
  163.  
  164. /*--------------------------------------------------------------------*/
  165. /*    m a i n                                                         */
  166. /*                                                                    */
  167. /*    main program                                                    */
  168. /*--------------------------------------------------------------------*/
  169.  
  170. void main(int  argc, char  **argv)
  171. {
  172.    int c;
  173.    extern char *optarg;
  174.    extern int   optind;
  175.    COMMAND_CLASS command = LIST_DEFAULT;
  176.  
  177.    char *system = NULL;
  178.    char *userid = NULL;
  179.    char *job    = NULL;
  180.  
  181. /*--------------------------------------------------------------------*/
  182. /*     Report our version number and date/time compiled               */
  183. /*--------------------------------------------------------------------*/
  184.  
  185.    debuglevel = 0;
  186.    banner( argv );
  187.  
  188. #if defined(__CORE__)
  189.    copywrong = strdup(copyright);
  190.    checkref(copywrong);
  191. #endif
  192.  
  193.    if (!configure( B_UUSTAT ))
  194.       exit(1);   /* system configuration failed */
  195.  
  196. /*--------------------------------------------------------------------*/
  197. /*                   Switch to the spool directory                    */
  198. /*--------------------------------------------------------------------*/
  199.  
  200.    tzset();                      /* Set up time zone information  */
  201.    PushDir( E_spooldir );
  202.    atexit( PopDir );
  203.  
  204. /*--------------------------------------------------------------------*/
  205. /*        Process our arguments                                       */
  206. /*--------------------------------------------------------------------*/
  207.  
  208.    while ((c = getopt(argc, argv, "amqk:r:s:u:x:P:")) !=  EOF)
  209.       switch(c) {
  210.       case 'a':
  211.          command = LIST_ALL;
  212.          break;
  213.  
  214.       case 'm':
  215.          command = LIST_ACCESS;
  216.          system = optarg;
  217.          break;
  218.  
  219.       case 'q':
  220.          command = LIST_QUEUE;
  221.          break;
  222.  
  223.       case 'k':
  224.          command = KILL_JOB;
  225.          job = optarg;
  226.          break;
  227.  
  228.       case 'r':
  229.          command = REFRESH_JOB;
  230.          job = optarg;
  231.          break;
  232.  
  233.       case 's':
  234.          if ( system != NULL )
  235.          {
  236.             printmsg(0,"Invalid or duplicate option -s %s",optarg);
  237.             usage();
  238.          }
  239.          system = optarg;
  240.          break;
  241.  
  242.       case 'u':
  243.          userid = optarg;
  244.          break;
  245.  
  246.       case 'x':
  247.          debuglevel = atoi( optarg );
  248.          break;
  249.  
  250.       case 'P':
  251.          command = FORCE_POLL;
  252.          if ( system != NULL )
  253.          {
  254.             printmsg(0,"Invalid or duplicate option -P %s",optarg);
  255.             usage();
  256.          }
  257.          system = optarg;
  258.          break;
  259.  
  260.       case '?':
  261.          usage();
  262.    }
  263.  
  264.    if (optind != argc) {
  265.       puts("Extra parameter(s) at end.");
  266.       exit(2);
  267.    }
  268.  
  269. /*--------------------------------------------------------------------*/
  270. /*                 Determine if we have a valid host                  */
  271. /*--------------------------------------------------------------------*/
  272.  
  273.    if( (system != NULL) && !equal( system , ALL ) )
  274.    {
  275.       struct HostTable *hostp = checkreal( system );
  276.  
  277.       if (hostp  ==  BADHOST)
  278.       {
  279.          printf("Unknown host \"%s\", program terminating.\n",
  280.                system );
  281.          panic();
  282.       }
  283.    } /* if */
  284.  
  285. /*--------------------------------------------------------------------*/
  286. /*                   Execute the requested command                    */
  287. /*--------------------------------------------------------------------*/
  288.  
  289.    switch ( command )
  290.    {
  291.       case LIST_DEFAULT:
  292.          if ( (system == NULL ) && ( userid == NULL) )
  293.          {
  294.             all( ALL, E_mailbox );
  295.             break;
  296.          }
  297.             /* Otherwise, fall through ... */
  298.  
  299.       case LIST_ALL:
  300.          if ( system == NULL )
  301.             system = ALL;
  302.          if ( userid == NULL )
  303.             userid = ALL;
  304.          all( system, userid );
  305.          break;
  306.  
  307.       case LIST_ACCESS:
  308.          if ( system == NULL )
  309.             system = ALL;
  310.          short_stats( system );
  311.          break;
  312.  
  313.       case LIST_QUEUE:
  314.          if ( system == NULL )
  315.             system = ALL;
  316.          long_stats( system );
  317.          break;
  318.  
  319.       case KILL_JOB:
  320.          kill_job( job );
  321.          break;
  322.  
  323.       case REFRESH_JOB:
  324.          refresh_job( job );
  325.          break;
  326.  
  327.       case FORCE_POLL:
  328.          poll( system );
  329.          break;
  330.  
  331.       default:
  332.          panic();
  333.  
  334.    } /* switch */
  335.    exit(0);
  336.  
  337. } /* main */
  338.  
  339. /*--------------------------------------------------------------------*/
  340. /*    a l l                                                           */
  341. /*                                                                    */
  342. /*    Report on all systems                                           */
  343. /*--------------------------------------------------------------------*/
  344.  
  345. void all( const char *system, const char *userid)
  346. {
  347.    char  canon[FILENAME_MAX];
  348.    long  size, ltime;
  349.    struct HostTable *hostp;
  350.    boolean hit = FALSE;
  351.  
  352.    if ( equal(system,ALL) )
  353.       hostp = nexthost( TRUE );
  354.    else
  355.       hostp = checkreal( system );
  356.  
  357. /*--------------------------------------------------------------------*/
  358. /*                  Scan one or all host directories                  */
  359. /*--------------------------------------------------------------------*/
  360.  
  361.    while  (hostp !=  BADHOST )
  362.    {
  363.       char fname[FILENAME_MAX];
  364.  
  365. /*--------------------------------------------------------------------*/
  366. /*   Examine all of the files in the spool directory for this host    */
  367. /*--------------------------------------------------------------------*/
  368.  
  369.       while( readnext(fname , hostp->hostname, "C", NULL) )
  370.       {
  371.          boolean display = equal( userid, ALL );
  372.          struct data_queue *data_link = NULL;
  373.          char user[MAXL];
  374.          char sys[MAXL];
  375.  
  376.          strcpy(user,userid);       /* Nice default for the user     */
  377.                                     /* Generates more output if
  378.                                        straight copies in the queue  */
  379.          strcpy(sys, E_nodename);   /* Nice default for node as well */
  380.  
  381.          printmsg(1,"ALL(%s)", fname);
  382.          exportpath(canon, fname, hostp->hostname);
  383.  
  384. /*--------------------------------------------------------------------*/
  385. /*               Determine what kind of Call file it is               */
  386. /*--------------------------------------------------------------------*/
  387.  
  388.          switch(open_call(fname, hostp->hostname,
  389.                            &data_link, user, sys, JOB_STATUS))
  390.          {
  391.             case POLL_CALL:
  392.                if ( display )
  393.                {
  394.                   ltime = stater(fname, &size);
  395.                   hit = TRUE;
  396.                   printf( "%s %s %s\n",canon+2,dater(ltime, NULL),
  397.                         "(POLL)");
  398.                }
  399.                break;
  400.  
  401.             case SEND_CALL:
  402.             case RECEIVE_CALL:
  403.                if( equal(userid , ALL) || equal(userid, user))
  404.                   display = TRUE;
  405.  
  406.                if(display)
  407.                {
  408.                    hit = TRUE;
  409.                    print_all( canon + 2, data_link, user,
  410.                               hostp->hostname );
  411.                }
  412.                break;
  413.          } /* switch */
  414.  
  415.       } /* while */
  416.  
  417. /*--------------------------------------------------------------------*/
  418. /*    If processing all hosts, step to the next host in the queue     */
  419. /*--------------------------------------------------------------------*/
  420.  
  421.       if( equal( system , ALL ))
  422.          hostp = nexthost( FALSE );
  423.       else
  424.          hostp = BADHOST;
  425.  
  426.    } /* while */
  427.  
  428.    if ( !hit )
  429.       printf("uustat: No jobs queued for system %s by user %s\n",
  430.                system , userid );
  431.  
  432. } /* all */
  433.  
  434. /*--------------------------------------------------------------------*/
  435. /*    p o l l                                                         */
  436. /*                                                                    */
  437. /*    Write a dummy call file to request a poll of a host             */
  438. /*--------------------------------------------------------------------*/
  439.  
  440. static void poll(const char *callee)
  441. {
  442.  
  443.    char tmfile[15];           /* Call file, UNIX format name         */
  444.    char msname[FILENAME_MAX];
  445.    FILE *stream;
  446.    static char *spool_fmt = SPOOLFMT;              /* spool file name */
  447.    struct HostTable *hostp;
  448.  
  449. /*--------------------------------------------------------------------*/
  450. /*              Determine first (only?) host to process               */
  451. /*--------------------------------------------------------------------*/
  452.  
  453.    if ( equal(callee,ALL) )
  454.       hostp = nexthost( TRUE );
  455.    else
  456.       hostp = checkreal( callee );
  457.  
  458. /*--------------------------------------------------------------------*/
  459. /*                  Scan one or all host directories                  */
  460. /*--------------------------------------------------------------------*/
  461.  
  462.    while  (hostp !=  BADHOST )
  463.    {
  464.       printmsg(1,"POLL(%s)", hostp->hostname);
  465.       sprintf(tmfile,"%.8s",hostp->hostname);
  466.  
  467.       if ( ValidDOSName( tmfile ) || !equal(callee, ALL))
  468.       {
  469.          sprintf(tmfile, spool_fmt, 'C', hostp->hostname,'Z' ,
  470.                   "000");
  471.          importpath( msname, tmfile, hostp->hostname);
  472.  
  473.          if ( access( msname , 0 ))   /* Does the host file exist?     */
  474.          {
  475.             stream = FOPEN( msname ,"w", BINARY);
  476.  
  477.             if ( stream == NULL )
  478.             {
  479.                printerr( msname );
  480.                printmsg(0,"uustat: problem creating poll file %s (%s)",
  481.                         tmfile, msname);
  482.                panic();
  483.             }
  484.  
  485.             fclose( stream );
  486.  
  487.             printmsg(0,"Created dummy job %s (%s) for system %s",
  488.                   tmfile, msname, hostp->hostname );
  489.  
  490.          } /* if */
  491.          else
  492.             printmsg(0,"Dummy job %s (%s) already exists for system %s",
  493.                   tmfile, msname, hostp->hostname );
  494.  
  495.       } /* if ( ValidDOSName( tmfile ) || !equal(callee, ALL)) */
  496.       else {
  497.          printmsg(0,"%s is not a valid host name, skipping",
  498.                      hostp->hostname);
  499.       } /* else */
  500.  
  501. /*--------------------------------------------------------------------*/
  502. /*    If processing all hosts, step to the next host in the queue     */
  503. /*--------------------------------------------------------------------*/
  504.  
  505.       if( equal( callee , ALL ))
  506.          hostp = nexthost( FALSE );
  507.       else
  508.          hostp = BADHOST;
  509.  
  510.    } /* while  (hostp !=  BADHOST ) */
  511.  
  512. } /* poll */
  513.  
  514.  
  515. /*--------------------------------------------------------------------*/
  516. /*    l o n g _ s t a t s                                             */
  517. /*                                                                    */
  518. /*    Report full information on jobs for a host                      */
  519. /*--------------------------------------------------------------------*/
  520.  
  521. /*--------------------------------------------------------------------*/
  522. /* fifi      1C     03/30-16:34 LOGIN FAILED Retry: 22:39 Count: 130  */
  523. /* helps     2C     03/30-16:32 SUCCESSFUL                            */
  524. /* im4u      5C(1)  03/30-06:50 LOGIN FAILED Retry: 11:15 Count: 9    */
  525. /* irasun           03/30-16:23 TALKING                               */
  526. /* killer    5C(6)  03/29-11:13 WRONG TIME TO CALL                    */
  527. /* radian    18C    03/30-16:42 WRONG TIME TO CALL                    */
  528. /* shemp      1C    03/30-14:43 LOGIN FAILED Retry: 20:48 Count: 107  */
  529. /* unisec     2C(1) 03/30-16:50 WRONG TIME TO CALL                    */
  530. /* ut-emx     1X    03/30-16:50 SUCCESSFUL                            */
  531. /*--------------------------------------------------------------------*/
  532.  
  533. static void long_stats( const char *system )
  534. {
  535.    struct HostTable *hostp;
  536.    time_t now = time( NULL );
  537.    boolean hit = FALSE;
  538.    time_t ltime;
  539.    long size;
  540.    char buf[BUFSIZ];
  541.  
  542.    HostStatus();              /* Load the host status table info     */
  543.  
  544. /*--------------------------------------------------------------------*/
  545. /*                  Get the first system to process                   */
  546. /*--------------------------------------------------------------------*/
  547.  
  548.    if ( equal(system,ALL) )
  549.       hostp = nexthost( TRUE );
  550.    else
  551.       hostp = checkreal( system );
  552.  
  553. /*--------------------------------------------------------------------*/
  554. /*              Begin loop to display status of systems               */
  555. /*--------------------------------------------------------------------*/
  556.  
  557.    while(hostp != BADHOST )
  558.    {
  559.       char fname[FILENAME_MAX];
  560.                               /* Get list of files in the directory  */
  561.       size_t jobs = 0;        /* Declare, reset counter              */
  562.       time_t oldest_file = now;  /* Make the "oldest" file new       */
  563.  
  564. /*--------------------------------------------------------------------*/
  565. /*           Inner loop to count files and determine oldest           */
  566. /*--------------------------------------------------------------------*/
  567.  
  568.       while( readnext(fname, hostp->hostname, "C", NULL) != NULL )
  569.       {
  570.          ltime = stater( fname , &size);
  571.          if ((ltime > -1) && (ltime < oldest_file ))
  572.             oldest_file = ltime;
  573.  
  574.          jobs++;
  575.  
  576.       } /* while */
  577.  
  578. /*--------------------------------------------------------------------*/
  579. /*  We have all the information for this system; summary and display  */
  580. /*--------------------------------------------------------------------*/
  581.  
  582.       if ( jobs > 0 )
  583.       {
  584.          if (oldest_file + DAY < now)     /* File older than 24 hours?  */
  585.             sprintf( buf , "(%d)", (now - oldest_file) / DAY );
  586.                                           /* Yes --> Format info        */
  587.          else
  588.             *buf = '\0';                  /* No --> No, empty display   */
  589.  
  590.          printf("%-8.8s  %3dC%-4s  %s  %s\n", hostp->hostname, jobs , buf,
  591.                dater( hostp->hstats->lconnect , NULL ),
  592.                hostp->hstatus < last_status ?
  593.                      host_status[ hostp->hstatus ] :
  594.                      "*** INVALID/UNDOCUMENTED STATUS ***");
  595.          hit = TRUE;
  596.  
  597.       } /* if ( jobs > 0 ) */
  598.  
  599.       if (equal(system, ALL))
  600.          hostp = nexthost( FALSE );
  601.       else
  602.          hostp = BADHOST;
  603.    } /* while */
  604.  
  605.    if ( !hit )
  606.       printf("uustat: No jobs queued for system %s\n", system );
  607. } /* long_stats */
  608.  
  609. /*--------------------------------------------------------------------*/
  610. /*    s h o r t _ s t a t s                                           */
  611. /*                                                                    */
  612. /*    Report access to a system                                       */
  613. /*--------------------------------------------------------------------*/
  614.  
  615. static void short_stats( const char *system )
  616. {
  617.    struct HostTable *hostp;
  618.  
  619.    HostStatus();              /* Load the host status table info     */
  620.  
  621. /*--------------------------------------------------------------------*/
  622. /*                  Get the first system to process                   */
  623. /*--------------------------------------------------------------------*/
  624.  
  625.    if ( equal(system,ALL) )
  626.       hostp = nexthost( TRUE );
  627.    else
  628.       hostp = checkreal( system );
  629.  
  630. /*--------------------------------------------------------------------*/
  631. /*              Begin loop to display status of systems               */
  632. /*--------------------------------------------------------------------*/
  633.  
  634.    while(hostp != BADHOST )
  635.    {
  636.       printf("%-8.8s  %s  %s\n", hostp->hostname,
  637.             dater( hostp->hstats->lconnect , NULL ),
  638.             hostp->hstatus < last_status ?
  639.                   host_status[ hostp->hstatus ] :
  640.                   "*** INVALID/UNDOCUMENTED STATUS ***");
  641.  
  642.       if (equal(system, ALL))
  643.          hostp = nexthost( FALSE );
  644.       else
  645.          hostp = BADHOST;
  646.    } /* while */
  647.  
  648. } /* short_stats */
  649.  
  650.  
  651. /*--------------------------------------------------------------------*/
  652. /*    k i l l _ j o b                                                 */
  653. /*                                                                    */
  654. /*    Kill a queued UUPC/extended job                                 */
  655. /*--------------------------------------------------------------------*/
  656.  
  657. static void kill_job(const char *jobid)
  658. {
  659.    char *system;              /* System name returned by is_job()    */
  660.    char host[FILENAME_MAX];
  661.    char canon[FILENAME_MAX];
  662.    char user[FILENAME_MAX];
  663.    char sys[FILENAME_MAX];
  664.  
  665.    strcpy(canon,"C.");
  666.    strcat(canon,jobid);
  667.    strcpy( user, "uucp");
  668.    system = is_job( canon ); /* Never returns if an error occurs     */
  669.    importpath( host, canon, system );
  670.                               /* Get the local name of the file      */
  671.    open_call(host, system, NULL, user, sys, JOB_KILL);
  672.    unlink( host );
  673.    printf("Deleted file %s (%s)\n", canon, host);
  674.    printf("Killed job %s (%s) queued for host %s by %s\n",
  675.             jobid, host, system, user);
  676.  
  677. } /* kill_job */
  678.  
  679. /*--------------------------------------------------------------------*/
  680. /*    r e f r e s h _ j o b                                           */
  681. /*                                                                    */
  682. /*    Refresh a job in the spool                                      */
  683. /*--------------------------------------------------------------------*/
  684.  
  685. static void refresh_job(const char *jobid)
  686. {
  687.    char *system;              /* System name returned by is_job()    */
  688.    char host[FILENAME_MAX];
  689.    char canon[FILENAME_MAX];
  690.    char user[FILENAME_MAX];
  691.    char sys[FILENAME_MAX];
  692.  
  693.    strcpy(canon,"C.");
  694.    strcat(canon,jobid);
  695.    strcpy( user, "uucp");
  696.    system = is_job( canon ); /* Never returns if an error occurs     */
  697.    importpath( host, canon, system );
  698.                               /* Get the local name of the file      */
  699.    open_call(host, system, NULL, user, sys, JOB_REFRESH);
  700.    touch( host );
  701.    printf("Rejuvenated job %s (%s) queued for host %s by %s\n",
  702.             jobid, host, system, user);
  703.  
  704. } /* refresh_job */
  705.  
  706. /*--------------------------------------------------------------------*/
  707. /*    o p e n _ c a l l                                               */
  708. /*                                                                    */
  709. /*    Process a call file                                             */
  710. /*--------------------------------------------------------------------*/
  711.  
  712. static CALLTYPE open_call( const char *callname,
  713.                            const char *remote,
  714.                                  struct data_queue **top,
  715.                                  char *user,
  716.                                  char *sys,
  717.                            const CALL_ACTION action )
  718. {
  719.    FILE *stream;
  720.    char buf[BUFSIZ];
  721.    struct data_queue *current;
  722.    CALLTYPE this_call = POLL_CALL;
  723.  
  724.    char host[FILENAME_MAX];
  725.    char type[FILENAME_MAX], fname[FILENAME_MAX], tname[FILENAME_MAX];
  726.    char flgs[FILENAME_MAX], dname[FILENAME_MAX];
  727.  
  728.    current = NULL;
  729.  
  730. /*--------------------------------------------------------------------*/
  731. /*                    Open the file for processing                    */
  732. /*--------------------------------------------------------------------*/
  733.  
  734.    printmsg(2,"Scanning call file \"%s\"", callname );
  735.    stream = FOPEN( callname, "r", TEXT );
  736.  
  737.    if ( stream == NULL )
  738.    {
  739.       printerr( callname );
  740.       panic();
  741.    }
  742.  
  743.    while( fgets( buf, BUFSIZ, stream) != NULL)
  744.    {
  745.       switch (*buf)
  746.       {
  747.          case '#':
  748.             break;
  749.  
  750.          case 'S':
  751.          case 'R':
  752.             *dname = '\0';
  753.             sscanf(buf, "%s %s %s %s %s %s",
  754.                          type, fname, tname, user , flgs, dname);
  755.             this_call = *type;
  756.  
  757.             if ( ! strlen( dname ) )
  758.             {
  759.                printmsg(0,"Invalid data in call file \"%s\"",callname);
  760.                panic();
  761.             }
  762.  
  763. /*--------------------------------------------------------------------*/
  764. /*    If returning job status, allocate next link in status queue     */
  765. /*--------------------------------------------------------------------*/
  766.  
  767.             if ( action == JOB_STATUS )
  768.             {
  769.               if ( current == NULL)
  770.                  *top = current = malloc( sizeof *current );
  771.               else {
  772.                  current->next_link = malloc( sizeof *current );
  773.                  current = current->next_link;
  774.               }
  775.               checkref( current );
  776.  
  777.               current->next_link = NULL;
  778.               current->type      = *buf;
  779.             } /* action */
  780.  
  781. /*--------------------------------------------------------------------*/
  782. /*    Get the UNIX and local filenames for data file to be            */
  783. /*    processed; we always report the original file name, but we      */
  784. /*    always look at and/or update the spool copy of the data; we     */
  785. /*    never alter or delete the original                              */
  786. /*--------------------------------------------------------------------*/
  787.  
  788.             if ( this_call == SEND_CALL )
  789.             {
  790.                if (equal(dname, "D.0"))
  791.                   importpath( host, fname, remote );
  792.                else
  793.                   importpath( host, dname, remote );
  794.  
  795.                switch  ( action )
  796.                {
  797.                   case JOB_STATUS:
  798.                      current->created   = stater( host,  ¤t->size);
  799.                      if ((current->created != -1) &&
  800.                          (equaln(tname ,"X.",2)))  /* Execute file?  */
  801.                      {
  802.                         open_data( host, user, sys, current->name );
  803.                         current->execute = TRUE;
  804.                      }
  805.                      else {
  806.                         current->execute = FALSE;
  807.                         strcpy( current->name , fname );
  808.                      }
  809.                      break;
  810.  
  811.                   case JOB_KILL:
  812.                      if ((current->created != -1) && !equal(dname, "D.0"))
  813.                      {
  814.                         unlink( host );
  815.                         printf("Deleted file %s (%s)\n", dname, host);
  816.                      }
  817.                      break;
  818.  
  819.                   case JOB_REFRESH:
  820.                      if ((current->created != -1) && !equal(dname, "D.0"))
  821.                         touch( host );
  822.                      break;
  823.  
  824.                   default:
  825.                      panic();
  826.                } /* switch */
  827.             }
  828.             else if ( action == JOB_STATUS )
  829.             {
  830.                current->execute = FALSE;
  831.                strcpy( current->name , fname );
  832.                current->created = stater( callname, ¤t->size);
  833.             }
  834.  
  835.             break;
  836.  
  837.          default:
  838.             printmsg(0,"Invalid line \"%s\" in call file \"%s\"",
  839.                               buf, callname);
  840.             break;
  841.       } /* switch */
  842.    } /* while */
  843.  
  844. /*--------------------------------------------------------------------*/
  845. /*                   Clean up and return to caller                    */
  846. /*--------------------------------------------------------------------*/
  847.  
  848.    if (ferror( stream ))
  849.    {
  850.       printerr( callname );
  851.       clearerr( stream );
  852.    }
  853.  
  854.    fclose( stream );
  855.    return this_call;
  856.  
  857. } /* open call */
  858.  
  859.  
  860. /*--------------------------------------------------------------------*/
  861. /*    o p e n _ d a t a                                               */
  862. /*                                                                    */
  863. /*    Process a single execute file                                   */
  864. /*--------------------------------------------------------------------*/
  865.  
  866. static void open_data(const char *file,
  867.                       char *user,
  868.                       char *sys,
  869.                       char *command)
  870. {
  871.    FILE  *data_fp;
  872.    char  data_buf[BUFSIZ];
  873.    char  *token = "(none)";
  874.    static char f_name[ FILENAME_MAX ];
  875.    size_t  bytes = 0;
  876.  
  877.    printmsg(1,"INSIDE OPEN_DATA(%s)",file);
  878.    *f_name = '\0';
  879.  
  880.    printmsg(1,"OPENING(%s)",file);
  881.    data_fp = FOPEN(file, "r", BINARY);
  882.    if(data_fp ==  NULL){
  883.       printerr( file );
  884.       panic();
  885.    }
  886.    printmsg(1,"OPEN (%s) SIZE (%d)", file, BUFSIZ);
  887.  
  888.    while(fgets(data_buf, (int) BUFSIZ, data_fp) !=  NULL){
  889.  
  890.       size_t len = strlen( data_buf );
  891.       bytes += strlen( data_buf );
  892.       if ( data_buf[ --len ] == '\n')
  893.          data_buf[ len ] = '\0';
  894.  
  895. /*--------------------------------------------------------------------*/
  896. /*                 Determine what kind of line it is                  */
  897. /*--------------------------------------------------------------------*/
  898.  
  899.       switch(data_buf[0]){
  900.          case '#':
  901.          printmsg(5,"COMMENT %s", data_buf);
  902.             break;
  903.  
  904.           case 'U':
  905.             token = strtok( data_buf + 1, WHITESPACE );
  906.             if ( token == NULL )
  907.                break;
  908.             strncpy( user, token , MAXL );
  909.             user[ MAXL-1 ] = '\0';
  910.             token = strtok( NULL , WHITESPACE );
  911.             if ( token == NULL )
  912.                break;
  913.             strncpy(  sys, token , MAXL);
  914.             sys[ MAXL-1 ] = '\0';
  915.             printmsg(1,"SYSTEM = (%s), USER = (%s)",sys,user);
  916.             break;
  917.  
  918.          case 'F':
  919.             printmsg(5,"File %s", data_buf);
  920.             token = strtok( data_buf + 1 , WHITESPACE );
  921.             strncpy( f_name, token , FILENAME_MAX );
  922.             f_name[ FILENAME_MAX - 1 ] = '\0';
  923.             break;
  924.  
  925.          case 'I':
  926.             printmsg(5,"Input %s", data_buf);
  927.             break;
  928.  
  929.          case 'C':
  930.             printmsg(1,"Command %s", data_buf);
  931.             token = strchr(data_buf,' ') + 1;
  932.             strncpy(command, token , FILENAME_MAX - 1);
  933.             command[ FILENAME_MAX - 1] = '\0';
  934.             break;
  935.          default:
  936.             printmsg(1,"UNKNOWN LINE %s", data_buf);
  937.             break;
  938.       } /* switch */
  939.    } /* while */
  940.  
  941.    printmsg(1,"CLOSED (%s), bytes = %d, d file = %s", file, bytes, token);
  942.    fclose(data_fp);
  943. } /* open_data */
  944.  
  945. /*--------------------------------------------------------------------*/
  946. /*    p r i n t _ a l l                                               */
  947. /*                                                                    */
  948. /*    Print all the information related to a job in the spool         */
  949. /*--------------------------------------------------------------------*/
  950.  
  951. static void print_all(       char *job,
  952.                              struct data_queue *current ,
  953.                        const char *user,
  954.                        const char *sys )
  955. {
  956.    char blanks[FILENAME_MAX];
  957.    size_t subscript = 0;
  958.  
  959. /*--------------------------------------------------------------------*/
  960. /*  Create a blank buffer for printing all of the first line of data  */
  961. /*--------------------------------------------------------------------*/
  962.  
  963.    while ( job[subscript] != '\0')
  964.       blanks[subscript++] = ' ';
  965.    blanks[ subscript ] = '\0';
  966.  
  967. /*--------------------------------------------------------------------*/
  968. /*                Loop through the files for this job                 */
  969. /*--------------------------------------------------------------------*/
  970.  
  971.    while ( current != NULL )
  972.    {
  973.       struct data_queue *save_data = current->next_link;
  974.       if ( current->execute)
  975.          printf("%s %s %c %-8.8s %-8.8s %s\n", job,
  976.                      dater( current->created, NULL ),
  977.                      current->type,
  978.                      sys, user, current->name );
  979.       else
  980.          printf("%s %s %c %-8.8s %-8.8s %ld %s\n", job,
  981.                      dater( current->created, NULL ),
  982.                      current->type,
  983.                      sys, user, current->size , current->name );
  984.  
  985.       job = blanks;           /* Don't print job name more than once */
  986.       free( current );        /* Release the abused storage          */
  987.       current = save_data;
  988.    } /* while */
  989.  
  990. } /* print_all */
  991.  
  992. /*--------------------------------------------------------------------*/
  993. /*    i s _ j o b                                                     */
  994. /*                                                                    */
  995. /*    Determine if a file is a valid job                              */
  996. /*                                                                    */
  997. /*    The task of finding a proper job made is harder by the fact     */
  998. /*    UUPC inserts all these jobs into separate directories.  We      */
  999. /*    brute force it, checking all known hosts for the wayward        */
  1000. /*    file.  This is relatively cheap, since most hosts will fail     */
  1001. /*    on the host name and thus we never look on disk.  Since job     */
  1002. /*    sequence numbers are global across UUPC/extende, a false hit    */
  1003. /*    on the host doesn't hurt because the job number will never      */
  1004. /*    match.                                                          */
  1005. /*--------------------------------------------------------------------*/
  1006.  
  1007. static char *is_job(const char *callfile)
  1008. {
  1009.    struct HostTable *hostp;
  1010.    char host[FILENAME_MAX];
  1011.    boolean hit = FALSE;
  1012.  
  1013. /*--------------------------------------------------------------------*/
  1014. /*                  Get the first system to process                   */
  1015. /*--------------------------------------------------------------------*/
  1016.  
  1017.       hostp = nexthost( TRUE );
  1018.  
  1019. /*--------------------------------------------------------------------*/
  1020. /*              Begin loop to display local system                    */
  1021. /*--------------------------------------------------------------------*/
  1022.  
  1023.    while(hostp != BADHOST )
  1024.    {
  1025.       if (equaln(hostp->hostname, &callfile[2],
  1026.                 min( strlen( hostp->hostname ), HOSTLEN)))
  1027.                               /* Right host?                         */
  1028.       {                       /* Maybe --> Look for the file         */
  1029.          importpath( host, callfile, hostp->hostname);
  1030.          if ( !access( host, 0 ))   /* Does the host file exist?     */
  1031.             return hostp->hostname; /* Yes --> Return success        */
  1032.          hit = TRUE;
  1033.       }
  1034.  
  1035.       hostp = nexthost( FALSE );
  1036.    } /* while */
  1037.  
  1038. /*--------------------------------------------------------------------*/
  1039. /*          We didn't get the host; report an error and exit          */
  1040. /*--------------------------------------------------------------------*/
  1041.  
  1042.    if ( hit )
  1043.       printf("Unable to locate call file %s -- run uustat -a\n", callfile );
  1044.    else
  1045.       printf("Unable to locate call file %s -- hostname may be incorrect\n",
  1046.                callfile);
  1047.    exit(1);
  1048.    return NULL;                  /* Make C compiler happy */
  1049. } /* is_job */
  1050.  
  1051. /*--------------------------------------------------------------------*/
  1052. /*    t o u c h                                                       */
  1053. /*                                                                    */
  1054. /*    Update list modified time for a file by opening and closing it  */
  1055. /*--------------------------------------------------------------------*/
  1056.  
  1057. static void touch( const char *fname)
  1058. {
  1059.    printf("touch: function not available.  Parameter was \"%s\"\n",
  1060.             fname);
  1061. } /* touch */
  1062.  
  1063.  
  1064. /*--------------------------------------------------------------------*/
  1065. /*    u s a g e                                                       */
  1066. /*                                                                    */
  1067. /*    Report how to use program                                       */
  1068. /*--------------------------------------------------------------------*/
  1069.  
  1070. static void usage( void )
  1071. {
  1072.    fputs("Usage:\tuustat\t[-m | -q | -a] [-u userid | all ] [-s system | all] [-x debug]\n\
  1073. \t\t[-k | -r ] jobid [-x debug]\n\
  1074. \t\t[-P system | all] [-x debug]\n\n\
  1075. \tDefault is to display jobs queued by current user\n\
  1076. \t-a\t\tDisplays all jobs for all users\n\
  1077. \t-k jobid\tKills job \"jobid\"\n\
  1078. \t-m\t\tDisplays accessability of known systems\n\
  1079. \t-q\t\tDisplays summary of queues for systems with work\n\
  1080. \t-r jobid\tRefreshes timestamps on job \"jobid\"\n\
  1081. \t-s system\tRestricts output to system (Ignored by -k -P -r)\n\
  1082. \t-u userid\tRestricts output to userid (Ignored by -k -m -P -q -r)\n\
  1083. \t-P system\tQueues dummy job for system \"system\" to force poll\n\
  1084. \t-x debuglevel",
  1085.             stdout );
  1086.    exit(1);
  1087. } /* usage */
  1088.