Skip to content
GitLab
Explore
Sign in
Register
Primary navigation
Search or go to…
Project
ion
Manage
Activity
Members
Labels
Plan
Issues
Issue boards
Milestones
Wiki
Code
Merge requests
Repository
Branches
Commits
Tags
Repository graph
Compare revisions
Snippets
Deploy
Releases
Container Registry
Model registry
Monitor
Incidents
Analyze
Value stream analytics
Contributor analytics
Repository analytics
Model experiments
Help
Help
Support
GitLab documentation
Compare GitLab plans
Community forum
Contribute to GitLab
Provide feedback
Keyboard shortcuts
?
Snippets
Groups
Projects
Show more breadcrumbs
redox-os
ion
Commits
3cac1eb4
Commit
3cac1eb4
authored
5 years ago
by
AdminXVII
Browse files
Options
Downloads
Patches
Plain Diff
Start implementing errors for pipes
parent
a66d3c64
No related branches found
No related tags found
1 merge request
!1024
Bubble execution errors instead of printing to stderr directly
Changes
2
Hide whitespace changes
Inline
Side-by-side
Showing
2 changed files
src/lib/shell/pipe_exec/fork.rs
+9
-6
9 additions, 6 deletions
src/lib/shell/pipe_exec/fork.rs
src/lib/shell/pipe_exec/mod.rs
+40
-50
40 additions, 50 deletions
src/lib/shell/pipe_exec/mod.rs
with
49 additions
and
56 deletions
src/lib/shell/pipe_exec/fork.rs
+
9
−
6
View file @
3cac1eb4
...
...
@@ -5,7 +5,6 @@ use super::{
job_control
::{
BackgroundProcess
,
ProcessState
},
};
use
crate
::
parser
::
pipelines
::
Pipeline
;
use
std
::
process
::
exit
;
impl
<
'a
>
Shell
<
'a
>
{
/// Ensures that the forked child is given a unique process ID.
...
...
@@ -31,7 +30,14 @@ impl<'a> Shell<'a> {
Self
::
create_process_group
(
0
);
// After execution of it's commands, exit with the last command's status.
sys
::
fork_exit
(
self
.pipe
(
pipeline
)
.as_os_code
());
sys
::
fork_exit
(
self
.pipe
(
pipeline
)
.unwrap_or_else
(|
err
|
{
eprintln!
(
"{}"
,
err
);
Status
::
COULD_NOT_EXEC
})
.as_os_code
(),
);
}
Ok
(
pid
)
=>
{
if
state
!=
ProcessState
::
Empty
{
...
...
@@ -40,10 +46,7 @@ impl<'a> Shell<'a> {
}
Status
::
SUCCESS
}
Err
(
why
)
=>
{
eprintln!
(
"ion: background fork failed: {}"
,
why
);
exit
(
1
);
}
Err
(
why
)
=>
Status
::
error
(
format!
(
"ion: background fork failed: {}"
,
why
)),
}
}
}
This diff is collapsed.
Click to expand it.
src/lib/shell/pipe_exec/mod.rs
+
40
−
50
View file @
3cac1eb4
...
...
@@ -41,9 +41,9 @@ use std::{
#[derive(Debug,
Error)]
pub
enum
InputError
{
#[error(display
=
"
ion:
failed to redirect '{}' to stdin: {}"
,
file,
why)]
#[error(display
=
"failed to redirect '{}' to stdin: {}"
,
file,
why)]
File
{
file
:
String
,
why
:
io
::
Error
},
#[error(display
=
"
ion:
failed to redirect herestring '{}' to stdin: {}"
,
string,
why)]
#[error(display
=
"failed to redirect herestring '{}' to stdin: {}"
,
string,
why)]
HereString
{
string
:
String
,
why
:
io
::
Error
},
}
...
...
@@ -54,17 +54,25 @@ pub struct OutputError {
why
:
io
::
Error
,
}
#[derive(Debug)]
#[derive(Debug
,
Error
)]
pub
enum
RedirectError
{
Input
(
InputError
),
Output
(
OutputError
),
#[error(display
=
"{}"
,
cause)]
Input
{
cause
:
InputError
},
#[error(display
=
"{}"
,
cause)]
Output
{
cause
:
OutputError
},
}
#[derive(Debug,
Error)]
pub
enum
PipeError
{
#[error(display
=
"ion: {}"
,
cause)]
RedirectError
{
cause
:
RedirectError
},
}
impl
fmt
::
Display
for
OutputError
{
fn
fmt
(
&
self
,
f
:
&
mut
fmt
::
Formatter
)
->
fmt
::
Result
{
write!
(
f
,
"
ion:
failed to redirect {} to file '{}': {}"
,
"failed to redirect {} to file '{}': {}"
,
match
self
.redirect
{
RedirectFrom
::
Both
=>
"both stdout and stderr"
,
RedirectFrom
::
Stdout
=>
"stdout"
,
...
...
@@ -81,30 +89,16 @@ impl std::error::Error for OutputError {
fn
source
(
&
self
)
->
Option
<&
(
dyn
std
::
error
::
Error
+
'static
)
>
{
Some
(
&
self
.why
)
}
}
impl
fmt
::
Display
for
RedirectError
{
fn
fmt
(
&
self
,
f
:
&
mut
fmt
::
Formatter
)
->
fmt
::
Result
{
match
self
{
RedirectError
::
Output
(
why
)
=>
write!
(
f
,
"ion: failed to redirect {}"
,
why
),
RedirectError
::
Input
(
why
)
=>
write!
(
f
,
"ion: failed to redirect {}"
,
why
),
}
}
}
impl
std
::
error
::
Error
for
RedirectError
{
fn
source
(
&
self
)
->
Option
<&
(
dyn
std
::
error
::
Error
+
'static
)
>
{
Some
(
match
self
{
RedirectError
::
Output
(
why
)
=>
why
,
RedirectError
::
Input
(
why
)
=>
why
,
})
}
}
impl
From
<
OutputError
>
for
RedirectError
{
fn
from
(
error
:
OutputError
)
->
Self
{
RedirectError
::
Output
(
error
)
}
fn
from
(
cause
:
OutputError
)
->
Self
{
RedirectError
::
Output
{
cause
}
}
}
impl
From
<
InputError
>
for
RedirectError
{
fn
from
(
error
:
InputError
)
->
Self
{
RedirectError
::
Input
(
error
)
}
fn
from
(
cause
:
InputError
)
->
Self
{
RedirectError
::
Input
{
cause
}
}
}
impl
From
<
RedirectError
>
for
PipeError
{
fn
from
(
cause
:
RedirectError
)
->
Self
{
PipeError
::
RedirectError
{
cause
}
}
}
/// Create an OS pipe and write the contents of a byte slice to one end
...
...
@@ -442,7 +436,10 @@ impl<'b> Shell<'b> {
PipeType
::
Background
=>
self
.fork_pipe
(
pipeline
,
command
,
ProcessState
::
Running
),
// Execute each command in the pipeline, giving each command the foreground.
PipeType
::
Normal
=>
{
let
exit_status
=
self
.pipe
(
pipeline
);
let
exit_status
=
self
.pipe
(
pipeline
)
.unwrap_or_else
(|
err
|
{
eprintln!
(
"{}"
,
err
);
Status
::
COULD_NOT_EXEC
});
// Set the shell as the foreground process again to regain the TTY.
if
!
self
.opts.is_background_shell
{
let
_
=
sys
::
tcsetpgrp
(
0
,
process
::
id
());
...
...
@@ -455,14 +452,8 @@ impl<'b> Shell<'b> {
/// Executes a piped job `job1 | job2 | job3`
///
/// This function will panic if called with an empty slice
fn
pipe
(
&
mut
self
,
pipeline
:
Pipeline
<
'b
>
)
->
Status
{
let
mut
commands
=
match
prepare
(
self
,
pipeline
)
{
Ok
(
c
)
=>
c
.into_iter
()
.peekable
(),
Err
(
why
)
=>
{
eprintln!
(
"{}"
,
why
);
return
Status
::
COULD_NOT_EXEC
;
}
};
fn
pipe
(
&
mut
self
,
pipeline
:
Pipeline
<
'b
>
)
->
Result
<
Status
,
PipeError
>
{
let
mut
commands
=
prepare
(
self
,
pipeline
)
?
.into_iter
()
.peekable
();
if
let
Some
((
mut
parent
,
mut
kind
))
=
commands
.next
()
{
if
kind
==
RedirectFrom
::
None
&&
!
parent
.needs_forking
()
{
...
...
@@ -471,7 +462,7 @@ impl<'b> Shell<'b> {
let
_
=
io
::
stdout
()
.flush
();
let
_
=
io
::
stderr
()
.flush
();
status
Ok
(
status
)
}
else
{
let
(
mut
pgid
,
mut
last_pid
,
mut
current_pid
)
=
(
0
,
0
,
0
);
...
...
@@ -510,18 +501,17 @@ impl<'b> Shell<'b> {
RedirectFrom
::
None
=>
(),
RedirectFrom
::
Stderr
=>
parent
.stderr
(
writer
),
RedirectFrom
::
Stdout
=>
parent
.stdout
(
writer
),
RedirectFrom
::
Both
=>
match
writer
.try_clone
()
{
Err
(
e
)
=>
{
eprintln!
(
"ion: failed to redirect stdout and stderr: {}"
,
e
);
}
Ok
(
duped
)
=>
{
parent
.stderr
(
writer
);
parent
.stdout
(
duped
);
}
},
RedirectFrom
::
Both
=>
{
let
duped
=
writer
.try_clone
()
.map_err
(|
why
|
{
RedirectError
::
from
(
OutputError
{
redirect
:
kind
,
file
:
"pipe"
.to_string
(),
why
,
})
})
?
;
parent
.stderr
(
writer
);
parent
.stdout
(
duped
);
}
}
}
}
...
...
@@ -554,10 +544,10 @@ impl<'b> Shell<'b> {
let
_
=
io
::
stdout
()
.flush
();
let
_
=
io
::
stderr
()
.flush
();
}
status
Ok
(
status
)
}
}
else
{
Status
::
SUCCESS
Ok
(
Status
::
SUCCESS
)
}
}
}
...
...
This diff is collapsed.
Click to expand it.
Preview
0%
Loading
Try again
or
attach a new file
.
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Save comment
Cancel
Please
register
or
sign in
to comment