Skip to content

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.log through action_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:

FieldTypeRequiredDescription
namestringtrueevent name
timestringtrueevent time (ISO 8601 with milliseconds)
styleNamestringfalsecurrently loaded style name
styleURLstringfalsecurrently loaded style URL
clientNamestringfalse
clientVersionstringfalse
eventobjectfalseevent 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.

swift
let options = MLNMapOptions()
        options.actionJournalOptions.enabled = true
        options.styleURL = AMERICANA_STYLE
        mapView = MLNMapView(frame: view.bounds, options: options)
swift
@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.