This document details the key steps needed to start using the Sight system.
- The Prerequisites section details all the steps needed to configure Sight to work within a given GCP project, including accounts, accessing the Sight codebase and launching the Sight server.
- The Logging API section summarizes Sight's APIs for logging the application's execution so that it can be visualized, searched or analyzed by external tools.
- Section Decision API describes Sight's APIs for using external ML libraries to control or optimize the application's execution to achieve the user's objectives.
- Section Sight Utilities details useful supporting tools that make it easier to use Sight.
# Install basic libraries
sudo apt-get update
sudo apt install git
sudo apt install virtualenv
sudo apt-get install python3-pip
# Add Docker's official GPG key:
sudo apt-get update
sudo apt-get install ca-certificates curl gnupg
sudo install -m 0755 -d /etc/apt/keyrings
curl -fsSL https://download.docker.com/linux/debian/gpg | sudo gpg --dearmor -o /etc/apt/keyrings/docker.gpg
sudo chmod a+r /etc/apt/keyrings/docker.gpg
# Add the repository to Apt sources:
echo \
"deb [arch="$(dpkg --print-architecture)" signed-by=/etc/apt/keyrings/docker.gpg] https://download.docker.com/linux/debian \
"$(. /etc/os-release && echo "$VERSION_CODENAME")" stable" | \
sudo tee /etc/apt/sources.list.d/docker.list > /dev/null
sudo apt-get update
# Install the Docker packages.
sudo apt-get install docker-ce docker-ce-cli containerd.io docker-buildx-plugin docker-compose-plugin
# Verify
sudo docker run hello-world
# Run Docker Commands Without Sudo
# Need to close and start the terminal again for this to get effect
sudo usermod -aG docker $USER
If you haven't generated a new identifier required for a partner-code domain in a while, Please generate a new identifier using this link and paste it in terminal to refresh the credentials.
#clone and fetch latest sigh code from gerrit
cd ~/
git clone https://partner-code.googlesource.com/x-sight
cd x-sight
git fetch https://partner-code.googlesource.com/x-sight refs/changes/63/609763/2 && git checkout -b change-609763 FETCH_HEAD
# Create and set-up new virtual environment
cd ~/
mkdir venvs
cd venvs
# 3.9 should work if 10 doesn't
virtualenv sight_env --python=python3.10
source ~/venvs/sight_env/bin/activate
# Install necessary dependencies from requirement.txt file
pip install -r ~/x-sight/sight/requirements.txt
# Set python path to x-sight directory and reload the bashrc file
echo 'export PYTHONPATH="$HOME/x-sight:$PYTHONPATH"' >> ~/.bashrc
source ~/.bashrc
source ~/venvs/sight_env/bin/activate
cd ~/x-sight
For completing rest of the task from prerequisites, either one needs owner role and directly continue to this section or one can create Sight Manager role as following and assign that role to any user to delegate the remaining tasks from prerequisites.
gcloud iam roles create sight_manager --project=$PROJECT_ID --file=sight-manager-role.yaml
gcloud projects add-iam-policy-binding $PROJECT_ID \
--member="user:$USER_ACCOUNT" \
--role="projects/$PROJECT_ID/roles/sight_manager"
We'll need one service account and 2 custom roles (for application user and for service account to be used by sight spawned workers) with necessary permissions to run the Sight.
Make sure following APIs are enabled in your gcp project
- aiplatform.googleapis.com
- lifesciences.googleapis.com
- run.googleapis.com
- cloudresourcemanager.googleapis.com
-
Create a custom role for User working with Sight from the sight-user-role.yaml file available in the root directory of repo.
gcloud iam roles create sight_user --project=$PROJECT_ID --file=sight-user-role.yaml
-
Assign the custom role to user.
gcloud projects add-iam-policy-binding $PROJECT_ID \ --member="user:$USER_ACCOUNT" \ --role="projects/$PROJECT_ID/roles/sight_user"
-
Create the service account for sight related work
gcloud iam service-accounts create sight-service-account \ --display-name="sight service account" --project=$PROJECT_ID
-
Create a custom role for service-account with necessary permissions to work with Sight from the sight-service-account-role.yaml file available in the root directory of repo.
gcloud iam roles create sight_service_account --project=$PROJECT_ID --file=sight-service-account-role.yaml
-
Assign the custom role to the newly created service account so that this service account can have required permissions.
gcloud projects add-iam-policy-binding $PROJECT_ID \ --member="serviceAccount:sight-service-account@$PROJECT_ID.iam.gserviceaccount.com" \ --role="projects/$PROJECT_ID/roles/sight_service_account"
-
Add your project details to .env file in the code
echo "PROJECT_ID=$PROJECT_ID" >> .env
-
Create service image from the code and host it on gcr.io
docker build --tag gcr.io/$PROJECT_ID/sight-service-default -f service/Dockerfile . gcloud auth print-access-token | docker login -u oauth2accesstoken --password-stdin https://gcr.io docker push gcr.io/$PROJECT_ID/sight-service-default
-
With the help of the image, launch cloud run service
gcloud run deploy sight-service-default --image=gcr.io/$PROJECT_ID/sight-service-default:latest --allow-unauthenticated --service-account=sight-service-account@$PROJECT_ID.iam.gserviceaccount.com --concurrency=default --cpu=2 --memory=8Gi --min-instances=1 --max-instances=1 --no-cpu-throttling --region=us-central1 --project=$PROJECT_ID
Host the worker image in a cloud which will be used as default image by the workers spawned using sight unless specified otherwise.
docker build --tag gcr.io/$PROJECT_ID/sight-worker -f sight_all/Dockerfile .
gcloud auth print-access-token | docker login -u oauth2accesstoken --password-stdin https://gcr.io
docker push gcr.io/$PROJECT_ID/sight-worker
Sight enables application developers to log the dynamic behavior of their application at a high level of abstraction, documenting the hierarchical nesting structure of its logic and the full details of the logged data structures. These logs are stored in BigQuery and can be viewed via a specialized GUI that clearly visualizes the log's structure and provides search and navigation capabilities, powered by BigQuery.
Logging is done via an object of type Sight, which exposes a range of logging APIs specialized to document various aspects of application behavior. A Sight object is created as follows:
def get_sight_instance():
params = sight_pb2.Params(
label="experiment", # Human-readable description of the application
project_id=PROJECT_ID, # Name of the user's GCP project
bucket_name=PROJECT_ID+"-sight", # The GCS bucket where the log is stored
)
sight_obj = Sight(params)
return sight_ob
Sight enables users to document the logical nesting structure of their application's execution via the Block object, which is used as follows:
with Block("A", sight):
with Block("A1.1", sight):
sight.text("A1.1 text")
Everything logged to sight while a given object is in scope is nested within that block and this containment relationship is visualized in the GUI.
And one can click on the arrow > button to close down the whole code block of particular region and can focus on the things which are required.
The Attribute API enables users to search for regions of the log where certain conditions hold. This mechanism lets users specify key-value pair bindings that hold while an Attribute object is in scope. Everything logged during this time period is marked with this key-value pair and the GUI lets users search for regions of the log where various constraints on these key-value pairs are satisfied.
with Block("B", sight):
sight.text("B preText")
with Attribute("key", "B1", sight):
with Block("B1", sight):
with Block("B1.1", sight):
sight.text("B1.1 text")
with Attribute("key", "B2", sight):
with Block("B2", sight):
with Block("B2.1", sight):
sight.text("B2.1 text")
sight.text("B postText")
Users typically need to log complex data structures and analyze specific components of these objects. Traditional logging systems typically require such data structures to be serialized into some human-readable string format and then log this string. In contrast, Sight's data_structure library logs the full structure of objects directly, capturing their internal details within the BigQuery database in a structured format, making it possible to visualize them or analyze them programmatically. To use the data_structure while logging using sight, where the first argument of log API can be any supported data_structure supported by python.
with Block("C", sight):
data = list(range(0, 12))
data_structures.log(
{
"x": 1,
(1, 2, 3): ["a", "b", {1: 2}],
1: [1, 2],
"1d": np.array(data),
"2d": np.array(data).reshape((3, 4))
},
sight,
)
The file sight/demo/demo.py
provides examples of how all of the above APIs may
be used and is an excellent starting point to understand the core capabilities
and how the resulting logs are visualized in Sight's GUI. To generate a log,
please edit this file to specify your own project and bucket and then run the
demo code:
python sight/demo/demo.py
Sight then prints out the GCS path where the log is stored, the BigQuery table that references it, and most importantly, the URL of the Sight GUI where this log may be viewed:
While the logging API helps application developers better understand the behavior of their applications, the decision API enables them to incorporate machine learning into their application logic, extending their own domain expertise with new tools from the ML community with minimal modifications to their code. The core idea of the Decision API is that when application developers are not sure what their application should do, they can ask Sight, which may use a wide range of ML algorithms (e.g. Bayesian Optimization, Reinforcement Learning, or Large Language Models) to suggest a course of action. At a later point in the application's execution it must use Sight's Decision API to let the ML algorithm know how effective that suggestion was so that such suggestions may be refined in the future. Sight can train these ML models to provide good advice by running many copies of the application in parallel on GCP's distributed compute clusters.
The Decision API may be used by general applications to make complex decisions,
such as the choice of a caching policy, the layout of widgets on a web page
(e.g. for A/B experiments), selecting the split points in hierarchical sorting
algorithms or choosing the parameters of simulations to minimize prediction
error. To use the Decision API application developers must package their
application into a form that allows Sight to run it many times as it trains the
model and then start execution via Sight's decision.run()
function. Currently
Sight supports the following package types:
- If the application logic is implemented via a dm_env-type environment (e.g. the OpenAI gym) users can provide this environment with no modification and Sight will call the gym's step and reward functions.
- Alternatively, application logic can be encapsulated in a call-back driver function that calls Sight's Decision API directly. Sight will then call this driver function one or more times during the model's training process, as well as during normal program execution.
-
To start the program users must create their environment object and pass it to the
decision.run()
function. This environment may be implemented by the application developer (e.g. ShowerEnv) or may be a ready-made environment from AI gym:decision.run(sight=sight, env=shower_env.ShowerEnv()) decision.run( sight=sight, env=wrappers.GymWrapper(gym.make(flags.FLAGS.env_name)) )
-
The code below provides an example implementation of a gym, which must implement methods reset(), step(), action_spec() and observation_spec().
class ShowerEnv(dm_env.Environment): """dm_env environment for managing shower temperature.""" def __init__(self): self.state = 38 + random.randint(-3, 3) self.shower_length = 60 def action_spec(self): return dm_env.specs.BoundedArray( shape=(), dtype=int, name='action', minimum=0, maximum=2 ) def observation_spec(self): return dm_env.specs.BoundedArray( shape=(), dtype=np.float32, name='observation', minimum=0, maximum=100, ) def step(self, action): self.state += action - 1 self.shower_length -= 1 # Calculate reward, based on whether the shower's temperature is # comfortable (between 37 and 39 degrees C). if self.state >= 37 and self.state <= 39: reward = 1 else: reward = -1 # Check if shower is done if self.shower_length <= 0: done = True else: done = False # Return step information if done: return dm_env.termination( reward, np.array([self.state], dtype=np.float32) ) else: return dm_env.transition(reward, np.array([self.state], dtype=np.float32)) def reset(self): # Reset shower temperature self.state = 38 + random.randint(-3, 3) self.shower_length = 60 return dm_env.restart(np.array([self.state], dtype=np.float32))
The Decision API can be used in generic code by using the Decision API to identify the questions that need to be asked, the format of the answers that need to be given and add explicit calls where the application asks for guidance and reports back on the outcomes of the ML algorithm's suggestions.
-
The first step is to implement the application's entry point as a driver function that is passed as a parameter to the
decision.run()
call. -
Next developers must document the properties of the application's dynamic execution that will be communicated to the ML algorithm as context for the decisions it makes. These are passed to the
state_attrs
parameter and document the range of possible values of each state parameter. -
Finally, developers must document the format of the guidance it needs back from the ML algorithm as a set of named attributes and their allowed value ranges. These are passed to the
action_attrs
parameter.decision.run( sight=sight, driver_fn=driver_fn, state_attrs={ "Temperature": sight_pb2.DecisionConfigurationStart.AttrProps( min_value=0, max_value=100, step_size=1, ), }, action_attrs={ "Direction": sight_pb2.DecisionConfigurationStart.AttrProps( min_value=0, max_value=2, step_size=1, ), }, )
-
The driver function performs all the logic of the application and calls the following functions to interact with the ML algorithm:
-
data_structures.log_var():
When the application logs a variable explicitly named as a state variable, it is communicated to the ML algorithm as part of the application's current state. -
decision.decision_point()
: Asks the ML algorithm for suggestions about how the application should execute. The response a dict with same keys as theaction_attrs
parameter in thedecision.run()
call and values in the allowed range for each such parameter.action = decision_point("label", sight)
-
decision.decision_outcome()
: Communicates to the ML algorithm the effectiveness of the preceding suggestion, with higher values of thereward
parameter indicating that thedecision_outcome("label",updated_timestep.reward,sight)
-
-
Driver function should looks something like:
def driver_fn(sight: Sight) -> None: # Initialize simulation state temperature = 38 + random.randint(-3, 3) shower_length = 60 data_structures.log_var("Temperature", temperature, sight) # Simulation loop for _ in range(shower_length): # Ask Sight's optimizer for the action to perform. chosen_action = decision.decision_point("DP_label", sight) direction = np.array(chosen_action["Direction"], dtype=np.int64) # Change temperature based on the Sight-recommended direction. temperature += direction - 1 data_structures.log_var("Temperature", temperature, sight) # Calculate reward based on whether the temperature target has # been achieved. if temperature >= 37 and temperature <= 39: current_reward = 1 else: current_reward = -1 # Inform Sight of the outcome of the recommended action. decision.decision_outcome( "DO_label", current_reward, sight, )
-
In the case of the dm_env type RL environment provided, the default driver function gets run multiple times while running the simulation, if the custom function is not provided. The usage of decision_point and decision_outcome call in this driver function is shown below:
Applications the use the Decision API may be run in two modes:
- Training: the ML algorithm observes many runs of the application to learn how to make high-quality decisions, and
- Run: the ML algorithm has been trained and the application is run normally, guided by the trained ML algorithm.
To run the application in training mode users must run the application's binary
while setting the command line flag --decision_mode
as train
and must use the
following flags to control the training process:
-
deployment_mode
: The procedure to use when training a model to drive applications that use the Decision API.- distributed: The application is executed in parallel on the GCP cloud.
- docker_local: The application is executed locally in a docker container
- local: The application is executed locally in the current OS environment
-
optimizer_type
: The optimizer to be used while training (vizier, dm-acme, exhaustive_search) -
num_train_workers: Number of workers to use on the GCP cloud in a training run in distributed mode.
-
num_trials: Total number of training trials to perform across all the local or distributed worker nodes.
-
docker_image: docker image to used by worker nodes while running the taks
-
log_path: path to store the logs of workers
Once the ML model has been trained users can use this model to guide ordinary
application runs by executing the application's binary while setting the command
line flag: --decision_mode
as run
and --trained_model_log_id
as $log_id
of generated
sight run while training).
To make it easier to experiment with the Decision API the sight/demo directory contains the following demo applications that use the Decision API in the different ways described above:
- shower_demo_without_env.py (dm_acme): driver function that uses the Decision API explicitly.
- shower_demo_with_env.py (dm_acme): uses the Decision API implicitly via an RL environment that is driven from Sight.
- gym_demo_env.py (dm_acme): uses the Decision API implicitly to drive an AI
gym environment that is specified via the
--env_name
command line flag. - sweetness.py: Simple program that tries to learn the level of sweetness a user likes and uses the explicit Decision API to describe the state and action attributes, as well as the decision point and outcome. Used most effectively with the following optimizers: vizier, exhaustive_search.
- volterra_lotka.py: Simulation of the Volterra-Lotka predator-prey model using the explicit Decision API. Used most effectively with the following optimizers: vizier, exhaustive_search.
To make it easier to start experimenting with Sight, below are some example commands for running demo files with different optimizers.
To use the sight for training optimizer, without any environment, run following command with all the mandatory flags mentioned here:
python sight/demo/shower_demo_without_env.py \
--decision_mode=train \
--deployment_mode=distributed \
--optimizer_type=dm_acme \
--num_train_workers=2 \
--num_trials=5 \
--docker_image=gcr.io/$PROJECT_ID/sight-worker
To use the sight for training optimizer with gym environment, add env_name flag in addition to all the mandatory flags for any other dm_env type environment, no need to pass env_name flag:
python sight/demo/gym_demo_env.py \
--decision_mode=train \
--deployment_mode=distributed \
--optimizer_type=dm_acme \
--num_train_workers=2 \
--num_trials=5 \
--docker_image=gcr.io/$PROJECT_ID/sight-worker \
--project_id=$PROJECT_ID \
--env_name=CartPole-v1
To use sight with vertex AI vizier for hyperparameter turning one can use the following
python sight/demo/sweetness.py \
--decision_mode=train \
--deployment_mode=distributed \
--optimizer_type=vizier \
--num_train_workers=2 \
--num_trials=5 \
--docker_image=gcr.io/$PROJECT_ID/sight-worker
python sight/demo/sweetness.py \
--decision_mode=train \
--deployment_mode=distributed \
--optimizer_type=exhaustive_search \
--num_train_workers=2 \
--num_trials=5 \
--docker_image=gcr.io/$PROJECT_ID/sight-worker
After running the larger experiment, If the user wants to check the status of their experiment, they can run the current_status.py file to check the same. This file calls the server and gets the latest status of the experiment. User can run this file as follows:
python sight/widgets/decision/current_status.py --project_id=$PROJECT_ID --log_id=SIGHT_LOG_ID
Sight applications are trained using a single default server process that is
hosted on CloudRun and is shared among all users of a given GCP project. As this
may cause contention and slowdowns users may also start one or more private
Sight servers that are dedicated to individual training or logging workloads.
This is done by passing the desired unique name of the server in the
--service_name
command line parameter while running the experiment. Users may
also use the --service_docker_file
to specify a specific custom build of Sight
(e.g. if they're experimenting with ML algorithms).
Taking example from here, and adding above mentioned flags:
python sight/demo/gym_demo_env.py \
--service_name=new-service \
--service_docker_file=service/Dockerfile \
--decision_mode=train \
--deployment_mode=distributed \
--optimizer_type=dm_acme \
--num_train_workers=2 \
--num_trials=5 \
--docker_image=gcr.io/cameltrain/sight-worker-meet \
--env_name=CartPole-v1