Action Journal
Learn about the MLNMapView methods for logging and viewing map actions.
The Action Journal provides functionality for persistent logging of top level map events.
Its primary use case is to assist in debugging problematic sessions and crashes by offering additional insight into the actions performed by the map at the time of failure. Data is stored in human readable format, which is useful for analyzing individual cases, but can also be easily translated and aggregated into a database, allowing for efficient analysis of multiple cases and helping to identify recurring patterns (Google BigQuery, AWS Glue + S3 + Athena, etc).
We are always interested in improving observability, so if you have a special use case, feel free to open an issue or pull request to extend the types of observability methods.
Logging implementation details
The logging is implemented using rolling files with a size based policy:
- A new file is created when the current log size exceeds
MLNActionJournalOptions/logFileSize. - When the maximum number of files exceeds
MLNActionJournalOptions/logFileCount:- The oldest one is deleted.
- The remaining files are renamed sequentially to maintain the naming convention
action_journal.0.logthroughaction_journal.{logFileCount - 1}.log.
- Each file contains one event per line.
- All files are stored in an umbrella "action_journal" directory at
MLNActionJournalOptions/path.
See also: MLNSettings, MLNActionJournalOptions.
Event format
Events are stored as JSON objects with the following format:
| Field | Type | Required | Description |
|---|---|---|---|
| name | string | true | event name |
| time | string | true | event time (ISO 8601 with milliseconds) |
| styleName | string | false | currently loaded style name |
| styleURL | string | false | currently loaded style URL |
| clientName | string | false | |
| clientVersion | string | false | |
| event | object | false | event specific data - consists of encoded values of the parameters passed to their MLNMapViewDelegate counterparts |
{
"name" : "onTileAction",
"time" : "2025-04-17T13:13:13.974Z",
"styleName" : "Streets",
"styleURL" : "maptiler://maps/streets",
"clientName" : "App",
"clientVersion" : "1.0",
"event" : {
"action" : "RequestedFromNetwork",
"tileX" : 0,
"tileY" : 0,
"tileZ" : 0,
"overscaledZ" : 0,
"sourceID" : "openmaptiles"
}
}Usage
Enabling the action journal.
let options = MLNMapOptions()
options.actionJournalOptions.enabled = true
options.styleURL = AMERICANA_STYLE
mapView = MLNMapView(frame: view.bounds, options: options)@objc func printActionJournal() {
print("ActionJournalLog files: \(mapView.getActionJournalLogFiles())")
print("ActionJournalLog : \(mapView.getActionJournalLog())")
// print only the newest events on each call
mapView.clearActionJournalLog()
}Alternative
The implementation is kept close to the core events to minimize additional locking and avoid platform-specific conversions and calls. As a result customization options and extensibility is limited.
For greater flexibility, consider using the MLNMapViewDelegate interface. It provides hooks for most Action Journal events and allows for more customizable querying and storage of map data. However, this comes at the cost of added complexity. See Observe Low-Level Events to learn about the map events that you can listen for, which mirror the events available in the action journal.