1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
use std::{
path::{Path, PathBuf},
sync::Arc,
};
use console::{Term, style};
use eyre::{Result, bail, eyre};
use itertools::Itertools;
use jiff::Timestamp;
use path_absolutize::Absolutize;
use crate::cli::args::{BackendArg, ToolArg};
use crate::config::config_file::ConfigFile;
use crate::config::{Config, ConfigPathOptions, Settings, config_file, resolve_target_config_path};
use crate::duration::parse_into_timestamp;
use crate::file::display_path;
use crate::registry::REGISTRY;
use crate::toolset::{
InstallOptions, ResolveOptions, ToolRequest, ToolSource, ToolVersion, ToolsetBuilder,
};
use crate::ui::ctrlc;
use crate::{config, env, exit, file};
/// Installs a tool and adds the version to mise.toml.
///
/// This will install the tool version if it is not already installed.
/// By default, this will use a `mise.toml` file in the current directory.
/// If multiple config files exist (e.g., both `mise.toml` and `mise.local.toml`),
/// the lowest precedence file (`mise.toml`) will be used.
/// See https://mise.jdx.dev/configuration.html#target-file-for-write-operations
///
/// In the following order:
/// - If `--global` is set, it will use the global config file.
/// - If `--path` is set, it will use the config file at the given path.
/// - If `--env` is set, it will use `mise.<env>.toml`.
/// - If `MISE_DEFAULT_CONFIG_FILENAME` is set, it will use that instead.
/// - If `MISE_OVERRIDE_CONFIG_FILENAMES` is set, it will the first from that list.
/// - Otherwise just "mise.toml" or global config if cwd is home directory.
///
/// Use the `--global` flag to use the global config file instead.
#[derive(Debug, clap::Args)]
#[clap(verbatim_doc_comment, visible_alias = "u", after_long_help = AFTER_LONG_HELP)]
pub struct Use {
/// Tool(s) to add to config file
///
/// e.g.: node@20, cargo:ripgrep@latest npm:prettier@3
/// If no version is specified, it will default to @latest
///
/// Tool options can be set with this syntax:
///
/// mise use ubi:BurntSushi/ripgrep[exe=rg]
#[clap(value_name = "TOOL@VERSION", verbatim_doc_comment)]
tool: Vec<ToolArg>,
/// Create/modify an environment-specific config file like .mise.<env>.toml
#[clap(long, short, overrides_with_all = & ["global", "path"])]
env: Option<String>,
/// Force reinstall even if already installed
#[clap(long, short, requires = "tool")]
force: bool,
/// Use the global config file (`~/.config/mise/config.toml`) instead of the local one
#[clap(short, long, overrides_with_all = & ["path", "env"])]
global: bool,
/// Number of jobs to run in parallel
/// [default: 4]
#[clap(long, short, env = "MISE_JOBS", verbatim_doc_comment)]
jobs: Option<usize>,
/// Perform a dry run, showing what would be installed and modified without making changes
#[clap(long, short = 'n', verbatim_doc_comment)]
dry_run: bool,
/// Specify a path to a config file or directory
///
/// If a directory is specified, it will look for a config file in that directory following
/// the rules above.
#[clap(short, long, overrides_with_all = & ["global", "env"], value_hint = clap::ValueHint::FilePath)]
path: Option<PathBuf>,
/// Only install versions released before this date
///
/// Supports absolute dates like "2024-06-01" and relative durations like "90d" or "1y".
#[clap(long, verbatim_doc_comment)]
before: Option<String>,
/// Like --dry-run but exits with code 1 if there are changes to make
///
/// This is useful for scripts to check if tools need to be added or removed.
#[clap(long, verbatim_doc_comment)]
dry_run_code: bool,
/// Save fuzzy version to config file
///
/// e.g.: `mise use --fuzzy node@20` will save 20 as the version
/// this is the default behavior unless `MISE_PIN=1`
#[clap(long, verbatim_doc_comment, overrides_with = "pin")]
fuzzy: bool,
/// Save exact version to config file
/// e.g.: `mise use --pin node@20` will save 20.0.0 as the version
/// Set `MISE_PIN=1` to make this the default behavior
///
/// Consider using mise.lock as a better alternative to pinning in mise.toml:
/// https://mise.jdx.dev/configuration/settings.html#lockfile
#[clap(long, verbatim_doc_comment, overrides_with = "fuzzy")]
pin: bool,
/// Directly pipe stdin/stdout/stderr from plugin to user
/// Sets `--jobs=1`
#[clap(long, overrides_with = "jobs")]
raw: bool,
/// Remove the plugin(s) from config file
#[clap(long, value_name = "PLUGIN", aliases = ["rm", "unset"])]
remove: Vec<BackendArg>,
}
impl Use {
fn is_dry_run(&self) -> bool {
self.dry_run || self.dry_run_code
}
pub async fn run(mut self) -> Result<()> {
if self.tool.is_empty() && self.remove.is_empty() {
self.tool = vec![self.tool_selector()?];
}
env::TOOL_ARGS.write().unwrap().clone_from(&self.tool);
let mut config = Config::get().await?;
let mut ts = ToolsetBuilder::new()
.with_global_only(self.global)
.build(&config)
.await?;
let cf = self.get_config_file().await?;
let mut resolve_options = ResolveOptions {
latest_versions: false,
use_locked_version: true,
before_date: self.get_before_date()?,
};
let versions: Vec<_> = self
.tool
.iter()
.cloned()
.map(|t| match t.tvr {
Some(tvr) => {
if tvr.version() == "latest" && !Settings::get().locked {
// user specified `@latest` so we should resolve the latest version
// TODO: this should only happen on this tool, not all of them
resolve_options.latest_versions = true;
resolve_options.use_locked_version = false;
}
Ok(tvr)
}
None => ToolRequest::new(
t.ba,
"latest",
ToolSource::MiseToml(cf.get_path().to_path_buf()),
),
})
.collect::<Result<_>>()?;
let mut versions = ts
.install_all_versions(
&mut config,
versions.clone(),
&InstallOptions {
reason: "use".to_string(),
force: self.force,
jobs: self.jobs,
raw: self.raw,
dry_run: self.is_dry_run(),
resolve_options,
..Default::default()
},
)
.await?;
let pin = self.pin || !self.fuzzy && (Settings::get().pin || Settings::get().asdf_compat);
for (ba, tvl) in &versions.iter().chunk_by(|tv| tv.ba()) {
let versions: Vec<_> = tvl
.into_iter()
.map(|tv| {
let mut request = tv.request.clone();
if pin
&& let ToolRequest::Version {
version: _version,
source,
options,
backend,
} = request
{
request = ToolRequest::Version {
version: tv.version.clone(),
source,
options,
backend,
};
}
request
})
.collect();
cf.replace_versions(ba, versions)?;
}
if self.global {
self.warn_if_hidden(&config, cf.get_path()).await;
}
for plugin_name in &self.remove {
cf.remove_tool(plugin_name)?;
}
if !self.is_dry_run() {
cf.save()?;
for tv in &mut versions {
// update the source so the lockfile is updated correctly
tv.request.set_source(cf.source());
}
let config = Config::reset().await?;
let ts = config.get_toolset().await?;
config::rebuild_shims_and_runtime_symlinks(&config, ts, &versions).await?;
}
self.render_success_message(cf.as_ref(), &versions, &self.remove)?;
Ok(())
}
async fn get_config_file(&self) -> Result<Arc<dyn ConfigFile>> {
let cwd = env::current_dir()?;
// Handle special case for --path that needs absolutize logic for compatibility
let path = if let Some(p) = &self.path {
let from_dir = config::config_file_from_dir(p).absolutize()?.to_path_buf();
if from_dir.starts_with(&cwd) {
from_dir
} else {
p.clone()
}
} else {
let opts = ConfigPathOptions {
global: self.global,
path: None, // handled above
env: self.env.clone(),
cwd: Some(cwd),
prefer_toml: false, // mise use supports .tool-versions and other formats
prevent_home_local: true, // When in HOME, use global config
};
resolve_target_config_path(opts)?
};
config_file::parse_or_init(&path).await
}
async fn warn_if_hidden(&self, config: &Arc<Config>, global: &Path) {
let ts = ToolsetBuilder::new()
.build(config)
.await
.unwrap_or_default();
let warn = |targ: &ToolArg, p| {
let plugin = &targ.ba;
let p = display_path(p);
let global = display_path(global);
warn!("{plugin} is defined in {p} which overrides the global config ({global})");
};
for targ in &self.tool {
if let Some(tv) = ts.versions.get(targ.ba.as_ref())
&& let ToolSource::MiseToml(p) | ToolSource::ToolVersions(p) = &tv.source
&& !file::same_file(p, global)
{
warn(targ, p);
}
}
}
fn render_success_message(
&self,
cf: &dyn ConfigFile,
versions: &[ToolVersion],
remove: &[BackendArg],
) -> Result<()> {
let path = display_path(cf.get_path());
if self.is_dry_run() {
let mut messages = vec![];
if !versions.is_empty() {
let tools = versions.iter().map(|t| t.style()).join(", ");
messages.push(format!("add: {tools}"));
}
if !remove.is_empty() {
let tools_to_remove = remove.iter().map(|r| r.to_string()).join(", ");
messages.push(format!("remove: {tools_to_remove}"));
}
if !messages.is_empty() {
miseprintln!(
"{} would update {} ({})",
style("mise").green(),
style(&path).cyan().for_stderr(),
messages.join(", ")
);
if self.dry_run_code {
exit::exit(1);
}
}
} else {
if !versions.is_empty() {
let tools = versions.iter().map(|t| t.style()).join(", ");
miseprintln!(
"{} {} tools: {tools}",
style("mise").green(),
style(&path).cyan().for_stderr(),
);
}
if !remove.is_empty() {
let tools_to_remove = remove.iter().map(|r| r.to_string()).join(", ");
miseprintln!(
"{} {} removed: {tools_to_remove}",
style("mise").green(),
style(&path).cyan().for_stderr(),
);
}
}
Ok(())
}
fn tool_selector(&self) -> Result<ToolArg> {
if !console::user_attended_stderr() {
bail!("No tool specified and not running interactively");
}
let theme = crate::ui::theme::get_theme();
let mut s = demand::Select::new("Tools")
.description("Select a tool to install")
.filtering(true)
.filterable(true)
.theme(&theme);
for rt in REGISTRY.values().unique_by(|r| r.short) {
if let Some(backend) = rt.backends().first() {
// TODO: populate registry with descriptions from aqua and other sources
// TODO: use the backend from the lockfile if available
let description = rt.description.unwrap_or(backend);
s = s.option(demand::DemandOption::new(rt).description(description));
}
}
ctrlc::show_cursor_after_ctrl_c();
match s.run() {
Ok(rt) => rt.short.parse(),
Err(err) => {
Term::stderr().show_cursor()?;
Err(eyre!(err))
}
}
}
/// Get the before_date from CLI flag or settings
fn get_before_date(&self) -> Result<Option<Timestamp>> {
if let Some(before) = &self.before {
return Ok(Some(parse_into_timestamp(before)?));
}
if let Some(before) = &Settings::get().install_before {
return Ok(Some(parse_into_timestamp(before)?));
}
Ok(None)
}
}
static AFTER_LONG_HELP: &str = color_print::cstr!(
r#"<bold><underline>Examples:</underline></bold>
# run with no arguments to use the interactive selector
$ <bold>mise use</bold>
# set the current version of node to 20.x in mise.toml of current directory
# will write the fuzzy version (e.g.: 20)
$ <bold>mise use node@20</bold>
# set the current version of node to 20.x in ~/.config/mise/config.toml
# will write the precise version (e.g.: 20.0.0)
$ <bold>mise use -g --pin node@20</bold>
# sets .mise.local.toml (which is intended not to be committed to a project)
$ <bold>mise use --env local node@20</bold>
# sets .mise.staging.toml (which is used if MISE_ENV=staging)
$ <bold>mise use --env staging node@20</bold>
"#
);