Home | History | Annotate | Download | only in posix
      1 /* paste.c - Replace newlines
      2  *
      3  * Copyright 2012 Felix Janda <felix.janda (at) posteo.de>
      4  *
      5  * http://pubs.opengroup.org/onlinepubs/9699919799/utilities/paste.html
      6  *
      7 USE_PASTE(NEWTOY(paste, "d:s", TOYFLAG_BIN))
      8 
      9 config PASTE
     10   bool "paste"
     11   default y
     12   help
     13     usage: paste [-s] [-d list] [file...]
     14 
     15     Replace newlines in files.
     16 
     17     -d list    list of delimiters to separate lines
     18     -s         process files sequentially instead of in parallel
     19 
     20     By default print corresponding lines separated by <tab>.
     21 */
     22 #define FOR_paste
     23 #include "toys.h"
     24 
     25 GLOBALS(
     26   char *delim;
     27 )
     28 
     29 void paste_main(void)
     30 {
     31   char *p, *buf = toybuf, **args = toys.optargs;
     32   size_t ndelim = 0;
     33   int i, j, c;
     34 
     35   // Process delimiter list
     36   // TODO: Handle multibyte characters
     37   if (!(toys.optflags & FLAG_d)) TT.delim = "\t";
     38   for (p = TT.delim; *p; p++, buf++, ndelim++) {
     39     if (*p == '\\') {
     40       p++;
     41       if (-1 == (i = stridx("nt\\0", *p)))
     42         error_exit("bad delimiter: \\%c", *p);
     43       *buf = "\n\t\\\0"[i];
     44     } else *buf = *p;
     45   }
     46   *buf = 0;
     47 
     48   if (toys.optflags & FLAG_s) { // Sequential
     49     FILE *f;
     50 
     51     for (; *args; args++) {
     52       if ((*args)[0] == '-' && !(*args)[1]) f = stdin;
     53       else if (!(f = fopen(*args, "r"))) perror_exit_raw(*args);
     54       for (i = 0, c = 0; c != EOF;) {
     55         switch(c = getc(f)) {
     56         case '\n':
     57           putchar(toybuf[i++ % ndelim]);
     58         case EOF:
     59           break;
     60         default:
     61           putchar(c);
     62         }
     63       }
     64       if (f != stdin) fclose(f);
     65       putchar('\n');
     66     }
     67   } else { // Parallel
     68     // Need to be careful not to print an extra line at the end
     69     FILE **files;
     70     int anyopen = 1;
     71 
     72     files = (FILE**)(buf + 1);
     73     for (; *args; args++, files++) {
     74       if ((*args)[0] == '-' && !(*args)[1]) *files = stdin;
     75       else if (!(*files = fopen(*args, "r"))) perror_exit_raw(*args);
     76     }
     77     while (anyopen) {
     78       anyopen = 0;
     79       for (i = 0; i < toys.optc; i++) {
     80         FILE **f = (FILE**)(buf + 1) + i;
     81 
     82         if (*f) for (;;) {
     83           c = getc(*f);
     84           if (c != EOF) {
     85             if (!anyopen++) for (j = 0; j < i; j++) putchar(toybuf[j % ndelim]);
     86             if (c != '\n') putchar(c);
     87             else break;
     88           }
     89           else {
     90             if (*f != stdin) fclose(*f);
     91             *f = 0;
     92             break;
     93           }
     94         }
     95         if (anyopen) putchar((i + 1 == toys.optc) ? toybuf[i % ndelim] : '\n');
     96       }
     97     }
     98   }
     99 }
    100