textselect

Interactively select lines and pipe it to a command
Log | Files | Refs | README | LICENSE

commit 9005c2eaffbd3438cb94ccda7dbbd401a4beef7c
parent 152cfc65daccf3ed64097f30af1848abb86485fe
Author: Friedel Schön <[email protected]>
Date:   Tue,  6 Aug 2024 14:45:55 +0200

xargs functionality

Diffstat:
Mtextselect.1 | 5++++-
Mtextselect.c | 410++++++++++++++++++++++++++++++++++++++++++++-----------------------------------
2 files changed, 231 insertions(+), 184 deletions(-)

diff --git a/textselect.1 b/textselect.1 @@ -16,7 +16,10 @@ is a utility that allows users to interactively select lines from a text file us Display usage information and exit. .TP .B \-v -Invert the selection of lines (selected lines become unselected and vice versa). +Invert the selection of lines. +.TP +.B \-x +Call command with lines as arguments. .TP .B \-o \fIoutput\fP Specify an output file to save the selected lines. If not specified, the selected lines are printed to stdout. diff --git a/textselect.c b/textselect.c @@ -5,268 +5,312 @@ #include <stdio.h> #include <stdlib.h> #include <string.h> -#include <unistd.h> #include <sys/wait.h> +#include <unistd.h> #define READBUFFER 512 #define BUFFERGROW 512 #define NORETURN __attribute__((noreturn)) -char *buffer = NULL; -size_t buffer_ptr = 0; -size_t buffer_alloc = 0; -int line_count = 0; -bool *chosen_lines = NULL; -bool invert_chosen = false; -char *argv0; +char* buffer = NULL; +size_t buffer_ptr = 0; +size_t buffer_alloc = 0; +int line_count = 0; +bool* chosen_lines = NULL; +bool invert_chosen = false; +char* argv0; /** * @brief Handles error messages and exits the program. - * + * * @param message Error message to print. */ -void die(const char *message) { - perror(message); - exit(EXIT_FAILURE); +void die(const char* message) { + perror(message); + exit(EXIT_FAILURE); } /** * @brief Allocates memory for the buffer, growing it by BUFFERGROW bytes. */ void grow_buffer(void) { - char *new_buf; - if ((new_buf = realloc(buffer, buffer_alloc += BUFFERGROW)) == NULL) { - die("allocating buffer"); - } - buffer = new_buf; + char* new_buf; + if ((new_buf = realloc(buffer, buffer_alloc += BUFFERGROW)) == NULL) { + die("allocating buffer"); + } + buffer = new_buf; } /** * @brief Loads a file into memory, storing its lines in the buffer. - * + * * @param filename Name of the file to load, or NULL to read from stdin. */ -void load_file(const char *filename) { - static char readbuf[READBUFFER]; - ssize_t nread; - int fd; - - fd = (filename == NULL) ? STDIN_FILENO : open(filename, O_RDONLY); - if (fd == -1) die("Failed to open file"); - - while ((nread = read(fd, readbuf, sizeof(readbuf))) > 0) { - for (ssize_t i = 0; i < nread; i++) { - if (buffer_ptr == buffer_alloc) grow_buffer(); - - if (readbuf[i] == '\n') { - if (buffer[buffer_ptr - 1] != '\0') { - buffer[buffer_ptr++] = '\0'; - line_count++; - } - } else { - buffer[buffer_ptr++] = readbuf[i]; - } - } - } - - buffer[buffer_ptr++] = '\0'; - if (fd > 2) close(fd); - - chosen_lines = calloc(line_count, sizeof(bool)); - if (chosen_lines == NULL) die("allocating chosen_lines"); +void load_file(const char* filename) { + static char readbuf[READBUFFER]; + ssize_t nread; + int fd; + + fd = (filename == NULL) ? STDIN_FILENO : open(filename, O_RDONLY); + if (fd == -1) die("Failed to open file"); + + while ((nread = read(fd, readbuf, sizeof(readbuf))) > 0) { + for (ssize_t i = 0; i < nread; i++) { + if (buffer_ptr == buffer_alloc) grow_buffer(); + + if (readbuf[i] == '\n') { + if (buffer[buffer_ptr - 1] != '\0') { + buffer[buffer_ptr++] = '\0'; + line_count++; + } + } else { + buffer[buffer_ptr++] = readbuf[i]; + } + } + } + + buffer[buffer_ptr++] = '\0'; + if (fd > 2) close(fd); + + chosen_lines = calloc(line_count, sizeof(bool)); + if (chosen_lines == NULL) die("allocating chosen_lines"); } /** * @brief Retrieves a line from the buffer. - * + * * @param line Line number to retrieve. * @return Pointer to the start of the line. */ -char *get_line(size_t line) { - size_t current = 0; - if (line == 0) return buffer; - for (size_t i = 0; i < buffer_ptr; i++) { - if (buffer[i] == '\0') { - current++; - if (current == line) return &buffer[i + 1]; - } - } - return NULL; +char* get_line(size_t line) { + size_t current = 0; + if (line == 0) return buffer; + for (size_t i = 0; i < buffer_ptr; i++) { + if (buffer[i] == '\0') { + current++; + if (current == line) return &buffer[i + 1]; + } + } + return NULL; } /** * @brief Prints lines to the ncurses window. - * + * * @param win ncurses window to print to. * @param current_line Line currently selected. * @param start_line Line to start printing from. * @param max_y Maximum number of lines to print. */ -void draw_lines(WINDOW *win, int current_line, int start_line, int max_y) { - werase(win); - for (int i = 0; i < max_y && (start_line + i) < line_count; i++) { - if ((start_line + i) == current_line) wattron(win, A_REVERSE); - if (chosen_lines[start_line + i] != invert_chosen) wattron(win, A_BOLD); - mvwprintw(win, i, 0, "%s", get_line(start_line + i)); - wattroff(win, A_REVERSE | A_BOLD); - } - wrefresh(win); +void draw_lines(WINDOW* win, int current_line, int start_line, int max_y) { + werase(win); + for (int i = 0; i < max_y && (start_line + i) < line_count; i++) { + if ((start_line + i) == current_line) wattron(win, A_REVERSE); + if (chosen_lines[start_line + i] != invert_chosen) wattron(win, A_BOLD); + mvwprintw(win, i, 0, "%s", get_line(start_line + i)); + wattroff(win, A_REVERSE | A_BOLD); + } + wrefresh(win); } /** * @brief Prints the chosen lines to the specified file descriptor. - * + * * @param fd File descriptor to print to. */ void output_chosen_lines_fd(int fd) { - size_t current = 0; - if (chosen_lines[0] != invert_chosen) - dprintf(fd, "%s\n", buffer); - - for (size_t i = 0; i < buffer_ptr; i++) { - if (buffer[i] == '\0') { - current++; - if (chosen_lines[current] != invert_chosen) - dprintf(fd, "%s\n", &buffer[i + 1]); - } - } + size_t current = 0; + if (chosen_lines[0] != invert_chosen) + dprintf(fd, "%s\n", buffer); + + for (size_t i = 0; i < buffer_ptr; i++) { + if (buffer[i] == '\0') { + current++; + if (chosen_lines[current] != invert_chosen) + dprintf(fd, "%s\n", &buffer[i + 1]); + } + } } /** * @brief Prints the chosen lines to the specified output file. - * + * * @param filename Name of the output file. */ -void output_chosen_lines(char *filename) { - int fd; +void output_chosen_lines(char* filename) { + int fd; - fd = (filename == NULL) ? STDOUT_FILENO : open(filename, O_WRONLY | O_TRUNC | O_CREAT, 0664); - if (fd == -1) die("Failed to open file"); + fd = (filename == NULL) ? STDOUT_FILENO : open(filename, O_WRONLY | O_TRUNC | O_CREAT, 0664); + if (fd == -1) die("Failed to open file"); - output_chosen_lines_fd(fd); + output_chosen_lines_fd(fd); } /** * @brief Displays usage information and exits the program. - * + * * @param exitcode Exit code. */ NORETURN void usage(int exitcode) { - fprintf(stderr, "Usage: %s [-hv] [-o output] <input> [command ...]\n", argv0); - exit(exitcode); + fprintf(stderr, "Usage: %s [-hvx] [-o output] <input> [command ...]\n", argv0); + exit(exitcode); } /** * @brief Initializes and handles the ncurses window for line selection. */ void display_window(void) { - initscr(); - cbreak(); - noecho(); - keypad(stdscr, TRUE); - - int current_line = 0; - int start_line = 0; - int ch; - int max_y = getmaxy(stdscr); - - draw_lines(stdscr, current_line, start_line, max_y); - - while ((ch = getch()) != 'q') { - switch (ch) { - case KEY_UP: - if (current_line > 0) { - (current_line)--; - if (current_line < start_line) - start_line--; - } - break; - case KEY_DOWN: - if (current_line < line_count - 1) { - (current_line)++; - if (current_line >= start_line + max_y) - start_line++; - } - break; - case 'v': - invert_chosen = !invert_chosen; - break; - case KEY_ENTER: - case ' ': - chosen_lines[current_line] = !chosen_lines[current_line]; - break; - } - draw_lines(stdscr, current_line, start_line, max_y); - } - - endwin(); + initscr(); + cbreak(); + noecho(); + keypad(stdscr, TRUE); + + int current_line = 0; + int start_line = 0; + int ch; + int max_y = getmaxy(stdscr); + + draw_lines(stdscr, current_line, start_line, max_y); + + while ((ch = getch()) != 'q') { + switch (ch) { + case KEY_UP: + case KEY_LEFT: + if (current_line > 0) { + (current_line)--; + if (current_line < start_line) + start_line--; + } + break; + case KEY_DOWN: + case KEY_RIGHT: + if (current_line < line_count - 1) { + (current_line)++; + if (current_line >= start_line + max_y) + start_line++; + } + break; + case 'v': + invert_chosen = !invert_chosen; + break; + case KEY_ENTER: + case ' ': + chosen_lines[current_line] = !chosen_lines[current_line]; + break; + } + draw_lines(stdscr, current_line, start_line, max_y); + } + + endwin(); } /** * @brief Executes a command and passes the chosen lines as stdin. - * + * * @param argv Command and its arguments. */ -void execute_command(char **argv) { - int pipefd[2]; - pid_t pid; - - if (pipe(pipefd) == -1) die("pipe"); - - if ((pid = fork()) == -1) die("fork"); - - if (pid == 0) { // Child process - close(pipefd[1]); // Close write end of the pipe - dup2(pipefd[0], STDIN_FILENO); // Redirect stdin to read end of the pipe - execvp(argv[0], argv); - die("execvp"); // If execvp fails - } else { // Parent process - close(pipefd[0]); // Close read end of the pipe - output_chosen_lines_fd(pipefd[1]); - close(pipefd[1]); // Close write end after writing - wait(NULL); // Wait for the child process to finish - } +void execute_command(char** argv) { + int pipefd[2]; + pid_t pid; + + if (pipe(pipefd) == -1) die("pipe"); + + if ((pid = fork()) == -1) die("fork"); + + if (pid == 0) { // Child process + close(pipefd[1]); // Close write end of the pipe + dup2(pipefd[0], STDIN_FILENO); // Redirect stdin to read end of the pipe + execvp(argv[0], argv); + die("execvp"); // If execvp fails + } else { // Parent process + close(pipefd[0]); // Close read end of the pipe + output_chosen_lines_fd(pipefd[1]); + close(pipefd[1]); // Close write end after writing + wait(NULL); // Wait for the child process to finish + } } -int main(int argc, char *argv[]) { - char *output = NULL; - - argv0 = argv[0]; - ARGBEGIN - switch (OPT) { - case 'h': - usage(0); - case 'v': - invert_chosen = true; - break; - case 'o': - output = EARGF(usage(1)); - break; - default: - fprintf(stderr, "error: unknown option '-%c'\n", OPT); - usage(1); +void execute_command_xargs(int argc, char** argv) { + char** newargv; + int chosencount = argc + 1; // + NULL + for (int i = 0; i < line_count; i++) + if (chosen_lines[i] != invert_chosen) + chosencount++; + + newargv = malloc(chosencount * sizeof(char*)); + int newargc; + for (newargc = 0; newargc < argc; newargc++) + newargv[newargc] = argv[newargc]; + + size_t current = 0; + if (chosen_lines[0] != invert_chosen) + newargv[newargc++] = buffer; + + for (size_t i = 0; i < buffer_ptr; i++) { + if (buffer[i] == '\0') { + current++; + if (chosen_lines[current] != invert_chosen) + newargv[newargc++] = &buffer[i + 1]; } - ARGEND; + } - if (argc == 0) { - fprintf(stderr, "error: missing input\n"); - usage(1); - } + newargv[newargc++] = NULL; - load_file(argv[0]); - SHIFT; + pid_t pid; - display_window(); - - if (argc == 0) { - output_chosen_lines(output); - } else { - execute_command(argv); - } - - free(buffer); - free(chosen_lines); + if ((pid = fork()) == -1) die("fork"); + if (pid == 0) { + execvp(newargv[0], newargv); + die("execvp"); + } + wait(NULL); +} - return 0; +int main(int argc, char* argv[]) { + char* output = NULL; + bool xargs = false; + + argv0 = argv[0]; + ARGBEGIN + switch (OPT) { + case 'h': + usage(0); + case 'v': + invert_chosen = true; + break; + case 'x': + xargs = true; + break; + case 'o': + output = EARGF(usage(1)); + break; + default: + fprintf(stderr, "error: unknown option '-%c'\n", OPT); + usage(1); + } + ARGEND; + + if (argc == 0) { + fprintf(stderr, "error: missing input\n"); + usage(1); + } + + load_file(argv[0]); + SHIFT; + + display_window(); + + if (argc == 0) { + output_chosen_lines(output); + } else if (!xargs) { + execute_command(argv); + } else { + execute_command_xargs(argc, argv); + } + + free(buffer); + free(chosen_lines); + + return 0; }