Build-async-logger
Build an async logger directly from AsyncLoggerBuildConfig. This is the config-driven async entry point that bridges synchronous logger config, sink creation, and async queue setup in one call.
Interface
pub fn build_async_logger(config : AsyncLoggerBuildConfig) -> AsyncLogger[@bitlogger.RuntimeSink] {}input
config : AsyncLoggerBuildConfig- Combined synchronous logger config plus async queue/flush config.
output
AsyncLogger[RuntimeSink]- A config-built async logger with runtime sink control preserved.
Explanation
Detailed rules explaining key parameters and behaviors
- The
loggersection is built through the same config machinery used by synchronous configured loggers. - That means
LoggerConfig.sinkand the optional synchronousLoggerConfig.queueare applied before the async layer is added. - The resulting async logger inherits
min_level,target, and timestamp behavior from the built synchronous logger. - File, formatter, and any configured synchronous queue choices all come from config rather than direct code-side sink wiring.
- The returned sink type is
RuntimeSink, which keeps configured control helpers available where relevant. - This builder returns the underlying
AsyncLogger[@bitlogger.RuntimeSink]value directly.build_application_async_logger(...)only re-exports the same result under theApplicationAsyncLoggeralias, whilebuild_library_async_logger(...)wraps the same result inLibraryAsyncLogger[@bitlogger.RuntimeSink]. - The returned async logger also keeps ordinary target rules unchanged:
log(..., target=...)can override the target for one call, while severity helpers such asdebug(...),info(...), anderror(...)continue to use the stored logger target unless a derived logger was created first withwith_target(...)orchild(...). - Because this path starts from
build_logger(config.logger), it preserves the broader runtime-sink build path, including sync-side queue decoration whenLoggerConfig.queueis present. - Because this path starts from
build_logger(config.logger),config.logger.sink.kindhas already selected the concreteRuntimeSinkvariant before async wrapping, and optional sync queue decoration can further turn that built sink into queued runtime variants such asQueuedConsoleorQueuedFile. - This runtime-sink path also wires the explicit async flush callback as
flush=fn(sink) { sink.flush() }, soBatchandShutdownpolicies invoke the built runtime sink's real flush behavior instead of the default no-op callback. - In the current direct builder coverage, the returned logger exposes the expected serialized async state snapshot, runtime-sink variant choice, queue counters, lifecycle flags, and later failure fields after
run()orshutdown(). - File-backed runtime helpers also stay directly available on the returned
RuntimeSinkwith the same behavior later observed through the application and library facade equivalence tests. - Use
build_async_text_logger(...)instead when you want the direct text-console async builder path withFormattedConsoleSinkand without the sync runtime-sink construction layer. - The
src-asynclibrary is designed to compile onnative / llvm / js / wasm / wasm-gc, but runtime mode differs by backend. - Current local release-facing verification is explicit for
native / js / wasm / wasm-gc. llvmremains experimental and did not complete local verification in this environment.- On non-native targets, the async library still compiles and exposes the same public surface through compatibility-mode behavior rather than native background-worker semantics.
How to Use
Here are some specific examples provided.
When Need Fully Config-driven Async Bootstrapping
When your application should build async logging entirely from configuration:
let config = parse_async_logger_build_config_text(raw) catch {
err => return
}
let logger = build_async_logger(config)In this example, parsing and async runtime wiring are separated cleanly.
And the returned logger can immediately be started with run().
When Need A Per-call Target Override After Typed Async Build
When typed async config should still build a logger that supports a one-call target override:
let logger = build_async_logger(config)
logger.log(@bitlogger.Level::Error, "boom", target="svc.audit")In this example, the emitted record uses svc.audit for that call.
And later debug(...), info(...), or error(...) calls still use the logger's stored target unless code derives another logger first with with_target(...) or child(...).
When Need Runtime Sink Features After Async Build
When the sink shape is configured but runtime features still matter:
let logger = build_async_logger(config)
println(stringify_async_logger_state(logger.state(), pretty=true))In this example, the built async logger remains introspectable even though construction was config-driven.
And any configured synchronous runtime sink controls are preserved inside the returned RuntimeSink.
And if AsyncFlushPolicy::Batch or Shutdown is configured, this builder uses the runtime sink's real flush() path rather than the default no-op callback used by the text-specific builder line.
Error Case
e.g.:
If the config text was invalid, error handling should happen earlier in
parse_async_logger_build_config_text(...).If the configured sink shape is unsupported for a specific capability, the resulting runtime behavior follows the existing sink/runtime rules rather than inventing a separate builder-only failure model.
If callers depend on queued runtime-sink helpers or file-backed runtime helpers, this builder is the direct API that preserves them rather than a reduced facade path.
If callers depend on the exact runtime-sink variant, they should read this builder as preserving the sync-first sink selection result from
build_logger(config.logger), not as deferring sink-kind branching until after the async layer is added.If callers need the text-console path where
BatchandShutdownkeep the default no-op flush callback,build_async_text_logger(...)is the different builder contract; this runtime-sink path intentionally wires the sink's realflush()behavior.
Notes
Prefer this API when applications externalize both sync sink choice and async queue behavior.
Use
async_logger(...)directly when you want explicit code-defined sink wiring.Library portability is broader than example portability: a runnable
async fn mainexample may still be target-limited even when the async library itself compiles for that backend.See target-verification.md for the current verification boundary.
If the next boundary is library-facing rather than application-facing, build here and then narrow with
build_library_async_logger(...)instead of documenting the broader runtime helper surface directly.