update to cli library

This commit is contained in:
2025-11-30 21:49:35 +01:00
parent 71a27177b4
commit ce4159cab2
+98 -27
View File
@@ -3,17 +3,46 @@
#include <stdbool.h> #include <stdbool.h>
#define UNIQUE __macro_internal_34bba35b8b9b20a75f9881e3795630e25d36e620d9c9741e2e9141ba82ec6ef7__
#define UNIQUE2 __macro_internal_34bba35b8b9b20a75f9881e3795630e25d36e620d9c9741e2e9141ba82ec6ef8__
typedef struct args { typedef struct args {
int count; int count;
char **values; const char **values;
} args_t; } args_t;
args_t make_args(int arg_count, char **args); args_t cli_make_args(int argc, const char **argv); // no crash
bool command(args_t *args, char *command_name); bool cli_command(args_t *args, const char *command_name); // no crash
bool parse_bool(args_t args, char *flag_name); bool cli_bool(args_t args, const char *flag_name); // no crash
char *parse_string(args_t args, char *flag_name); bool cli_opt_str_func(args_t args, const char *flag_name, const char **output, const char *file, int line);
int parse_int(args_t args, char *flag_name); const char *cli_req_str_func(args_t args, const char *flag_name, const char *file, int line);
bool cli_opt_int_func(args_t args, const char *flag_name, int *output, const char *file, int line);
int cli_req_int_func(args_t args, const char *flag_name, const char *file, int line);
#define cli_opt_str(ARGS, FLAG_NAME, OUTPUT) cli_opt_str_func(ARGS, FLAG_NAME, OUTPUT, __FILE__, __LINE__)
#define cli_req_str(ARGS, FLAG_NAME) cli_req_str_func(ARGS, FLAG_NAME, __FILE__, __LINE__)
#define cli_opt_int(ARGS, FLAG_NAME, OUTPUT) cli_opt_int_func(ARGS, FLAG_NAME, OUTPUT, __FILE__, __LINE__)
#define cli_req_int(ARGS, FLAG_NAME) cli_req_int_func(ARGS, FLAG_NAME, __FILE__, __LINE__)
#define with_opt_str(ARGS, FLAG_NAME, VAR) \
for (const char *VAR = NULL; VAR == NULL;) \
for (bool UNIQUE = true; UNIQUE;) \
for ( \
bool UNIQUE2 = (UNIQUE = cli_opt_str(ARGS, FLAG_NAME, &VAR), true); \
UNIQUE2; \
UNIQUE2 = (VAR = ((const char *) 1), UNIQUE = false, false) \
) \
if (UNIQUE)
#define with_opt_int(ARGS, FLAG_NAME, VAR) \
for (int VAR = 0; VAR == 0;) \
for (bool UNIQUE = true; UNIQUE;) \
for ( \
bool UNIQUE2 = (UNIQUE = cli_opt_int(ARGS, FLAG_NAME, &VAR), true); \
UNIQUE2; \
UNIQUE2 = (VAR = 1, UNIQUE = false, false) \
) \
if (UNIQUE)
#ifdef CLI_IMPL #ifdef CLI_IMPL
@@ -21,8 +50,7 @@ int parse_int(args_t args, char *flag_name);
#include <stdlib.h> #include <stdlib.h>
#include <string.h> #include <string.h>
bool cli_command(args_t *args, const char *command_name) {
bool command(args_t *args, char *command_name) {
if (args->count == 0) { if (args->count == 0) {
return false; return false;
} }
@@ -34,7 +62,7 @@ bool command(args_t *args, char *command_name) {
return false; return false;
} }
bool parse_bool(args_t args, char *flag_name) { bool cli_bool(args_t args, const char *flag_name) {
for ( int i = 0; i < args.count; i++ ) { for ( int i = 0; i < args.count; i++ ) {
if (strcmp(args.values[i], flag_name) == 0) { if (strcmp(args.values[i], flag_name) == 0) {
return true; return true;
@@ -43,44 +71,87 @@ bool parse_bool(args_t args, char *flag_name) {
return false; return false;
} }
char *parse_string(args_t args, char *flag_name) { bool cli_opt_str_func(args_t args, const char *flag_name, const char **output, const char *file, int line) {
bool is_found = false; bool is_flag_name_found = false;
int flag_name_index; int flag_name_index;
for ( int i = 0; i < args.count; i++ ) { for ( int i = 0; i < args.count; i++ ) {
if (strcmp(args.values[i], flag_name) == 0) { if (strcmp(args.values[i], flag_name) == 0) {
is_found = true; is_flag_name_found = true;
flag_name_index = i; flag_name_index = i;
break; break;
} }
} }
if (!is_found) { static const char *dummy = "";
fprintf(stderr, "Arg %s not found!\n", flag_name); if (!is_flag_name_found) {
exit(1); *output = dummy;
return false;
} }
int flag_value_index = flag_name_index + 1; int flag_value_index = flag_name_index + 1;
if (flag_value_index >= args.count) { if (args.count <= flag_value_index) {
fprintf(stderr, "No value provided for %s!\n", flag_name); fprintf(stderr, "%s:%d: No value provided for %s!\n", file, line, flag_name);
exit(1); exit(1);
} }
return args.values[flag_value_index]; *output = args.values[flag_value_index];
return true;
} }
int parse_int(args_t args, char *flag_name) { const char *cli_req_str_func(args_t args, const char *flag_name, const char *file, int line) {
char *text = parse_string(args, flag_name); const char *value;
for ( char *c = text; (*c) != '\0'; c++ ) { if (cli_opt_str_func(args, flag_name, &value, file, line)) {
return value;
} else {
fprintf(stderr, "%s:%d: Required string %s not provided!\n", file, line, flag_name);
exit(1);
}
return "";
}
bool cli_opt_int_func(args_t args, const char *flag_name, int *output, const char *file, int line) {
const char *text = "";
if (cli_opt_str_func(args, flag_name, &text, file, line)) {
if (
text[0] != '-' &&
text[0] < '0' &&
'9' < text[0]
) {
fprintf(stderr, "%s:%d: Invalid integer %s!\n", file, line, text);
exit(1);
}
if (
text[0] == '-' &&
text[1] == '\0'
) {
fprintf(stderr, "%s:%d: Invalid integer %s!\n", file, line, text);
exit(1);
}
for ( const char *c = &text[1]; (*c) != '\0'; c++ ) {
if ((*c) < '0' || '9' < (*c)) { if ((*c) < '0' || '9' < (*c)) {
fprintf(stderr, "Invalid integer %s!\n", text); fprintf(stderr, "%s:%d: Invalid integer %s!\n", file, line, text);
exit(1); exit(1);
} }
} }
int result = atoi(text); *output = atoi(text);
return result; return true;
}
*output = 0;
return false;
} }
args_t make_args(int arg_count, char **args) { int cli_req_int_func(args_t args, const char *flag_name, const char *file, int line) {
int value;
if (cli_opt_int_func(args, flag_name, &value, file, line)) {
return value;
} else {
fprintf(stderr, "%s:%d: Required int %s not provided!\n", file, line, flag_name);
exit(1);
}
return 0;
}
args_t cli_make_args(int argc, const char **argv) {
return (args_t) { return (args_t) {
.count = arg_count-1, .count = argc-1,
.values = args+1, .values = argv+1,
}; };
} }