Z# (Zee-sharp)

A new .NET language

Errors

In light of .NET support we may have to switch to using Exceptions. We could allow use of the error mechanism for Z# code only as an alternative to exceptions although exceptions will always be part of the possible error conditions. A .NET assembly will see the Err<T> type as a ‘OneOf' of the Z# Error object and T.

The error mechanism in Z# is not based on exceptions. On the other hand, naked error value passing is tedious, error prone and does not guarantee handling of the error.

To create a robust error handling mechanism, it has been incorporated into the language. There are two keywords that deal with handling errors and syntax to specify errors can be returned from a function.

Errors are values returned from a function that cannot be ignored by the caller (When .NET exceptions are used, this becomes a problem).

The story starts when an error condition is detected and a function returns an error:

errorFn: (): U8!
    return Error("Failed")

The ! on the return type indicates that you should pay attention, because this function can return an error!

Details on using the Error type can be read here

The calling code needs to have a way to deal with either a good return value from a function or handling the error. The primary keyword to use is catch, like so:

couldWork: (): U8!
    return Error("Sorry, I can't do that Dave.")

v = couldWork() catch(err)
    print(err.message())    // access error message
    return                  // exit control flow
use(v)

The catch keyword is specified after the function that could return the error, and it introduces a scope. This scope is where the handler code goes in case there is an error. The variable name that is used to hand the code the Error is specified inside the parentheses.

TBD: use a more lambda-like syntax: v = fn() catch -> (err)? That would align more with Error Handlers.

Alternate way of handling more complex error conditions using a match expression.

FnErr: (): U8!
    ...

a := match FnErr()
    err: Error -> 0
    custom: MyError -> return custom.fld    // return exits the function
    n: U8 -> n

A (predicted) common pattern is that a function will call many functions itself and as soon as one errors out, the function itself will simply stop and propagate the error to its caller. The try keyword is syntactic sugar for ‘catch(err) return err’ and is specified in front of the function call. It can be used as follows:

MyFunc: (): Bool!
    // propagate error from function
    b := try couldWork() // try -> catch(err) return err
    // b is the plain type - without the Err<> wrapper type.
    use(b)

It takes away some of the noise of simple error handling.

The explicit keywords catch and try -and in a lesser sense match are explicitly chosen to make it clear how these errors are handled in the code.

Note that both catch and try strip of the Err<T> part from the return value of the function. So variable has no Err<T> component/decorator type, it’s just its plain Type (T).

TBD: use the try keyword for converting errors to exceptions?

TBD: Use the catch keyword for convering exceptions to errors?

throwsException: ()
    throw ...
returnsError: (): Void!
    return Error(...)

// using errors
throwsException() catch(err)

// using exceptions
try returnsError()
// what type of exception?

This would require the previous examples to use different keywords. Perhaps tryError and catchError? (then rename try and catch to tryException and catchException?) (try-error and catch-error / try-expection and catch-expection)

Or leave try and catch for exceptions (like in C#) and use catchErr and tryErr for errors?

Use ! to propagate errors instead of try? What to use for exceptions?

actionOrError: (): Void!
    // use ! to propagete errors
    v := couldWork()!
    use(v)

What if multiple transitions between errors and exceptions take place? How do you preserve the original error/exception type? Have an Exception field in Error and store error instances in context data (under a fixed key) in an exception.

It would be benificial to have the option to handle exceptions on per call basis as this would blend better with the (Z#) error handling mechanism.

The catch and try keywords can only be used on functions that actually return errors.

myFunc: (): Bool
    ...

b := myFunc() catch(err)  // error! myFunc does not return errors
    ...

It is also possible to return an Error from a Void function - a function that has no return value.

voidFn: (): Void!       // Void with Err
    return Error("Failed")  // ok, exit function

// handle error on Void! fn
voidFn() catch(err)
    // handle error
    ...

It is also possible to trigger a FatalError anywhere.

noErrorFn: ()       // no Err return value specified
    return Error("Failed")  // error! no error return type
    FatalError("Abort!")    // ok, exit program

You can use the return keyword in a function to exit its execution of course.


Error Scopes

For .NET Interop, we need a way to use the try and catch keywords for a block of code. Perhaps we could pin this to Captures? Although both the capture and the catch keyword introduce an indent.

fn: (): U8!     // can return Error
    a := 42
    try [a]     // any Error out of this block is forwarded
        // code here inside capture
    
    try [a]
        // capture code
    catch(err)
        // capture error code

Alternate idea:

The Err<T> type is not used at all*. try is not used on a per function basis but as a scope (typical try-catch usage) optionally used as a capture block. catch can be still per function or as a try-handler - implicitly wrapping the function call in a try-block. The ‘finally’ block is implicit by using the defer keyword.

*) Error<T> can instead be (re)defined as Error<T>: Exception or T and used as a way to return exceptions or return values from a function invocation.

// starts scope (optionally a capture)
try
    // catch-lambda (with exception type) for local handling
    handle := File.Open("file.txt") catch -> (err: FileNotFoundException)
        // handle FileNotFound
    ...
catch -> (err)      // err: Exception
    ...

Use multiple catch blocks.

try
    ...
// order of appearance is important
catch -> (err: FileNotFoundException)
    ...
catch -> (err: EndOfFileException)
    ...
// Exception is propagated if not caught 

A ‘finally’ handler?

try
    ...
finally 
    ...     // always executed even if error

...         // executed but not if error

Not the same as ‘defer’

try
    defer ...
    errdefer ...

...     // executed but not if error
// defer list always executed even if error
// errdefer list only executed if error

Throw a new Exception:

err = ArgumentException("Not good", "param1")

// throw exception
Error(err)
err.throw()
exit err    // implies exit-fn at minimum
// params in the same order as ctor of Exception (brittle)
Error<ArgumentException>("Not good", "param1")

// use named parameters
Error<ArgumentException>(message="Not good", paramName="param1")

// as tuple
p = { message="Not good", paramName="param1" }
Error<ArgumentException>(p)

Are param names available on imported external assemblies?

Rethrow an existing Exception:

// catch (Exception e) {
//     throw;
// }
catch -> (err)
    err.throw()     // can we track usage to generate 'throw'?
    Error(err)
    err.rethrow()   // make it explicit?

throw and rethrow functions are bound to the Exception type that will be emitted as a C# throw statement.


Error Handlers

We’ll probably drop this for it does not add any value in a .NET context.

Use catch with a handler function instead of an inline handler? Function Interface needed for the error-handler function.

How can the handler-function direct control-flow?

errorHandler: (err: Error): Bool!
    return err          // try functionality, same as...
    return false        // could not handle, propagate err
    return true         // error is handled, continue
    return Error("New Error", err)

errorFn: (): U8!
    return Error("Failed")

v := errorFn() catch(errorHandler)

Test for error type

How to test for the type of error or if a field is available. Cannot use compile-time extensions because type of error is determined at runtime.

errorFn: (p: U8) U8!
    if p = 42
        err42 = MyError     // custom error type
            ...
        return Error("Custom Error Type", err42)
    else
        return Error("Standard Error")

v := errorFn(42) catch(err)
    // control flow: using typeid
    if err#typeId = MyError#typeId
        ...

v := errorFn(42) catch(err)
    // value: match
    a := match err
        myErr: MyError -> ...
        _ -> ...

v := match errorFn(42)
    n: U8 -> // use normal return value
    myErr: MyError -> ...
    _ -> ...

We may need to capture local variables for the error handler to access. How will that work?


Fatal Errors

What some call ‘panic’ also known as exit().

For some type of errors you want to crash and have diagnostics.

// exits program
FatalError("panic!")
FatalError("panic!", err)

What errors are fatal? DivideByZero, StackOverflow, OutOfMemory, … ??


Error Trace

When an error is returned from a function and it naturally bubbles up the call stack, a trace can be made of all the code sites it visits.

This diagnostic information can be useful for tracking down problems.


How to access standard errors (make a list of standard errors?)?