add __paginate_file

This commit is contained in:
Gavin Hayes 2024-04-16 04:47:54 -04:00
parent 7eaa6a0061
commit 346a78ddc0
2 changed files with 64 additions and 23 deletions

View file

@ -103,6 +103,7 @@ int verynice(void);
void __warn_if_powersave(void); void __warn_if_powersave(void);
void _Exit1(int) libcesque wontreturn; void _Exit1(int) libcesque wontreturn;
void __paginate(int, const char *); void __paginate(int, const char *);
void __paginate_file(int, const char *);
/* memory management */ /* memory management */
void _weakfree(void *); void _weakfree(void *);
void *_mapanon(size_t) attributeallocsize((1)) mallocesque; void *_mapanon(size_t) attributeallocsize((1)) mallocesque;

View file

@ -24,31 +24,30 @@
#include "libc/mem/gc.h" #include "libc/mem/gc.h"
#include "libc/runtime/runtime.h" #include "libc/runtime/runtime.h"
#include "libc/str/str.h" #include "libc/str/str.h"
#include "libc/sysv/consts/o.h"
#include "libc/temp.h" #include "libc/temp.h"
#include "libc/x/x.h" #include "libc/x/x.h"
/** static char *get_pagerpath(char *pathbuf, size_t pathbufsz) {
* Displays wall of text in terminal with pagination. char *pagerpath;
*/
void __paginate(int fd, const char *s) {
int tfd, pid;
char *args[3] = {0};
char tmppath[] = "/tmp/paginate.XXXXXX";
char progpath[PATH_MAX];
char16_t widepath[PATH_MAX];
int n;
if (strcmp(nulltoempty(getenv("TERM")), "dumb") && isatty(0) && isatty(1) && if (strcmp(nulltoempty(getenv("TERM")), "dumb") && isatty(0) && isatty(1) &&
((args[0] = commandv("less", progpath, sizeof(progpath))) || ((pagerpath = commandv("less", pathbuf, pathbufsz)) ||
(args[0] = commandv("more", progpath, sizeof(progpath))) || (pagerpath = commandv("more", pathbuf, pathbufsz)) ||
(args[0] = commandv("more.exe", progpath, sizeof(progpath))) || (pagerpath = commandv("more.exe", pathbuf, pathbufsz)) ||
(args[0] = commandv("more.com", progpath, sizeof(progpath))))) { (pagerpath = commandv("more.com", pathbuf, pathbufsz)))) {
if ((tfd = mkstemp(tmppath)) != -1) { return pagerpath;
write(tfd, s, strlen(s)); }
close(tfd); return 0;
args[1] = tmppath; }
if (!IsWindows() || strcasecmp(args[0], "/C/Windows/System32/more.com") ||
(((n = __mkntpath(tmppath, widepath)) != -1) && static bool run_pager(char *args[hasatleast 3]) {
(args[1] = gc(utf16to8(widepath, n, 0))))) { char16_t widepath[PATH_MAX];
int n, pid;
if (IsWindows() && !strcasecmp(args[0], "/C/Windows/System32/more.com") &&
(((n = __mkntpath(args[1], widepath)) == -1) ||
!(args[1] = gc(utf16to8(widepath, n, 0))))) {
return false;
}
if ((pid = fork()) != -1) { if ((pid = fork()) != -1) {
putenv("LC_ALL=C.UTF-8"); putenv("LC_ALL=C.UTF-8");
putenv("LESSCHARSET=utf-8"); putenv("LESSCHARSET=utf-8");
@ -58,12 +57,53 @@ void __paginate(int fd, const char *s) {
_Exit(127); _Exit(127);
} }
waitpid(pid, 0, 0); waitpid(pid, 0, 0);
return true;
}
return false;
}
/**
* Displays wall of text in terminal with pagination.
*/
void __paginate(int fd, const char *s) {
int tfd;
char *args[3] = {0};
char tmppath[] = "/tmp/paginate.XXXXXX";
char progpath[PATH_MAX];
if ((args[0] = get_pagerpath(progpath, sizeof(progpath)))) {
if ((tfd = mkstemp(tmppath)) != -1) {
// defer((void*)unlink, tmppath);
write(tfd, s, strlen(s));
close(tfd);
args[1] = tmppath;
if (run_pager(args)) {
unlink(tmppath); unlink(tmppath);
return; return;
} }
}
unlink(tmppath); unlink(tmppath);
} }
} }
write(fd, s, strlen(s)); write(fd, s, strlen(s));
} }
/**
* Displays a file in terminal with pagination
*/
void __paginate_file(int fd, const char *path) {
char *args[3] = {0};
char progpath[PATH_MAX];
if ((args[0] = get_pagerpath(progpath, sizeof(progpath)))) {
args[1] = (char *)path;
if (run_pager(args)) {
return;
}
}
int sfd = open(path, O_RDONLY);
if (sfd != -1) {
ssize_t n;
while ((n = read(sfd, progpath, sizeof(progpath)) > 0)) {
write(fd, progpath, n);
}
}
close(sfd);
}