/* * Copyright (c) 2023 DTB * Copyright (c) 2023 Marceline Cramer * SPDX-License-Identifier: AGPL-3.0-or-later * * This program is free software: you can redistribute it and/or modify it under * the terms of the GNU Affero General Public License as published by the Free * Software Foundation, either version 3 of the License, or (at your option) any * later version. * * This program is distributed in the hope that it will be useful, but WITHOUT * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS * FOR A PARTICULAR PURPOSE. See the GNU Affero General Public License for more * details. * * You should have received a copy of the GNU Affero General Public License * along with this program. If not, see https://www.gnu.org/licenses/. */ #include #include /* NULL */ #include /* fprintf(3), perror(3) */ #include /* size_t, EXIT_FAILURE */ #include /* strcmp(3) */ #include /* EX_OSERR, EX_USAGE */ #ifdef __OpenBSD__ # include /* pledge(2) */ #endif char *program_name = "str"; static struct { char *name; int (*f)(int); } ctypes[] = { { "isalnum", isalnum }, { "isalpha", isalpha }, { "isblank", isblank }, { "iscntrl", iscntrl }, { "isdigit", isdigit }, { "isxdigit", isxdigit }, { "isgraph", isgraph }, { "islower", islower }, { "isprint", isprint }, { "ispunct", ispunct }, { "isspace", isspace }, { "isupper", isupper }, { NULL, NULL } /* marks end */ }; static int usage(char *argv0) { (void)fprintf(stderr, "Usage: %s type string...\n", argv0); return EX_USAGE; } int main(int argc, char *argv[]) { size_t ctype; // selected from ctypes.h; index of ctype int retval; // initially fail but becomes success on the first valid char program_name = argv[0] == NULL ? program_name : argv[0]; #ifdef __OpenBSD__ if (pledge("stdio", NULL) == -1) { perror(program_name); return EX_OSERR; } #endif if (argc < 3) { return usage(program_name); } for ( /* iterate ctypes */ ctype = 0; ctypes[ctype].f != NULL /* break at the end of ctypes */ && strcmp(argv[1], ctypes[ctype].name) != 0; /* break at match */ ++ctype ); if (ctypes[ctype].f == NULL) { return usage(argv[0]); } /* iterate args */ for (argv += 2, retval = EXIT_FAILURE; *argv != NULL; ++argv) { for (size_t i = 0; argv[0][i] != '\0'; ++i) { /* iterate arg bytes */ /* First checks if argv[0][i] is valid ASCII; ctypes(3) don't * handle non-ASCII. This is bad. */ if( (unsigned char)argv[0][i] < 0x80 // argv[0][i] is ASCII, && !ctypes[ctype].f(argv[0][i]) // so use ctypes(3) ) { return EXIT_FAILURE; } else { retval = EXIT_SUCCESS; } } } return retval; }