flags.rs 10.6 KB
Newer Older
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
// SPDX-FileCopyrightText: Copyright (c) 2024-2025 NVIDIA CORPORATION & AFFILIATES. All rights reserved.
// SPDX-License-Identifier: Apache-2.0
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.

16
use std::collections::HashMap;
17
18
use std::path::PathBuf;

19
use clap::ValueEnum;
20
use dynamo_llm::entrypoint::RouterConfig;
21
use dynamo_llm::entrypoint::input::Input;
22
use dynamo_llm::kv_router::KvRouterConfig;
23
use dynamo_llm::local_model::LocalModel;
24
use dynamo_llm::mocker::protocols::MockEngineArgs;
25
use dynamo_runtime::pipeline::RouterMode as RuntimeRouterMode;
26

27
28
use crate::Output;

29
30
31
32
/// Required options depend on the in and out choices
#[derive(clap::Parser, Debug, Clone)]
#[command(version, about, long_about = None)]
pub struct Flags {
33
34
35
36
37
38
39
    /// The model. The options depend on the engine.
    ///
    /// The full list - only mistralrs supports all three currently:
    /// - Full path to a GGUF file
    /// - Full path of a checked out Hugging Face repository containing safetensor files
    /// - Name of a Hugging Face repository, e.g 'google/flan-t5-small'. The model will be
    ///   downloaded and cached.
40
41
42
    #[arg(index = 1)]
    pub model_path_pos: Option<PathBuf>,

43
    // `--model-path`. The one above is `dynamo-run <positional-model-path>`
44
45
46
47
    #[arg(long = "model-path")]
    pub model_path_flag: Option<PathBuf>,

    /// HTTP port. `in=http` only
Graham King's avatar
Graham King committed
48
    /// If tls_cert_path and tls_key_path are provided, this will be TLS/HTTPS.
49
50
51
    #[arg(long, default_value = "8080")]
    pub http_port: u16,

Graham King's avatar
Graham King committed
52
53
54
55
56
57
58
59
    /// TLS certificate file
    #[arg(long, requires = "tls_key_path")]
    pub tls_cert_path: Option<PathBuf>,

    /// TLS certificate key file
    #[arg(long, requires = "tls_cert_path")]
    pub tls_key_path: Option<PathBuf>,

60
61
62
63
    /// The name of the model we are serving
    #[arg(long)]
    pub model_name: Option<String>,

64
65
66
67
    /// Verbose output (-v for debug, -vv for trace)
    #[arg(short = 'v', action = clap::ArgAction::Count, default_value_t = 0)]
    pub verbosity: u8,

68
69
70
71
72
73
74
75
76
    /// llamacpp only
    ///
    /// The path to the tokenizer and model config because:
    /// - llama_cpp only runs GGUF files
    /// - our engine is a 'core' engine in that we do the tokenization, so we need the vocab
    /// - TODO: we don't yet extract that from the GGUF. Once we do we can remove this flag.
    #[arg(long)]
    pub model_config: Option<PathBuf>,

77
    /// If using `out=dyn` with multiple instances, this says how to route the requests.
78
79
    ///
    /// Mostly interesting for KV-aware routing.
80
81
    /// Defaults to RouterMode::RoundRobin
    #[arg(long, default_value = "round-robin")]
82
83
    pub router_mode: RouterMode,

84
85
86
87
88
89
    /// Maximum number of batched tokens for KV routing
    /// Needed for informing the KV router
    /// NOTE: this is not actually used for now
    #[arg(long, default_value = "8192")]
    pub max_num_batched_tokens: Option<u32>,

90
    /// KV Router: Weight for overlap score in worker selection.
91
    /// Higher values prioritize KV cache reuse. Default: 1.0
92
93
94
    #[arg(long)]
    pub kv_overlap_score_weight: Option<f64>,

95
96
    /// KV Router: Temperature for worker sampling via softmax.
    /// Higher values promote more randomness, and 0 fallbacks to deterministic.
97
    /// Default: 0.0
98
    #[arg(long)]
99
    pub router_temperature: Option<f64>,
100

101
102
103
104
105
106
107
    /// KV Router: Whether to use KV events to maintain the view of cached blocks
    /// If false, would use ApproxKvRouter for predicting block creation / deletion
    /// based only on incoming requests at a timer.
    /// Default: true
    #[arg(long)]
    pub use_kv_events: Option<bool>,

108
109
110
111
112
113
    /// KV Router: Whether to enable replica synchronization across multiple router instances.
    /// When true, routers will publish and subscribe to events to maintain consistent state.
    /// Default: false
    #[arg(long)]
    pub router_replica_sync: Option<bool>,

114
115
116
117
    /// Max model context length. Reduce this if you don't have enough VRAM for the full model
    /// context length (e.g. Llama 4).
    /// Defaults to the model's max, which is usually model_max_length in tokenizer_config.json.
    #[arg(long)]
118
    pub context_length: Option<u32>,
119

120
    /// KV cache block size (is this used? Maybe by Python vllm worker?)
121
    #[arg(long)]
122
    pub kv_cache_block_size: Option<u32>,
123

124
    /// Mocker engine only.
125
126
127
128
129
    /// Additional engine-specific arguments from a JSON file.
    /// Contains a mapping of parameter names to values.
    #[arg(long)]
    pub extra_engine_args: Option<PathBuf>,

130
131
132
133
134
135
136
137
138
139
140
    /// Path to a JSON file containing default request fields.
    /// These fields will be merged with each request, but can be overridden by the request.
    /// Example file contents:
    /// {
    ///     "model": "Qwen2.5-3B-Instruct",
    ///     "temperature": 0.7,
    ///     "max_completion_tokens": 4096
    /// }
    #[arg(long)]
    pub request_template: Option<PathBuf>,

141
142
143
144
145
    /// How many times a request can be migrated to another worker if the HTTP server lost
    /// connection to the current worker.
    #[arg(long, value_parser = clap::value_parser!(u32).range(0..1024))]
    pub migration_limit: Option<u32>,

146
147
148
149
150
151
    /// Make this a static worker.
    /// Do not connect to or advertise self on etcd.
    /// in=dyn://x.y.z only
    #[arg(long, default_value = "false")]
    pub static_worker: bool,

152
153
154
155
156
157
158
    /// Everything after a `--`.
    /// These are the command line arguments to the python engine when using `pystr` or `pytok`.
    #[arg(index = 2, last = true, hide = true, allow_hyphen_values = true)]
    pub last: Vec<String>,
}

impl Flags {
159
160
    /// For each Output variant, check if it would be able to run.
    /// This takes validation out of the main engine creation path.
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
    pub fn validate(
        &self,
        local_model: &LocalModel,
        in_opt: &Input,
        out_opt: &Output,
    ) -> anyhow::Result<()> {
        match in_opt {
            Input::Endpoint(_) => {}
            _ => {
                if self.static_worker {
                    anyhow::bail!("'--static-worker true' only applies to in=dyn://x.y.z");
                }
            }
        }

176
        match out_opt {
177
            Output::Auto => {
178
                if self.context_length.is_some() {
179
180
181
                    anyhow::bail!(
                        "'--context-length' flag should only be used on the worker node, not on the ingress"
                    );
182
183
                }
                if self.kv_cache_block_size.is_some() {
184
185
186
                    anyhow::bail!(
                        "'--kv-cache-block-size' flag should only be used on the worker node, not on the ingress"
                    );
187
                }
188
                if self.migration_limit.is_some() {
189
190
191
                    anyhow::bail!(
                        "'--migration-limit' flag should only be used on the worker node, not on the ingress"
                    );
192
                }
193
            }
194
195
196
197
198
199
200
201
202
203
204
205
206
            Output::Static(_) => {
                if self.model_name.is_none()
                    || self
                        .model_path_pos
                        .as_ref()
                        .or(self.model_path_flag.as_ref())
                        .is_none()
                {
                    anyhow::bail!(
                        "out=dyn://<path> requires --model-name and --model-path, which are the name and path on disk of the model we expect to serve."
                    );
                }
            }
207
208
209
210
211
212
213
214
215
216
217
218
219
            Output::EchoFull => {}
            Output::EchoCore => {
                if !local_model.card().has_tokenizer() {
                    anyhow::bail!(
                        "out=echo_core need to find the tokenizer. Pass flag --model-path <path>"
                    );
                };
            }
            #[cfg(feature = "mistralrs")]
            Output::MistralRs => {}
            #[cfg(feature = "llamacpp")]
            Output::LlamaCpp => {
                if !local_model.path().is_file() {
220
221
222
                    anyhow::bail!(
                        "--model-path should refer to a GGUF file. llama_cpp does not support safetensors."
                    );
223
224
                }
            }
225
226
227
            Output::Mocker => {
                // nothing to check here
            }
228
        }
229
230
231
232
233
234
235
236
237
238

        match out_opt {
            Output::Mocker => {}
            _ => {
                if self.extra_engine_args.is_some() {
                    anyhow::bail!("`--extra-engine-args` is only for the mocker engine");
                }
            }
        }

239
        Ok(())
240
241
    }

242
243
244
245
246
    pub fn router_config(&self) -> RouterConfig {
        RouterConfig::new(
            self.router_mode.into(),
            KvRouterConfig::new(
                self.kv_overlap_score_weight,
247
                self.router_temperature,
248
                self.use_kv_events,
249
                self.router_replica_sync,
250
                self.max_num_batched_tokens,
251
252
            ),
        )
253
    }
254
255
256
257
258
259
260
261
262
263
264
265
266
267

    /// Load extra engine arguments from a JSON file
    /// Returns a HashMap of parameter names to values
    pub fn load_extra_engine_args(
        &self,
    ) -> anyhow::Result<Option<HashMap<String, serde_json::Value>>> {
        if let Some(path) = &self.extra_engine_args {
            let file_content = std::fs::read_to_string(path)?;
            let args: HashMap<String, serde_json::Value> = serde_json::from_str(&file_content)?;
            Ok(Some(args))
        } else {
            Ok(None)
        }
    }
268
269
270
271
272
273
274
275
276

    pub fn mocker_config(&self) -> MockEngineArgs {
        let Some(path) = &self.extra_engine_args else {
            tracing::warn!("Did not specify extra engine args. Using default mocker args.");
            return MockEngineArgs::default();
        };
        MockEngineArgs::from_json_file(path)
            .unwrap_or_else(|e| panic!("Failed to build mocker engine args from {path:?}: {e}"))
    }
277
278
}

279
#[derive(Default, PartialEq, Eq, ValueEnum, Clone, Debug, Copy)]
280
281
282
283
pub enum RouterMode {
    #[default]
    #[value(name = "round-robin")]
    RoundRobin,
284
    Random,
285
286
287
288
    #[value(name = "kv")]
    KV,
}

289
290
291
292
293
294
impl From<RouterMode> for RuntimeRouterMode {
    fn from(r: RouterMode) -> RuntimeRouterMode {
        match r {
            RouterMode::RoundRobin => RuntimeRouterMode::RoundRobin,
            RouterMode::Random => RuntimeRouterMode::Random,
            RouterMode::KV => RuntimeRouterMode::KV,
295
296
297
        }
    }
}