How to Collect Telemetry Events
Telemetry
Emitting events from your BXL script
In BXL, you can emit custom events via ctx.instant_event()
, which takes in two
named parameters:
id
: string, identifies your event. Helpful to identify your event when looking through event logs. Ids do not have to be unique in a single BXL script.metadata
: dict, where keys are strings, and values are strings, bools, ints, or lists/dicts of the mentioned types. You can put any metadata you wish here.
Example:
def _impl(ctx):
ctx.instant_event(id = "id1", metadata = {"foo": "bar"})
my_script = bxl_main(
impl = _impl,
cli_args = {},
)
Only instant events can be manually created within BXL at this time, which means
that the event represents a single point in time. If you need something similar
to spans (start and end events which encompass a range of time) for measuring
the duration of a particular section (excluding actions - see below for more
information), you could couple instant events with the global now()
function
to measure the duration yourself:
def _impl(ctx):
instant = now()
# do something time intensive
end = instant.elapsed_millis()
ctx.instant_event(id = "id1", metadata = {"duration": end})
# do something else time intensive
end = instant.elapsed_millis()
ctx.instant_event(id = "id2", metadata = {"duration": end})
my_script = bxl_main(
impl = _impl,
cli_args = {},
)
Measuring time for actions and ensuring artifacts
You cannot use now()
to measure the time it takes to run actions and ensure
artifacts because these processes occur asynchronously outside of the BXL script
execution. For BXL user telemetry, we emit action events via the buck2 core
automatically. Events around ensuring the artifacts are not emitted currently,
but will be added soon.
User event log
To write to your own event log when running BXL, you can run your BXL command
with the --user-event-log
flag to tell buck2 where to write the events to.
Buck2 is aware of the following file extensions: .json-lines
,
json-lines.zst
, .json-lines.gz
, and will compress the files automatically
for you depending on the extension. If the extension is not one of these, the
logs will always be written in JSONL format, uncompressed.
Example:
buck2 bxl path//to/my_script/script.bxl:my_script --user-event-log my_file.json-lines.gz
When using this flag to write to a custom event log, it is up to you to clean up
these log files. In addition, if the same filename is used with subsequent BXL
invocations, events are always appended to the existing file contents, which is
the same behavior as buck2 <any command> --event-log <path>
. If you tell buck2
to write to a compressed file, you are responsible for decompressing them.
Getting a user event log from a normal event log
buck2 log show-user
can be used to convert a normal event log (regardless of
encoding/compression) to a user event. Similar to buck2 log show
, you can
choose the most recent invocation, or the nth invocation, or provide a path to
the normal user event log. Note that user event logs are not able to be passed
into buck2 log show
or buck2 log show-user
.
Event log output
The first line of your event log will always be the invocation record, which contains useful things like command line args used, working directory, etc. The subsequent lines are either instant events and/or action events, depending on your BXL script's contents.
Instant event
Sample:
{
"StarlarkUserEvent": {
"id": "foo",
"metadata": {
"bool_value": true,
"string_value": "str",
"int_value": 123,
"list_value": [
"a",
"b",
"c"
],
"dict_value": {
"foo": "bar"
}
},
},
"epoch_millis": 123456789 # when the event was emitted
}
Action event
{
"ActionExecutionEvent": {
"kind": "Write", # kind of action, like write or run
"name": { # name of the action, for user display. Unique within the execution of a particular target
"category": "write", # category for the action
"identifier": "my_output" # identifier for the action
},
"duration_millis": 0, # duration of the action in millis, excluding input materialization time
"output_size": 10, # size in bytes of the action's outputs
"input_materialization_duration_millis": 0, # how long it took to materialize any inputs to the action
"execution_kind": "Simple", # how the action was executed
"owner": "cell//path/to/script.bxl:function_name" # owner of the action execution (target label, anon target label, bxl label)
},
"epoch_millis": 123456789 # when the event was emitted
}
execution_kind
includes:
- Local: action was executed locally
- Remote: action was executed via a remote executor
- ActionCache: action was served by the action cache and not executed
- Simple: action is simple and executed inline within buck2 (ex: write, symlink_dir)
- Skipped: action was not executed at all
- Deferred: action logically executed, but didn't do all the work
- LocalDepFile: action was served by the local dep file cache and not executed.
- LocalWorker: action was executed via a local worker
- NotSet: action execution kind was not set
Ensure artifact event
{
"BxlEnsureArtifactsEvent": {
"duration_millis": 0, # duration of ensuring the artifact
},
"epoch_millis": 123456789 # when the event was emitted
}