Pipeline Logs MLOps API Tutorial

How to retrieve pipeline logs through the Wallaroo MLOps API.

This tutorial and the assets can be downloaded as part of the Wallaroo Tutorials repository.

Pipeline API Log Tutorial

This tutorial demonstrates Wallaroo Pipeline MLOps API for pipeline log retrieval.

This tutorial will demonstrate how to:

  1. Select or create a workspace, pipeline and upload the control model, and additional testing models.
  2. Add a pipeline step with the champion model, then deploy the pipeline and perform sample inferences.
  3. Retrieve the logs via the Wallaroo MLOps API. These steps will be simplified to only show the API log retrieval method. See the Wallaroo Documentation site for full details.
  4. Swap out the pipeline step with the champion model with a shadow deploy step that compares the champion model against two competitors.
  5. Perform sample inferences with a shadow deployed step, then display the log files through the MLOps API for a shadow deployed pipeline.
  6. Swap out the shadow deployed pipeline step with an A/B pipeline step.
  7. Perform sample inferences with a A/B pipeline step, then display the log files through the MLOps API for an A/B pipeline step.
  8. Undeploy the pipeline.

This tutorial provides the following:

  • Models:
    • models/rf_model.onnx: The champion model that has been used in this environment for some time.
    • models/xgb_model.onnx and models/gbr_model.onnx: Rival models that will be tested against the champion.
  • Data:
    • data/xtest-1.df.json and data/xtest-1k.df.json: DataFrame JSON inference inputs with 1 input and 1,000 inputs.
    • data/xtest-1k.arrow: Apache Arrow inference inputs with 1 input and 1,000 inputs.

Prerequisites

  • A deployed Wallaroo instance
  • The following Python libraries installed:
    • wallaroo: The Wallaroo SDK. Included with the Wallaroo JupyterHub service by default.
    • pandas: Pandas, mainly used for Pandas DataFrame
    • pyarrow: Pyarrow for Apache Arrow support

Initial Steps

Import libraries

The first step is to import the libraries needed for this notebook.

import wallaroo
from wallaroo.object import EntityNotFoundError

import pyarrow as pa

from IPython.display import display

# used to display DataFrame information without truncating
from IPython.display import display
import pandas as pd
pd.set_option('display.max_colwidth', None)
pd.set_option('display.max_columns', None)

import datetime
import requests

Connect to the Wallaroo Instance

The first step is to connect to Wallaroo through the Wallaroo client. The Python library is included in the Wallaroo install and available through the Jupyter Hub interface provided with your Wallaroo environment.

This is accomplished using the wallaroo.Client() command, which provides a URL to grant the SDK permission to your specific Wallaroo environment. When displayed, enter the URL into a browser and confirm permissions. Store the connection into a variable that can be referenced later.

If logging into the Wallaroo instance through the internal JupyterHub service, use wl = wallaroo.Client(). For more details on logging in through Wallaroo, see the Wallaroo SDK Essentials Guide: Client Connection.

# Login through local Wallaroo instance

wl = wallaroo.Client()

Wallaroo MLOps API URL

API URL

The variable APIURL is used to specify the connection to the Wallaroo instance’s MLOps API URL, and is composed of the Wallaroo DNS prefix and suffix. For full details, see the Wallaroo API Connection Guide
.

For our examples, we will use the Wallaroo SDK to retrieve the API endpoint via the wl.api_endpoint() method.

display(wl.api_endpoint)
'https://doc-test.wallarooexample.ai'

Create Workspace

We will create a workspace to manage our pipeline and models. The following variables will set the name of our sample workspace then set it as the current workspace.

workspace_name = 'logapiworkspace'
main_pipeline_name = 'logapipipeline'
model_name_control = 'logapicontrol'
model_file_name_control = './models/rf_model.onnx'
workspace = wl.get_workspace(name=workspace_name, create_if_not_exist=True)

wl.set_current_workspace(workspace)

workspace_id = workspace.id()

Standard Pipeline

Upload The Champion Model

For our example, we will upload the champion model that has been trained to derive house prices from a variety of inputs. The model file is rf_model.onnx, and is uploaded with the name housingcontrol.

housing_model_control = (wl.upload_model(model_name_control, 
                                         model_file_name_control, 
                                         framework=wallaroo.framework.Framework.ONNX)
                                         .configure(tensor_fields=["tensor"])
                        )

Build the Pipeline

This pipeline is made to be an example of an existing situation where a model is deployed and being used for inferences in a production environment. We’ll call it housepricepipeline, set housingcontrol as a pipeline step, then run a few sample inferences.

mainpipeline = wl.build_pipeline(main_pipeline_name)
# in case this pipeline was run before
mainpipeline.undeploy()
mainpipeline.clear()
mainpipeline.add_model_step(housing_model_control)
namelogapipipeline
created2024-07-29 20:06:53.925863+00:00
last_updated2024-07-29 20:08:32.511667+00:00
deployedFalse
workspace_id7
workspace_namelogapiworkspace
archx86
accelnone
tags
versions61c710b2-cd83-44b2-a15b-bff7891d06ac, 815ba571-17b6-457a-8bc9-bead597abcd2, 437eca05-f161-495a-8129-687e406d292b
stepslogapicontrol
publishedFalse
mainpipeline.deploy()
namelogapipipeline
created2024-07-29 20:06:53.925863+00:00
last_updated2024-07-29 20:09:10.465319+00:00
deployedTrue
workspace_id7
workspace_namelogapiworkspace
archx86
accelnone
tags
versionsfd47a339-fd43-43d6-bf3c-dbbf701f1998, 61c710b2-cd83-44b2-a15b-bff7891d06ac, 815ba571-17b6-457a-8bc9-bead597abcd2, 437eca05-f161-495a-8129-687e406d292b
stepslogapicontrol
publishedFalse

Testing

We’ll pass in two DataFrame formatted inference requests which are returned as a pandas DataFrame. Then roughly 1,000 inferences as a batch as an Apache Arrow table, which is returned as an arrow table, which we’ll convert into a pandas DataFrame to display the first 20 results.

dataframe_start = datetime.datetime.now(datetime.timezone.utc)

normal_input = pd.DataFrame.from_records({"tensor": [[4.0, 2.5, 2900.0, 5505.0, 2.0, 0.0, 0.0, 3.0, 8.0, 2900.0, 0.0, 47.6063, -122.02, 2970.0, 5251.0, 12.0, 0.0, 0.0]]})
result = mainpipeline.infer(normal_input)
display(result)

large_house_input = pd.DataFrame.from_records({'tensor': [[4.0, 3.0, 3710.0, 20000.0, 2.0, 0.0, 2.0, 5.0, 10.0, 2760.0, 950.0, 47.6696, -122.261, 3970.0, 20000.0, 79.0, 0.0, 0.0]]})
large_house_result = mainpipeline.infer(large_house_input)
display(large_house_result)

import time
time.sleep(10)
dataframe_end = datetime.datetime.now(datetime.timezone.utc)

# generating multiple log entries
batch_inferences = mainpipeline.infer_from_file('./data/xtest-1k.arrow')
batch_inferences = mainpipeline.infer_from_file('./data/xtest-1k.arrow')
batch_inferences = mainpipeline.infer_from_file('./data/xtest-1k.arrow')

large_inference_result = batch_inferences.to_pandas()
display(large_inference_result.head(20))
timein.tensorout.variableanomaly.count
02024-07-29 20:09:34.342[4.0, 2.5, 2900.0, 5505.0, 2.0, 0.0, 0.0, 3.0, 8.0, 2900.0, 0.0, 47.6063, -122.02, 2970.0, 5251.0, 12.0, 0.0, 0.0][718013.7]0
timein.tensorout.variableanomaly.count
02024-07-29 20:09:34.577[4.0, 3.0, 3710.0, 20000.0, 2.0, 0.0, 2.0, 5.0, 10.0, 2760.0, 950.0, 47.6696, -122.261, 3970.0, 20000.0, 79.0, 0.0, 0.0][1514079.4]0
timein.tensorout.variableanomaly.count
02024-07-29 20:09:45.423[4.0, 2.5, 2900.0, 5505.0, 2.0, 0.0, 0.0, 3.0, 8.0, 2900.0, 0.0, 47.6063, -122.02, 2970.0, 5251.0, 12.0, 0.0, 0.0][718013.75]0
12024-07-29 20:09:45.423[2.0, 2.5, 2170.0, 6361.0, 1.0, 0.0, 2.0, 3.0, 8.0, 2170.0, 0.0, 47.7109, -122.017, 2310.0, 7419.0, 6.0, 0.0, 0.0][615094.56]0
22024-07-29 20:09:45.423[3.0, 2.5, 1300.0, 812.0, 2.0, 0.0, 0.0, 3.0, 8.0, 880.0, 420.0, 47.5893, -122.317, 1300.0, 824.0, 6.0, 0.0, 0.0][448627.72]0
32024-07-29 20:09:45.423[4.0, 2.5, 2500.0, 8540.0, 2.0, 0.0, 0.0, 3.0, 9.0, 2500.0, 0.0, 47.5759, -121.994, 2560.0, 8475.0, 24.0, 0.0, 0.0][758714.2]0
42024-07-29 20:09:45.423[3.0, 1.75, 2200.0, 11520.0, 1.0, 0.0, 0.0, 4.0, 7.0, 2200.0, 0.0, 47.7659, -122.341, 1690.0, 8038.0, 62.0, 0.0, 0.0][513264.7]0
52024-07-29 20:09:45.423[3.0, 2.0, 2140.0, 4923.0, 1.0, 0.0, 0.0, 4.0, 8.0, 1070.0, 1070.0, 47.6902, -122.339, 1470.0, 4923.0, 86.0, 0.0, 0.0][668288.0]0
62024-07-29 20:09:45.423[4.0, 3.5, 3590.0, 5334.0, 2.0, 0.0, 2.0, 3.0, 9.0, 3140.0, 450.0, 47.6763, -122.267, 2100.0, 6250.0, 9.0, 0.0, 0.0][1004846.5]0
72024-07-29 20:09:45.423[3.0, 2.0, 1280.0, 960.0, 2.0, 0.0, 0.0, 3.0, 9.0, 1040.0, 240.0, 47.602, -122.311, 1280.0, 1173.0, 0.0, 0.0, 0.0][684577.2]0
82024-07-29 20:09:45.423[4.0, 2.5, 2820.0, 15000.0, 2.0, 0.0, 0.0, 4.0, 9.0, 2820.0, 0.0, 47.7255, -122.101, 2440.0, 15000.0, 29.0, 0.0, 0.0][727898.1]0
92024-07-29 20:09:45.423[3.0, 2.25, 1790.0, 11393.0, 1.0, 0.0, 0.0, 3.0, 8.0, 1790.0, 0.0, 47.6297, -122.099, 2290.0, 11894.0, 36.0, 0.0, 0.0][559631.1]0
102024-07-29 20:09:45.423[3.0, 1.5, 1010.0, 7683.0, 1.5, 0.0, 0.0, 5.0, 7.0, 1010.0, 0.0, 47.72, -122.318, 1550.0, 7271.0, 61.0, 0.0, 0.0][340764.53]0
112024-07-29 20:09:45.423[3.0, 2.0, 1270.0, 1323.0, 3.0, 0.0, 0.0, 3.0, 8.0, 1270.0, 0.0, 47.6934, -122.342, 1330.0, 1323.0, 8.0, 0.0, 0.0][442168.06]0
122024-07-29 20:09:45.423[4.0, 1.75, 2070.0, 9120.0, 1.0, 0.0, 0.0, 4.0, 7.0, 1250.0, 820.0, 47.6045, -122.123, 1650.0, 8400.0, 57.0, 0.0, 0.0][630865.6]0
132024-07-29 20:09:45.423[4.0, 1.0, 1620.0, 4080.0, 1.5, 0.0, 0.0, 3.0, 7.0, 1620.0, 0.0, 47.6696, -122.324, 1760.0, 4080.0, 91.0, 0.0, 0.0][559631.1]0
142024-07-29 20:09:45.423[4.0, 3.25, 3990.0, 9786.0, 2.0, 0.0, 0.0, 3.0, 9.0, 3990.0, 0.0, 47.6784, -122.026, 3920.0, 8200.0, 10.0, 0.0, 0.0][909441.1]0
152024-07-29 20:09:45.423[4.0, 2.0, 1780.0, 19843.0, 1.0, 0.0, 0.0, 3.0, 7.0, 1780.0, 0.0, 47.4414, -122.154, 2210.0, 13500.0, 52.0, 0.0, 0.0][313096.0]0
162024-07-29 20:09:45.423[4.0, 2.5, 2130.0, 6003.0, 2.0, 0.0, 0.0, 3.0, 8.0, 2130.0, 0.0, 47.4518, -122.12, 1940.0, 4529.0, 11.0, 0.0, 0.0][404040.8]0
172024-07-29 20:09:45.423[3.0, 1.75, 1660.0, 10440.0, 1.0, 0.0, 0.0, 3.0, 7.0, 1040.0, 620.0, 47.4448, -121.77, 1240.0, 10380.0, 36.0, 0.0, 0.0][292859.5]0
182024-07-29 20:09:45.423[3.0, 2.5, 2110.0, 4118.0, 2.0, 0.0, 0.0, 3.0, 8.0, 2110.0, 0.0, 47.3878, -122.153, 2110.0, 4044.0, 25.0, 0.0, 0.0][338357.88]0
192024-07-29 20:09:45.423[4.0, 2.25, 2200.0, 11250.0, 1.5, 0.0, 0.0, 5.0, 7.0, 1300.0, 900.0, 47.6845, -122.201, 2320.0, 10814.0, 94.0, 0.0, 0.0][682284.6]0

Standard Pipeline Logs

Pipeline logs are retrieved through the Wallaroo MLOps API with the following request.

  • REQUEST URL
    • v1/api/pipelines/get_logs
  • Headers
    • Accept:
      • application/json; format=pandas-records: For the logs returned as pandas DataFrame
      • application/vnd.apache.arrow.file: for the logs returned as Apache Arrow
  • PARAMETERS
    • pipeline_name (String Required): The name of the pipeline.
    • workspace_id (Integer Required): The numerical identifier of the workspace.
    • cursor (String Optional): Cursor returned with a previous page of results from a pipeline log request, used to retrieve the next page of information.
    • order (String Optional Default: Desc): The order for log inserts returned. Valid values are:
      • Asc: In chronological order of inserts.
      • Desc: In reverse chronological order of inserts.
    • page_size (Integer Optional Default: 1000.): Max records per page.
    • start_time (String Optional): The start time of the period to retrieve logs for in RFC 3339 format for DateTime. Must be combined with end_time.
    • end_time (String Optional): The end time of the period to retrieve logs for in RFC 3339 format for DateTime. Must be combined with start_time.
  • RETURNS
    • The logs are returned by default as 'application/json; format=pandas-records' format. To request the logs as Apache Arrow tables, set the submission header Accept to application/vnd.apache.arrow.file.
    • Headers:
      • x-iteration-cursor: Used to retrieve the next page of results. This is not included if x-iteration-status is All.
      • x-iteration-status: Informs whether there are more records available outside of this log request parameters.
        • All: This page includes all logs available from this request. If x-iteration-status is All, then x-iteration-cursor is not provided.
        • SchemaChange: A change in the log schema caused by actions such as pipeline version, etc.
        • RecordLimited: The number of records exceeded from the page size, more records can be requested as the next page. There may be more records available to retrieve OR the record limit was reached for this request even if no more records are available in next cursor request.
        • ByteLimited: The number of records exceeded the pipeline log limit which is around 100K.
# retrieve the authorization token
headers = wl.auth.auth_header()

url = f"{wl.api_endpoint}/v1/api/pipelines/get_logs"

# Standard log retrieval

data = {
    'pipeline_name': main_pipeline_name,
    'workspace_id': workspace_id
}

response = requests.post(url, headers=headers, json=data)
standard_logs = pd.DataFrame.from_records(response.json())

display(len(standard_logs))
display(standard_logs.head(5).loc[:, ["time", "in", "out"]])
cursor = response.headers['x-iteration-cursor']
2
timeinout
01722283774342{'tensor': [4.0, 2.5, 2900.0, 5505.0, 2.0, 0.0, 0.0, 3.0, 8.0, 2900.0, 0.0, 47.6063, -122.02, 2970.0, 5251.0, 12.0, 0.0, 0.0]}{'variable': [718013.7]}
11722283774577{'tensor': [4.0, 3.0, 3710.0, 20000.0, 2.0, 0.0, 2.0, 5.0, 10.0, 2760.0, 950.0, 47.6696, -122.261, 3970.0, 20000.0, 79.0, 0.0, 0.0]}{'variable': [1514079.4]}
# Get next page of results as an arrow table

# retrieve the authorization token
headers = wl.auth.auth_header()
headers['Accept']="application/vnd.apache.arrow.file"

url = f"{wl.api_endpoint}/v1/api/pipelines/get_logs"

# Standard log retrieval

data = {
    'pipeline_name': main_pipeline_name,
    'workspace_id': workspace_id,
    'cursor': cursor
}

response = requests.post(url, headers=headers, json=data)

# Arrow table is retrieved 
with pa.ipc.open_file(response.content) as reader:
    arrow_table = reader.read_all()

# convert to Polars DataFrame and display the first 5 rows
display(arrow_table.to_pandas().head(5).loc[:,["time", "out"]])
timeout
01722283784868{'variable': [718013.75]}
11722283784868{'variable': [615094.56]}
21722283784868{'variable': [448627.72]}
31722283784868{'variable': [758714.2]}
41722283784868{'variable': [513264.7]}
# Retrieve logs from specific date/time to only get the two DataFrame input inferences in ascending format

# retrieve the authorization token
headers = wl.auth.auth_header()

url = f"{wl.api_endpoint}/v1/api/pipelines/get_logs"

# Standard log retrieval

data = {
    'pipeline_name': main_pipeline_name,
    'workspace_id': workspace_id,
    'order': 'Asc',
    'start_time': f'{dataframe_start.isoformat()}',
    'end_time': f'{dataframe_end.isoformat()}'
}

response = requests.post(url, headers=headers, json=data)
standard_logs = pd.DataFrame.from_records(response.json())

display(standard_logs.head(5).loc[:, ["time", "in", "out"]])
display(response.headers)
timeinout
01722283774342{'tensor': [4.0, 2.5, 2900.0, 5505.0, 2.0, 0.0, 0.0, 3.0, 8.0, 2900.0, 0.0, 47.6063, -122.02, 2970.0, 5251.0, 12.0, 0.0, 0.0]}{'variable': [718013.7]}
11722283774577{'tensor': [4.0, 3.0, 3710.0, 20000.0, 2.0, 0.0, 2.0, 5.0, 10.0, 2760.0, 950.0, 47.6696, -122.261, 3970.0, 20000.0, 79.0, 0.0, 0.0]}{'variable': [1514079.4]}
{'content-type': 'application/json; format=pandas-records', 'x-iteration-status': 'All', 'date': 'Mon, 29 Jul 2024 20:09:45 GMT', 'content-length': '946', 'x-envoy-upstream-service-time': '4', 'server': 'opscenter-https'}

Shadow Deploy Pipelines

Let’s assume that after analyzing the assay information we want to test two challenger models to our control. We do that with the Shadow Deploy pipeline step.

In Shadow Deploy, the pipeline step is added with the add_shadow_deploy method, with the champion model listed first, then an array of challenger models after. All inference data is fed to all models, with the champion results displayed in the out.variable column, and the shadow results in the format out_{model name}.variable. For example, since we named our challenger models housingchallenger01 and housingchallenger02, the columns out_housingchallenger01.variable and out_housingchallenger02.variable have the shadow deployed model results.

For this example, we will remove the previous pipeline step, then replace it with a shadow deploy step with rf_model.onnx as our champion, and models xgb_model.onnx and gbr_model.onnx as the challengers. We’ll deploy the pipeline and prepare it for sample inferences.

# Upload the challenger models

model_name_challenger01 = 'logcontrolchallenger01'
model_file_name_challenger01 = './models/xgb_model.onnx'

model_name_challenger02 = 'logcontrolchallenger02'
model_file_name_challenger02 = './models/gbr_model.onnx'

housing_model_challenger01 = (wl.upload_model(model_name_challenger01, 
                                              model_file_name_challenger01, 
                                              framework=wallaroo.framework.Framework.ONNX)
                                              .configure(tensor_fields=["tensor"])
                            )
housing_model_challenger02 = (wl.upload_model(model_name_challenger02, 
                                              model_file_name_challenger02, 
                                              framework=wallaroo.framework.Framework.ONNX).configure(tensor_fields=["tensor"])
                                )
# Undeploy the pipeline
mainpipeline.undeploy()

mainpipeline.clear()

# Add the new shadow deploy step with our challenger models
mainpipeline.add_shadow_deploy(housing_model_control, [housing_model_challenger01, housing_model_challenger02])

# Deploy the pipeline with the new shadow step
mainpipeline.deploy()
namelogapipipeline
created2024-07-29 20:06:53.925863+00:00
last_updated2024-07-29 20:10:26.503094+00:00
deployedTrue
workspace_id7
workspace_namelogapiworkspace
archx86
accelnone
tags
versionsd84ec60d-0527-48e7-b4f0-f5eec93a2831, fd47a339-fd43-43d6-bf3c-dbbf701f1998, 61c710b2-cd83-44b2-a15b-bff7891d06ac, 815ba571-17b6-457a-8bc9-bead597abcd2, 437eca05-f161-495a-8129-687e406d292b
stepslogapicontrol
publishedFalse

Shadow Deploy Sample Inference

We’ll now use our same sample data for an inference to our shadow deployed pipeline, then display the first 20 results with just the comparative outputs.

shadow_date_start = datetime.datetime.now(datetime.timezone.utc)

shadow_result = mainpipeline.infer_from_file('./data/xtest-1k.arrow')

shadow_outputs =  shadow_result.to_pandas()
display(shadow_outputs.loc[0:20,['out.variable','out_logcontrolchallenger01.variable','out_logcontrolchallenger02.variable']])

shadow_date_end = datetime.datetime.now(datetime.timezone.utc)
out.variableout_logcontrolchallenger01.variableout_logcontrolchallenger02.variable
0[718013.75][659806.0][704901.9]
1[615094.56][732883.5][695994.44]
2[448627.72][419508.84][416164.8]
3[758714.2][634028.8][655277.2]
4[513264.7][427209.44][426854.66]
5[668288.0][615501.9][632556.1]
6[1004846.5][1139732.5][1100465.2]
7[684577.2][498328.88][528278.06]
8[727898.1][722664.4][659439.94]
9[559631.1][525746.44][534331.44]
10[340764.53][376337.1][377187.2]
11[442168.06][382053.12][403964.3]
12[630865.6][505608.97][528991.3]
13[559631.1][603260.5][612201.75]
14[909441.1][969585.4][893874.7]
15[313096.0][313633.75][318054.94]
16[404040.8][360413.56][357816.75]
17[292859.5][316674.94][294034.7]
18[338357.88][299907.44][323254.3]
19[682284.6][811896.75][770916.7]
20[583765.94][573618.5][549141.4]

Shadow Deploy Logs

Pipelines with a shadow deployed step include the shadow inference result in the same format as the inference result: inference results from shadow deployed models are displayed as out_{model name}.{output variable}.

# Retrieve logs from specific date/time to only get the two DataFrame input inferences in ascending format

# retrieve the authorization token
headers = wl.auth.auth_header()

url = f"{wl.api_endpoint}/v1/api/pipelines/get_logs"

# Standard log retrieval

data = {
    'pipeline_name': main_pipeline_name,
    'workspace_id': workspace_id,
    'order': 'Asc',
    'start_time': f'{shadow_date_start.isoformat()}',
    'end_time': f'{shadow_date_end.isoformat()}'
}

response = requests.post(url, headers=headers, json=data)
standard_logs = pd.DataFrame.from_records(response.json())

display(standard_logs.head(5).loc[:, ["time", "out", "out_logcontrolchallenger01", "out_logcontrolchallenger02"]])
timeoutout_logcontrolchallenger01out_logcontrolchallenger02
01722283857826{'variable': [718013.75]}{'variable': [659806.0]}{'variable': [704901.9]}
11722283857826{'variable': [615094.56]}{'variable': [732883.5]}{'variable': [695994.44]}
21722283857826{'variable': [448627.72]}{'variable': [419508.84]}{'variable': [416164.8]}
31722283857826{'variable': [758714.2]}{'variable': [634028.8]}{'variable': [655277.2]}
41722283857826{'variable': [513264.7]}{'variable': [427209.44]}{'variable': [426854.66]}

A/B Testing Pipeline

A/B testing allows inference requests to be split between a control model and one or more challenger models. For full details, see the Pipeline Management Guide: A/B Testing.

When the inference results and log entries are displayed, they include the column out._model_split which displays:

FieldTypeDescription
nameStringThe model name used for the inference.
versionStringThe version of the model.
shaStringThe sha hash of the model version.

For this example, the shadow deployed step will be removed and replaced with an A/B Testing step with the ratio 1:1:1, so the control and each of the challenger models will be split randomly between inference requests. A set of sample inferences will be run, then the pipeline logs displayed.

pipeline = (wl.build_pipeline(“randomsplitpipeline-demo”)
.add_random_split([(2, control), (1, challenger)], “session_id”))

ab_date_start = datetime.datetime.now(datetime.timezone.utc)
mainpipeline.undeploy()

# remove the shadow deploy steps
mainpipeline.clear()

# Add the a/b test step to the pipeline
mainpipeline.add_random_split([(1, housing_model_control), 
                               (1, housing_model_challenger01), 
                               (1, housing_model_challenger02)], 
                               "session_id")

mainpipeline.deploy()

# Perform sample inferences of 20 rows and display the results

abtesting_inputs = pd.read_json('./data/xtest-1k.df.json')

for index, row in abtesting_inputs.sample(20).iterrows():
    display(mainpipeline.infer(
        row.to_frame('tensor')
        .reset_index())
        .loc[:,["out._model_split", "out.variable"]]
    )

ab_date_end = datetime.datetime.now(datetime.timezone.utc)
out._model_splitout.variable
0[{"model_version":{"name":"logapicontrol","visibility":"private","workspace_id":7,"conversion":{"arch":"x86","accel":"none","python_version":"3.8","requirements":[],"framework":"onnx"},"id":4,"image_path":null,"status":"ready","task_id":null,"file_info":{"version":"196423d7-72d6-437b-8b9f-da0994a1e31a","sha":"e22a0831aafd9917f3cc87a15ed267797f80e2afa12ad7d8810ca58f173b8cc6","file_name":"rf_model.onnx","size":225818},"created_on_version":"2024.2.0","created_by":"john.hansarick@wallaroo.ai","created_at":null,"deployed":false},"config":{"id":5,"model_version_id":4,"runtime":"onnx","filter_threshold":null,"tensor_fields":["tensor"],"input_schema":null,"output_schema":null,"batch_config":null,"sidekick_uri":null}}][413013.38]
out._model_splitout.variable
0[{"model_version":{"name":"logcontrolchallenger02","visibility":"private","workspace_id":7,"conversion":{"arch":"x86","accel":"none","python_version":"3.8","requirements":[],"framework":"onnx"},"id":6,"image_path":null,"status":"ready","task_id":null,"file_info":{"version":"d6f20d57-946a-4079-88f9-e908c085669c","sha":"ed6065a79d841f7e96307bb20d5ef22840f15da0b587efb51425c7ad60589d6a","file_name":"gbr_model.onnx","size":214380},"created_on_version":"2024.2.0","created_by":"john.hansarick@wallaroo.ai","created_at":null,"deployed":false},"config":{"id":9,"model_version_id":6,"runtime":"onnx","filter_threshold":null,"tensor_fields":["tensor"],"input_schema":null,"output_schema":null,"batch_config":null,"sidekick_uri":null}}][290710.4]
out._model_splitout.variable
0[{"model_version":{"name":"logcontrolchallenger01","visibility":"private","workspace_id":7,"conversion":{"arch":"x86","accel":"none","python_version":"3.8","requirements":[],"framework":"onnx"},"id":5,"image_path":null,"status":"ready","task_id":null,"file_info":{"version":"da42f143-0dd7-44d2-a7ab-2694a3f2eb70","sha":"31e92d6ccb27b041a324a7ac22cf95d9d6cc3aa7e8263a229f7c4aec4938657c","file_name":"xgb_model.onnx","size":171121},"created_on_version":"2024.2.0","created_by":"john.hansarick@wallaroo.ai","created_at":null,"deployed":false},"config":{"id":7,"model_version_id":5,"runtime":"onnx","filter_threshold":null,"tensor_fields":["tensor"],"input_schema":null,"output_schema":null,"batch_config":null,"sidekick_uri":null}}][420434.13]
out._model_splitout.variable
0[{"model_version":{"name":"logcontrolchallenger02","visibility":"private","workspace_id":7,"conversion":{"arch":"x86","accel":"none","python_version":"3.8","requirements":[],"framework":"onnx"},"id":6,"image_path":null,"status":"ready","task_id":null,"file_info":{"version":"d6f20d57-946a-4079-88f9-e908c085669c","sha":"ed6065a79d841f7e96307bb20d5ef22840f15da0b587efb51425c7ad60589d6a","file_name":"gbr_model.onnx","size":214380},"created_on_version":"2024.2.0","created_by":"john.hansarick@wallaroo.ai","created_at":null,"deployed":false},"config":{"id":9,"model_version_id":6,"runtime":"onnx","filter_threshold":null,"tensor_fields":["tensor"],"input_schema":null,"output_schema":null,"batch_config":null,"sidekick_uri":null}}][637474.1]
out._model_splitout.variable
0[{"model_version":{"name":"logcontrolchallenger01","visibility":"private","workspace_id":7,"conversion":{"arch":"x86","accel":"none","python_version":"3.8","requirements":[],"framework":"onnx"},"id":5,"image_path":null,"status":"ready","task_id":null,"file_info":{"version":"da42f143-0dd7-44d2-a7ab-2694a3f2eb70","sha":"31e92d6ccb27b041a324a7ac22cf95d9d6cc3aa7e8263a229f7c4aec4938657c","file_name":"xgb_model.onnx","size":171121},"created_on_version":"2024.2.0","created_by":"john.hansarick@wallaroo.ai","created_at":null,"deployed":false},"config":{"id":7,"model_version_id":5,"runtime":"onnx","filter_threshold":null,"tensor_fields":["tensor"],"input_schema":null,"output_schema":null,"batch_config":null,"sidekick_uri":null}}][314528.34]
out._model_splitout.variable
0[{"model_version":{"name":"logcontrolchallenger02","visibility":"private","workspace_id":7,"conversion":{"arch":"x86","accel":"none","python_version":"3.8","requirements":[],"framework":"onnx"},"id":6,"image_path":null,"status":"ready","task_id":null,"file_info":{"version":"d6f20d57-946a-4079-88f9-e908c085669c","sha":"ed6065a79d841f7e96307bb20d5ef22840f15da0b587efb51425c7ad60589d6a","file_name":"gbr_model.onnx","size":214380},"created_on_version":"2024.2.0","created_by":"john.hansarick@wallaroo.ai","created_at":null,"deployed":false},"config":{"id":9,"model_version_id":6,"runtime":"onnx","filter_threshold":null,"tensor_fields":["tensor"],"input_schema":null,"output_schema":null,"batch_config":null,"sidekick_uri":null}}][848861.3]
out._model_splitout.variable
0[{"model_version":{"name":"logcontrolchallenger02","visibility":"private","workspace_id":7,"conversion":{"arch":"x86","accel":"none","python_version":"3.8","requirements":[],"framework":"onnx"},"id":6,"image_path":null,"status":"ready","task_id":null,"file_info":{"version":"d6f20d57-946a-4079-88f9-e908c085669c","sha":"ed6065a79d841f7e96307bb20d5ef22840f15da0b587efb51425c7ad60589d6a","file_name":"gbr_model.onnx","size":214380},"created_on_version":"2024.2.0","created_by":"john.hansarick@wallaroo.ai","created_at":null,"deployed":false},"config":{"id":9,"model_version_id":6,"runtime":"onnx","filter_threshold":null,"tensor_fields":["tensor"],"input_schema":null,"output_schema":null,"batch_config":null,"sidekick_uri":null}}][878232.2]
out._model_splitout.variable
0[{"model_version":{"name":"logcontrolchallenger01","visibility":"private","workspace_id":7,"conversion":{"arch":"x86","accel":"none","python_version":"3.8","requirements":[],"framework":"onnx"},"id":5,"image_path":null,"status":"ready","task_id":null,"file_info":{"version":"da42f143-0dd7-44d2-a7ab-2694a3f2eb70","sha":"31e92d6ccb27b041a324a7ac22cf95d9d6cc3aa7e8263a229f7c4aec4938657c","file_name":"xgb_model.onnx","size":171121},"created_on_version":"2024.2.0","created_by":"john.hansarick@wallaroo.ai","created_at":null,"deployed":false},"config":{"id":7,"model_version_id":5,"runtime":"onnx","filter_threshold":null,"tensor_fields":["tensor"],"input_schema":null,"output_schema":null,"batch_config":null,"sidekick_uri":null}}][517068.47]
out._model_splitout.variable
0[{"model_version":{"name":"logapicontrol","visibility":"private","workspace_id":7,"conversion":{"arch":"x86","accel":"none","python_version":"3.8","requirements":[],"framework":"onnx"},"id":4,"image_path":null,"status":"ready","task_id":null,"file_info":{"version":"196423d7-72d6-437b-8b9f-da0994a1e31a","sha":"e22a0831aafd9917f3cc87a15ed267797f80e2afa12ad7d8810ca58f173b8cc6","file_name":"rf_model.onnx","size":225818},"created_on_version":"2024.2.0","created_by":"john.hansarick@wallaroo.ai","created_at":null,"deployed":false},"config":{"id":5,"model_version_id":4,"runtime":"onnx","filter_threshold":null,"tensor_fields":["tensor"],"input_schema":null,"output_schema":null,"batch_config":null,"sidekick_uri":null}}][518869.03]
out._model_splitout.variable
0[{"model_version":{"name":"logcontrolchallenger02","visibility":"private","workspace_id":7,"conversion":{"arch":"x86","accel":"none","python_version":"3.8","requirements":[],"framework":"onnx"},"id":6,"image_path":null,"status":"ready","task_id":null,"file_info":{"version":"d6f20d57-946a-4079-88f9-e908c085669c","sha":"ed6065a79d841f7e96307bb20d5ef22840f15da0b587efb51425c7ad60589d6a","file_name":"gbr_model.onnx","size":214380},"created_on_version":"2024.2.0","created_by":"john.hansarick@wallaroo.ai","created_at":null,"deployed":false},"config":{"id":9,"model_version_id":6,"runtime":"onnx","filter_threshold":null,"tensor_fields":["tensor"],"input_schema":null,"output_schema":null,"batch_config":null,"sidekick_uri":null}}][670369.3]
out._model_splitout.variable
0[{"model_version":{"name":"logcontrolchallenger01","visibility":"private","workspace_id":7,"conversion":{"arch":"x86","accel":"none","python_version":"3.8","requirements":[],"framework":"onnx"},"id":5,"image_path":null,"status":"ready","task_id":null,"file_info":{"version":"da42f143-0dd7-44d2-a7ab-2694a3f2eb70","sha":"31e92d6ccb27b041a324a7ac22cf95d9d6cc3aa7e8263a229f7c4aec4938657c","file_name":"xgb_model.onnx","size":171121},"created_on_version":"2024.2.0","created_by":"john.hansarick@wallaroo.ai","created_at":null,"deployed":false},"config":{"id":7,"model_version_id":5,"runtime":"onnx","filter_threshold":null,"tensor_fields":["tensor"],"input_schema":null,"output_schema":null,"batch_config":null,"sidekick_uri":null}}][400925.9]
out._model_splitout.variable
0[{"model_version":{"name":"logcontrolchallenger02","visibility":"private","workspace_id":7,"conversion":{"arch":"x86","accel":"none","python_version":"3.8","requirements":[],"framework":"onnx"},"id":6,"image_path":null,"status":"ready","task_id":null,"file_info":{"version":"d6f20d57-946a-4079-88f9-e908c085669c","sha":"ed6065a79d841f7e96307bb20d5ef22840f15da0b587efb51425c7ad60589d6a","file_name":"gbr_model.onnx","size":214380},"created_on_version":"2024.2.0","created_by":"john.hansarick@wallaroo.ai","created_at":null,"deployed":false},"config":{"id":9,"model_version_id":6,"runtime":"onnx","filter_threshold":null,"tensor_fields":["tensor"],"input_schema":null,"output_schema":null,"batch_config":null,"sidekick_uri":null}}][365585.75]
out._model_splitout.variable
0[{"model_version":{"name":"logapicontrol","visibility":"private","workspace_id":7,"conversion":{"arch":"x86","accel":"none","python_version":"3.8","requirements":[],"framework":"onnx"},"id":4,"image_path":null,"status":"ready","task_id":null,"file_info":{"version":"196423d7-72d6-437b-8b9f-da0994a1e31a","sha":"e22a0831aafd9917f3cc87a15ed267797f80e2afa12ad7d8810ca58f173b8cc6","file_name":"rf_model.onnx","size":225818},"created_on_version":"2024.2.0","created_by":"john.hansarick@wallaroo.ai","created_at":null,"deployed":false},"config":{"id":5,"model_version_id":4,"runtime":"onnx","filter_threshold":null,"tensor_fields":["tensor"],"input_schema":null,"output_schema":null,"batch_config":null,"sidekick_uri":null}}][449699.88]
out._model_splitout.variable
0[{"model_version":{"name":"logapicontrol","visibility":"private","workspace_id":7,"conversion":{"arch":"x86","accel":"none","python_version":"3.8","requirements":[],"framework":"onnx"},"id":4,"image_path":null,"status":"ready","task_id":null,"file_info":{"version":"196423d7-72d6-437b-8b9f-da0994a1e31a","sha":"e22a0831aafd9917f3cc87a15ed267797f80e2afa12ad7d8810ca58f173b8cc6","file_name":"rf_model.onnx","size":225818},"created_on_version":"2024.2.0","created_by":"john.hansarick@wallaroo.ai","created_at":null,"deployed":false},"config":{"id":5,"model_version_id":4,"runtime":"onnx","filter_threshold":null,"tensor_fields":["tensor"],"input_schema":null,"output_schema":null,"batch_config":null,"sidekick_uri":null}}][340764.53]
out._model_splitout.variable
0[{"model_version":{"name":"logcontrolchallenger02","visibility":"private","workspace_id":7,"conversion":{"arch":"x86","accel":"none","python_version":"3.8","requirements":[],"framework":"onnx"},"id":6,"image_path":null,"status":"ready","task_id":null,"file_info":{"version":"d6f20d57-946a-4079-88f9-e908c085669c","sha":"ed6065a79d841f7e96307bb20d5ef22840f15da0b587efb51425c7ad60589d6a","file_name":"gbr_model.onnx","size":214380},"created_on_version":"2024.2.0","created_by":"john.hansarick@wallaroo.ai","created_at":null,"deployed":false},"config":{"id":9,"model_version_id":6,"runtime":"onnx","filter_threshold":null,"tensor_fields":["tensor"],"input_schema":null,"output_schema":null,"batch_config":null,"sidekick_uri":null}}][473311.16]
out._model_splitout.variable
0[{"model_version":{"name":"logapicontrol","visibility":"private","workspace_id":7,"conversion":{"arch":"x86","accel":"none","python_version":"3.8","requirements":[],"framework":"onnx"},"id":4,"image_path":null,"status":"ready","task_id":null,"file_info":{"version":"196423d7-72d6-437b-8b9f-da0994a1e31a","sha":"e22a0831aafd9917f3cc87a15ed267797f80e2afa12ad7d8810ca58f173b8cc6","file_name":"rf_model.onnx","size":225818},"created_on_version":"2024.2.0","created_by":"john.hansarick@wallaroo.ai","created_at":null,"deployed":false},"config":{"id":5,"model_version_id":4,"runtime":"onnx","filter_threshold":null,"tensor_fields":["tensor"],"input_schema":null,"output_schema":null,"batch_config":null,"sidekick_uri":null}}][450867.7]
out._model_splitout.variable
0[{"model_version":{"name":"logapicontrol","visibility":"private","workspace_id":7,"conversion":{"arch":"x86","accel":"none","python_version":"3.8","requirements":[],"framework":"onnx"},"id":4,"image_path":null,"status":"ready","task_id":null,"file_info":{"version":"196423d7-72d6-437b-8b9f-da0994a1e31a","sha":"e22a0831aafd9917f3cc87a15ed267797f80e2afa12ad7d8810ca58f173b8cc6","file_name":"rf_model.onnx","size":225818},"created_on_version":"2024.2.0","created_by":"john.hansarick@wallaroo.ai","created_at":null,"deployed":false},"config":{"id":5,"model_version_id":4,"runtime":"onnx","filter_threshold":null,"tensor_fields":["tensor"],"input_schema":null,"output_schema":null,"batch_config":null,"sidekick_uri":null}}][756045.9]
out._model_splitout.variable
0[{"model_version":{"name":"logcontrolchallenger02","visibility":"private","workspace_id":7,"conversion":{"arch":"x86","accel":"none","python_version":"3.8","requirements":[],"framework":"onnx"},"id":6,"image_path":null,"status":"ready","task_id":null,"file_info":{"version":"d6f20d57-946a-4079-88f9-e908c085669c","sha":"ed6065a79d841f7e96307bb20d5ef22840f15da0b587efb51425c7ad60589d6a","file_name":"gbr_model.onnx","size":214380},"created_on_version":"2024.2.0","created_by":"john.hansarick@wallaroo.ai","created_at":null,"deployed":false},"config":{"id":9,"model_version_id":6,"runtime":"onnx","filter_threshold":null,"tensor_fields":["tensor"],"input_schema":null,"output_schema":null,"batch_config":null,"sidekick_uri":null}}][334796.13]
out._model_splitout.variable
0[{"model_version":{"name":"logapicontrol","visibility":"private","workspace_id":7,"conversion":{"arch":"x86","accel":"none","python_version":"3.8","requirements":[],"framework":"onnx"},"id":4,"image_path":null,"status":"ready","task_id":null,"file_info":{"version":"196423d7-72d6-437b-8b9f-da0994a1e31a","sha":"e22a0831aafd9917f3cc87a15ed267797f80e2afa12ad7d8810ca58f173b8cc6","file_name":"rf_model.onnx","size":225818},"created_on_version":"2024.2.0","created_by":"john.hansarick@wallaroo.ai","created_at":null,"deployed":false},"config":{"id":5,"model_version_id":4,"runtime":"onnx","filter_threshold":null,"tensor_fields":["tensor"],"input_schema":null,"output_schema":null,"batch_config":null,"sidekick_uri":null}}][757403.4]
out._model_splitout.variable
0[{"model_version":{"name":"logapicontrol","visibility":"private","workspace_id":7,"conversion":{"arch":"x86","accel":"none","python_version":"3.8","requirements":[],"framework":"onnx"},"id":4,"image_path":null,"status":"ready","task_id":null,"file_info":{"version":"196423d7-72d6-437b-8b9f-da0994a1e31a","sha":"e22a0831aafd9917f3cc87a15ed267797f80e2afa12ad7d8810ca58f173b8cc6","file_name":"rf_model.onnx","size":225818},"created_on_version":"2024.2.0","created_by":"john.hansarick@wallaroo.ai","created_at":null,"deployed":false},"config":{"id":5,"model_version_id":4,"runtime":"onnx","filter_threshold":null,"tensor_fields":["tensor"],"input_schema":null,"output_schema":null,"batch_config":null,"sidekick_uri":null}}][941029.0]

Retrieve A/B Testing Log Files through API

The log files for A/B Testing pipeline inference results contain the model information with the model outputs in the out field.

# Retrieve logs from specific date/time to only get the two DataFrame input inferences in ascending format

# retrieve the authorization token
headers = wl.auth.auth_header()

url = f"{wl.api_endpoint}/v1/api/pipelines/get_logs"

# Standard log retrieval

data = {
    'pipeline_name': main_pipeline_name,
    'workspace_id': workspace_id,
    'order': 'Asc',
    'start_time': f'{ab_date_start.isoformat()}',
    'end_time': f'{ab_date_end.isoformat()}'
}

response = requests.post(url, headers=headers, json=data)
standard_logs = pd.DataFrame.from_records(response.json())

display(standard_logs.head(5).loc[:, ["time", "out"]])
timeout
01722283909751{'_model_split': ['{"model_version":{"name":"logapicontrol","visibility":"private","workspace_id":7,"conversion":{"arch":"x86","accel":"none","python_version":"3.8","requirements":[],"framework":"onnx"},"id":4,"image_path":null,"status":"ready","task_id":null,"file_info":{"version":"196423d7-72d6-437b-8b9f-da0994a1e31a","sha":"e22a0831aafd9917f3cc87a15ed267797f80e2afa12ad7d8810ca58f173b8cc6","file_name":"rf_model.onnx","size":225818},"created_on_version":"2024.2.0","created_by":"john.hansarick@wallaroo.ai","created_at":null,"deployed":false},"config":{"id":5,"model_version_id":4,"runtime":"onnx","filter_threshold":null,"tensor_fields":["tensor"],"input_schema":null,"output_schema":null,"batch_config":null,"sidekick_uri":null}}'], 'variable': [413013.38]}
11722283909979{'_model_split': ['{"model_version":{"name":"logcontrolchallenger02","visibility":"private","workspace_id":7,"conversion":{"arch":"x86","accel":"none","python_version":"3.8","requirements":[],"framework":"onnx"},"id":6,"image_path":null,"status":"ready","task_id":null,"file_info":{"version":"d6f20d57-946a-4079-88f9-e908c085669c","sha":"ed6065a79d841f7e96307bb20d5ef22840f15da0b587efb51425c7ad60589d6a","file_name":"gbr_model.onnx","size":214380},"created_on_version":"2024.2.0","created_by":"john.hansarick@wallaroo.ai","created_at":null,"deployed":false},"config":{"id":9,"model_version_id":6,"runtime":"onnx","filter_threshold":null,"tensor_fields":["tensor"],"input_schema":null,"output_schema":null,"batch_config":null,"sidekick_uri":null}}'], 'variable': [290710.4]}
21722283910154{'_model_split': ['{"model_version":{"name":"logcontrolchallenger01","visibility":"private","workspace_id":7,"conversion":{"arch":"x86","accel":"none","python_version":"3.8","requirements":[],"framework":"onnx"},"id":5,"image_path":null,"status":"ready","task_id":null,"file_info":{"version":"da42f143-0dd7-44d2-a7ab-2694a3f2eb70","sha":"31e92d6ccb27b041a324a7ac22cf95d9d6cc3aa7e8263a229f7c4aec4938657c","file_name":"xgb_model.onnx","size":171121},"created_on_version":"2024.2.0","created_by":"john.hansarick@wallaroo.ai","created_at":null,"deployed":false},"config":{"id":7,"model_version_id":5,"runtime":"onnx","filter_threshold":null,"tensor_fields":["tensor"],"input_schema":null,"output_schema":null,"batch_config":null,"sidekick_uri":null}}'], 'variable': [420434.13]}
31722283910381{'_model_split': ['{"model_version":{"name":"logcontrolchallenger02","visibility":"private","workspace_id":7,"conversion":{"arch":"x86","accel":"none","python_version":"3.8","requirements":[],"framework":"onnx"},"id":6,"image_path":null,"status":"ready","task_id":null,"file_info":{"version":"d6f20d57-946a-4079-88f9-e908c085669c","sha":"ed6065a79d841f7e96307bb20d5ef22840f15da0b587efb51425c7ad60589d6a","file_name":"gbr_model.onnx","size":214380},"created_on_version":"2024.2.0","created_by":"john.hansarick@wallaroo.ai","created_at":null,"deployed":false},"config":{"id":9,"model_version_id":6,"runtime":"onnx","filter_threshold":null,"tensor_fields":["tensor"],"input_schema":null,"output_schema":null,"batch_config":null,"sidekick_uri":null}}'], 'variable': [637474.1]}
41722283910598{'_model_split': ['{"model_version":{"name":"logcontrolchallenger01","visibility":"private","workspace_id":7,"conversion":{"arch":"x86","accel":"none","python_version":"3.8","requirements":[],"framework":"onnx"},"id":5,"image_path":null,"status":"ready","task_id":null,"file_info":{"version":"da42f143-0dd7-44d2-a7ab-2694a3f2eb70","sha":"31e92d6ccb27b041a324a7ac22cf95d9d6cc3aa7e8263a229f7c4aec4938657c","file_name":"xgb_model.onnx","size":171121},"created_on_version":"2024.2.0","created_by":"john.hansarick@wallaroo.ai","created_at":null,"deployed":false},"config":{"id":7,"model_version_id":5,"runtime":"onnx","filter_threshold":null,"tensor_fields":["tensor"],"input_schema":null,"output_schema":null,"batch_config":null,"sidekick_uri":null}}'], 'variable': [314528.34]}

Undeploy Main Pipeline

With the examples and tutorial complete, we will undeploy the main pipeline and return the resources back to the Wallaroo instance.

mainpipeline.undeploy()
namelogapipipeline
created2024-07-29 20:06:53.925863+00:00
last_updated2024-07-29 20:11:37.199597+00:00
deployedFalse
workspace_id7
workspace_namelogapiworkspace
archx86
accelnone
tags
versions2bd6b3a3-0044-459b-8b91-3796eb64be47, d84ec60d-0527-48e7-b4f0-f5eec93a2831, fd47a339-fd43-43d6-bf3c-dbbf701f1998, 61c710b2-cd83-44b2-a15b-bff7891d06ac, 815ba571-17b6-457a-8bc9-bead597abcd2, 437eca05-f161-495a-8129-687e406d292b
stepslogapicontrol
publishedFalse