2023-04-11 00:14:18 +00:00
|
|
|
#include <libgen.h>
|
2023-04-10 04:58:48 +00:00
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
2023-04-10 18:36:45 +00:00
|
|
|
#include <string.h>
|
2023-04-10 18:51:49 +00:00
|
|
|
#include <unistd.h>
|
2023-04-10 04:58:48 +00:00
|
|
|
|
2023-04-13 03:46:53 +00:00
|
|
|
#include "arg.h"
|
2023-04-11 00:14:18 +00:00
|
|
|
#include "file.h"
|
2023-04-11 03:08:11 +00:00
|
|
|
#include "util.h"
|
2023-04-10 04:58:48 +00:00
|
|
|
|
2023-04-10 16:51:42 +00:00
|
|
|
#define INVERT_T "\x1b[7m"
|
|
|
|
#define UINVERT_T "\x1b[27m"
|
2023-04-10 18:36:45 +00:00
|
|
|
#define GREY "\x1b[90m"
|
|
|
|
#define RESET "\x1b[0m"
|
2023-04-10 16:51:42 +00:00
|
|
|
|
2023-04-13 13:24:41 +00:00
|
|
|
void run(FILE *fp, char *filename, bool tty) {
|
2023-04-12 13:38:38 +00:00
|
|
|
const char *invert_t = conf.color ? INVERT_T : "";
|
|
|
|
const char *uinvert_t = conf.color ? UINVERT_T : "";
|
|
|
|
const char *grey = conf.color ? GREY : "";
|
|
|
|
const char *reset = conf.color ? RESET : "";
|
2023-04-12 02:53:21 +00:00
|
|
|
|
2023-04-11 00:14:18 +00:00
|
|
|
struct filedata f;
|
|
|
|
f = readfile(fp);
|
2023-04-10 16:51:42 +00:00
|
|
|
|
2023-04-11 23:17:36 +00:00
|
|
|
if (tty) {
|
2023-04-13 13:03:33 +00:00
|
|
|
char *addon = f.binary ? "<binary>" : "";
|
|
|
|
fprintf(stderr, "\r\x1b[2K%s%s%s%s\r\n", invert_t, basename(filename),
|
|
|
|
addon, uinvert_t);
|
2023-04-11 23:17:36 +00:00
|
|
|
}
|
2023-04-10 18:36:45 +00:00
|
|
|
|
2023-04-11 00:14:18 +00:00
|
|
|
int lcpad = intlen(f.lc);
|
2023-04-10 18:36:45 +00:00
|
|
|
|
2023-04-11 00:14:18 +00:00
|
|
|
f.lc = 0;
|
2023-04-10 18:36:45 +00:00
|
|
|
char pc = '\0';
|
2023-04-11 00:14:18 +00:00
|
|
|
char c;
|
2023-04-11 16:32:38 +00:00
|
|
|
for (size_t i = 0; i < f.len; i++) {
|
2023-04-11 00:14:18 +00:00
|
|
|
c = f.buf[i];
|
2023-04-10 04:58:48 +00:00
|
|
|
|
2023-04-13 03:46:53 +00:00
|
|
|
if ((conf.lines && tty && !f.binary) && (pc == '\n' || i == 0)) {
|
2023-04-11 00:14:18 +00:00
|
|
|
f.lc++;
|
2023-04-11 16:32:38 +00:00
|
|
|
|
2023-04-11 00:14:18 +00:00
|
|
|
int padlen = lcpad - intlen(f.lc);
|
2023-04-10 18:36:45 +00:00
|
|
|
char padding[padlen];
|
2023-04-11 23:17:36 +00:00
|
|
|
|
|
|
|
if (padlen)
|
|
|
|
memset(padding, ' ', padlen);
|
2023-04-11 00:14:18 +00:00
|
|
|
|
2023-04-12 02:53:21 +00:00
|
|
|
fprintf(stderr, "\r%s%s%d:%s ", grey, padlen > 0 ? padding : "", f.lc,
|
|
|
|
reset); // padlen < 1 causes undefined
|
2023-04-10 18:36:45 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
pc = c;
|
|
|
|
printf("%c", c);
|
|
|
|
}
|
2023-04-10 16:51:42 +00:00
|
|
|
|
2023-04-11 23:17:36 +00:00
|
|
|
fflush(stdout); // prevent timing inconsistencies between stdout and stderr
|
|
|
|
|
2023-04-11 00:49:43 +00:00
|
|
|
if (tty) {
|
|
|
|
float rounded;
|
2023-04-11 23:17:36 +00:00
|
|
|
char *format = formatbytes(f.len, &rounded);
|
2023-04-10 16:51:42 +00:00
|
|
|
|
2023-04-12 02:53:21 +00:00
|
|
|
char *cnewline = c == '\n' ? "" : "\n";
|
|
|
|
fprintf(stderr, "\r%s%s%.2f %s%s\r\n", cnewline, invert_t, rounded, format,
|
|
|
|
uinvert_t);
|
2023-04-11 00:49:43 +00:00
|
|
|
}
|
2023-04-14 13:43:05 +00:00
|
|
|
|
|
|
|
free(f.buf);
|
2023-04-10 18:36:45 +00:00
|
|
|
}
|
|
|
|
|
2023-04-13 03:46:53 +00:00
|
|
|
void initconf(void) {
|
|
|
|
conf.color = true;
|
|
|
|
conf.lines = true;
|
2023-04-13 13:24:41 +00:00
|
|
|
conf.has_read_stdin = false;
|
|
|
|
}
|
|
|
|
|
2023-04-13 13:37:24 +00:00
|
|
|
void clearstdin(void) {
|
2023-04-13 13:24:41 +00:00
|
|
|
// from
|
|
|
|
// https://stackoverflow.com/questions/7898215/how-to-clear-input-buffer-in-c
|
|
|
|
fseek(stdin, 0, SEEK_END);
|
2023-04-13 03:46:53 +00:00
|
|
|
}
|
2023-04-12 13:38:38 +00:00
|
|
|
|
2023-04-10 18:36:45 +00:00
|
|
|
int main(int argc, char *argv[]) {
|
2023-04-12 13:38:38 +00:00
|
|
|
initconf();
|
|
|
|
|
2023-04-13 03:46:53 +00:00
|
|
|
// init no_color first so that args take priority
|
2023-04-12 02:53:21 +00:00
|
|
|
char *no_color = getenv("NO_COLOR");
|
|
|
|
|
|
|
|
if (no_color != NULL && no_color[0] != '\0') {
|
2023-04-13 03:46:53 +00:00
|
|
|
conf.color = false;
|
2023-04-12 02:53:21 +00:00
|
|
|
}
|
|
|
|
|
2023-04-13 13:24:41 +00:00
|
|
|
bool tty = isatty(STDOUT_FILENO);
|
2023-04-11 03:06:08 +00:00
|
|
|
if (argc > 1) {
|
2023-04-13 03:46:53 +00:00
|
|
|
int offset = parseargs(argc, argv);
|
2023-04-13 13:03:33 +00:00
|
|
|
for (int i = offset; i < argc; i++) {
|
2023-04-13 13:24:41 +00:00
|
|
|
if (*argv[i] == '-') {
|
|
|
|
if (conf.has_read_stdin)
|
|
|
|
clearstdin();
|
|
|
|
conf.has_read_stdin = true;
|
|
|
|
run(stdin, "stdin", tty);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2023-04-11 03:06:08 +00:00
|
|
|
FILE *fp = fopen(argv[i], "rb");
|
2023-04-11 03:47:45 +00:00
|
|
|
if (fp == NULL)
|
|
|
|
die(argv[i]);
|
2023-04-11 23:17:36 +00:00
|
|
|
run(fp, argv[i], tty);
|
2023-04-11 03:06:08 +00:00
|
|
|
fclose(fp);
|
2023-04-10 18:36:45 +00:00
|
|
|
|
2023-04-13 13:03:33 +00:00
|
|
|
if (tty && (i + 1 != argc)) {
|
|
|
|
fprintf(stderr, "\r\n"); // separate concurrent files in tty
|
2023-04-11 03:06:08 +00:00
|
|
|
}
|
2023-04-10 18:36:45 +00:00
|
|
|
}
|
2023-04-13 13:03:33 +00:00
|
|
|
|
|
|
|
if (offset == argc) {
|
2023-04-13 13:24:41 +00:00
|
|
|
run(stdin, "stdin", tty);
|
2023-04-13 13:03:33 +00:00
|
|
|
}
|
2023-04-11 03:06:08 +00:00
|
|
|
} else {
|
2023-04-13 13:24:41 +00:00
|
|
|
run(stdin, "stdin", tty); // for piped-input or repl-like behavior
|
2023-04-10 18:36:45 +00:00
|
|
|
}
|
2023-04-10 04:58:48 +00:00
|
|
|
|
2023-04-13 03:46:53 +00:00
|
|
|
return EXIT_SUCCESS;
|
2023-04-09 17:42:44 +00:00
|
|
|
}
|