// // 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 #include "draw.h" #include "terminal.h" #include "types.h" #include "utils.h" #ifndef BB_NAME #define BB_NAME "bb" #endif #define BB_VERSION "0.31.0" #define MAX_BINDINGS 1024 #define SCROLLOFF MIN(5, (winsize.ws_row-4)/2) #define ONSCREEN (winsize.ws_row - 3) #define LOG(...) do { FILE *f = fopen("log.txt", "a"); fprintf(f, __VA_ARGS__); fclose(f); } while (0) // Functions void bb_browse(bb_t *bb, int argc, char *argv[]); static void check_cmdfile(bb_t *bb); static void cleanup(void); static void cleanup_and_raise(int sig); static int compare_files(const void *v1, const void *v2); __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 char* normalize_path(const char *path, char *pbuf); static int populate_files(bb_t *bb, const char *path); static void print_bindings(FILE *f); 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 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; // Redirect stderr/stdout to these files during execution, and dump them on exit typedef struct { int orig_fd, dup_fd, tmp_fd; const char *name; char filename[PATH_MAX]; } outbuf_t; outbuf_t output_buffers[] = { {.name="stdout", .orig_fd=STDOUT_FILENO, .dup_fd=-1, .tmp_fd=-1}, {.name="stderr", .orig_fd=STDERR_FILENO, .dup_fd=-1, .tmp_fd=-1}, }; // // Use bb to browse the filesystem. // void bb_browse(bb_t *bb, int argc, char *argv[]) { const char *initial_path; if (argc >= 3 && streq(argv[argc-2], "--")) { initial_path = argv[argc-1]; argc -= 2; } else if (argc >= 2 && argv[argc-1][0] != '-' && argv[argc-1][0] != '+') { initial_path = argv[argc-1]; argc -= 1; } else { initial_path = "."; } char full_initial_path[PATH_MAX]; normalize_path(initial_path, full_initial_path); struct stat path_stat; const char *goto_file = NULL; nonnegative(stat(full_initial_path, &path_stat), "Could not find initial path: \"%s\"", initial_path); if (!S_ISDIR(path_stat.st_mode)) { char *slash = strrchr(full_initial_path, '/'); *slash = '\0'; goto_file = slash+1; } if (populate_files(bb, full_initial_path)) errx(EXIT_FAILURE, "Could not find initial path: \"%s\"", full_initial_path); // Emergency fallback: bindings[0].key = KEY_CTRL_C; bindings[0].script = check_strdup("kill -INT $PPID"); bindings[0].description = check_strdup("Kill the bb process"); system("bbstartup"); FILE *cmdfile = fopen(cmdfilename, "a"); if (goto_file) fprintf(cmdfile, "%cgoto:%s", '\0', goto_file); for (int i = 0; i < argc; i++) { if (argv[i][0] == '+') { char *cmd = argv[i] + 1; char *colon = strchr(cmd, ':'); if (colon && !colon[1]) { for (++i; i < argc; i++) fprintf(cmdfile, "%c%s%s", '\0', cmd, argv[i]); } else { fprintf(cmdfile, "%c%s", '\0', cmd); } } } fclose(cmdfile); check_cmdfile(bb); while (!bb->should_quit) { render(tty_out, bb); handle_next_key_binding(bb); } system("bbshutdown"); check_cmdfile(bb); } // // 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; } delete(&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, delete the cmdfile, and print the stdout/stderr buffers // 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); } FOREACH(outbuf_t*, ob, output_buffers) { if (ob->tmp_fd == -1) continue; fflush(ob->orig_fd == STDOUT_FILENO ? stdout : stderr); dup2(ob->dup_fd, ob->orig_fd); lseek(ob->tmp_fd, 0, SEEK_SET); char buf[256]; for (ssize_t len; (len = read(ob->tmp_fd, buf, LEN(buf))) > 0; ) write(ob->orig_fd, buf, len); close(ob->tmp_fd); ob->tmp_fd = ob->dup_fd = -1; unlink(ob->filename); } } // // Used for sorting, this function compares files according to the sorting-related options, // like bb->sort // static int compare_files(const void *v1, const void *v2) { #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 = current_bb; 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(get_mtime(e1->info), get_mtime(e2->info)); break; case COL_CTIME: COMPARE_TIME(get_ctime(e1->info), get_ctime(e2->info)); break; case COL_ATIME: COMPARE_TIME(get_atime(e1->info), get_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 { struct winsize prevsize = winsize; key = bgetkey(tty_in, &mouse_x, &mouse_y); // Window size changed while waiting for keypress: if (winsize.ws_row != prevsize.ws_row || winsize.ws_col != prevsize.ws_col) bb->dirty = 1; if (key == -1 && bb->dirty) return; } while (key == -1); binding = NULL; FOREACH(binding_t*, b, bindings) { if (key == b->key) { binding = b; 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(&orig_termios); run_script(bb, binding->script); for (entry_t *next, *e = bb->selected; e; e = next) { next = e->selected.next; struct stat buf; if (stat(e->fullname, &buf) != 0) set_selected(bb, e, 0); } 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) { nonnegative(tcsetattr(fileno(tty_out), TCSANOW, &bb_termios)); 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 = nonnegative(readlink(pbuf, linkbuf, sizeof(linkbuf)), "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 = new_bytes(entry_size); char *end = stpcpy(entry->fullname, pbuf); if (linkpathlen >= 0) entry->linkname = strcpy(end + 1, linkbuf); if (streq(entry->fullname, "/")) { entry->name = entry->fullname; } else { if (strncmp(entry->fullname, bb->path, strlen(bb->path)) == 0) entry->name = entry->fullname + strlen(bb->path); 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 == ':'; } // // Prepend `./` to relative paths, replace "~" with $HOME. // The normalized path is stored in `normalized`. // static char *normalize_path(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, "./"); } 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) return -1; if (bb->history->prev) bb->history = bb->history->prev; path = bb->history->path; } else if (streq(path, "+")) { if (!bb->history) return -1; 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(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 ? bb->history->next : NULL; h; h = next) { next = h->next; delete(&h); } bb_history_t *h = new(bb_history_t); strcpy(h->path, pbuf); h->prev = bb->history; if (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; } delete(&bb->files); } 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 = check_strdup(bb->globpats); while ((pat = strsep(&tmpglob, " ")) != NULL) glob(pat, GLOB_NOSORT|GLOB_APPEND, NULL, &globbuf); delete(&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 = grow(bb->files, space += 100); 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(FILE *f) { FOREACH(binding_t*, b, bindings) { if (!b->description) break; if (b->key == -1) { const char *label = b->description; fprintf(f, "\n\033[33;1;4m\033[%dG%s\033[0m\n", (winsize.ws_col-(int)strlen(label))/2, label); continue; } char buf[1000]; char *p = buf; for (binding_t *next = b; next < &bindings[LEN(bindings)] && next->script && streq(b->description, next->description); next++) { if (next > b) p = stpcpy(p, ", "); p = bkeyname(next->key, p); b = next; } *p = '\0'; fprintf(f, "\033[1m\033[%dG%s\033[0m", winsize.ws_col/2 - 1 - (int)strlen(buf), buf); fprintf(f, "\033[1m\033[%dG\033[34m%s\033[0m", winsize.ws_col/2 + 1, b->description); fprintf(f, "\033[0m\n"); } fprintf(f, "\n"); } // // 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; if (matches_cmd(cmd, "bind:")) { // +bind::