| /* cut.c - Cut from a file. |
| * |
| * Copyright 2012 Ranjan Kumar <ranjankumar.bth@gmail.com> |
| * Copyright 2012 Kyungwan Han <asura321@gmail.com> |
| * |
| * http://pubs.opengroup.org/onlinepubs/9699919799/utilities/cut.html |
| * |
| * TODO: cleanup |
| |
| USE_CUT(NEWTOY(cut, "b:|c:|f:|d:sn[!cbf]", TOYFLAG_USR|TOYFLAG_BIN)) |
| |
| config CUT |
| bool "cut" |
| default y |
| help |
| usage: cut OPTION... [FILE]... |
| |
| Print selected parts of lines from each FILE to standard output. |
| |
| -b LIST select only these bytes from LIST. |
| -c LIST select only these characters from LIST. |
| -f LIST select only these fields. |
| -d DELIM use DELIM instead of TAB for field delimiter. |
| -s do not print lines not containing delimiters. |
| -n don't split multibyte characters (Ignored). |
| */ |
| #define FOR_cut |
| #include "toys.h" |
| |
| GLOBALS( |
| char *delim; |
| char *flist; |
| char *clist; |
| char *blist; |
| |
| void *slist_head; |
| unsigned nelem; |
| void (*do_cut)(int fd); |
| ) |
| |
| struct slist { |
| struct slist *next; |
| int start, end; |
| }; |
| |
| static void add_to_list(int start, int end) |
| { |
| struct slist *current, *head_ref, *temp1_node; |
| |
| head_ref = TT.slist_head; |
| temp1_node = xzalloc(sizeof(struct slist)); |
| temp1_node->start = start; |
| temp1_node->end = end; |
| |
| /* Special case for the head end */ |
| if (!head_ref || head_ref->start >= start) { |
| temp1_node->next = head_ref; |
| head_ref = temp1_node; |
| } else { |
| /* Locate the node before the point of insertion */ |
| current = head_ref; |
| while (current->next && current->next->start < temp1_node->start) |
| current = current->next; |
| temp1_node->next = current->next; |
| current->next = temp1_node; |
| } |
| TT.slist_head = head_ref; |
| } |
| |
| // parse list and add to slist. |
| static void parse_list(char *list) |
| { |
| for (;;) { |
| char *ctoken = strsep(&list, ","), *dtoken; |
| int start = 0, end = INT_MAX; |
| |
| if (!ctoken) break; |
| if (!*ctoken) continue; |
| |
| //Get start position. |
| if (*(dtoken = strsep(&ctoken, "-"))) { |
| start = atolx_range(dtoken, 0, INT_MAX); |
| start = (start?(start-1):start); |
| } |
| |
| //Get end position. |
| if (!ctoken) end = -1; //case e.g. 1,2,3 |
| else if (*ctoken) {//case e.g. N-M |
| end = atolx_range(ctoken, 0, INT_MAX); |
| if (!end) end = INT_MAX; |
| end--; |
| if(end == start) end = -1; |
| } |
| add_to_list(start, end); |
| TT.nelem++; |
| } |
| //if list is missing in command line. |
| if (!TT.nelem) error_exit("missing positions list"); |
| } |
| |
| /* |
| * retrive data from the file/s. |
| */ |
| static void get_data(void) |
| { |
| char **argv = toys.optargs; //file name. |
| toys.exitval = EXIT_SUCCESS; |
| |
| if(!*argv) TT.do_cut(0); //for stdin |
| else { |
| for(; *argv; ++argv) { |
| if(strcmp(*argv, "-") == 0) TT.do_cut(0); //for stdin |
| else { |
| int fd = open(*argv, O_RDONLY, 0); |
| if(fd < 0) {//if file not present then continue with other files. |
| perror_msg("%s", *argv); |
| continue; |
| } |
| TT.do_cut(fd); |
| xclose(fd); |
| } |
| } |
| } |
| } |
| |
| // perform cut operation on the given delimiter. |
| static void do_fcut(int fd) |
| { |
| char *buff, *pfield = 0, *delimiter = TT.delim; |
| |
| for (;;) { |
| unsigned cpos = 0; |
| int start, ndelimiters = -1; |
| int nprinted_fields = 0; |
| struct slist *temp_node = TT.slist_head; |
| |
| free(pfield); |
| pfield = 0; |
| |
| if (!(buff = get_line(fd))) break; |
| |
| //does line have any delimiter?. |
| if (strrchr(buff, (int)delimiter[0]) == NULL) { |
| //if not then print whole line and move to next line. |
| if (!(toys.optflags & FLAG_s)) xputs(buff); |
| continue; |
| } |
| |
| pfield = xzalloc(strlen(buff) + 1); |
| |
| if (temp_node) { |
| //process list on each line. |
| while (cpos < TT.nelem && buff) { |
| if (!temp_node) break; |
| start = temp_node->start; |
| do { |
| char *field = 0; |
| |
| //count number of delimeters per line. |
| while (buff) { |
| if (ndelimiters < start) { |
| ndelimiters++; |
| field = strsep(&buff, delimiter); |
| } else break; |
| } |
| //print field (if not yet printed). |
| if (!pfield[ndelimiters]) { |
| if (ndelimiters == start) { |
| //put delimiter. |
| if (nprinted_fields++ > 0) xputc(delimiter[0]); |
| if (field) fputs(field, stdout); |
| //make sure this field won't print again. |
| pfield[ndelimiters] = (char) 0x23; //put some char at this position. |
| } |
| } |
| start++; |
| if ((temp_node->end < 0) || !buff) break; |
| } while(start <= temp_node->end); |
| temp_node = temp_node->next; |
| cpos++; |
| } |
| } |
| xputc('\n'); |
| } |
| } |
| |
| // perform cut operation char or byte. |
| static void do_bccut(int fd) |
| { |
| char *buff; |
| |
| while ((buff = get_line(fd)) != NULL) { |
| unsigned cpos = 0; |
| int buffln = strlen(buff); |
| char *pfield = xzalloc(buffln + 1); |
| struct slist *temp_node = TT.slist_head; |
| |
| if (temp_node != NULL) { |
| while (cpos < TT.nelem) { |
| int start; |
| |
| if (!temp_node) break; |
| start = temp_node->start; |
| while (start < buffln) { |
| //to avoid duplicate field printing. |
| if (pfield[start]) { |
| if (++start <= temp_node->end) continue; |
| temp_node = temp_node->next; |
| break; |
| } else { |
| //make sure this field won't print again. |
| pfield[start] = (char) 0x23; //put some char at this position. |
| xputc(buff[start]); |
| } |
| if (++start > temp_node->end) { |
| temp_node = temp_node->next; |
| break; |
| } |
| } |
| cpos++; |
| } |
| xputc('\n'); |
| } |
| free(pfield); |
| pfield = NULL; |
| } |
| } |
| |
| void cut_main(void) |
| { |
| char delimiter = '\t'; //default delimiter. |
| char *list; |
| |
| TT.nelem = 0; |
| TT.slist_head = NULL; |
| |
| //Get list and assign the function. |
| if (toys.optflags & FLAG_f) { |
| list = TT.flist; |
| TT.do_cut = do_fcut; |
| } else if (toys.optflags & FLAG_c) { |
| list = TT.clist; |
| TT.do_cut = do_bccut; |
| } else { |
| list = TT.blist; |
| TT.do_cut = do_bccut; |
| } |
| |
| if (toys.optflags & FLAG_d) { |
| //delimiter must be 1 char. |
| if(TT.delim[0] && TT.delim[1]) |
| perror_exit("the delimiter must be a single character"); |
| delimiter = TT.delim[0]; |
| } |
| |
| if(!(toys.optflags & FLAG_d) && (toys.optflags & FLAG_f)) { |
| TT.delim = xzalloc(2); |
| TT.delim[0] = delimiter; |
| } |
| |
| //when field is not specified, cutting has some special handling. |
| if (!(toys.optflags & FLAG_f)) { |
| if (toys.optflags & FLAG_s) |
| perror_exit("suppressing non-delimited lines operating on fields"); |
| if (delimiter != '\t') |
| perror_exit("an input delimiter may be specified only when operating on fields"); |
| } |
| |
| parse_list(list); |
| get_data(); |
| if (!(toys.optflags & FLAG_d) && (toys.optflags & FLAG_f)) { |
| free(TT.delim); |
| TT.delim = NULL; |
| } |
| llist_traverse(TT.slist_head, free); |
| } |