- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我有一个采用不同选项的命令,这些选项的相对顺序对于命令的语义很重要。例如,在 command --config A --some-option --config-file B --random-option --config C --another-option --more-options --config-file D
,A、B、C、D
的相对顺序很重要,因为它会影响命令的含义。
如果我只定义选项如下:
#[derive(Debug, StructOpt)]
pub struct Command {
#[structopt(long = "config")]
configs: Vec<String>,
#[structopt(long = "config-file")]
config_files: Vec<String>,
}
然后我会得到两个向量,configs = [A, C]
和 config_files = [B, D]
但 中元素之间的相对顺序configs
和 config_files
已丢失。
我们的想法是提供一个自定义的解析函数,并在每个选项被解析时使用一个计数器来记录索引。不幸的是,解析函数没有按照命令定义的原始顺序调用。
fn get_next_atomic_int() -> usize {
static ATOMIC_COUNTER: Lazy<AtomicUsize> = Lazy::new(|| AtomicUsize::new(0));
ATOMIC_COUNTER.fetch_add(1, Ordering::Relaxed)
}
fn parse_passthrough_string_ordered(arg: &str) -> (String, usize) {
(arg.to_owned(), get_next_atomic_int())
}
#[derive(Debug, StructOpt)]
#[structopt(name = "command"]
pub struct Command {
#[structopt(long = "config-file", parse(from_str = parse_passthrough_string_ordered))]
config_files: Vec<(String, usize)>,
#[structopt(short = "c", long = "config", parse(from_str = parse_passthrough_string_ordered))]
configs: Vec<(String, usize)>,
}
我可以为选项添加别名,如下所示:
#[derive(Debug, StructOpt)]
pub struct Command {
#[structopt(long = "config", visible_alias = "config-file")]
configs: Vec<String>,
}
这种方法有两个问题:
--config
还是 --config-file
传递的(并不总是能够弄清楚值是如何传递的)仅通过检查值即可通过)。另一个想法是附加多个 structopt
指令,这样两个选项将使用相同的底层向量。不幸的是,它不起作用 - structopt
只使用最后一个指令。像这样的东西:
#[derive(Debug)]
enum Config {
File(String),
Literal(String),
}
fn parse_config_literal(arg: &str) -> Config {
Config::Literal(arg.to_owned())
}
fn parse_config_file(arg: &str) -> Config {
Config::File(arg.to_owned())
}
#[derive(Debug, StructOpt)]
#[structopt(name = "example")]
struct Opt {
#[structopt(long = "--config-file", parse(from_str = parse_config_file))]
#[structopt(short = "-c", long = "--config", parse(from_str = parse_config_literal))]
options: Vec<Config>,
}
我可以尝试通过搜索已解析的值来恢复原始顺序。但这意味着我将不得不重复相当多的解析逻辑(例如,需要支持传递 --config=X
、--config X
,需要处理 X
显示为另一个选项的输入,等等)。
我宁愿有一种方法来可靠地获取原始数据,而不是丢失订单并尝试以可能脆弱的方式恢复它。
最佳答案
如 @TeXitoi 所述,我错过了 ArgMatches::indices_of()
函数,它为我们提供了所需的信息。
use structopt::StructOpt;
#[derive(Debug)]
enum Config {
File(String),
Literal(String),
}
fn parse_config_literal(arg: &str) -> Config {
Config::Literal(arg.to_owned())
}
fn parse_config_file(arg: &str) -> Config {
Config::File(arg.to_owned())
}
#[derive(Debug, StructOpt)]
#[structopt(name = "example")]
struct Opt {
#[structopt(short = "c", long = "config", parse(from_str = parse_config_literal))]
configs: Vec<Config>,
#[structopt(long = "config-file", parse(from_str = parse_config_file))]
config_files: Vec<Config>,
}
fn with_indices<'a, I: IntoIterator + 'a>(
collection: I,
name: &str,
matches: &'a structopt::clap::ArgMatches,
) -> impl Iterator<Item = (usize, I::Item)> + 'a {
matches
.indices_of(name)
.into_iter()
.flatten()
.zip(collection)
}
fn main() {
let args = vec!["example", "--config", "A", "--config-file", "B", "--config", "C", "--config-file", "D"];
let clap = Opt::clap();
let matches = clap.get_matches_from(args);
let opt = Opt::from_clap(&matches);
println!("configs:");
for (i, c) in with_indices(&opt.configs, "configs", &matches) {
println!("{}: {:#?}", i, c);
}
println!("\nconfig-files:");
for (i, c) in with_indices(&opt.config_files, "config-files", &matches) {
println!("{}: {:#?}", i, c);
}
}
关于rust - 使用 clap 和 structopt 获取不同命令行选项的相对顺序,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/67290849/
有没有办法让我们在 clap 的帮助消息中换行? 我尝试了多行注释,还尝试在混合中插入 \n。但两者都不起作用。 #[derive(Parser, Debug)] #[clap(author, ver
我想使用 clap derive API 来解析 Ipv4Addr。 #![allow(unused)] use clap; // 3.1.6 use clap::Parser; use std::n
我正在使用 Clap,我的 YAML 文件包含以下内容: args: - DIRECTORY help: one or more directories required: true
我正在构建一个可以调用其他底层程序的 CLI,这些程序有自己的选项和参数。我希望能够通过 CLI 将这些选项传递给程序。 $ cli --program [PROGRAM] --programOpts
我正在使用 clap,但在尝试解析参数时出现意外行为。 我的命令行工具应该像这样工作 foo -u 例如: foo -u jack echo s foo -u paul ls -al 我需要获取诸
这是一个例子: extern crate clap; use clap::{Arg, App}; fn main() { let args = App::new("test") .arg(Ar
即使阅读了关于引用所有权和借用的章节,我仍然无法理解以下代码中的某些内容,这实际上阻止了我从 clap::App 调用多个方法! extern crate clap; use clap::App; f
假设我有以下命令行: ./app foo bar baz 我想从中取出这个数组: ["foo", "bar", "baz"] 考虑到位置参数可以任意计数,有没有办法在 clap 中做到这一点? 最佳答
作为我的 CLI 工具的一部分,我有一个 clap::Arg取多个值,表示 (x, y)协调。我希望使用能够将值作为 -p/--position 1 0 传递 .arg( clap::Arg:
我正在使用 Clap我正在努力使子命令可以为参数取多个值。我追求的界面是: just use repo [files] 一个例子: just use radar/dot-files gitaliase
我正在使用 the Clap crate用于解析命令行参数。我定义了一个子命令 ls那应该列出文件。 Clap 还定义了一个 help显示有关应用程序及其使用情况的信息的子命令。 如果没有提供命令,则
我目前正在研究 Rust port证券tool .根据 Rust 的指南,我想将核心库隔离到它自己的箱子中,这样我们就可以创建与核心库交互的各种工具(CLI、API、流等),而无需将它们耦合在一起。
关闭。这个问题是opinion-based 。目前不接受答案。 想要改进这个问题吗?更新问题,以便 editing this post 可以用事实和引文来回答它。 . 去年关闭。 Improve th
我有一个使用 Clap 的程序 foo处理命令参数解析。 foo 调用另一个程序 bar。最近,我决定 foo 的用户如果愿意,应该能够将参数传递给 bar。我向 Clap 添加了 bar 命令: l
我正在尝试在 Clap 和 YAML 的帮助下用 Rust 编写 CLI。我的输入将需要一个参数(文件路径)和标志之一 -s、-r 或 -g。标志 -s 和 -r 将需要两个标志之一 -t 和 -m,
问题 我有一个采用不同选项的命令,这些选项的相对顺序对于命令的语义很重要。例如,在 command --config A --some-option --config-file B --random-
我遇到了与 Is there any straightforward way for Clap to display help when no command is provided? 相同的问题,但
我正在学习 Rust 并尝试找到类似实用程序的工具(是的,另一个),我正在使用 clap 并尝试支持程序参数的命令行和配置文件(这与 clap yml 文件无关) . 我正在尝试解析命令,如果没有命令
我有一个鼓掌App像这样: let m = App::new("test") .arg( Arg::with_name("INPUT") .help("
clap crate 为 -h 选项实现了内置行为,但它似乎没有为 -? 做同样的事情。有没有办法告诉它这样做? 最佳答案 我在 clap repository 上开了一个问题.作者/主要贡献者已在那
我是一名优秀的程序员,十分优秀!