2023-12-27 15:44:19 -07:00
|
|
|
/*
|
|
|
|
* Copyright (c) 2023 Emma Tebibyte <emma@tebibyte.media>
|
|
|
|
* SPDX-License-Identifier: AGPL-3.0-or-later
|
|
|
|
*
|
|
|
|
* This program is free software: you can redistribute it and/or modify it under
|
|
|
|
* the terms of the GNU Affero General Public License as published by the Free
|
|
|
|
* Software Foundation, either version 3 of the License, or (at your option) any
|
|
|
|
* later version.
|
|
|
|
*
|
|
|
|
* This program is distributed in the hope that it will be useful, but WITHOUT
|
|
|
|
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
|
|
|
|
* FOR A PARTICULAR PURPOSE. See the GNU Affero General Public License for more
|
|
|
|
* details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU Affero General Public License
|
|
|
|
* along with this program. If not, see https://www.gnu.org/licenses/.
|
|
|
|
*/
|
|
|
|
|
|
|
|
use std::{
|
|
|
|
env::args,
|
2023-12-27 22:42:50 -07:00
|
|
|
io::{ stdin, Write },
|
2023-12-27 15:44:19 -07:00
|
|
|
process::{ Command, exit, Stdio },
|
|
|
|
};
|
|
|
|
|
2023-12-27 22:42:50 -07:00
|
|
|
extern crate sysexits;
|
2023-12-27 22:47:06 -07:00
|
|
|
use sysexits::{ EX_DATAERR, EX_USAGE };
|
2023-12-27 22:42:50 -07:00
|
|
|
|
2023-12-27 15:44:19 -07:00
|
|
|
fn main() {
|
|
|
|
let argv = args().collect::<Vec<String>>();
|
|
|
|
|
2023-12-27 22:42:50 -07:00
|
|
|
let usage = format!("Usage: {} index command [args...]", argv[0]);
|
|
|
|
|
2023-12-27 15:44:19 -07:00
|
|
|
let index = match argv.get(1) {
|
|
|
|
Some(i) => {
|
|
|
|
i.parse::<usize>().unwrap_or_else(|_| {
|
|
|
|
eprintln!("{}: {}: Not an integer.", argv[0], i);
|
|
|
|
exit(1);
|
|
|
|
})
|
|
|
|
},
|
|
|
|
None => {
|
2023-12-27 22:42:50 -07:00
|
|
|
eprintln!("{}", usage);
|
|
|
|
exit(EX_USAGE);
|
2023-12-27 15:44:19 -07:00
|
|
|
},
|
|
|
|
};
|
|
|
|
|
|
|
|
let mut buf = String::new();
|
2023-12-27 22:42:50 -07:00
|
|
|
stdin().read_line(&mut buf).unwrap();
|
2023-12-27 15:44:19 -07:00
|
|
|
let mut fields = buf.split('␞').collect::<Vec<&str>>();
|
|
|
|
|
|
|
|
argv.get(2).unwrap_or_else(|| {
|
2023-12-27 22:42:50 -07:00
|
|
|
eprintln!("{}", usage);
|
|
|
|
exit(EX_USAGE);
|
2023-12-27 15:44:19 -07:00
|
|
|
});
|
|
|
|
|
|
|
|
let opts = argv.iter().clone().skip(3).collect::<Vec<&String>>();
|
|
|
|
|
|
|
|
let mut spawned = Command::new(argv.get(2).unwrap())
|
|
|
|
.args(opts)
|
|
|
|
.stdin(Stdio::piped())
|
|
|
|
.stdout(Stdio::piped())
|
|
|
|
.spawn()
|
|
|
|
.unwrap();
|
|
|
|
|
|
|
|
let field = fields.get(index).unwrap_or_else(|| {
|
2023-12-27 22:42:50 -07:00
|
|
|
eprintln!(
|
|
|
|
"{}: {}: No such index in input.",
|
|
|
|
argv[0],
|
|
|
|
index.to_string()
|
|
|
|
);
|
2023-12-27 22:47:06 -07:00
|
|
|
exit(EX_DATAERR);
|
2023-12-27 15:44:19 -07:00
|
|
|
});
|
|
|
|
|
|
|
|
if let Some(mut child_stdin) = spawned.stdin.take() {
|
|
|
|
child_stdin.write_all(field.as_bytes()).unwrap();
|
|
|
|
drop(child_stdin);
|
|
|
|
}
|
|
|
|
|
|
|
|
let output = spawned.wait_with_output().unwrap();
|
|
|
|
|
|
|
|
let new_field = String::from_utf8(output.stdout).unwrap();
|
|
|
|
|
|
|
|
fields[index] = &new_field;
|
|
|
|
|
|
|
|
print!("{}", fields.join("␞"));
|
|
|
|
}
|