Cube API Guide

Late Data Handling 

Disadvantages of using conventional method

  • Relatively hard to maintain

  • Raw data are not aggregated. Self aggregation and comparison are required. Raw files not combined to branch levels if one of the counters is offline, it would result in lower branch level traffic

  • No visibility on whether late data will be re included next day

  • Data Integrity: Inaccurate data especially when one offline counter affects the overall accuracy of counting, in terms of sales conversion, visitor counts etc. No visibility if data retrieved is partially completed or fully completed, if it is clean data with no offline counters and no unverified data

Handling: 

  • Ensure the data collected is fully verified

  • Ensure the branch level data collected has no missing data due to one of the counter offline

  • Ensure the data is fully uploaded to the server and get the complete set to compare with sales conversion

Video Guide

Step 1: Generate AToken

method: POST
url: https://v9.footfallcam.com/account/GenerateAccessToken
Body(payload):
{
"email": "[email protected]",
"password": "123456",
"expiration": "2024-03-15"
}

After clicking send, you may able to get the AToken in the Body Response.

Step 2: Generate site detail

method: GET
url: https://data.footfallcam.com/api/Sites
Hearder: add the KEYWORD AToken in key, and paste the AToken generated before into value

After clicking send, you can get the BranchId in the Body Response.

Step 3: Call Cube Example

Method: POST
URL: https://cube.footfallcam.com/API/v1/load
Authorization: Bearer Token: {{your access token}}
KEY Content-Type VALUE: application/json
Body:

{
    "query": {
        "measures": [
            "sitearea_footfallcounting_hour.FC01_1_SUM",
            "sitearea_footfallcounting_hour.FC02_1_SUM"
        ],
        "dimensions": [
            "sitearea_footfallcounting_hour.BranchName",
            "sitearea_footfallcounting_hour.Time"
        ],
        "filters": [
            {
                "member": "sitearea_footfallcounting_hour.BranchId",
                "operator": "equals",
                "values": [
                    "6903"
                ]
            },
            {
                "member": "sitearea_footfallcounting_hour.isoperating",
                "operator": "equals",
                "values": [
                    "true"
                ]
            }
        ],
        "timeDimensions": [
            {
                "dimension": "sitearea_footfallcounting_hour.Time",
                "dateRange": [
                    "2023-07-27T00:00:00.000",
                    "2023-07-27"
                ]
            }
        ],
        "order": [
            [
                "sitearea_footfallcounting_hour.BranchId",
                "asc"
            ],
            [
                "sitearea_footfallcounting_hour.Time",
                "asc"
            ]
        ],
        "offset": 0,
        "limit": 100
    }
}

You can get the data in the Body Response after clicking send.

Please refer List of Cube to use other cube and refer Metric Documentation for the metrics.

FAQ: How to Handle Late Data?

Why Late Data can happen?

Late data can happen when some devices are offline before the data is retrieved via the API. This prevents the data from being sent to our database in real-time. 

Solution

There is a column called 'AggregationStatus' in the following two cubes:

  • ffc_site_summary

  • ffc_area_summary

This column can only be used when filtering "granularity": "day" in the timeDimensions section.

There are four types of AggregationStatus: 

  • Complete - This indicates that all the data has been fully aggregated. 

  • Precheck - This normally indicates that this particular hour/day is still ongoing and thus the aggregation status has not yet been confirmed. 

  • Late Data - This indicates there exists late data and the data is not yet fully aggregated.

  • Missing Data - This indicates that there exist missing data that are not late. This shouldn't appear under normal circumstances.

Below is an example payload and the corresponding output: 
Payload

{
    "query": {
        "measures": [
            "ffc_site_summary.A01",
            "ffc_site_summary.A02"
        ],
        "dimensions": [
            "ffc_site_summary.AggregationStatus",
            "ffc_site_summary.AggregationStatusLastUpdateUTCDateTime"
        ],
        "filters": [
            {
                "member": "ffc_site_summary.BranchId",
                "operator": "equals",
                "values": [
                    "123"
                ]
            }
        ],
        "timeDimensions": [
            {
                "dimension": "ffc_site_summary.Time",
                "dateRange": [
                    "2024-08-17T00:00:00.000",
                    "2024-08-22T23:59:59.999"
                ],
                "granularity": "day"
            }
        ],
        "order": [
            [
                "ffc_site_summary.Time",
                "asc"
            ]
        ],
        "offset": 0,
        "limit": 100
    }
}


Output Data
"data": [
        {
            "ffc_site_summary.AggregationStatus": "Complete",
            "ffc_site_summary.AggregationStatusLastUpdateUTCDateTime": "2024-08-18T00:13:59.000"
            "ffc_site_summary.Time.day": "2024-08-17T00:00:00.000",
            "ffc_site_summary.Time": "2024-08-17T00:00:00.000",
            "ffc_site_summary.A01": 234,
            "ffc_site_summary.A02": 214
        },
        {
            "ffc_site_summary.AggregationStatus": "Missing Data",
            "ffc_site_summary.AggregationStatusLastUpdateUTCDateTime": "2024-08-18T22:32:02.000"
            "ffc_site_summary.Time.day": "2024-08-18T00:00:00.000",
            "ffc_site_summary.Time": "2024-08-18T00:00:00.000",
            "ffc_site_summary.A01": 112,
            "ffc_site_summary.A02": 84
        },
        {
            "ffc_site_summary.AggregationStatus": "Missing Data",
            "ffc_site_summary.AggregationStatusLastUpdateUTCDateTime": "2024-08-19T21:21:13.000"
            "ffc_site_summary.Time.day": "2024-08-19T00:00:00.000",
            "ffc_site_summary.Time": "2024-08-19T00:00:00.000",
            "ffc_site_summary.A01": 23,
            "ffc_site_summary.A02": 52
        },
        {
            "ffc_site_summary.AggregationStatus": "Missing Data",
            "ffc_site_summary.AggregationStatusLastUpdateUTCDateTime": "2024-08-20T22:13:25.000"
            "ffc_site_summary.Time.day": "2024-08-20T00:00:00.000",
            "ffc_site_summary.Time": "2024-08-20T00:00:00.000",
            "ffc_site_summary.A01": 142,
            "ffc_site_summary.A02": 121
        },
        {
            "ffc_site_summary.AggregationStatus": "Late Data",
            "ffc_site_summary.AggregationStatusLastUpdateUTCDateTime": "2024-08-21T23:58:43.000"
            "ffc_site_summary.Time.day": "2024-08-21T00:00:00.000",
            "ffc_site_summary.Time": "2024-08-21T00:00:00.000",
            "ffc_site_summary.A01": 187,
            "ffc_site_summary.A02": 193
        },
        {
            "ffc_site_summary.AggregationStatus": "Precheck",
            "ffc_site_summary.AggregationStatusLastUpdateUTCDateTime": "2024-08-22T14:23:10.000"
            "ffc_site_summary.Time.day": "2024-08-22T00:00:00.000",
            "ffc_site_summary.Time": "2024-08-22T00:00:00.000",
            "ffc_site_summary.A01": 12,
            "ffc_site_summary.A02": 23
        }
    ],

Updated on January 20, 2025