rustfmt'd the codebase
This commit is contained in:
@@ -1,44 +1,52 @@
|
||||
use std::collections::VecDeque;
|
||||
|
||||
use crate::{
|
||||
getopt::{Opt, OptSpec, get_opts_from_tokens}, libsh::error::{ShErr, ShErrKind, ShResult, ShResultExt}, parse::{NdRule, Node, execute::prepare_argv}, prelude::*, procio::borrow_fd, state::{self, VarFlags, VarKind, write_vars}
|
||||
getopt::{Opt, OptSpec, get_opts_from_tokens},
|
||||
libsh::error::{ShErr, ShErrKind, ShResult, ShResultExt},
|
||||
parse::{NdRule, Node, execute::prepare_argv},
|
||||
prelude::*,
|
||||
procio::borrow_fd,
|
||||
state::{self, VarFlags, VarKind, write_vars},
|
||||
};
|
||||
|
||||
fn arr_op_optspec() -> Vec<OptSpec> {
|
||||
vec![
|
||||
OptSpec {
|
||||
opt: Opt::Short('c'),
|
||||
takes_arg: true
|
||||
},
|
||||
OptSpec {
|
||||
opt: Opt::Short('r'),
|
||||
takes_arg: false
|
||||
},
|
||||
OptSpec {
|
||||
opt: Opt::Short('v'),
|
||||
takes_arg: true
|
||||
}
|
||||
]
|
||||
vec![
|
||||
OptSpec {
|
||||
opt: Opt::Short('c'),
|
||||
takes_arg: true,
|
||||
},
|
||||
OptSpec {
|
||||
opt: Opt::Short('r'),
|
||||
takes_arg: false,
|
||||
},
|
||||
OptSpec {
|
||||
opt: Opt::Short('v'),
|
||||
takes_arg: true,
|
||||
},
|
||||
]
|
||||
}
|
||||
|
||||
pub struct ArrOpOpts {
|
||||
count: usize,
|
||||
reverse: bool,
|
||||
var: Option<String>,
|
||||
count: usize,
|
||||
reverse: bool,
|
||||
var: Option<String>,
|
||||
}
|
||||
|
||||
impl Default for ArrOpOpts {
|
||||
fn default() -> Self {
|
||||
Self {
|
||||
count: 1,
|
||||
reverse: false,
|
||||
var: None,
|
||||
}
|
||||
}
|
||||
fn default() -> Self {
|
||||
Self {
|
||||
count: 1,
|
||||
reverse: false,
|
||||
var: None,
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
#[derive(Clone, Copy)]
|
||||
enum End { Front, Back }
|
||||
enum End {
|
||||
Front,
|
||||
Back,
|
||||
}
|
||||
|
||||
fn arr_pop_inner(node: Node, end: End) -> ShResult<()> {
|
||||
let NdRule::Command {
|
||||
@@ -49,40 +57,42 @@ fn arr_pop_inner(node: Node, end: End) -> ShResult<()> {
|
||||
unreachable!()
|
||||
};
|
||||
|
||||
let (argv, opts) = get_opts_from_tokens(argv, &arr_op_optspec())?;
|
||||
let arr_op_opts = get_arr_op_opts(opts)?;
|
||||
let (argv, opts) = get_opts_from_tokens(argv, &arr_op_optspec())?;
|
||||
let arr_op_opts = get_arr_op_opts(opts)?;
|
||||
let mut argv = prepare_argv(argv)?;
|
||||
if !argv.is_empty() { argv.remove(0); }
|
||||
if !argv.is_empty() {
|
||||
argv.remove(0);
|
||||
}
|
||||
let stdout = borrow_fd(STDOUT_FILENO);
|
||||
let mut status = 0;
|
||||
let mut status = 0;
|
||||
|
||||
for (arg,_) in argv {
|
||||
for _ in 0..arr_op_opts.count {
|
||||
let pop = |arr: &mut std::collections::VecDeque<String>| match end {
|
||||
End::Front => arr.pop_front(),
|
||||
End::Back => arr.pop_back(),
|
||||
};
|
||||
let Some(popped) = write_vars(|v| v.get_arr_mut(&arg).ok().and_then(pop)) else {
|
||||
status = 1;
|
||||
break;
|
||||
};
|
||||
status = 0;
|
||||
for (arg, _) in argv {
|
||||
for _ in 0..arr_op_opts.count {
|
||||
let pop = |arr: &mut std::collections::VecDeque<String>| match end {
|
||||
End::Front => arr.pop_front(),
|
||||
End::Back => arr.pop_back(),
|
||||
};
|
||||
let Some(popped) = write_vars(|v| v.get_arr_mut(&arg).ok().and_then(pop)) else {
|
||||
status = 1;
|
||||
break;
|
||||
};
|
||||
status = 0;
|
||||
|
||||
if let Some(ref var) = arr_op_opts.var {
|
||||
write_vars(|v| v.set_var(var, VarKind::Str(popped), VarFlags::NONE))?;
|
||||
} else {
|
||||
write(stdout, popped.as_bytes())?;
|
||||
write(stdout, b"\n")?;
|
||||
}
|
||||
}
|
||||
}
|
||||
if let Some(ref var) = arr_op_opts.var {
|
||||
write_vars(|v| v.set_var(var, VarKind::Str(popped), VarFlags::NONE))?;
|
||||
} else {
|
||||
write(stdout, popped.as_bytes())?;
|
||||
write(stdout, b"\n")?;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
state::set_status(status);
|
||||
Ok(())
|
||||
}
|
||||
|
||||
fn arr_push_inner(node: Node, end: End) -> ShResult<()> {
|
||||
let blame = node.get_span().clone();
|
||||
let blame = node.get_span().clone();
|
||||
let NdRule::Command {
|
||||
assignments: _,
|
||||
argv,
|
||||
@@ -91,49 +101,60 @@ fn arr_push_inner(node: Node, end: End) -> ShResult<()> {
|
||||
unreachable!()
|
||||
};
|
||||
|
||||
let (argv, opts) = get_opts_from_tokens(argv, &arr_op_optspec())?;
|
||||
let _arr_op_opts = get_arr_op_opts(opts)?;
|
||||
let (argv, opts) = get_opts_from_tokens(argv, &arr_op_optspec())?;
|
||||
let _arr_op_opts = get_arr_op_opts(opts)?;
|
||||
let mut argv = prepare_argv(argv)?;
|
||||
if !argv.is_empty() { argv.remove(0); }
|
||||
if !argv.is_empty() {
|
||||
argv.remove(0);
|
||||
}
|
||||
|
||||
let mut argv = argv.into_iter();
|
||||
let Some((name, _)) = argv.next() else {
|
||||
return Err(ShErr::at(ShErrKind::ExecFail, blame, "push: missing array name".to_string()));
|
||||
};
|
||||
let mut argv = argv.into_iter();
|
||||
let Some((name, _)) = argv.next() else {
|
||||
return Err(ShErr::at(
|
||||
ShErrKind::ExecFail,
|
||||
blame,
|
||||
"push: missing array name".to_string(),
|
||||
));
|
||||
};
|
||||
|
||||
for (val, span) in argv {
|
||||
let push_val = val.clone();
|
||||
write_vars(|v| {
|
||||
if let Ok(arr) = v.get_arr_mut(&name) {
|
||||
match end {
|
||||
End::Front => arr.push_front(push_val),
|
||||
End::Back => arr.push_back(push_val),
|
||||
}
|
||||
Ok(())
|
||||
} else {
|
||||
v.set_var(&name, VarKind::Arr(VecDeque::from([push_val])), VarFlags::NONE)
|
||||
}
|
||||
}).blame(span)?;
|
||||
}
|
||||
for (val, span) in argv {
|
||||
let push_val = val.clone();
|
||||
write_vars(|v| {
|
||||
if let Ok(arr) = v.get_arr_mut(&name) {
|
||||
match end {
|
||||
End::Front => arr.push_front(push_val),
|
||||
End::Back => arr.push_back(push_val),
|
||||
}
|
||||
Ok(())
|
||||
} else {
|
||||
v.set_var(
|
||||
&name,
|
||||
VarKind::Arr(VecDeque::from([push_val])),
|
||||
VarFlags::NONE,
|
||||
)
|
||||
}
|
||||
})
|
||||
.blame(span)?;
|
||||
}
|
||||
|
||||
state::set_status(0);
|
||||
Ok(())
|
||||
}
|
||||
|
||||
pub fn arr_pop(node: Node) -> ShResult<()> {
|
||||
arr_pop_inner(node, End::Back)
|
||||
arr_pop_inner(node, End::Back)
|
||||
}
|
||||
|
||||
pub fn arr_fpop(node: Node) -> ShResult<()> {
|
||||
arr_pop_inner(node, End::Front)
|
||||
arr_pop_inner(node, End::Front)
|
||||
}
|
||||
|
||||
pub fn arr_push(node: Node) -> ShResult<()> {
|
||||
arr_push_inner(node, End::Back)
|
||||
arr_push_inner(node, End::Back)
|
||||
}
|
||||
|
||||
pub fn arr_fpush(node: Node) -> ShResult<()> {
|
||||
arr_push_inner(node, End::Front)
|
||||
arr_push_inner(node, End::Front)
|
||||
}
|
||||
|
||||
pub fn arr_rotate(node: Node) -> ShResult<()> {
|
||||
@@ -145,52 +166,63 @@ pub fn arr_rotate(node: Node) -> ShResult<()> {
|
||||
unreachable!()
|
||||
};
|
||||
|
||||
let (argv, opts) = get_opts_from_tokens(argv, &arr_op_optspec())?;
|
||||
let arr_op_opts = get_arr_op_opts(opts)?;
|
||||
let (argv, opts) = get_opts_from_tokens(argv, &arr_op_optspec())?;
|
||||
let arr_op_opts = get_arr_op_opts(opts)?;
|
||||
let mut argv = prepare_argv(argv)?;
|
||||
if !argv.is_empty() { argv.remove(0); }
|
||||
if !argv.is_empty() {
|
||||
argv.remove(0);
|
||||
}
|
||||
|
||||
for (arg, _) in argv {
|
||||
write_vars(|v| -> ShResult<()> {
|
||||
let arr = v.get_arr_mut(&arg)?;
|
||||
if arr_op_opts.reverse {
|
||||
arr.rotate_right(arr_op_opts.count.min(arr.len()));
|
||||
} else {
|
||||
arr.rotate_left(arr_op_opts.count.min(arr.len()));
|
||||
}
|
||||
Ok(())
|
||||
})?;
|
||||
}
|
||||
for (arg, _) in argv {
|
||||
write_vars(|v| -> ShResult<()> {
|
||||
let arr = v.get_arr_mut(&arg)?;
|
||||
if arr_op_opts.reverse {
|
||||
arr.rotate_right(arr_op_opts.count.min(arr.len()));
|
||||
} else {
|
||||
arr.rotate_left(arr_op_opts.count.min(arr.len()));
|
||||
}
|
||||
Ok(())
|
||||
})?;
|
||||
}
|
||||
|
||||
state::set_status(0);
|
||||
Ok(())
|
||||
}
|
||||
|
||||
pub fn get_arr_op_opts(opts: Vec<Opt>) -> ShResult<ArrOpOpts> {
|
||||
let mut arr_op_opts = ArrOpOpts::default();
|
||||
for opt in opts {
|
||||
match opt {
|
||||
Opt::ShortWithArg('c', count) => {
|
||||
arr_op_opts.count = count.parse::<usize>().map_err(|_| {
|
||||
ShErr::simple(ShErrKind::ParseErr, format!("invalid count: {}", count))
|
||||
})?;
|
||||
}
|
||||
Opt::Short('c') => {
|
||||
return Err(ShErr::simple(ShErrKind::ParseErr, "missing count for -c".to_string()));
|
||||
}
|
||||
Opt::Short('r') => {
|
||||
arr_op_opts.reverse = true;
|
||||
}
|
||||
Opt::ShortWithArg('v', var) => {
|
||||
arr_op_opts.var = Some(var);
|
||||
}
|
||||
Opt::Short('v') => {
|
||||
return Err(ShErr::simple(ShErrKind::ParseErr, "missing variable name for -v".to_string()));
|
||||
}
|
||||
_ => {
|
||||
return Err(ShErr::simple(ShErrKind::ParseErr, format!("invalid option: {}", opt)));
|
||||
}
|
||||
}
|
||||
}
|
||||
Ok(arr_op_opts)
|
||||
let mut arr_op_opts = ArrOpOpts::default();
|
||||
for opt in opts {
|
||||
match opt {
|
||||
Opt::ShortWithArg('c', count) => {
|
||||
arr_op_opts.count = count
|
||||
.parse::<usize>()
|
||||
.map_err(|_| ShErr::simple(ShErrKind::ParseErr, format!("invalid count: {}", count)))?;
|
||||
}
|
||||
Opt::Short('c') => {
|
||||
return Err(ShErr::simple(
|
||||
ShErrKind::ParseErr,
|
||||
"missing count for -c".to_string(),
|
||||
));
|
||||
}
|
||||
Opt::Short('r') => {
|
||||
arr_op_opts.reverse = true;
|
||||
}
|
||||
Opt::ShortWithArg('v', var) => {
|
||||
arr_op_opts.var = Some(var);
|
||||
}
|
||||
Opt::Short('v') => {
|
||||
return Err(ShErr::simple(
|
||||
ShErrKind::ParseErr,
|
||||
"missing variable name for -v".to_string(),
|
||||
));
|
||||
}
|
||||
_ => {
|
||||
return Err(ShErr::simple(
|
||||
ShErrKind::ParseErr,
|
||||
format!("invalid option: {}", opt),
|
||||
));
|
||||
}
|
||||
}
|
||||
}
|
||||
Ok(arr_op_opts)
|
||||
}
|
||||
|
||||
Reference in New Issue
Block a user