2020-05-09 22:08:53 +02:00
|
|
|
use nu_errors::ShellError;
|
2020-05-30 19:41:25 +02:00
|
|
|
use nu_protocol::{CallInfo, Value};
|
2020-05-09 22:08:53 +02:00
|
|
|
use nu_source::{Tag, Tagged, TaggedItem};
|
2020-05-08 20:19:48 +02:00
|
|
|
use std::path::Path;
|
|
|
|
#[cfg(not(target_os = "windows"))]
|
|
|
|
use std::process::{Command, Stdio};
|
|
|
|
|
2020-05-09 22:08:53 +02:00
|
|
|
#[derive(Default)]
|
2020-05-08 20:19:48 +02:00
|
|
|
pub struct Start {
|
2020-05-09 22:08:53 +02:00
|
|
|
pub tag: Tag,
|
|
|
|
pub filenames: Vec<Tagged<String>>,
|
2020-05-08 20:19:48 +02:00
|
|
|
pub application: Option<String>,
|
|
|
|
}
|
|
|
|
|
2020-05-09 22:08:53 +02:00
|
|
|
impl Start {
|
|
|
|
pub fn new() -> Start {
|
|
|
|
Start {
|
|
|
|
tag: Tag::unknown(),
|
|
|
|
filenames: vec![],
|
|
|
|
application: None,
|
2020-05-08 20:19:48 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-05-09 22:08:53 +02:00
|
|
|
pub fn parse(&mut self, call_info: CallInfo) -> Result<(), ShellError> {
|
|
|
|
self.tag = call_info.name_tag.clone();
|
2021-04-24 16:33:17 +02:00
|
|
|
self.parse_input_parameters(&call_info)?;
|
2020-05-08 20:19:48 +02:00
|
|
|
self.parse_application(&call_info);
|
2020-05-09 22:08:53 +02:00
|
|
|
Ok(())
|
2020-05-08 20:19:48 +02:00
|
|
|
}
|
|
|
|
|
2020-05-09 22:08:53 +02:00
|
|
|
fn add_filename(&mut self, filename: Tagged<String>) -> Result<(), ShellError> {
|
|
|
|
if Path::new(&filename.item).exists() || url::Url::parse(&filename.item).is_ok() {
|
2020-05-08 20:19:48 +02:00
|
|
|
self.filenames.push(filename);
|
2020-05-09 22:08:53 +02:00
|
|
|
Ok(())
|
2020-05-08 20:19:48 +02:00
|
|
|
} else {
|
2020-05-09 22:08:53 +02:00
|
|
|
Err(ShellError::labeled_error(
|
|
|
|
format!("The file '{}' does not exist", filename.item),
|
|
|
|
"doesn't exist",
|
|
|
|
filename.tag,
|
|
|
|
))
|
2020-05-08 20:19:48 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-05-30 19:41:25 +02:00
|
|
|
fn glob_to_values(&self, value: &Value) -> Result<Vec<Tagged<String>>, ShellError> {
|
|
|
|
let mut result = vec![];
|
|
|
|
match glob::glob(&value.as_string()?) {
|
|
|
|
Ok(paths) => {
|
|
|
|
for path_result in paths {
|
|
|
|
match path_result {
|
|
|
|
Ok(path) => result
|
|
|
|
.push(path.to_string_lossy().to_string().tagged(value.tag.clone())),
|
|
|
|
Err(glob_error) => {
|
|
|
|
return Err(ShellError::labeled_error(
|
|
|
|
format!("{}", glob_error),
|
|
|
|
"glob error",
|
|
|
|
value.tag.clone(),
|
|
|
|
));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
Err(pattern_error) => {
|
|
|
|
return Err(ShellError::labeled_error(
|
|
|
|
format!("{}", pattern_error),
|
|
|
|
"invalid pattern",
|
|
|
|
value.tag.clone(),
|
|
|
|
))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Ok(result)
|
|
|
|
}
|
|
|
|
|
2021-04-24 16:33:17 +02:00
|
|
|
fn parse_input_parameters(&mut self, call_info: &CallInfo) -> Result<(), ShellError> {
|
2020-05-08 20:19:48 +02:00
|
|
|
let candidates = match &call_info.args.positional {
|
2020-05-09 22:08:53 +02:00
|
|
|
Some(values) => {
|
|
|
|
let mut result = vec![];
|
|
|
|
|
|
|
|
for value in values.iter() {
|
2021-04-24 16:33:17 +02:00
|
|
|
let val_str = value.as_string();
|
|
|
|
match val_str {
|
|
|
|
Ok(s) => {
|
|
|
|
if s.to_ascii_lowercase().starts_with("http")
|
|
|
|
|| s.to_ascii_lowercase().starts_with("https")
|
|
|
|
{
|
|
|
|
if webbrowser::open(&s).is_ok() {
|
|
|
|
result.push("http/web".to_string().tagged_unknown())
|
|
|
|
} else {
|
|
|
|
return Err(ShellError::labeled_error(
|
|
|
|
&format!("error opening {}", &s),
|
|
|
|
"error opening url",
|
|
|
|
self.tag.span,
|
|
|
|
));
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
let res = self.glob_to_values(value)?;
|
|
|
|
result.extend(res);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
Err(e) => {
|
|
|
|
return Err(ShellError::labeled_error(
|
|
|
|
e.to_string(),
|
|
|
|
"no input given",
|
|
|
|
self.tag.span,
|
|
|
|
));
|
|
|
|
}
|
|
|
|
}
|
2020-05-09 22:08:53 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
if result.is_empty() {
|
|
|
|
return Err(ShellError::labeled_error(
|
2021-04-24 16:33:17 +02:00
|
|
|
"No input given",
|
|
|
|
"no input given",
|
2020-05-09 22:08:53 +02:00
|
|
|
self.tag.span,
|
|
|
|
));
|
|
|
|
}
|
|
|
|
result
|
|
|
|
}
|
|
|
|
None => {
|
|
|
|
return Err(ShellError::labeled_error(
|
2021-04-24 16:33:17 +02:00
|
|
|
"No input given",
|
|
|
|
"no input given",
|
2020-05-09 22:08:53 +02:00
|
|
|
self.tag.span,
|
|
|
|
))
|
|
|
|
}
|
2020-05-08 20:19:48 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
for candidate in candidates {
|
2021-04-24 16:33:17 +02:00
|
|
|
if !candidate.contains("http/web") {
|
|
|
|
self.add_filename(candidate)?;
|
|
|
|
}
|
2020-05-08 20:19:48 +02:00
|
|
|
}
|
2020-05-09 22:08:53 +02:00
|
|
|
|
|
|
|
Ok(())
|
2020-05-08 20:19:48 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
fn parse_application(&mut self, call_info: &CallInfo) {
|
|
|
|
self.application = if let Some(app) = call_info.args.get("application") {
|
|
|
|
match app.as_string() {
|
|
|
|
Ok(name) => Some(name),
|
|
|
|
Err(_) => None,
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
None
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
#[cfg(target_os = "macos")]
|
2020-05-09 22:08:53 +02:00
|
|
|
pub fn exec(&mut self) -> Result<(), ShellError> {
|
2020-05-08 20:19:48 +02:00
|
|
|
let mut args = vec![];
|
2020-05-09 22:08:53 +02:00
|
|
|
args.append(
|
|
|
|
&mut self
|
|
|
|
.filenames
|
|
|
|
.iter()
|
|
|
|
.map(|x| x.item.clone())
|
|
|
|
.collect::<Vec<_>>(),
|
|
|
|
);
|
2020-05-08 20:19:48 +02:00
|
|
|
|
|
|
|
if let Some(app_name) = &self.application {
|
|
|
|
args.append(&mut vec![String::from("-a"), app_name.to_string()]);
|
|
|
|
}
|
2020-05-09 22:08:53 +02:00
|
|
|
exec_cmd("open", &args, self.tag.clone())
|
2020-05-08 20:19:48 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
#[cfg(target_os = "windows")]
|
2020-05-09 22:08:53 +02:00
|
|
|
pub fn exec(&mut self) -> Result<(), ShellError> {
|
2020-05-08 20:19:48 +02:00
|
|
|
if let Some(app_name) = &self.application {
|
|
|
|
for file in &self.filenames {
|
2020-05-09 22:08:53 +02:00
|
|
|
match open::with(&file.item, app_name) {
|
2020-05-08 20:19:48 +02:00
|
|
|
Ok(_) => continue,
|
|
|
|
Err(_) => {
|
2020-05-09 22:08:53 +02:00
|
|
|
return Err(ShellError::labeled_error(
|
2020-05-08 20:19:48 +02:00
|
|
|
"Failed to open file with specified application",
|
2020-05-09 22:08:53 +02:00
|
|
|
"can't open with specified application",
|
|
|
|
file.tag.span,
|
2020-05-08 20:19:48 +02:00
|
|
|
))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
for file in &self.filenames {
|
2020-05-09 22:08:53 +02:00
|
|
|
match open::that(&file.item) {
|
2020-05-08 20:19:48 +02:00
|
|
|
Ok(_) => continue,
|
|
|
|
Err(_) => {
|
2020-05-09 22:08:53 +02:00
|
|
|
return Err(ShellError::labeled_error(
|
2020-05-08 20:19:48 +02:00
|
|
|
"Failed to open file with default application",
|
2020-05-09 22:08:53 +02:00
|
|
|
"can't open with default application",
|
|
|
|
file.tag.span,
|
2020-05-08 20:19:48 +02:00
|
|
|
))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
|
|
|
#[cfg(not(any(target_os = "windows", target_os = "macos")))]
|
2020-05-09 22:08:53 +02:00
|
|
|
pub fn exec(&mut self) -> Result<(), ShellError> {
|
2020-05-08 20:19:48 +02:00
|
|
|
let mut args = vec![];
|
2020-05-09 22:08:53 +02:00
|
|
|
args.append(
|
|
|
|
&mut self
|
|
|
|
.filenames
|
|
|
|
.iter()
|
|
|
|
.map(|x| x.item.clone())
|
|
|
|
.collect::<Vec<_>>(),
|
|
|
|
);
|
2020-05-08 20:19:48 +02:00
|
|
|
|
|
|
|
if let Some(app_name) = &self.application {
|
2020-05-09 22:08:53 +02:00
|
|
|
exec_cmd(&app_name, &args, self.tag.clone())
|
2020-05-08 20:19:48 +02:00
|
|
|
} else {
|
|
|
|
for cmd in &["xdg-open", "gnome-open", "kde-open", "wslview"] {
|
2020-05-09 22:08:53 +02:00
|
|
|
if exec_cmd(cmd, &args, self.tag.clone()).is_err() {
|
2020-05-08 20:19:48 +02:00
|
|
|
continue;
|
2020-05-09 19:28:57 +02:00
|
|
|
} else {
|
|
|
|
return Ok(());
|
2020-05-08 20:19:48 +02:00
|
|
|
}
|
|
|
|
}
|
2020-05-09 22:08:53 +02:00
|
|
|
Err(ShellError::labeled_error(
|
2020-05-08 20:19:48 +02:00
|
|
|
"Failed to open file(s) with xdg-open. gnome-open, kde-open, and wslview",
|
2020-05-09 22:08:53 +02:00
|
|
|
"failed to open xdg-open. gnome-open, kde-open, and wslview",
|
|
|
|
self.tag.span,
|
2020-05-08 20:19:48 +02:00
|
|
|
))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[cfg(not(target_os = "windows"))]
|
2020-05-09 22:08:53 +02:00
|
|
|
fn exec_cmd(cmd: &str, args: &[String], tag: Tag) -> Result<(), ShellError> {
|
2020-05-08 20:19:48 +02:00
|
|
|
if args.is_empty() {
|
2020-05-09 22:08:53 +02:00
|
|
|
return Err(ShellError::labeled_error(
|
|
|
|
"No file(s) or application provided",
|
|
|
|
"no file(s) or application provided",
|
|
|
|
tag,
|
|
|
|
));
|
2020-05-08 20:19:48 +02:00
|
|
|
}
|
|
|
|
let status = match Command::new(cmd)
|
|
|
|
.stdout(Stdio::null())
|
|
|
|
.stderr(Stdio::null())
|
|
|
|
.args(args)
|
|
|
|
.status()
|
|
|
|
{
|
|
|
|
Ok(exit_code) => exit_code,
|
2020-05-09 22:08:53 +02:00
|
|
|
Err(_) => {
|
|
|
|
return Err(ShellError::labeled_error(
|
|
|
|
"Failed to run native open syscall",
|
|
|
|
"failed to run native open call",
|
|
|
|
tag,
|
|
|
|
))
|
|
|
|
}
|
2020-05-08 20:19:48 +02:00
|
|
|
};
|
|
|
|
if status.success() {
|
|
|
|
Ok(())
|
|
|
|
} else {
|
2020-05-09 22:08:53 +02:00
|
|
|
Err(ShellError::labeled_error(
|
2020-05-08 20:19:48 +02:00
|
|
|
"Failed to run start. Hint: The file(s)/application may not exist",
|
2020-05-09 22:08:53 +02:00
|
|
|
"failed to run",
|
|
|
|
tag,
|
2020-05-08 20:19:48 +02:00
|
|
|
))
|
|
|
|
}
|
|
|
|
}
|