home *** CD-ROM | disk | FTP | other *** search
/ rtsi.com / 2014.01.www.rtsi.com.tar / www.rtsi.com / OS9 / OSK / APPS / elm.lzh / ELM / SRC / FILE.C < prev    next >
Text File  |  1991-01-11  |  11KB  |  346 lines

  1.  
  2. static char rcsid[] = "@(#)$Id: file.c,v 4.1 90/04/28 22:43:02 syd Exp $";
  3.  
  4. /*******************************************************************************
  5.  *  The Elm Mail System  -  $Revision: 4.1 $   $State: Exp $
  6.  *
  7.  *             Copyright (c) 1986, 1987 Dave Taylor
  8.  *             Copyright (c) 1988, 1989, 1990 USENET Community Trust
  9.  *******************************************************************************
  10.  * Bug reports, patches, comments, suggestions should be sent to:
  11.  *
  12.  *    Syd Weinstein, Elm Coordinator
  13.  *    elm@DSI.COM            dsinc!elm
  14.  *
  15.  *******************************************************************************
  16.  * $Log:    file.c,v $
  17.  * Revision 4.1  90/04/28  22:43:02  syd
  18.  * checkin of Elm 2.3 as of Release PL0
  19.  * 
  20.  *
  21.  ******************************************************************************/
  22.  
  23. /** File I/O routines, mostly the save to file command...
  24.  
  25. **/
  26.  
  27. #ifdef PWDINSYS
  28. #  include <sys/pwd.h>
  29. #else
  30. #  include <pwd.h>
  31. #endif
  32.  
  33. #include "headers.h"
  34. #include <ctype.h>
  35. #include <errno.h>
  36.  
  37. #ifdef BSD
  38. #undef tolower
  39. #endif
  40.  
  41. extern int errno;
  42.  
  43. char *error_name(), *error_description(), *strcpy(), *getenv(), *nameof();
  44. unsigned long sleep();
  45.  
  46. int
  47. save(redraw, silently, delete)
  48. int *redraw, silently, delete;
  49. {
  50.     /** Save all tagged messages + current in a folder.  If no messages
  51.         are tagged, save the current message instead!  This routine
  52.         will return ZERO if the operation failed.
  53.         'redraw' is set to TRUE iff we use the '?' and mess up
  54.         the screen.  Pretty reasonable, eh?  If "silently" is set,
  55.         then don't output the "D" character upon marking for
  56.         deletion...
  57.         If delete is set, then delete the saved messages, else
  58.         we are just copying the messages without deletion.
  59.     **/
  60.  
  61.     register int tagged = 0, i, oldstat, appending = 0;
  62.     int mesgnum;    /* message whose address is used for save-by-name fn */
  63.     char filename[SLEN], address[LONG_STRING], buffer[LONG_STRING];
  64.     static char helpmsg[LONG_STRING];
  65.     FILE *save_file;
  66.  
  67.     oldstat = headers[current-1]->status;    /* remember */
  68.     *redraw = FALSE;
  69.  
  70.     for (i=0; i < message_count; i++) {
  71.       if (ison(headers[i]->status, TAGGED)) {
  72.         if(!tagged)
  73.           mesgnum = i;    /* first tagged msg -  use this one for
  74.                  * save-by-name folder name */
  75.         tagged++;
  76.       }
  77.     }
  78.  
  79.     if (tagged == 0) {
  80.       mesgnum = current-1;    /* use this one for save-by-name folder name */
  81.       tagged = 1;
  82.       setit(headers[current-1]->status, TAGGED);
  83.     }
  84.  
  85.     dprint(4, (debugfile, "%d message%s tagged for saving (save)\n", tagged,
  86.         plural(tagged)));
  87.  
  88.     while (1) {
  89.  
  90.       PutLine2(LINES-2, 0, "%s message%s to: ",
  91.           (delete ? "Save" : "Copy"), plural(tagged));
  92.  
  93.       if (save_by_name) {
  94.         /** build default filename to save to **/
  95.         get_return(address, mesgnum);
  96.         get_return_name(address, buffer, TRUE);
  97.         sprintf(filename, "=%s", buffer);
  98.       }
  99.       else
  100.         filename[0] = '\0';
  101.  
  102.       if (tagged > 1)
  103.         optionally_enter(filename, LINES-2, 19, FALSE, FALSE);
  104.       else    
  105.         optionally_enter(filename, LINES-2, 18, FALSE, FALSE);
  106.   
  107.  
  108.       if (strlen(filename) == 0) {  /** <return> means 'cancel', right? **/
  109.         headers[current-1]->status = oldstat;    /* BACK! */
  110.         return(0);
  111.       }
  112.      
  113.       if (strcmp(filename,"?") == 0) {    /* user asked for listing */
  114.         *redraw = TRUE;    /* set the flag so we know what to do later */
  115.         if(!*helpmsg) {    /* format helpmsg if not yet done */
  116.  
  117.           strcpy(helpmsg, "\l\r\l\rEnter: <nothing> to not ");
  118.           strcat(helpmsg, (delete ? "save" : "copy"));
  119.           strcat(helpmsg, " your message");
  120.           strcat(helpmsg, (plural(tagged) ? "s" : ""));
  121.           strcat(helpmsg, "\l\r       '>' to ");
  122.           strcat(helpmsg, (delete ? "save" : "copy"));
  123.           strcat(helpmsg, " your message");
  124.           strcat(helpmsg, (plural(tagged) ? "s" : ""));
  125.           strcat(helpmsg, " to your \"received\" folder (");
  126.           strcat(helpmsg, nameof(recvd_mail));
  127.           strcat(helpmsg, ")\l\r       '<' to ");
  128.           strcat(helpmsg, (delete ? "save" : "copy"));
  129.           strcat(helpmsg, " your message");
  130.           strcat(helpmsg, (plural(tagged) ? "s" : ""));
  131.           strcat(helpmsg, " to your \"sent\" folder (");
  132.           strcat(helpmsg, nameof(sent_mail));
  133.           strcat(helpmsg, ") \l\r       a filename");
  134.           strcat(helpmsg, " (leading '=' denotes your folder directory ");
  135.           strcat(helpmsg, folders);
  136.           strcat(helpmsg, ").\l\r");
  137.         }
  138.  
  139.         list_folders(4, helpmsg);
  140.         continue;
  141.       }
  142.  
  143.       /* else - got a folder name - check it out */
  144.       if (! expand_filename(filename, TRUE)) {
  145.         dprint(2, (debugfile,
  146.           "Error: Failed on expansion of filename %s (save)\n", 
  147.           filename));
  148.         continue;
  149.       }
  150.       if ((errno = can_open(filename, "a"))) {
  151.         error2("Cannot %s message to folder %s!",
  152.           delete ? "save":"copy", filename);
  153.         continue;
  154.       }
  155.       break;    /* got a valid filename */
  156.     }
  157.  
  158.     save_file_stats(filename);
  159.  
  160.     if (access(filename,ACCESS_EXISTS)== 0)     /* already there!! */
  161.       appending = 1;
  162.       
  163.     dprint(4,(debugfile, "Saving mail to folder '%s'...\n", filename));
  164.  
  165.     if ((save_file = fopen(filename,"a")) == NULL) {
  166.       dprint(2, (debugfile,
  167.         "Error: couldn't append to specified folder %s (save)\n", 
  168.         filename));
  169.       error1("Couldn't append to folder %s!", filename);
  170.       headers[current-1]->status = oldstat;    /* BACK! */
  171.       return(0); 
  172.     }
  173.  
  174.     /* if we need a redraw that means index screen no longer present
  175.      * so whatever silently was, now it's true - we can't show those
  176.      * delete markings.
  177.      */
  178.     if(*redraw) silently = TRUE;
  179.  
  180.     for (i=0; i < message_count; i++)     /* save each tagged msg */
  181.       if (headers[i]->status & TAGGED)
  182.         save_message(i, filename, save_file, (tagged > 1), appending++, 
  183.              silently, delete);
  184.  
  185.     fclose(save_file);
  186.  
  187.     restore_file_stats(filename);
  188.  
  189.     if (tagged > 1)
  190.       error2("Message%s %s.", plural(tagged), delete ? "saved": "copied");
  191.     return(1);
  192. }
  193.  
  194. int
  195. save_message(number, filename, fd, pause, appending, silently, delete)
  196. int number, pause, appending, silently, delete;
  197. char *filename;
  198. FILE *fd;
  199. {
  200.     /** Save an actual message to a folder.  This is called by 
  201.         "save()" only!  The parameters are the message number,
  202.         and the name and file descriptor of the folder to save to.
  203.         If 'pause' is true, a sleep(2) will be done after the
  204.         saved message appears on the screen...
  205.         'appending' is only true if the folder already exists 
  206.         If 'delete' is true, mark the message for deletion.
  207.     **/
  208.  
  209.     register int save_current, is_new;
  210.     
  211.     dprint(4, (debugfile, "\tSaving message %d to folder...\n", number));
  212.  
  213.     save_current = current;
  214.     current = number+1;
  215.  
  216.     /* change status from NEW before copy and reset to what it was
  217.      * so that copy doesn't look new, but we can preserve new status
  218.      * of message in this mailfile. This is important because if
  219.      * user does a resync, we don't want NEW status to be lost.
  220.      * I.e. NEW becomes UNREAD when we "really" leave a mailfile.
  221.      */
  222.     if(is_new = ison(headers[number]->status, NEW))
  223.       clearit(headers[number]->status, NEW);
  224.     copy_message("", fd, FALSE, FALSE, TRUE, FALSE);
  225.     if(is_new)
  226.       setit(headers[number]->status, NEW);
  227.     current = save_current;
  228.  
  229.     if (delete)
  230.       setit(headers[number]->status, DELETED); /* deleted, but ...   */
  231.     clearit(headers[number]->status, TAGGED);  /* not tagged anymore */
  232.  
  233.     if (appending)
  234.       error2("Message %d appended to folder %s.", number+1, filename);
  235.     else
  236.       error3("Message %d %s to folder %s.", number+1,
  237.          delete ? "saved" : "copied", filename);
  238.  
  239.     if (! silently)
  240.       show_new_status(number);    /* update screen, if needed */
  241.  
  242.     if (pause && (!silently) && (!appending))
  243.       sleep(2);
  244. }
  245.  
  246. int
  247. expand_filename(filename, use_cursor_control)
  248. char *filename;
  249. int use_cursor_control;
  250. {
  251.     /** Expands    ~/    to the current user's home directory
  252.             ~user/    to the home directory of "user"
  253.             =,+,%    to the user's folder's directory
  254.             !    to the user's incoming mailbox
  255.             >    to the user's received folder
  256.             <    to the user's sent folder
  257.             shell variables (begun with $)
  258.  
  259.         Returns     1    upon proper expansions
  260.             0    upon failed expansions
  261.      **/
  262.  
  263.     char temp_filename[SLEN], varname[SLEN],
  264.         env_value[SLEN], logname[SLEN], *ptr;
  265.     register int iindex;
  266.     struct passwd *pass, *getpwnam();
  267.     char *getenv();
  268.  
  269.     ptr = filename;
  270.     while (*ptr == ' ') ptr++;    /* leading spaces GONE! */
  271.     strcpy(temp_filename, ptr);
  272.  
  273.     /** New stuff - make sure no illegal char as last **/
  274.     if (lastch(temp_filename) == '\n' || lastch(temp_filename) == '\r')
  275.       lastch(temp_filename) = '\0';
  276.       
  277.     /** Strip off any trailing backslashes **/
  278.     while (lastch(temp_filename) == '\\')
  279.         lastch(temp_filename) = '\0';
  280.  
  281.     if (temp_filename[0] == '~') {
  282.       if(temp_filename[1] == '/')
  283.         sprintf(filename, "%s%s%s",
  284.           home, (lastch(home) != '/' ? "/" : ""), &temp_filename[2]);
  285.       else {
  286.         for(ptr = &temp_filename[1], iindex = 0; *ptr && *ptr != '/'; ptr++, iindex++)
  287.           logname[iindex] = *ptr;
  288.         logname[iindex] = '\0';
  289.         if((pass = getpwnam(logname)) == NULL) {
  290.           dprint(3,(debugfile, 
  291.               "Error: Can't get home directory for %s (%s)\n",
  292.               logname, "expand_filename"));
  293.           if(use_cursor_control)
  294.         error1("Don't know what the home directory of \"%s\" is!",
  295.             logname);
  296.           else
  297.         printf(
  298.             "\l\rDon't know what the home directory of \"%s\" is!\l\r",
  299.             logname);
  300.           return(0);
  301.         }
  302.         sprintf(filename, "%s%s", pass->pw_dir, ptr);
  303.       }
  304.  
  305.     }
  306.     else if (temp_filename[0] == '=' || temp_filename[0] == '+' || 
  307.           temp_filename[0] == '%') {
  308.       sprintf(filename, "%s%s%s", folders, 
  309.         (temp_filename[1] != '/' && lastch(folders) != '/')? "/" : "",
  310.           &temp_filename[1]);
  311.     }
  312.     else if (temp_filename[0] == '$') {    /* env variable! */
  313.       for(ptr = &temp_filename[1], iindex = 0; isalnum(*ptr); ptr++, iindex++)
  314.         varname[iindex] = *ptr;
  315.       varname[iindex] = '\0';
  316.  
  317.       env_value[0] = '\0';            /* null string for strlen! */
  318.       if (getenv(varname) != NULL)
  319.         strcpy(env_value, getenv(varname));
  320.  
  321.       if (strlen(env_value) == 0) {
  322.         dprint(3,(debugfile, 
  323.             "Error: Can't expand environment variable $%s (%s)\n",
  324.             varname, "expand_filename"));
  325.         if(use_cursor_control)
  326.           error1("Don't know what the value of $%s is!", varname);
  327.         else
  328.           printf("\l\rDon't know what the value of $%s is!\l\r", varname);
  329.         return(0);
  330.       }
  331.  
  332.       sprintf(filename, "%s%s%s", env_value, 
  333.         (*ptr != '/' && lastch(env_value) != '/')? "/" : "", ptr);
  334.  
  335.     } else if (strcmp(temp_filename, "!") == 0) {
  336.       strcpy(filename, defaultfile);
  337.     } else if (strcmp(temp_filename, ">") == 0) {
  338.       strcpy(filename, recvd_mail);
  339.     } else if (strcmp(temp_filename, "<") == 0) {
  340.       strcpy(filename, sent_mail);
  341.     } else
  342.       strcpy(filename, temp_filename);
  343.       
  344.     return(1);
  345. }
  346.