Skip to content

STB-style helper library for console input with error handling.

License

Notifications You must be signed in to change notification settings

5cover/stb_cori

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 

History

13 Commits
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 

Repository files navigation

Cori

STB-style helper library for console input with error handling.

Usage

Cori is basically a bunch of functions.

To choose the appropriate function, you can follow these steps:

1. What type of data do I need from the user?

Cori currently supports the following conversions:

Note

For integer conversions, type names ending in _base indicate a base integer argument. The supported values are same as for the standard library's string to integer function family (strtol, strtoul...).
Other type names assume 0 as the base, meaning the base is detected automatically.

Signed integers

Type name Expected input Actual type Additional parameters
intmax An integer in range [INTMAX_MIN ; INTMAX_MAX] intmax_t
longlong An integer in range [LLONG_MIN ; LLONG_MAX] long long
long An integer in range [LONG_MIN ; LONG_MAX] long
int An integer in range [INT_MIN ; INT_MAX] int
short An integer in range [SHRT_MIN ; SHRT_MAX] short
intmax_base An integer in range [INTMAX_MIN ; INTMAX_MAX] intmax_t the number base (int)
longlong_base An integer in range [LLONG_MIN ; LLONG_MAX] long long the number base (int)
long_base An integer in range [LONG_MIN ; LONG_MAX] long the number base (int)
int_base An integer in range [INT_MIN ; INT_MAX] int the number base (int)
short_base An integer in range [SHRT_MIN ; SHRT_MAX] short the number base (int)

Unsigned integers

Type name Expected input Actual type Additional parameters
uintmax An integer in range [0 ; UINTMAX_MAX] uintmax_t
ulonglong An integer in range [0 ; ULLONG_MAX] unsigned long long
ulong An integer in range [0 ; ULONG_MAX] unsigned long
uint An integer in range [0 ; UINT_MAX] unsigned int
ushort An integer in range [0 ; USHRT_MAX] unsigned short
uintmax_base An integer in range [0 ; UINTMAX_MAX] uintmax_t the number base (int)
ulonglong_base An integer in range [0 ; ULLONG_MAX] unsigned long long the number base (int)
ulong_base An integer in range [0 ; ULONG_MAX] unsigned long the number base (int)
uint_base An integer in range [0 ; UINT_MAX] unsigned int the number base (int)
ushort_base An integer in range [0 ; USHRT_MAX] unsigned short the number base (int)

Floating point

Type name Expected input Actual type Additional parameters
longdouble A real number long double
double A real number double
float A real number float

Text

Type name Expected input Actual type Additional parameters
character Any character except a newline char
until A string ended by a specific character char * (must be freed) the delimiter (marks the end of the string, not included) (char)
line A string ended by a newline char * (must be freed)
linel A string of a specific length terminated by a newline char * the buffer for the string char *, the size of the buffer (thus the maximum length of the string plus 1) (size_t)

Other

Type name Expected input Actual type Additional parameters
boolean A character, present in one of the specified sets of characters representing either true or false bool the set of characters representing true (char const *), the set of characters representing false (char const *)

2. How do I handle erroneous input?

Cori offers several possibilities for responding to erroneous input. Different functions for each conversion exist based on how much control you want over the error handling.

Input can be erroneous in 2 ways:

  • Type error: the issued string cannot be meaningfully converted to the target type without assuming the intent of the user.
    For example, while it may technically be possible to retrieve a single character if the user entered multiple, it's best to cause an error as it's certainly a mistake from the user.
    When the input cannot be converted at all, such as trying to convert dfsfds into an integer, it's also a type error.
  • Logic error: the issued string has been converted successfully, but the resulting value isn't valid according to the logic of the program.
    For example, a file path that does not exist for a text editor.

1. I don't care about erroneous input

This is the simplest approach. Useful for quick scripts and test programs.

Function name prefix: read_<typename>

Behavior on erroneous input: calls the NO-OP input error handler and retries.

Sample code:

printf("How old are you ? ");
int result = read_int();

Errors are ignored. This means the user will get successive prompts with no message between them.

2. I want to call a function on erroneous input and retry

This approach is useful for sharing error handling logic across the program.

Function name: read_<typename>_or

Behavior on erroneous input: calls the specified input error handler and retries.

These functions take a function pointer parameter of type InputErrorHandler, which is equivalent to void (*InputErrorHandler)(InputError).

So input error handlers take a single argument, the error that occured, and return nothing.

InputError is an enumeration representing all possible error conditions. Its values are prefixed with IE_.

Sample code:

printf("How old are you ? ");
int result = read_int_or(&inputErrorHandler);

Sample input eror handler:

void inputErrorHandler(InputError error)
{
    char *s = NULL;
    switch (error) {
    case IE_EMPTY: s = "Empty input"; break;
    case IE_EOF: s = "EOF reached"; break;
    case IE_INVALID_BOOLEAN_CHAR: s = "Invalid boolean char"; break;
    case IE_MULTIPLE_CHARS: s = "Multiple chars"; break;
    case IE_NOT_A_NUMBER: s = "Not a number"; break;
    case IE_NUMBER_OUT_OF_BOUNDS: s = "Number out of bounds"; break;
    case IE_OUT_OF_MEMORY: s = "Out of memory"; break;
    case IE_UNSUPPORTED_BASE: s = "Unsupported numeric base"; break;
    case IE_OK: return; // don't print if IE_OK
    }
    puts(s);
}

Two predefined input error handler functions are available:

name description
cori_handle_error_noop Does nothing
cori_handle_error_stderr_msg Prints a message to standard error

3. I want maximum control: read the input once, and let me do the rest

This approach is mainly useful for logic error handling and custom input restrictions.

Function name: tryRead_<typename>

Behavior on erroneous input: returns the appropriate InputError value.

These functions return an error code instead of the converted result, meaning that the result variable has to be passed as a pointer argument.

They follow the usual "try-function" pattern (link?) with an out argument.

They are especially useful in loops. Indeed, an InputError value can be treated as a boolean indicating whether or not an error occured, as:

  • IE_OK, the value that indicates that no error occured, has the numeric value 0, which means it evaluates to false
  • Any other value has a non-zero numeric value, meaning it evaluates to true

Sample code:

InputError error;
int result;
do {
    printf("How old are you ? ");
    error = tryRead_int(&result);
    inputErrorHandler(error);
} while (error);
printf("You're %d.\n", (int)result);

Warning

Since tryRead functions take a pointer, you must pass a pointer to a type identical to or of the same size as the actual result type.

3. Where do I want to read from?

You may want to read from either stdin, or some other stream.

Functions that end with _from take a FILE * parameter, the stream to read from.

Other functions read from stdin.

Reserved identifers

Any identifier starting with _cori_ is defined as private and should not be referenced.

About

STB-style helper library for console input with error handling.

Resources

License

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published