crossfire is a package created to give easier access to Fogo Cruzado's datasets, which is a digital collaborative platform of gun shooting occurrences in the metropolitan areas of Rio de Janeiro and Recife.
The package facilitates data extraction from Fogo Cruzado's open API.
- Python 3.9 or newer
$ pip install crossfireIf you want to have access to the data as Pandas DataFrames:
$ pip install crossfire[df]If you want to have access to the data as GeoPandas GeoDataFrames:
$ pip install crossfire[geodf]To have access to the API data, registration is required.
The email and password used in the registration can be configured as FOGOCRUZADO_EMAIL and FOGOCRUZADO_PASSWORD environment variables in a .env file:
FOGOCRUZADO_EMAIL=[email protected]
FOGOCRUZADO_PASSWORD=YOUR_PASSWORDIf you prefer not to use these environment variables, you can still use the credentials when instantiating a client.
Get all states with at least one city covered by the Fogo Cruzado project:
from crossfire import states
states()It is possible to get results in DataFrae:
states(format='df')Get cities from a specific state covered by the Fogo Cruzado project.
from crossfire import cities
cities()It is possible to get results in DataFrae:
cities(format='df')| Name | Required | Description | Type | Default value | Example | 
|---|---|---|---|---|---|
| state_id | ❌ | ID of the state | string | None | 'b112ffbe-17b3-4ad0-8f2a-2038745d1d14' | 
| city_id | ❌ | ID of the city | string | None | '88959ad9-b2f5-4a33-a8ec-ceff5a572ca5' | 
| city_name | ❌ | Name of the city | string | None | 'Rio de Janeiro' | 
| format | ❌ | Format of the result | string | 'dict' | 'dict','df'or'geodf' | 
To get shooting occurrences from Fogo Cruzado dataset it is necessary to specify a state id in id_state parameter:
from crossfire import occurrences
occurrences('813ca36b-91e3-4a18-b408-60b27a1942ef')It is possible to get results in DataFrae:
occurrences('813ca36b-91e3-4a18-b408-60b27a1942ef', format='df')Or as GeoDataFrame:
occurrences('813ca36b-91e3-4a18-b408-60b27a1942ef', format='geodf')| Name | Required | Description | Type | Default value | Example | 
|---|---|---|---|---|---|
| id_state | ✅ | ID of the state | string | None | 'b112ffbe-17b3-4ad0-8f2a-2038745d1d14' | 
| id_cities | ❌ | ID of the city | string or list of strings | None | '88959ad9-b2f5-4a33-a8ec-ceff5a572ca5'or['88959ad9-b2f5-4a33-a8ec-ceff5a572ca5', '9d7b569c-ec84-4908-96ab-3706ec3bfc57'] | 
| type_occurrence | ❌ | Type of occurrence | string | 'all' | 'all','withVictim'or'withoutVictim' | 
| initial_date | ❌ | Initial date of the occurrences | string, dateordatetime | None | '2020-01-01','2020/01/01','20200101',datetime.datetime(2023, 1, 1)ordatetime.date(2023, 1, 1) | 
| final_date | ❌ | Final date of the occurrences | string, dateordatetime | None | '2020-01-01','2020/01/01','20200101',datetime.datetime(2023, 1, 1)ordatetime.date(2023, 1, 1) | 
| max_parallel_requests | ❌ | Maximum number of parallel requests to the API | int | 16 | 32 | 
| format | ❌ | Format of the result | string | 'dict' | 'dict','df'or'geodf' | 
| flat | ❌ | Return nested columns as separate columns | bool | False | TrueorFalse | 
Note on Date Parameters: When using initial_date and final_date parameters, be aware that the API operates in Brazil timezone (America/Sao_Paulo, UTC-3). All occurrence timestamps and date filtering are processed according to Brazil time. Make sure to account for timezone differences when filtering data by date ranges.
Occurrence data often contains nested information in several columns. By setting the parameter flat=True, you can simplify the analysis by separating nested data into individual columns. This feature is particularly useful for columns such as contextInfo, state, region, city, neighborhood, and locality.
For example, to access detailed information about the context of occurrences, such as identifying the main reason, you would typically need to access the contextInfo column and then look for the mainReason key. With the flat=True parameter, this nested information is automatically split into separate columns, making the data easier to work with.
When flat=True is set, the function returns occurrences with the flattened columns. Each new column retains the original column name as a prefix and the nested key as a suffix. For instance, the contextInfo column will be split into the following columns: contextInfo_mainReason, contextInfo_complementaryReasons, contextInfo_clippings, contextInfo_massacre, and contextInfo_policeUnit.
from crossfire import occurrences
from crossfire.clients.occurrences import flatten
occs = occurrences('813ca36b-91e3-4a18-b408-60b27a1942ef')
occs[0].keys()
# dict_keys(['id', 'documentNumber', 'address', 'state', 'region', 'city', 'neighborhood', 'subNeighborhood', 'locality', 'latitude', 'longitude', 'date', 'policeAction', 'agentPresence', 'relatedRecord', 'contextInfo', 'transports', 'victims', 'animalVictims'])
flattened_occs = occurrences('813ca36b-91e3-4a18-b408-60b27a1942ef', flat=True)
occs[0].keys()
# dict_keys(['id', 'documentNumber', 'address', 'state', 'region', 'city', 'neighborhood', 'subNeighborhood', 'locality', 'latitude', 'longitude', 'date', 'policeAction', 'agentPresence', 'relatedRecord', 'transports', 'victims', 'animalVictims', 'contextInfo', 'contextInfo_mainReason', 'contextInfo_complementaryReasons', 'contextInfo_clippings', 'contextInfo_massacre', 'contextInfo_policeUnit'])By using the flat=True parameter, you ensure that all nested data is expanded into individual columns, simplifying data analysis and making it more straightforward to access specific details within your occurrence data.
Starting with API version 2.2.1, the Fogo Cruzado API returns additional metadata headers on /occurrences endpoints to help you track data freshness and implement intelligent caching strategies.
Important: All timestamps are returned in Brazil timezone (America/Sao_Paulo, UTC-3), not UTC.
| Header Name | Type | Description | 
|---|---|---|
| X-Last-Update | string | ISO 8601 timestamp with timezone offset (e.g., 2025-10-20T14:30:00-03:00) | 
| X-Last-Update-Timestamp | integer | Unix timestamp in seconds representing the last update | 
| X-Last-Update-State-Id | string | UUID of the filtered state (only present when filtering by state) | 
| X-Last-Update-State | string | ISO 8601 timestamp for state-specific last update | 
| X-Last-Update-State-Timestamp | integer | Unix timestamp in seconds for state-specific last update | 
These headers are automatically parsed and available in the metadata object returned by the occurrences() function.
Example 1: Accessing metadata fields
from crossfire import occurrences
data, metadata = occurrences('813ca36b-91e3-4a18-b408-60b27a1942ef', format='df')
# Access last update information
print(f"Last update: {metadata.last_update}")
print(f"Last update timestamp: {metadata.last_update_timestamp}")
print(f"State ID: {metadata.last_update_state_id}")
print(f"State last update: {metadata.last_update_state}")
# Example output:
# Last update: 2025-10-20T14:30:00-03:00
# Last update timestamp: 1729445400
# State ID: 813ca36b-91e3-4a18-b408-60b27a1942ef
# State last update: 2025-10-20T14:25:00-03:00Example 2: Checking if data needs refreshing
from crossfire import occurrences
from time import time
# Get current data
data, metadata = occurrences('813ca36b-91e3-4a18-b408-60b27a1942ef', format='df')
# Store the last update timestamp
last_sync = metadata.last_update_timestamp
# Later, check if data needs refreshing (e.g., after 1 hour)
current_time = int(time())
if current_time - last_sync > 3600:  # 3600 seconds = 1 hour
    print("Data is older than 1 hour, refreshing...")
    data, metadata = occurrences('813ca36b-91e3-4a18-b408-60b27a1942ef', format='df')
else:
    print("Data is still fresh, using cached version")Example 3: State-specific synchronization
from crossfire import occurrences
# Get data for Rio de Janeiro state
rio_state_id = '813ca36b-91e3-4a18-b408-60b27a1942ef'
data, metadata = occurrences(rio_state_id, format='df')
# Use state-specific timestamp for more accurate tracking
if metadata.last_update_state_timestamp:
    print(f"Rio de Janeiro state last updated at: {metadata.last_update_state}")
    print(f"Unix timestamp: {metadata.last_update_state_timestamp}")
    
    # Store this timestamp for state-specific cache invalidation
    # This is more accurate than the general last_update timestamp
    # when working with specific statesNote on Timezone: Remember that all timestamps returned by the API are in Brazil timezone (UTC-3). When comparing with local timestamps or implementing time-based logic, ensure you account for the timezone difference.
If not using the environment variables for authentication, it is recommended to use a custom client:
from crossfire import Client
client = Client(email="[email protected]", password="Rio&Pernambuco") # credentials are optional, the default are the environment variables
client.states()
client.cities()
client.occurrences('813ca36b-91e3-4a18-b408-60b27a1942ef')from crossfire import AsyncClient
client = AsyncClient()  # credentials are optional, the default are the environment variables
await client.states()
await client.cities()
await client.occurrences('813ca36b-91e3-4a18-b408-60b27a1942ef')@FelipeSBarros is the creator of the Python package. This implementation was funded by CYTED project number 520RT0010 redGeoLIBERO.