// // bb.c // Copyright 2020 Bruce Hill // Released under the MIT license with the Commons Clause // // This file contains the main source code of `bb` the Bitty Browser. // #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include "draw.h" #include "terminal.h" #include "types.h" #include "utils.h" #ifndef BB_NAME #define BB_NAME "bb" #endif #define BB_VERSION "0.30.0" #define MAX_BINDINGS 1024 #define SCROLLOFF MIN(5, (winsize.ws_row-4)/2) // Functions void bb_browse(bb_t *bb, const char *initial_path); static void check_cmdfile(bb_t *bb); static void cleanup(void); static void cleanup_and_raise(int sig); #ifdef __APPLE__ static int compare_files(void *v, const void *v1, const void *v2); #else static int compare_files(const void *v1, const void *v2, void *v); #endif __attribute__((format(printf,2,3))) void flash_warn(bb_t *bb, const char *fmt, ...); static void handle_next_key_binding(bb_t *bb); static void init_term(void); static int is_simple_bbcmd(const char *s); static entry_t* load_entry(bb_t *bb, const char *path); static int matches_cmd(const char *str, const char *cmd); static void* memcheck(void *p); static char* normalize_path(const char *root, const char *path, char *pbuf); static int populate_files(bb_t *bb, const char *path); static void print_bindings(int fd); static void run_bbcmd(bb_t *bb, const char *cmd); static void restore_term(const struct termios *term); static int run_script(bb_t *bb, const char *cmd); static void set_columns(bb_t *bb, const char *cols); static void set_cursor(bb_t *bb, int i); static void set_globs(bb_t *bb, const char *globs); static void set_interleave(bb_t *bb, int interleave); static void set_selected(bb_t *bb, entry_t *e, int selected); static void set_scroll(bb_t *bb, int i); static void set_sort(bb_t *bb, const char *sort); static void set_title(bb_t *bb); static void sort_files(bb_t *bb); static char *trim(char *s); static int try_free_entry(entry_t *e); static void update_term_size(int sig); static int wait_for_process(proc_t **proc); // Constants static const char *T_ENTER_BBMODE = T_OFF(T_SHOW_CURSOR ";" T_WRAP) T_ON(T_ALT_SCREEN ";" T_MOUSE_XY ";" T_MOUSE_CELL ";" T_MOUSE_SGR); static const char *T_LEAVE_BBMODE = T_OFF(T_MOUSE_XY ";" T_MOUSE_CELL ";" T_MOUSE_SGR ";" T_ALT_SCREEN) T_ON(T_SHOW_CURSOR ";" T_WRAP); static const char *T_LEAVE_BBMODE_PARTIAL = T_OFF(T_MOUSE_XY ";" T_MOUSE_CELL ";" T_MOUSE_SGR) T_ON(T_WRAP); static const struct termios default_termios = { .c_iflag = ICRNL, .c_oflag = OPOST | ONLCR | NL0 | CR0 | TAB0 | BS0 | VT0 | FF0, .c_lflag = ISIG | ICANON | IEXTEN | ECHO | ECHOE | ECHOK | ECHOCTL | ECHOKE, .c_cflag = CS8 | CREAD, .c_cc[VINTR] = '', .c_cc[VQUIT] = '', .c_cc[VERASE] = 127, .c_cc[VKILL] = '', .c_cc[VEOF] = '', .c_cc[VSTART] = '', .c_cc[VSTOP] = '', .c_cc[VSUSP] = '', .c_cc[VREPRINT] = '', .c_cc[VWERASE] = '', .c_cc[VLNEXT] = '', .c_cc[VDISCARD] = '', .c_cc[VMIN] = 1, .c_cc[VTIME] = 0, }; static const char *description_str = BB_NAME" - an itty bitty console TUI file browser\n"; static const char *usage_str = "Usage: "BB_NAME" (-h/--help | -v/--version | -s | -d | -0 | +command)* [[--] directory]\n"; // Variables used within this file to track global state static binding_t bindings[MAX_BINDINGS]; static struct termios orig_termios, bb_termios; static FILE *tty_out = NULL, *tty_in = NULL; static struct winsize winsize = {0}; static char cmdfilename[PATH_MAX] = {0}; static bb_t *current_bb = NULL; // // Use bb to browse the filesystem. // void bb_browse(bb_t *bb, const char *initial_path) { if (populate_files(bb, initial_path)) clean_err("Could not find initial path: \"%s\"", initial_path); run_script(bb, "bbstartup"); check_cmdfile(bb); while (!bb->should_quit) { render(tty_out, bb); handle_next_key_binding(bb); } run_script(bb, "bbshutdown"); } // // Check the bb command file and run any and all commands that have been // written to it. // static void check_cmdfile(bb_t *bb) { FILE *cmdfile = fopen(cmdfilename, "r"); if (!cmdfile) return; char *cmd = NULL; size_t space = 0; while (getdelim(&cmd, &space, '\0', cmdfile) >= 0) { if (!cmd[0]) continue; run_bbcmd(bb, cmd); if (bb->should_quit) break; } free(cmd); fclose(cmdfile); unlink(cmdfilename); } // // Clean up the terminal before going to the default signal handling behavior. // static void cleanup_and_raise(int sig) { cleanup(); int childsig = (sig == SIGTSTP || sig == SIGSTOP) ? sig : SIGHUP; if (current_bb) { for (proc_t *p = current_bb->running_procs; p; p = p->running.next) { kill(p->pid, childsig); LL_REMOVE(p, running); } } raise(sig); // This code will only ever be run if sig is SIGTSTP/SIGSTOP, otherwise, raise() won't return: init_term(); struct sigaction sa = {.sa_handler = &cleanup_and_raise, .sa_flags = (int)(SA_NODEFER | SA_RESETHAND)}; sigaction(sig, &sa, NULL); } // // Reset the screen and delete the cmdfile // static void cleanup(void) { if (cmdfilename[0]) { unlink(cmdfilename); cmdfilename[0] = '\0'; } if (tty_out) { fputs(T_LEAVE_BBMODE, tty_out); fflush(tty_out); tcsetattr(fileno(tty_out), TCSANOW, &orig_termios); } } // // Used for sorting, this function compares files according to the sorting-related options, // like bb->sort // #ifdef __APPLE__ static int compare_files(void *v, const void *v1, const void *v2) #else static int compare_files(const void *v1, const void *v2, void *v) #endif { #define COMPARE(a, b) if ((a) != (b)) { return sign*((a) < (b) ? 1 : -1); } #define COMPARE_TIME(t1, t2) COMPARE((t1).tv_sec, (t2).tv_sec) COMPARE((t1).tv_nsec, (t2).tv_nsec) bb_t *bb = (bb_t*)v; const entry_t *e1 = *((const entry_t**)v1), *e2 = *((const entry_t**)v2); int sign = 1; if (!bb->interleave_dirs) { COMPARE(E_ISDIR(e1), E_ISDIR(e2)); } for (char *sort = bb->sort + 1; *sort; sort += 2) { sign = sort[-1] == '-' ? -1 : 1; switch (*sort) { case COL_SELECTED: COMPARE(IS_SELECTED(e1), IS_SELECTED(e2)); break; case COL_NAME: { // This sorting method is not identical to strverscmp(). Notably, bb's sort // will order: [0, 1, 9, 00, 01, 09, 10, 000, 010] instead of strverscmp()'s // order: [000, 00, 01, 010, 09, 0, 1, 9, 10]. I believe bb's sort is consistent // with how people want their files grouped: all files padded to n digits // will be grouped together, and files with the same padding will be sorted // ordinally. This version also does case-insensitivity by lowercasing words, // so the following characters come before all letters: [\]^_` const char *n1 = e1->name, *n2 = e2->name; while (*n1 && *n2) { char c1 = tolower(*n1), c2 = tolower(*n2); if ('0' <= c1 && c1 <= '9' && '0' <= c2 && c2 <= '9') { long i1 = strtol(n1, (char**)&n1, 10); long i2 = strtol(n2, (char**)&n2, 10); // Shorter numbers always go before longer. In practice, I assume // filenames padded to the same number of digits should be grouped // together, instead of // [1.png, 0001.png, 2.png, 0002.png, 3.png], it makes more sense to have: // [1.png, 2.png, 3.png, 0001.png, 0002.png] COMPARE((n2 - e2->name), (n1 - e1->name)); COMPARE(i2, i1); } else { COMPARE(c2, c1); ++n1; ++n2; } } COMPARE(tolower(*n2), tolower(*n1)); break; } case COL_PERM: COMPARE((e1->info.st_mode & 0x3FF), (e2->info.st_mode & 0x3FF)); break; case COL_SIZE: COMPARE(e1->info.st_size, e2->info.st_size); break; case COL_MTIME: COMPARE_TIME(mtime(e1->info), mtime(e2->info)); break; case COL_CTIME: COMPARE_TIME(ctime(e1->info), ctime(e2->info)); break; case COL_ATIME: COMPARE_TIME(atime(e1->info), atime(e2->info)); break; case COL_RANDOM: COMPARE(e2->shufflepos, e1->shufflepos); break; default: break; } } return 0; #undef COMPARE #undef COMPARE_TIME } // // Flash a warning message at the bottom of the screen. // void flash_warn(bb_t *bb, const char *fmt, ...) { move_cursor(tty_out, 0, winsize.ws_row-1); fputs("\033[41;33;1m", tty_out); va_list args; va_start(args, fmt); vfprintf(tty_out, fmt, args); va_end(args); fputs(" Press any key to continue...\033[0m ", tty_out); fflush(tty_out); while (bgetkey(tty_in, NULL, NULL) == -1) usleep(100); bb->dirty = 1; } // // Wait until the user has pressed a key with an associated key binding and run // that binding. // static void handle_next_key_binding(bb_t *bb) { int key, mouse_x, mouse_y; binding_t *binding; do { do { key = bgetkey(tty_in, &mouse_x, &mouse_y); if (key == -1 && bb->dirty) return; } while (key == -1); binding = NULL; for (size_t i = 0; bindings[i].script && i < sizeof(bindings)/sizeof(bindings[0]); i++) { if (key == bindings[i].key) { binding = &bindings[i]; break; } } } while (!binding); char bbmousecol[2] = {0, 0}, bbclicked[PATH_MAX]; if (mouse_x != -1 && mouse_y != -1) { int *colwidths = get_column_widths(bb->columns, winsize.ws_col-1); // Get bb column: for (int col = 0, x = 0; bb->columns[col]; col++, x++) { x += colwidths[col]; if (x >= mouse_x) { bbmousecol[0] = bb->columns[col]; break; } } if (mouse_y == 1) { strcpy(bbclicked, ""); } else if (2 <= mouse_y && mouse_y <= winsize.ws_row - 2 && bb->scroll + (mouse_y - 2) <= bb->nfiles - 1) { strcpy(bbclicked, bb->files[bb->scroll + (mouse_y - 2)]->fullname); } else { bbclicked[0] = '\0'; } setenv("BBMOUSECOL", bbmousecol, 1); setenv("BBCLICKED", bbclicked, 1); } if (is_simple_bbcmd(binding->script)) { run_bbcmd(bb, binding->script); } else { move_cursor(tty_out, 0, winsize.ws_row-1); fputs("\033[K", tty_out); restore_term(&default_termios); run_script(bb, binding->script); init_term(); set_title(bb); check_cmdfile(bb); } if (mouse_x != -1 && mouse_y != -1) { setenv("BBMOUSECOL", "", 1); setenv("BBCLICKED", "", 1); } } // // Initialize the terminal files for /dev/tty and set up some desired // attributes like passing Ctrl-c as a key instead of interrupting // static void init_term(void) { if (tcsetattr(fileno(tty_out), TCSANOW, &bb_termios) == -1) clean_err("Couldn't tcsetattr"); update_term_size(0); // Initiate mouse tracking and disable text wrapping: fputs(T_ENTER_BBMODE, tty_out); fflush(tty_out); } // // Return whether or not 's' is a simple bb command that doesn't need // a full shell instance (e.g. "bbcmd cd:.." or "bbcmd move:+1"). // static int is_simple_bbcmd(const char *s) { if (!s) return 0; while (*s == ' ') ++s; if (strncmp(s, "bbcmd ", strlen("bbcmd ")) != 0) return 0; const char *special = ";$&<>|\n*?\\\"'"; for (const char *p = special; *p; ++p) { if (strchr(s, *p)) return 0; } return 1; } // // Load a file's info into an entry_t and return it (if found). // The returned entry must be free()ed by the caller. // Warning: this does not deduplicate entries, and it's best if there aren't // duplicate entries hanging around. // static entry_t* load_entry(bb_t *bb, const char *path) { struct stat linkedstat, filestat; if (!path || !path[0]) return NULL; if (lstat(path, &filestat) == -1) return NULL; char pbuf[PATH_MAX]; if (path[0] == '/') strcpy(pbuf, path); else sprintf(pbuf, "%s%s", bb->path, path); if (pbuf[strlen(pbuf)-1] == '/' && pbuf[1]) pbuf[strlen(pbuf)-1] = '\0'; // Check for pre-existing: for (entry_t *e = bb->hash[(int)filestat.st_ino & HASH_MASK]; e; e = e->hash.next) { if (e->info.st_ino == filestat.st_ino && e->info.st_dev == filestat.st_dev // Need to check filename in case of hard links && streq(pbuf, e->fullname)) return e; } ssize_t linkpathlen = -1; char linkbuf[PATH_MAX]; if (S_ISLNK(filestat.st_mode)) { linkpathlen = readlink(pbuf, linkbuf, sizeof(linkbuf)); if (linkpathlen < 0) clean_err("Couldn't read link: '%s'", pbuf); linkbuf[linkpathlen] = '\0'; while (linkpathlen > 0 && linkbuf[linkpathlen-1] == '/') linkbuf[--linkpathlen] = '\0'; if (stat(pbuf, &linkedstat) == -1) memset(&linkedstat, 0, sizeof(linkedstat)); } size_t pathlen = strlen(pbuf); size_t entry_size = sizeof(entry_t) + (pathlen + 1) + (size_t)(linkpathlen + 1); entry_t *entry = xcalloc(entry_size, 1); char *end = stpcpy(entry->fullname, pbuf); if (linkpathlen >= 0) entry->linkname = strcpy(end + 1, linkbuf); if (streq(entry->fullname, "/")) { entry->name = entry->fullname; } else { entry->name = strrchr(entry->fullname, '/') + 1; // Last path component } if (S_ISLNK(filestat.st_mode)) entry->linkedmode = linkedstat.st_mode; entry->info = filestat; LL_PREPEND(bb->hash[(int)filestat.st_ino & HASH_MASK], entry, hash); entry->index = -1; bb->hash[(int)filestat.st_ino & HASH_MASK] = entry; return entry; } // // Return whether a string matches a command // e.g. matches_cmd("sel:x", "select:") == 1, matches_cmd("q", "quit") == 1 // static int matches_cmd(const char *str, const char *cmd) { if ((strchr(cmd, ':') == NULL) != (strchr(str, ':') == NULL)) return 0; while (*str == *cmd && *cmd && *cmd != ':') ++str, ++cmd; return *str == '\0' || *str == ':'; } // // Memory allocation failures are unrecoverable in bb, so this wrapper just // prints an error message and exits if that happens. // static void* memcheck(void *p) { if (!p) clean_err("Allocation failure"); return p; } // // Prepend `root` to relative paths, replace "~" with $HOME. // The normalized path is stored in `normalized`. // static char *normalize_path(const char *root, const char *path, char *normalized) { char pbuf[PATH_MAX] = {0}; if (path[0] == '~' && (path[1] == '\0' || path[1] == '/')) { char *home; if (!(home = getenv("HOME"))) return NULL; strcpy(pbuf, home); ++path; } else if (path[0] != '/') { strcpy(pbuf, root); if (root[strlen(root)-1] != '/') strcat(pbuf, "/"); } strcat(pbuf, path); if (realpath(pbuf, normalized) == NULL) { strcpy(normalized, pbuf); // TODO: normalize better? return NULL; } return normalized; } // // Remove all the files currently stored in bb->files and if `bb->path` is // non-NULL, update `bb` with a listing of the files in `path` // static int populate_files(bb_t *bb, const char *path) { int clear_future_history = 0; if (path == NULL) ; else if (streq(path, "-")) { if (bb->history->prev) bb->history = bb->history->prev; path = bb->history->path; } else if (streq(path, "+")) { if (bb->history->next) bb->history = bb->history->next; path = bb->history->path; } else clear_future_history = 1; int samedir = path && streq(bb->path, path); int old_scroll = bb->scroll; int old_cursor = bb->cursor; char old_selected[PATH_MAX] = ""; if (samedir && bb->nfiles > 0) strcpy(old_selected, bb->files[bb->cursor]->fullname); char pbuf[PATH_MAX] = {0}, prev[PATH_MAX] = {0}; strcpy(prev, bb->path); if (path != NULL) { if (!normalize_path(bb->path, path, pbuf)) flash_warn(bb, "Could not normalize path: \"%s\"", path); if (pbuf[strlen(pbuf)-1] != '/') strcat(pbuf, "/"); if (chdir(pbuf)) { flash_warn(bb, "Could not cd to: \"%s\"", pbuf); return -1; } } if (clear_future_history && !samedir) { for (bb_history_t *next, *h = bb->history->next; h; h = next) { next = h->next; free(h); } bb_history_t *h = new(bb_history_t); strcpy(h->path, pbuf); h->prev = bb->history; bb->history->next = h; bb->history = h; } bb->dirty = 1; strcpy(bb->path, pbuf); set_title(bb); // Clear old files (if any) if (bb->files) { for (int i = 0; i < bb->nfiles; i++) { bb->files[i]->index = -1; try_free_entry(bb->files[i]); bb->files[i] = NULL; } free(bb->files); bb->files = NULL; } bb->nfiles = 0; bb->cursor = 0; bb->scroll = 0; if (!bb->path[0]) return 0; size_t space = 0; glob_t globbuf = {0}; char *pat, *tmpglob = memcheck(strdup(bb->globpats)); while ((pat = strsep(&tmpglob, " ")) != NULL) glob(pat, GLOB_NOSORT|GLOB_APPEND, NULL, &globbuf); free(tmpglob); for (size_t i = 0; i < globbuf.gl_pathc; i++) { // Don't normalize path so we can get "." and ".." entry_t *entry = load_entry(bb, globbuf.gl_pathv[i]); if (!entry) { flash_warn(bb, "Failed to load entry: '%s'", globbuf.gl_pathv[i]); continue; } entry->index = bb->nfiles; if ((size_t)bb->nfiles + 1 > space) bb->files = xrealloc(bb->files, (space += 100)*sizeof(void*)); bb->files[bb->nfiles++] = entry; } globfree(&globbuf); // RNG is seeded with a hash of all the inodes in the current dir // This hash algorithm is based on Python's frozenset hashing unsigned long seed = (unsigned long)bb->nfiles * 1927868237UL; for (int i = 0; i < bb->nfiles; i++) seed ^= ((bb->files[i]->info.st_ino ^ 89869747UL) ^ (bb->files[i]->info.st_ino << 16)) * 3644798167UL; srand((unsigned int)seed); for (int i = 0; i < bb->nfiles; i++) { int j = rand() % (i+1); // This introduces some RNG bias, but it's not important here bb->files[i]->shufflepos = bb->files[j]->shufflepos; bb->files[j]->shufflepos = i; } sort_files(bb); if (samedir) { set_scroll(bb, old_scroll); bb->cursor = old_cursor > bb->nfiles-1 ? bb->nfiles-1 : old_cursor; if (old_selected[0]) { entry_t *e = load_entry(bb, old_selected); if (e) set_cursor(bb, e->index); } } else { entry_t *p = load_entry(bb, prev); if (p) { if (IS_VIEWED(p)) set_cursor(bb, p->index); else try_free_entry(p); } } return 0; } // // Print the current key bindings // static void print_bindings(int fd) { char buf[1000], buf2[1024]; for (size_t i = 0; bindings[i].script && i < sizeof(bindings)/sizeof(bindings[0]); i++) { if (bindings[i].key == -1) { const char *label = bindings[i].description; sprintf(buf, "\n\033[33;1;4m\033[%dG%s\033[0m\n", (winsize.ws_col-(int)strlen(label))/2, label); write(fd, buf, strlen(buf)); continue; } size_t shared = 0; char *p = buf; for (size_t j = i; bindings[j].script && streq(bindings[j].description, bindings[i].description); j++) { if (j > i) p = stpcpy(p, ", "); ++shared; int key = bindings[j].key; p = bkeyname(key, p); } *p = '\0'; sprintf(buf2, "\033[1m\033[%dG%s\033[0m", winsize.ws_col/2 - 1 - (int)strlen(buf), buf); write(fd, buf2, strlen(buf2)); sprintf(buf2, "\033[1m\033[%dG\033[34m%s\033[0m", winsize.ws_col/2 + 1, bindings[i].description); write(fd, buf2, strlen(buf2)); write(fd, "\033[0m\n", strlen("\033[0m\n")); i += shared - 1; } write(fd, "\n", 1); } // // Run a bb internal command (e.g. "+refresh") and return an indicator of what // needs to happen next. // static void run_bbcmd(bb_t *bb, const char *cmd) { while (*cmd == ' ' || *cmd == '\n') ++cmd; if (strncmp(cmd, "bbcmd ", strlen("bbcmd ")) == 0) cmd = &cmd[strlen("bbcmd ")]; const char *value = strchr(cmd, ':'); if (value) ++value; #define set_bool(target) do { if (!value) { target = !target; } else { target = value[0] == '1'; } } while (0) if (matches_cmd(cmd, "bind:")) { // +bind::