Previous
Trigger on events
This section describes the structure of captured data and how to query it effectively.
All tabular data is stored in a single table called readings in the sensorData database. Each row represents one capture event from one resource. Here is a complete document:
{
"organization_id": "ab1c2d3e-1234-5678-abcd-ef1234567890",
"location_id": "loc-1234-5678-abcd-ef1234567890",
"robot_id": "robot-1234-5678-abcd-ef1234567890",
"part_id": "part-1234-5678-abcd-ef1234567890",
"component_type": "rdk:component:sensor",
"component_name": "my-sensor",
"method_name": "Readings",
"time_requested": "2025-03-15T14:30:00.000Z",
"time_received": "2025-03-15T14:30:01.234Z",
"tags": ["production", "floor-2"],
"additional_parameters": {},
"data": {
"readings": {
"temperature": 23.5,
"humidity": 61.2
}
}
}
Your actual sensor values are inside the data column, nested under a key that depends on the component type and capture method. Everything outside of data is metadata that Viam adds automatically.
| Column | Type | Description |
|---|---|---|
organization_id | String | Organization UUID. Set automatically from your machine’s config. |
location_id | String | Location UUID. The location this machine belongs to. |
robot_id | String | Machine UUID. Identifies which machine captured this data. |
part_id | String | Machine part UUID. Identifies which part of the machine. |
component_type | String | Resource type as a triplet (for example, rdk:component:sensor). |
component_name | String | The name you gave this component in your config (for example, my-sensor). This is what you filter on most often. |
method_name | String | The capture method (for example, Readings, GetImages, EndPosition). |
time_requested | Timestamp | When the capture was requested on the machine (machine’s clock). |
time_received | Timestamp | When the cloud received and stored the data. Use this for time-range queries since it’s indexed. |
tags | Array | User-applied tags from your data management config. |
additional_parameters | JSON | Method-specific parameters you configured (for example, pin_name, reader_name). |
data | JSON | Your actual captured values. Structure varies by component type and method. |
The readings table does not include robot_name or part_name columns. These fields appear in data export responses but are not part of the queryable schema. To filter by machine, use robot_id or part_id.
The data column contains your actual captured values as nested JSON. Its structure depends on what component and method captured the data. To find out what’s inside data for your specific components, run:
SELECT data FROM readings WHERE component_name = 'my-sensor' LIMIT 1
Then use the field names you see to build more specific queries.
Common data structures:
Keys inside data.readings are whatever your sensor returns. Each sensor is different.
{
"readings": {
"temperature": 23.5,
"humidity": 61.2
}
}
To query: data.readings.temperature
Example:
SELECT time_received,
data.readings.temperature AS temp,
data.readings.humidity AS humidity
FROM readings
WHERE component_name = 'my-sensor'
ORDER BY time_received DESC
LIMIT 10
Detections include bounding box coordinates and confidence scores.
{
"detections": [
{
"class_name": "person",
"confidence": 0.94,
"x_min": 120,
"y_min": 50,
"x_max": 340,
"y_max": 480
}
]
}
To query detections, use MQL since SQL cannot easily traverse arrays:
[
{ "$match": { "component_name": "my-vision" } },
{ "$unwind": "$data.detections" },
{ "$match": { "data.detections.confidence": { "$gt": 0.8 } } },
{
"$project": {
"class": "$data.detections.class_name",
"confidence": "$data.detections.confidence",
"time": "$time_received"
}
}
]
{
"position": 145.7
}
To query: data.position
{
"position": 12450,
"position_type": 1
}
To query: data.position
If you’re unsure what fields your component produces:
SELECT DISTINCT component_name FROM readings to see what components have captured data.SELECT data FROM readings WHERE component_name = 'YOUR-COMPONENT' LIMIT 1.data. followed by the path to the field you want (for example, data.readings.temperature).You can improve query performance by filtering on indexed fields early in your query. Viam stores data in blob storage using the path pattern:
/organization_id/location_id/robot_id/part_id/component_type/component_name/method_name/capture_day/*
The more specific you can be, starting with the beginning of the path, the faster your query. These fields are indexed:
organization_idlocation_idrobot_idpart_idcomponent_typecomponent_namemethod_namecapture_dayAdditional optimization techniques:
$match (MQL) or WHERE (SQL) before expensive operations like grouping or sorting.$project early to drop unneeded fields from the processing pipeline.$limit or LIMIT while developing queries to avoid scanning your entire dataset.Viam supports a subset of MongoDB aggregation pipeline stages. Operators not on this list will return an error.
$addFields$bucket$bucketAuto$count$densify$fill$geoNear$group$limit$match$project$redact$replaceRoot$replaceWith$sample$set$setWindowFields$skip$sort$sortByCount$unset$unwindSee the MQL documentation for syntax details.
Viam supports the MongoDB Atlas SQL dialect:
`) or double quotes (").o''clock to represent o'clock).date data type is not supported. Use timestamp instead.For a full list of limitations, see the MongoDB Atlas SQL Interface Language Reference.
MQL time-range queries perform better with the BSON date type than with $toDate. Use JavaScript Date() constructors in mongosh:
use sensorData
const startTime = new Date('2024-02-10T19:45:07.000Z')
const endTime = new Date()
db.readings.aggregate([
{ $match: {
time_received: {
$gte: startTime,
$lte: endTime
}
}}
])
Users with owner or operator roles at the organization, location, or machine level can query data. See Role-Based Access Control for details.
This section describes the configuration fields for data capture and cloud sync.
The data management service controls sync behavior, storage paths, and deletion policies. Most of these settings are configured through the Viam app UI. Edit JSON directly for settings not exposed in the UI, such as deletion thresholds, sync thread limits, and MongoDB capture.
{
"services": [
{
"name": "my-data-manager",
"api": "rdk:service:data_manager",
"model": "rdk:builtin:builtin",
"attributes": {
"sync_interval_mins": 1,
"capture_dir": "",
"tags": [],
"capture_disabled": false,
"sync_disabled": true,
"delete_every_nth_when_disk_full": 5,
"maximum_num_sync_threads": 250
}
}
]
}
{
"services": [
{
"name": "my-data-manager",
"api": "rdk:service:data_manager",
"model": "rdk:builtin:builtin",
"attributes": {
"capture_dir": "",
"tags": [],
"additional_sync_paths": [],
"sync_interval_mins": 3
}
}
]
}
| Name | Type | Required? | Description | viam-micro-server Support |
|---|---|---|---|---|
capture_disabled | bool | Optional | Toggle data capture on or off for the entire machine part. Even if capture is enabled for the whole part, data is only captured from components that have capture individually configured. Default: false | |
capture_dir | string | Optional | Path to the directory where captured data is stored. If you change this, only new data goes to the new directory; existing data stays where it was. Default: ~/.viam/capture | |
tags | array of strings | Optional | Tags applied to all data captured by this machine part. May include alphanumeric characters, underscores, and dashes. | |
sync_disabled | bool | Optional | Toggle cloud sync on or off for the entire machine part. Default: false | |
additional_sync_paths | string array | Optional | Additional directories to sync to the cloud. Data is deleted from the directory after syncing. Use absolute paths. | |
sync_interval_mins | float | Optional | Minutes between sync attempts. Your hardware or network speed may impose practical limits. Default: 0.1 (every 6 seconds). | |
selective_syncer_name | string | Optional | Name of the sensor that controls selective sync. Also add this sensor to the depends_on field. See Conditional sync. | |
delete_every_nth_when_disk_full | int | Optional | When local storage meets the fullness criteria, the service deletes every Nth captured file. Default: 5 | |
maximum_num_sync_threads | int | Optional | Max CPU threads for syncing to the cloud. Higher values may improve throughput but can cause instability on constrained devices. Default: runtime.NumCPU/2 | |
mongo_capture_config.uri | string | Optional | MongoDB URI for writing tabular data alongside disk capture. | |
mongo_capture_config.database | string | Optional | Database name for MongoDB capture. Default: "sensorData" | |
mongo_capture_config.collection | string | Optional | Collection name for MongoDB capture. Default: "readings" | |
maximum_capture_file_size_bytes | int | Optional | Maximum size in bytes of each capture file on disk. When a capture file reaches this size, a new file is created. Default: 262144 (256 KB) | |
file_last_modified_millis | int | Optional | How long (in ms) an arbitrary file must be unmodified before it is eligible for sync. Normal .capture files sync immediately.Default: 10000 | |
disk_usage_deletion_threshold | float | Optional | Disk usage ratio (0-1) at or above which captured files are deleted, provided the capture directory also meets capture_dir_deletion_threshold.Default: 0.9 | |
capture_dir_deletion_threshold | float | Optional | Ratio (0-1) of disk usage attributable to the capture directory, at or above which deletion occurs (if disk_usage_deletion_threshold is also met).Default: 0.5 |
The following settings appear in your machine’s configuration but are not processed by viam-server on your machine. They are read and enforced by the Viam cloud platform:
| Name | Type | Description |
|---|---|---|
delete_data_on_part_deletion | bool | Whether deleting this machine or machine part also deletes all its captured cloud data. Default: false. |
Data capture is configured per-resource in the service_configs array of a component or service. When you configure capture through the Viam app UI, these fields are set automatically. The table below is the JSON-level reference for manual configuration.
Avoid configuring capture rates higher than your hardware can handle. This leads to performance degradation.
| Name | Type | Required? | Description |
|---|---|---|---|
name | string | Required | Fully qualified resource name (for example, rdk:component:sensor/my-sensor). |
method | string | Required | Depends on the component or service type. See Supported resources. Individual tabular readings larger than 4 MB are rejected at upload time and will not sync to the cloud. |
capture_frequency_hz | float | Required | Frequency in hertz. For example, 0.5 = one reading every 2 seconds. |
additional_params | object | Optional | Method-specific parameters. For example, DoCommand requires a docommand_input object; GetImages accepts a filter_source_names list. |
disabled | boolean | Optional | Whether capture is disabled for this method. |
tags | array of strings | Optional | Tags applied to data captured by this specific method. Added alongside any tags set at the service level. |
capture_directory | string | Optional | Override the capture directory for this specific resource. If not set, uses the service-level capture_dir. |
capture_queue_size | int | Optional | Size of the buffer between the capture goroutine and the file writer. Default: 250 |
capture_buffer_size | int | Optional | Size in bytes of the buffer used when writing captured data to disk. Default: 4096 |
cache_size_kb | float | Optional | viam-micro-server only. Max storage (KB) per data collector.Default: 1 |
The following capture method settings are processed by the Viam cloud platform, not by viam-server:
| Name | Type | Description |
|---|---|---|
retention_policy | object | How long captured data is retained in the cloud. Options: "days": <int>, "binary_limit_gb": <int>, "tabular_limit_gb": <int>. Days are in UTC. |
recent_data_store | object | Store a rolling window of recent data in a hot data store for faster queries. Example: { "stored_hours": 24 } |
For remote parts capture, see Capture from remote parts.
The following components and services support data capture and cloud sync. The table shows which capture methods are available for each resource type. Not all models support all methods listed for their type.
| Type | Method |
|---|---|
| Arm | EndPosition, JointPositions, DoCommand |
| Base | Position, DoCommand |
| Board | Analogs, Gpios, DoCommand |
| Button | DoCommand |
| Camera | GetImages, ReadImage (deprecated), NextPointCloud, DoCommand |
| Encoder | TicksCount, DoCommand |
| Gantry | Lengths, Position, DoCommand |
| Gripper | DoCommand |
| Input Controller | DoCommand |
| Motor | Position, IsPowered, DoCommand |
| Movement sensor | AngularVelocity, CompassHeading, LinearAcceleration, LinearVelocity, Orientation, Position, DoCommand |
| Sensor | Readings, DoCommand |
| Power sensor | Voltage, Current, Power, DoCommand |
| Servo | Position, DoCommand |
| Switch | DoCommand |
| Generic | DoCommand |
| Base remote control service | DoCommand |
| Discovery service | DoCommand |
| Vision service | CaptureAllFromCamera, DoCommand |
| SLAM service | Position, PointCloudMap, DoCommand |
| Motion service | DoCommand |
| Navigation service | DoCommand |
| Shell service | DoCommand |
This section describes how captured data is stored on the machine before syncing.
By default, captured data is stored in ~/.viam/capture.
The actual path depends on your platform:
| Platform | Default directory |
|---|---|
| Windows | With viam-agent: |
| Linux | With root or sudo: |
| macOS |
You can change the capture directory with the capture_dir attribute in the data management service attributes.
After data syncs successfully, it is automatically deleted from local storage. While a machine is offline, captured data accumulates locally.
If your machine is offline and its disk fills up, the data management service will delete captured data to free space and keep the machine running.
Automatic deletion triggers when all of these conditions are met:
disk_usage_deletion_threshold (default: 90%)capture_dir_deletion_threshold proportion of disk usage (default: 50%)Control deletion behavior with the delete_every_nth_when_disk_full attribute.
The micro-RDK (for ESP32 and similar microcontrollers) supports data capture with a smaller set of resources than viam-server. See the Micro-RDK tab in the supported resources table for the specific methods available.
On micro-RDK devices, captured data is stored in the ESP32’s flash memory until it is uploaded to the cloud. If the machine restarts before all data is synced, unsynced data since the last sync point is lost.
Was this page helpful?
Glad to hear it! If you have any other feedback please let us know:
We're sorry about that. To help us improve, please tell us what we can do better:
Thank you!