lat/src/main.c

204 lines
4.2 KiB
C
Raw Normal View History

#include "arg.h"
2023-04-11 00:14:18 +00:00
#include "file.h"
2023-04-17 14:23:52 +00:00
#include "process.h"
#include "types.h"
2023-04-11 03:08:11 +00:00
#include "util.h"
2023-05-01 22:07:53 +00:00
#include <stdio.h>
#include <stdlib.h>
#include <unistd.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-19 14:46:17 +00:00
FILE *st;
FILE *err;
2023-04-29 19:08:36 +00:00
struct colors {
char *invert_t;
char *grey;
char *reset;
};
struct colors c;
void initcolor(void) {
c.invert_t = conf.color ? INVERT_T : "";
c.grey = conf.color ? GREY : "";
c.reset = conf.color ? RESET : "";
}
void printheadertop(char *filename, bool binary) {
char *name = conf.name == NULL ? filename : conf.name;
char *addon = binary ? "<binary>" : "";
if (conf.isstdin && !conf.pager)
fprintf(err, "\x1b[2K\r%s%s%s%s\n", c.invert_t, name, addon, c.reset);
else
fprintf(err, "%s%s%s%s\n", c.invert_t, name, addon, c.reset);
}
void printheaderbottom(size_t buflen) {
float rounded;
char *format = formatbytes(buflen, &rounded);
fprintf(err, "%s%.2f %s%s\n", c.invert_t, rounded, format, c.reset);
}
2023-04-13 13:24:41 +00:00
void run(FILE *fp, char *filename, bool tty) {
2023-04-29 19:08:36 +00:00
initcolor();
2023-04-12 02:53:21 +00:00
2023-04-11 00:14:18 +00:00
struct filedata f;
2023-04-29 19:08:36 +00:00
f = readfile(fp, conf.isstdin);
2023-04-10 16:51:42 +00:00
2023-04-19 14:46:17 +00:00
if (conf.pager) {
st = popen("less", "w");
err = st;
if (st == NULL)
die("popen 'less'");
}
if (conf.literal) {
err = st;
}
2023-04-18 12:58:23 +00:00
if (conf.force_binary > 0) {
f.binary = true;
} else if (conf.force_binary == 0) {
f.binary = false;
2023-04-18 01:07:24 +00:00
}
2023-04-18 02:04:03 +00:00
if (conf.headers) {
2023-04-29 19:08:36 +00:00
printheadertop(filename, f.binary);
}
2023-04-10 18:36:45 +00:00
2023-04-29 19:08:36 +00:00
// any/all processing to be done
2023-04-17 14:23:52 +00:00
conf.process = (tty && !f.binary);
if (conf.process) { // file display processing
loadlines(&f);
}
2023-04-10 18:36:45 +00:00
2023-04-17 14:23:52 +00:00
if (conf.process) {
2023-04-18 01:07:24 +00:00
int linecount = 1;
2023-04-17 14:23:52 +00:00
for (int i = 0; i < f.lc; i++) {
if (conf.lines) {
2023-04-29 19:42:42 +00:00
{ // line numbers
char *padding = linepad(linecount, f.lc);
fprintf(err, "%s%s%d│%s ", c.grey, padding, i + 1, c.reset);
free(padding);
}
2023-04-19 14:46:17 +00:00
fwrite(f.lines[i].buf, 1, f.lines[i].len, st);
fprintf(st, "\n");
2023-04-17 14:23:52 +00:00
linecount++;
} else {
2023-04-19 14:46:17 +00:00
fprintf(st, "%s\n", f.lines[i].buf);
2023-04-17 14:23:52 +00:00
}
free(f.lines[i].buf);
}
} else {
2023-04-19 14:46:17 +00:00
fwrite(f.buf, 1, f.buflen, st);
2023-04-19 16:43:54 +00:00
fflush(st);
2023-04-29 19:42:42 +00:00
if (tty)
fwrite("\n", 1, 1, err);
2023-04-17 14:23:52 +00:00
}
free(f.buf);
2023-05-01 22:07:53 +00:00
free(f.lines);
2023-04-10 16:51:42 +00:00
2023-04-19 14:46:17 +00:00
fflush(st); // prevent timing inconsistencies between st and err
2023-04-19 16:43:54 +00:00
fflush(err);
2023-04-18 02:04:03 +00:00
if (conf.headers) {
2023-04-29 19:08:36 +00:00
printheaderbottom(f.buflen);
2023-04-19 14:46:17 +00:00
}
if (conf.pager) {
pclose(st); // err is already the same as st
2023-04-11 00:49:43 +00:00
}
2023-04-10 18:36:45 +00:00
}
void initconf(void) {
2023-04-18 12:58:23 +00:00
conf.force_binary = -1;
2023-04-29 19:08:36 +00:00
conf.isstdin = false;
2023-04-18 02:04:03 +00:00
conf.has_read_stdin = false;
2023-04-19 14:46:17 +00:00
conf.pager = false;
conf.literal = false;
2023-04-17 14:23:52 +00:00
conf.process = true;
2023-04-18 02:04:03 +00:00
conf.headers = true;
conf.color = true;
conf.lines = true;
conf.name = NULL;
2023-04-29 19:42:42 +00:00
conf.extension = NULL;
2023-04-13 13:24:41 +00:00
}
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-10 18:36:45 +00:00
int main(int argc, char *argv[]) {
initconf();
2023-04-19 14:46:17 +00:00
st = stdout;
err = stderr;
// 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') {
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-29 19:42:42 +00:00
int offset = parseargs(argc, argv);
conf.headers = conf.headers && tty;
printf("set extension to '%s'\n", conf.extension);
2023-04-11 03:06:08 +00:00
if (argc > 1) {
2023-04-29 19:08:36 +00:00
tty = tty || conf.literal;
2023-04-19 14:46:17 +00:00
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;
2023-04-29 19:08:36 +00:00
conf.isstdin = true;
2023-04-13 13:24:41 +00:00
run(stdin, "stdin", tty);
if (tty && (i + 1 != argc)) {
fprintf(err, "\n"); // separate concurrent files in tty
}
2023-04-13 13:24:41 +00:00
continue;
}
2023-04-29 19:08:36 +00:00
conf.isstdin = false;
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]);
run(fp, argv[i], tty);
2023-04-11 03:06:08 +00:00
fclose(fp);
2023-04-13 13:03:33 +00:00
if (tty && (i + 1 != argc)) {
fprintf(err, "\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-29 19:08:36 +00:00
conf.isstdin = true;
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-29 19:08:36 +00:00
conf.isstdin = true;
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
return EXIT_SUCCESS;
2023-04-09 17:42:44 +00:00
}