Color Mixing#

This example demonstrates how EOS can be used to implement a virtual color mixing experiment. In this experiment, we mix CMYK ingredient colors to produce a target color. By employing Bayesian optimization, the goal is to find task input parameters to synthesize a target color with a secondary objective of minimizing the amount of color ingredients used. To make it easy to try out, this example uses no physical devices, but instead uses virtual ones. Color mixing is simulated using real-time fluid simulation running in a web browser.

The example is implemented in an EOS package called color_lab, and can be found here.

Installation#

  1. Clone the eos-examples repository inside the EOS user directory:

cd eos/user
git clone https://github.com/UNC-Robotics/eos-examples eos_examples
  1. Install the package’s dependencies in the EOS venv:

uv pip install -r user/eos_examples/color_lab/pyproject.toml
  1. Load the package in EOS:

Edit the config.yml file to have the following for user_dir, labs, and experiments:

user_dir: ./user
labs:
  - color_lab
experiments:
  - color_mixing

Sample Usage#

  1. cd into the eos directory

  2. Run python3 user/eos_examples/color_lab/device_drivers.py to start the fluid simulation and simulated device drivers. Browser windows will open automatically.

    On dual-GPU systems (e.g., NVIDIA + integrated GPU on Wayland), pass --browser chrome --nvidia to launch the browser with NVIDIA GPU offload and X11 mode for correct WebGL rendering.

  3. Start EOS.

  4. Submit tasks, experiments, or campaigns through the REST API.

You can submit a request to run a campaign through the REST API with curl as follows:

curl -X POST http://localhost:8070/api/campaigns \
     -H "Content-Type: application/json" \
     -d '{
          "name": "color_mixing",
          "experiment_type": "color_mixing",
          "owner": "name",
          "priority": 0,
          "max_experiments": 100,
          "max_concurrent_experiments": 3,
          "optimize": true,
          "optimizer_ip": "127.0.0.1",
          "global_parameters": {
            "score_color": {
                "target_color": [47, 181, 49]
            }
          }
    }'

Note

Do not minimize the fluid simulation browser windows while the campaign is running as the simulation may pause running.

Package Structure#

The top-level structure of the color_lab package is as follows:

color_lab/
├── common/ <-- contains shared code
├── devices/ <-- contains the device implementations
├── experiments/ <-- contains the color mixing experiment definitions
├── labs/ <-- contains the laboratory definition
├── tasks/ <-- contains the task definitions
├── fluid_simulation/ <-- contains the source code for the fluid simulation web app
└── device_drivers.py <-- a script for starting the fluid simulation and socket servers for the devices

Devices#

The package contains the following device implementations:

  • Color station: Sends commands to the fluid simulation to dispense and mix colors, and queries it to get the average fluid color.

  • Robot arm: Moves sample containers between other devices.

  • Cleaning station: Cleans sample containers (by erasing their stored metadata).

This is the Python code for the color station device:

device.py

from typing import Any

from eos.resources.entities.resource import Resource
from eos.devices.base_device import BaseDevice
from user.eos_examples.color_lab.common.device_client import DeviceClient


class ColorStation(BaseDevice):
    async def _initialize(self, init_parameters: dict[str, Any]) -> None:
        port = int(init_parameters["port"])
        self.client = DeviceClient(port)
        self.client.open_connection()

    async def _cleanup(self) -> None:
        self.client.close_connection()

    async def _report(self) -> dict[str, Any]:
        return {}

    def mix(
        self,
        container: Resource,
        cyan_volume: float,
        cyan_strength: float,
        magenta_volume: float,
        magenta_strength: float,
        yellow_volume: float,
        yellow_strength: float,
        black_volume: float,
        black_strength: float,
        mixing_time: int,
        mixing_speed: int,
    ) -> Resource:
        ...

    def analyze(self, container: Resource) -> tuple[Resource, tuple[int, int, int]]:
        rgb = self.client.send_command("analyze", {})
        return container, rgb

You will notice that the color station combines both mixing and analysis into a single device. This ensures that a single allocation connects to a single fluid simulation window, so the color that is mixed is the same one that gets analyzed.

The device implementation communicates with another process over a socket. This is a common pattern when integrating devices in the laboratory, as device drivers are usually provided by a 3rd party, such as the device manufacturer. So often the device implementation simply uses the existing driver. In some cases, the device implementation may include a full driver implementation.

The device implementation initializes a client that connects to the device driver over a socket. The device implements a mix function for dispensing and mixing colors, and an analyze function that returns the average RGB value of the fluid color from the fluid simulation.

The device YAML file for the color station device is:

device.yml

type: color_station
desc: Color mixing and analysis station backed by a fluid simulation

init_parameters:
  port: 5003

Tasks#

The package contains the following tasks:

  • Retrieve container: Retrieves a beaker from storage and moves it to a color station using the robot arm.

  • Mix colors: Dispenses and mixes colors using a color station (fluid simulation).

  • Analyze color: Analyzes the color of the fluid using a color station (fluid simulation).

  • Score color: Calculates a loss function taking into account how close the mixed color is to the target color and how much color ingredients were used.

  • Empty container: Empties a beaker with the robot arm.

  • Clean container: Cleans a beaker with the cleaning station.

  • Store container: Stores a beaker in storage with the robot arm.

This is the Python code for the “Analyze color” task:

task.py

from eos.tasks.base_task import BaseTask


class AnalyzeColor(BaseTask):
    async def _execute(
        self,
        devices: BaseTask.DevicesType,
        parameters: BaseTask.ParametersType,
        resources: BaseTask.ResourcesType,
    ) -> BaseTask.OutputType:
        color_station = devices["color_station"]

        resources["beaker"], rgb = color_station.analyze(resources["beaker"])

        output_parameters = {
            "red": rgb[0],
            "green": rgb[1],
            "blue": rgb[2],
        }

        return output_parameters, resources, None

The task implementation is straightforward. We first get a reference to the color station device. Then, we call the analyze function from the color station device we saw earlier. Finally, we construct and return the dict of output parameters and the resources.

The task YAML file is the following:

task.yml

type: Analyze Color
desc: Analyze the color of a solution

devices:
  color_station:
    type: color_station

input_resources:
  beaker:
    type: beaker

output_parameters:
  red:
    type: int
    unit: n/a
    desc: The red component of the color
  green:
    type: int
    unit: n/a
    desc: The green component of the color
  blue:
    type: int
    unit: n/a
    desc: The blue component of the color

Laboratory#

The laboratory YAML definition is shown below.

We define the devices we discussed earlier. Note that we define three color stations so the laboratory can support up to three simultaneous color mixing experiments.

We also define the resource types and the actual resources (beakers) with their initial locations.

lab.yml

name: color_lab
desc: A laboratory for color analysis and mixing

devices:
  robot_arm:
    desc: Robotic arm for moving containers
    type: robot_arm
    computer: eos_computer

    init_parameters:
      locations:
        - container_storage
        - color_station_1
        - color_station_2
        - color_station_3
        - cleaning_station
        - emptying_location

  cleaning_station:
    desc: Station for cleaning containers
    type: cleaning_station
    computer: eos_computer

    meta:
      location: cleaning_station

  color_station_1:
    desc: Color mixing and analysis station backed by a fluid simulation
    type: color_station
    computer: eos_computer

    init_parameters:
      port: 5003

    meta:
      location: color_station_1

  color_station_2:
    desc: Color mixing and analysis station backed by a fluid simulation
    type: color_station
    computer: eos_computer

    init_parameters:
      port: 5004

    meta:
      location: color_station_2

  color_station_3:
    desc: Color mixing and analysis station backed by a fluid simulation
    type: color_station
    computer: eos_computer

    init_parameters:
      port: 5005

    meta:
      location: color_station_3


resource_types:
  beaker:
    meta:
      capacity: 300

resources:
  c_a:
    type: beaker
    meta:
      location: container_storage
  c_b:
    type: beaker
    meta:
      location: container_storage
  c_c:
    type: beaker
    meta:
      location: container_storage
  c_d:
    type: beaker
    meta:
      location: container_storage
  c_e:
    type: beaker
    meta:
      location: container_storage

Experiment#

The color mixing experiment is a linear sequence of the following tasks:

  1. retrieve_container: Get a beaker from storage and move it to a color station.

  2. mix_colors: Iteratively dispense and mix the colors in the beaker.

  3. analyze_color: Analyze the color of the solution in the beaker and output the RGB values.

  4. score_color: Score the color (compute the loss function) based on the RGB values.

  5. empty_container: Empty the beaker and move it to the cleaning station.

  6. clean_container: Clean the beaker by rinsing it with distilled water.

  7. store_container: Store the beaker back in the storage.

The YAML definition of the experiment is shown below:

experiment.yml

type: color_mixing
desc: Experiment to find optimal parameters to synthesize a desired color

labs:
  - color_lab

tasks:
  - name: retrieve_container
    type: Retrieve Container
    desc: Get a container from storage and move it to the color dispenser
    duration: 5
    devices:
      robot_arm:
        lab_name: color_lab
        name: robot_arm
      color_station:
        allocation_type: dynamic
        device_type: color_station
    resources:
      beaker:
        allocation_type: dynamic
        resource_type: beaker
    dependencies: []

  - name: mix_colors
    type: Mix Colors
    desc: Mix the colors in the container
    duration: 20
    devices:
      color_station: retrieve_container.color_station
    resources:
      beaker: retrieve_container.beaker
    parameters:
      cyan_volume: eos_dynamic
      cyan_strength: eos_dynamic
      magenta_volume: eos_dynamic
      magenta_strength: eos_dynamic
      yellow_volume: eos_dynamic
      yellow_strength: eos_dynamic
      black_volume: eos_dynamic
      black_strength: eos_dynamic
      mixing_time: eos_dynamic
      mixing_speed: eos_dynamic
    dependencies: [retrieve_container]

  - name: analyze_color
    type: Analyze Color
    desc: Analyze the color of the solution in the container and output the RGB values
    duration: 2
    devices:
      color_station: mix_colors.color_station
    resources:
      beaker: mix_colors.beaker
    dependencies: [mix_colors]

  - name: score_color
    type: Score Color
    desc: Score the color based on the RGB values
    duration: 1
    parameters:
      red: analyze_color.red
      green: analyze_color.green
      blue: analyze_color.blue
      total_color_volume: mix_colors.total_color_volume
      max_total_color_volume: 300.0
      target_color: eos_dynamic
    dependencies: [analyze_color]

  - name: empty_container
    type: Empty Container
    desc: Empty the container and move it to the cleaning station
    duration: 5
    devices:
      robot_arm:
        lab_name: color_lab
        name: robot_arm
      cleaning_station:
        allocation_type: dynamic
        device_type: cleaning_station
        allowed_labs: [color_lab]
    resources:
      beaker: analyze_color.beaker
    parameters:
      emptying_location: emptying_location
    dependencies: [analyze_color]

  - name: clean_container
    type: Clean Container
    desc: Clean the container by rinsing it with distilled water
    duration: 5
    devices:
      cleaning_station: empty_container.cleaning_station
    resources:
      beaker: empty_container.beaker
    parameters:
      duration: 2
    dependencies: [empty_container]

  - name: store_container
    type: Store Container
    desc: Store the container back in the container storage
    duration: 5
    devices:
      robot_arm:
        lab_name: color_lab
        name: robot_arm
    resources:
      beaker: clean_container.beaker
    parameters:
      storage_location: container_storage
    dependencies: [clean_container]

Dynamic Parameters and Optimization#

Dynamic parameters are specified using the special value eos_dynamic in the experiment. For campaigns with optimization (optimize: true), EOS uses the experiment’s optimizer to propose values for the input dynamic parameters. Some dynamic parameters may still need to be provided by the user. In this experiment, score_color.target_color must be provided. Provide it via global_parameters or experiment_parameters in the campaign submission as shown above.

The optimizer used for this experiment is defined in optimizer.py adjacent to the experiment YAML and uses Bayesian optimization to minimize score_color.loss.

References Between Tasks#

EOS experiments commonly link tasks together by referencing devices, resources, and parameters from earlier tasks. The color mixing experiment demonstrates each kind of reference.

Device references: reuse the same physical device across tasks by referencing a named device handle from a prior task.

Example:

- name: mix_colors
  devices:
    color_station: retrieve_container.color_station

- name: analyze_color
  devices:
    color_station: mix_colors.color_station

In the first snippet, the mix_colors task uses the exact color_station allocated during retrieve_container. In the second, analyze_color uses the same color_station, ensuring that the mixed color is analyzed on the same simulation window.

Resource references: pass the same physical resource instance (e.g., a beaker) downstream.

Example:

- name: mix_colors
  resources:
    beaker: retrieve_container.beaker

- name: analyze_color
  resources:
    beaker: mix_colors.beaker

The beaker chosen (dynamically) in retrieve_container is reused by mix_colors, then reused by analyze_color on the same station.

Parameter references: feed outputs from one task as inputs to another by referencing output parameters.

Example:

- name: score_color
  parameters:
    red: analyze_color.red
    green: analyze_color.green
    blue: analyze_color.blue
    total_color_volume: mix_colors.total_color_volume
    max_total_color_volume: 300.0
    target_color: eos_dynamic

The score_color task consumes the RGB outputs from analyze_color and the total color volume from mix_colors.