# Planner The planner is a component that monitors the state of the system and makes adjustments to the number of workers to ensure that the system is running efficiently. It can dynamically scale prefill/decode workers up and down based on a variety of KV metrics. You can find documentation and benchmarking examples in the [planner docs](../../docs/planner.md). ## Usage After you've deployed a dynamo graph, you can start the planner with the following command: ```bash PYTHONPATH=/workspace/examples/llm python components/planner.py --namespace ``` ## Backends 1. `local` - uses circus to start/stop worker subprocesses 2. `kubernetes` - uses the kubernetes API to adjust replicas of the DynamoGraphDeployment resource, which automatically scales the corresponding worker pods up or down ## Local Backend (LocalPlanner) The LocalPlanner is built on top of circus, which is what we use to manage component subprocesses when running dynamo serve. LocalPlanner allows the planner component to scale workers up and down based on system metrics. **Current limitations** 1. Single node only 2. Workers must be using only a single GPU 3. Your initial deployment must be replicas=1 for both prefill and decode We are working on addressing these as fast as possible. ### Under the Hood Circus has a concept of an arbiter and a watcher: - **Arbiter**: The supervisor process that manages all watchers - **Watcher**: A process that encodes environment variables, command, name, and other information needed to run a component When a service is started, each worker process is spun up as a watcher. For example, when starting a VllmWorker, a watcher is created that looks like: ```json { "dynamo_VllmWorker": { "watcher_name": "dynamo_VllmWorker", "cmd": "/opt/dynamo/venv/bin/python3 -m dynamo.sdk.cli.serve_dynamo graphs.agg_router:Frontend --service-name VllmWorker --worker-id $(CIRCUS.WID) --worker-env [{\"CUDA_VISIBLE_DEVICES\": \"0\"}]", "resources": { "allocated_gpus": [ 0 ] }, "lease": 7587886183172559418 } } ``` The arbiter exposes an endpoint allowing messages to add/remove/change watchers. The LocalPlanner leverages this functionality to dynamically adjust worker counts. ### Implementation The planner architecture is designed to be simple and extensible: - An abstract class supports basic add/remove component operations - This is implemented in `local_connector.py` - Circus interaction logic is in `circusd.py`, which reads the statefile, connects to the endpoint, and provides add/remove functionality - Planner starts an instance of `LocalConnector` and uses it to modify the deployment topology ### Statefile The statefile maintains the current state of all running workers and is used by the LocalPlanner to track and modify the deployment. It's stored at `~/.dynamo/state/{namespace}.json` (or in the directory specified by `DYN_LOCAL_STATE_DIR`). The statefile is automatically created when you run dynamo serve and is cleaned up when the arbiter terminates. Each worker is identified as `{namespace}_{component_name}` with an optional numeric suffix for additional instances. #### Example: Adding and Removing Workers Starting with a single decode worker: ```json { "dynamo_VllmWorker": {..., "resources":{...}} } ``` After adding a worker: ```json { "dynamo_VllmWorker": {..., "resources":{...}}, "dynamo_VllmWorker_1": {..., "resources":{...}} } ``` After removing a worker (removes the highest suffix): ```json { "dynamo_VllmWorker": {..., "resources":{...}} } ``` If scaled to zero, the initial entry is kept without resources to maintain configuration information: ```json { "dynamo_VllmWorker": {...} } ``` ### Looking forward - Support for a multinode LocalPlanner - Storing the statefile (and initial configurations) in ETCD using the the new `EtcdKvCache`. ### Testing For manual testing, you can use the controller_test.py file to add/remove components after you've run a serve command on a Dynamo pipeline where the planner is linked. ## Kubernetes Backend The Kubernetes backend works by updating the replicas count of the DynamoGraphDeployment custom resource. When the planner determines that workers need to be scaled up or down based on workload metrics, it uses the Kubernetes API to patch the DynamoGraphDeployment resource specification, changing the replicas count for the appropriate worker component. The Kubernetes operator then reconciles this change by creating or terminating the necessary pods. This provides a seamless autoscaling experience in Kubernetes environments without requiring manual intervention. The Kubernetes backend will automatically be used by Planner when your pipeline is deployed with `dynamo deployment create`. By default, the planner will run in no-op mode, which means it will monitor metrics but not take scaling actions. To enable actual scaling, you should also specify `--Planner.no-operation=false`.