Merge pull request #73 from nushell/forgiving_def_parse

More forgiving def parse
This commit is contained in:
JT 2021-09-27 14:06:51 +13:00 committed by GitHub
commit 3b134a1ae2
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23

View file

@ -47,35 +47,49 @@ pub fn parse_def(
let mut error = None; let mut error = None;
let name = working_set.get_span_contents(spans[0]); let name = working_set.get_span_contents(spans[0]);
if name == b"def" && spans.len() >= 4 { if name == b"def" {
//FIXME: don't use expect here let def_decl_id = working_set
let (name_expr, err) = parse_string(working_set, spans[1]); .find_decl(b"def")
error = error.or(err); .expect("internal error: missing def command");
working_set.enter_scope(); let mut call = Box::new(Call {
let (sig, err) = parse_signature(working_set, spans[2]); head: spans[0],
error = error.or(err); decl_id: def_decl_id,
positional: vec![],
named: vec![],
});
let (block, err) = let call = if let Some(name_span) = spans.get(1) {
parse_block_expression(working_set, &SyntaxShape::Block(Some(vec![])), spans[3]); let (name_expr, err) = parse_string(working_set, *name_span);
error = error.or(err); error = error.or(err);
working_set.exit_scope();
if error.is_some() {
return (
Statement::Pipeline(Pipeline::from_vec(vec![garbage(span(spans))])),
error,
);
}
let name = name_expr.as_string(); let name = name_expr.as_string();
call.positional.push(name_expr);
if let Some(sig_span) = spans.get(2) {
working_set.enter_scope();
let (sig, err) = parse_signature(working_set, *sig_span);
error = error.or(err);
let signature = sig.as_signature(); let signature = sig.as_signature();
call.positional.push(sig);
if let Some(block_span) = spans.get(3) {
let (block, err) = parse_block_expression(
working_set,
&SyntaxShape::Block(Some(vec![])),
*block_span,
);
error = error.or(err);
let block_id = block.as_block(); let block_id = block.as_block();
match (name, signature, block_id) { call.positional.push(block);
(Some(name), Some(mut signature), Some(block_id)) => {
if let (Some(name), Some(mut signature), Some(block_id)) =
(name, signature, block_id)
{
let decl_id = working_set let decl_id = working_set
.find_decl(name.as_bytes()) .find_decl(name.as_bytes())
.expect("internal error: predeclaration failed to add definition"); .expect("internal error: predeclaration failed to add definition");
@ -85,18 +99,46 @@ pub fn parse_def(
signature.name = name; signature.name = name;
*declaration = signature.into_block_command(block_id); *declaration = signature.into_block_command(block_id);
}
} else {
let err_span = Span {
start: sig_span.end,
end: sig_span.end,
};
let def_decl_id = working_set error = error
.find_decl(b"def") .or_else(|| Some(ParseError::MissingPositional("block".into(), err_span)));
.expect("internal error: missing def command"); }
working_set.exit_scope();
let call = Box::new(Call { call
head: spans[0], } else {
decl_id: def_decl_id, let err_span = Span {
positional: vec![name_expr, sig, block], start: name_span.end,
named: vec![], end: name_span.end,
};
error = error
.or_else(|| Some(ParseError::MissingPositional("parameters".into(), err_span)));
call
}
} else {
let err_span = Span {
start: spans[0].end,
end: spans[0].end,
};
error = error.or_else(|| {
Some(ParseError::MissingPositional(
"definition name".into(),
err_span,
))
}); });
call
};
( (
Statement::Pipeline(Pipeline::from_vec(vec![Expression { Statement::Pipeline(Pipeline::from_vec(vec![Expression {
expr: Expr::Call(call), expr: Expr::Call(call),
@ -106,12 +148,6 @@ pub fn parse_def(
}])), }])),
error, error,
) )
}
_ => (
Statement::Pipeline(Pipeline::from_vec(vec![garbage(span(spans))])),
error,
),
}
} else { } else {
( (
garbage_statement(spans), garbage_statement(spans),