* Added endopint to return the logs from a run.
* Refactored the `pathVar & "/" & pathvar` pattern into `pathVar / pathVar`
using the `ospaths` module. Cleaner code, more resistant to extra `/` bugs.
* Added endpoints and core methods to list artifacts for a build, as well as to
retrieve specific artifacts.
* Fixed a problem with the `complete` status being overloaded. The problem was
that in the case of a multi-step build all of the prerequisite steps will
return a state of `complete` which will get recorded in the status file for
the run. The run will continue, but anyone watching the run state file (via
the API for example) had no definitive way to tell the difference between a
sub-step completing and the requested (last) step completing. This was caught
in the functional tests (race condition based on when it polls for the run
status). The fix was to introduce a new build state: `stepComplete`. The
inner `doRun` procedure uses this instead of `complete`. Now the only place
that `complete` is set is at the end of the original call to `run`, right
before the worker terminates. It checks the last result (from the originally
requested step) and if this result is `stepComplete` it "finishes" the build
by setting the state to `complete`. Because this is the only place where
`complete` is set, an observer is now guaranteed not to see `complete` until
all steps have run successfully.
* Fixed a long-standing bug with the request handling logic in error cases
(like requested resources not being available). Issue has something to do
with the way that `except` blocks become special when in an async context.
The main jester routes block invokes the handlers in an async context. The
effect is that one `except` block is fine, but adding more than one (to catch
different exception types, for example) causes the return type of the route
handler to change and not match what the outer block is expecting (a Future).
The fix here is to wrap any exception discrimination within a single outer
except block, re-raise the exception, and catch it inside this new
synchronous context. Ex:
```nim
try: someCall(mayFail)
except:
try: raise getCurrentException()
except ExceptionType1:
# do whatever...
except ExceptionType2:
# do whatever
except:
# general catch-all
return true
```
The return at the end is also part of the story. Jester's match handler
allows a route to defer making a decision about whether it matches. If you
return true from a route block Jester accepts the result as a matched route.
If you return false, Jester discards the result and looks for another
matching route. Normally this is taken care of by the `resp` templates
provided by Jester, but invoking those templates within the except blocks
also causes problems, so we manually setup the response and `return true` to
tell Jester that, yes this route matched, use this response.
* Moved the `/service/debug/ping` endpoint back to `/ping` and removed the
debug-only fence. I envision this as being useful as a simple healthcheck URL.