Skip to content

geode-sdk/result

Repository files navigation

Result

A result class for C++ implemented mainly for Geode.

The purpose of the class is wrapping the return value of a function which may fail, thus passing the error as part of the return type.

The design of this library is heavily inspired by rust's Result type

Example

Result<int> integerDivision(int a, int b) {
    if (b == 0) {
        return Err("Division by zero");
    }
    return Ok(a / b);
}

int main() {
    int value = integerDivision(3, 2).unwrapOrDefault();
    assert(value == 1);

    value = integerDivision(3, 0).unwrapOr(0);
    assert(value == 0);
}

Extracting the values

There are tons of conveniency tools to error check and extract the value from a Result. You are encouraged to use them.

Returning early

Here are the convenience utils for returning early from a Result returning function:

Result<int> myFunction() {
    // !! Only on Clang
    // Returns Err early from the function if the result is an error,
    // otherwise passes the value
    int p1 = GEODE_UNWRAP(integerDivision(3, 2));
    assert(p1 == 1);

    // You can use this macro on MSVC as well if you don't need the value
    // it will return Err early if the result is an error
    GEODE_UNWRAP(integerDivision(3, 2));

    // Returns Err early from the function if the result is an error,
    // otherwise sets the value into the variable
    GEODE_UNWRAP_INTO(int p2, integerDivision(3, 2));
    assert(p2 == 1);

    return Ok(0);
}

Entering into an if block

Here are the convenience utils for entering into an if block with the underlying value:

int main() {
    // Only enters the block if the result is ok,
    // setting the value into the variable
    // Requires ok value to be default constructible
    if (GEODE_UNWRAP_IF_OK(p3, integerDivision(3, 2))) {
        assert(p3 == 1);
    }

    int p4 = 0;
    if (GEODE_UNWRAP_INTO_IF_OK(p4, integerDivision(3, 2))) {
        assert(p4 == 1);
    }

    // Only enters the block if the result is an error,
    // setting the value into the variable
    // Requires err value to be default constructible
    if (GEODE_UNWRAP_IF_ERR(e1, integerDivision(3, 0))) {
        assert(e1 == "Division by zero");
    }

    std::string e2;
    if (GEODE_UNWRAP_INTO_IF_ERR(e2, integerDivision(3, 0))) {
        assert(e2 == "Division by zero");
    }

    // Enters the first block if the result is ok,
    // otherwise enters the second block
    // Requires both ok and err values to be default constructible
    if (GEODE_UNWRAP_EITHER(p5, err, integerDivision(3, 2))) {
        assert(p5 == 1);
    } else {
        assert(false);
    }

    if (GEODE_UNWRAP_EITHER(p6, err, integerDivision(3, 0))) {
        assert(false);
    } else {
        assert(err == "Division by zero");
    }

    int p7 = 0;
    std::string e3;
    if (GEODE_UNWRAP_INTO_EITHER(p7, e3, integerDivision(3, 2))) {
        assert(p7 == 1);
    } else {
        assert(false);
    }
}

Setting the value inline

Here are the convenience utils for setting a value inline with manually handling the error:

int main() {
    // Enters the trailing block if the result is an error,
    // otherwise sets the value into the variable
    // Requires both ok and err values to be default constructible
    GEODE_UNWRAP_OR_ELSE(p8, err, integerDivision(3, 2)) {
        return -1;
    }
    assert(p8 == 1);

    GEODE_UNWRAP_OR_ELSE(p9, err, integerDivision(3, 0)) {
        p9 = -1;
    }
    assert(p9 == -1);

    int p10 = 0;
    GEODE_UNWRAP_INTO_OR_ELSE(p10, err, integerDivision(3, 2)) {
        return -1;
    }
    assert(p10 == 1);
}

Builtin functions

And here are the functions built into the Result to extract the value:

int main() {
    // Returns the value if the result is ok,
    // otherwise returns the default value for type
    int p11 = integerDivision(3, 0).unwrapOrDefault();
    assert(p11 == 0);

    // Returns the value if the result is ok,
    // otherwise returns the passed value
    int p12 = integerDivision(3, 0).unwrapOr(-1);
    assert(p12 == -1);

    // Returns the value if the result is ok,
    // otherwise returns the result of the operation
    int p13 = integerDivision(3, 0).unwrapOrElse([](){
        return -1;
    });
    assert(p13 == -1);

    // NOT RECOMMENDED!!!
    // Returns the value if the result is ok,
    // otherwise **throws an exception**
    int p14 = integerDivision(3, 2).unwrap();
    std::string e4 = integerDivision(3, 0).unwrapErr();
}

Manipulating the values

There are lots of ways to manipulate a Result to better fit the required use case. Feel free to use as you please.

Chaining results

Here are the convenience utils for chaining results:

int main() {
    // Returns the passed result if the result is ok,
    // otherwise returns the result of itself
    int v1 = integerDivision(3, 2).and_(integerDivision(5, 2)).unwrapOrDefault();
    assert(v1 == 2);

    // Returns the passed result if the result is err,
    // otherwise returns the result of itself
    int v2 = integerDivision(3, 2).or_(integerDivision(5, 2)).unwrapOrDefault();
    assert(v2 == 1);

    // Returns the result of the operation if the result is ok,
    // otherwise returns the result of itself
    int v3 = integerDivision(3, 2).andThen([](){
        return Ok(10);
    }).unwrapOrDefault();
    assert(v3 == 10);

    // Returns the result of the operation if the result is err,
    // otherwise returns the result of itself
    int v4 = integerDivision(3, 0).orElse([](){
        return Ok(-10);
    }).unwrapOrDefault();
    assert(v4 == 10);
}

Mapping values

Here are the convenience utils for mapping values:

int main() {
    // Maps the value if the result is ok,
    // otherwise returns the result of itself
    if (GEODE_UNWRAP_IF_OK(v5, integerDivision(3, 2).map([](int v){
        return v * 2;
    }))) {
        assert(v5 == 2);
    }

    // Maps the error if the result is err,
    // otherwise returns the result of itself
    if (GEODE_UNWRAP_IF_ERR(e1, integerDivision(3, 0).mapErr([](auto const& e){
        return e + "!";
    }))) {
        assert(e1 == "Division by zero!");
    }

    // Maps the value if the result is ok,
    // otherwise returns the result of the operation
    int v6 = integerDivision(3, 2).mapOrElse([](){
        return -1;
    }, [](int v){
        return v * 2;
    });
    assert(v6 == 2);

    // Maps the value if the result is ok,
    // otherwise returns the given value
    int v7 = integerDivision(3, 2).mapOr(-1, [](int v){
        return v * 2;
    });
    assert(v7 == 2);
}