1
0
Fork 0

cat(1p): made simpler, added error handling

This commit is contained in:
Emma Tebibyte 2023-08-10 15:24:07 -06:00
parent 27a227e365
commit b360b823ac
Signed by untrusted user: emma
GPG Key ID: 6D661C738815E7DD
1 changed files with 41 additions and 19 deletions

View File

@ -30,18 +30,12 @@
int main(int argc, char *argv[]) {
bool u = false;
int opt;
int i = 1;
FILE *file;
char *usage_text = "(-u) [file...]";
int buf_size = strlen(argv[0]) + strlen("Usage: ") + strlen(usage_text) + 3;
char *usage = calloc(buf_size, buf_size);
if ((
snprintf(usage, buf_size, "Usage: %s (-u) [file...]\n", argv[0])
) < 0 ) {}
extern int optind;
while ((opt = getopt(argc, argv, "u")) != -1) {
switch(opt) {
switch (opt) {
/*
* From cat(1p):
*
@ -52,14 +46,15 @@ int main(int argc, char *argv[]) {
u = true;
break;
default:
break;
printf("Usage: %s (-u) file...\n", argv[0]);
return EX_USAGE;
}
}
argv += optind;
argc -= optind;
for (int i = 0; i < argc; i++) {
int byte = 0;
for (i = optind; i < argc; i++) {
int byte = 0; // variable for storing bytes as they are read
char buf[4096]; // buffer for buffered
int p = 0; // index counter for bytes in buffered reading
/*
* From cat(1p):
*
@ -71,23 +66,50 @@ int main(int argc, char *argv[]) {
* but shall accept multiple occurrences of '-' as a file
* operand.
*/
if (argv[i] == "-" || argc == 0) {
if (argv[i] == "-" || argc == 1) {
file = stdin;
} else if ((file = fopen(argv[i], "r")) == NULL) {
// TODO: Add error handling
fputs(usage, stderr);
return EX_NOINPUT;
switch (errno) {
case EACCES:
printf("%s: %s: Permission denied.\n", argv[0], argv[i]);
return EX_NOINPUT;
case EISDIR:
printf("%s: %s: Is a directory.\n", argv[0], argv[i]);
return EX_NOINPUT;
case ELOOP:
printf("%s: %s: Is a symbolic link loop.\n", argv[0], argv[i]);
return EX_UNAVAILABLE;
case EMFILE:
printf("%s: Internal error.\n", argv[0]);
return EX_SOFTWARE;
case ENOENT: case ENOTDIR: case ENXIO:
printf("%s: %s: No such file or directory.\n", argv[0], argv[i]);
return EX_NOINPUT;
default:
printf("%s: Unknown error.\n", argv[0]);
return EX_UNAVAILABLE;
}
}
for (char *buf = calloc(4096, 1); byte != EOF; fputs(buf, stdout)) {
while (byte != EOF) {
byte = fgetc(file);
if (u) {
putchar(byte);
} else {
snprintf(buf, 4096, "%c", byte);
if (p > sizeof(buf)) {
fputs(buf, stdout);
p = 0;
}
buf[p] = byte;
p += 1;
}
}
fputs(buf, stdout);
fflush(stdout);
if (file != stdin) { fclose(file); }
}