diff --git a/engine/runtime-integration-tests/src/test/java/org/enso/compiler/test/ErrorCompilerTest.java b/engine/runtime-integration-tests/src/test/java/org/enso/compiler/test/ErrorCompilerTest.java index db52d244659..5f2a64cce16 100644 --- a/engine/runtime-integration-tests/src/test/java/org/enso/compiler/test/ErrorCompilerTest.java +++ b/engine/runtime-integration-tests/src/test/java/org/enso/compiler/test/ErrorCompilerTest.java @@ -623,6 +623,28 @@ public class ErrorCompilerTest extends CompilerTests { ir, Syntax.UnexpectedExpression$.MODULE$, "Unexpected expression", 0, 41); } + @Test + public void inlineDocCommentIsNotAllowed_1() { + var ir = parse(""" + main args = + v = 42 ## meh + v + """); + assertSingleSyntaxError( + ir, Syntax.UnexpectedExpression$.MODULE$, "Unexpected expression", 23, 29); + } + + @Test + public void inlineDocCommentIsNotAllowed_2() { + var ir = parse(""" + main args = + v = 42 + v ## meh + """); + assertSingleSyntaxError( + ir, Syntax.UnexpectedExpression$.MODULE$, "Unexpected expression", 29, 35); + } + private void assertSingleSyntaxError( Module ir, Syntax.Reason type, String msg, int start, int end) { var errors = assertIR(ir, Syntax.class, 1); diff --git a/lib/java/test-utils/src/main/java/org/enso/test/utils/ContextUtils.java b/lib/java/test-utils/src/main/java/org/enso/test/utils/ContextUtils.java index b59709b4047..28f182885d8 100644 --- a/lib/java/test-utils/src/main/java/org/enso/test/utils/ContextUtils.java +++ b/lib/java/test-utils/src/main/java/org/enso/test/utils/ContextUtils.java @@ -189,6 +189,9 @@ public final class ContextUtils { var source = Source.newBuilder(LanguageInfo.ID, src, moduleName + ".enso").buildLiteral(); var module = ctx.eval(source); var runtimeMod = (org.enso.interpreter.runtime.Module) unwrapValue(ctx, module); + if (runtimeMod.getIr() == null) { + runtimeMod.compileScope(leakContext(ctx)); + } return runtimeMod.getIr(); } diff --git a/lib/rust/parser/debug/tests/parse.rs b/lib/rust/parser/debug/tests/parse.rs index 80702b393e3..c4a23d11fd9 100644 --- a/lib/rust/parser/debug/tests/parse.rs +++ b/lib/rust/parser/debug/tests/parse.rs @@ -169,6 +169,7 @@ fn doc_comments() { (Documented (#((Section " Test indent handling")) #(() ())) (Function (Ident foo) #((() (Ident bar) () ())) () (Ident foo)))))); + expect_invalid_node("expression ## unexpected doc comment on same line"); } diff --git a/lib/rust/parser/src/syntax/tree.rs b/lib/rust/parser/src/syntax/tree.rs index 2e34e14929f..6ed3aca150f 100644 --- a/lib/rust/parser/src/syntax/tree.rs +++ b/lib/rust/parser/src/syntax/tree.rs @@ -830,6 +830,7 @@ pub enum SyntaxError { PatternUnexpectedExpression, PatternUnexpectedDot, CaseOfInvalidCase, + DocumentationUnexpectedNonInitial, } impl From for Cow<'static, str> { @@ -864,6 +865,7 @@ impl From for Cow<'static, str> { PatternUnexpectedExpression => "Expression invalid in a pattern", PatternUnexpectedDot => "In a pattern, the dot operator can only be used in a qualified name", CaseOfInvalidCase => "Invalid case expression.", + DocumentationUnexpectedNonInitial => "Unexpected documentation at end of line", }) .into() } diff --git a/lib/rust/parser/src/syntax/treebuilding/compound_token.rs b/lib/rust/parser/src/syntax/treebuilding/compound_token.rs index a845167849e..103de827b98 100644 --- a/lib/rust/parser/src/syntax/treebuilding/compound_token.rs +++ b/lib/rust/parser/src/syntax/treebuilding/compound_token.rs @@ -6,6 +6,7 @@ use crate::syntax::consumer::TokenConsumer; use crate::syntax::consumer::TreeConsumer; use crate::syntax::maybe_with_error; use crate::syntax::token; +use crate::syntax::tree::SyntaxError; use crate::syntax::GroupHierarchyConsumer; use crate::syntax::Token; use crate::syntax::Tree; @@ -19,20 +20,22 @@ use crate::syntax::Tree; /// Recognizes lexical tokens that are indivisible, and assembles them into trees. #[derive(Default, Debug)] pub struct CompoundTokens<'s, Inner> { - compounding: Option>, - inner: Inner, + compounding: Option>, + inner: Inner, + has_preceding_item: bool, } impl<'s, Inner: TreeConsumer<'s> + TokenConsumer<'s>> CompoundTokens<'s, Inner> { fn try_start(&mut self, token: Token<'s>) { - match CompoundToken::start(token) { + match CompoundToken::start(token, self.has_preceding_item) { StartStep::Start(compounding) => self.compounding = Some(compounding), StartStep::RejectButStart(compounding, token) => { self.inner.push_token(token); - self.compounding = Some(compounding) + self.compounding = Some(compounding); } StartStep::Reject(token) => self.inner.push_token(token), } + self.has_preceding_item = true; } } @@ -89,6 +92,7 @@ impl<'s, Inner: TreeConsumer<'s>> CompoundTokens<'s, Inner> { if let Some(tree) = self.compounding.take().and_then(|builder| builder.flush()) { self.inner.push_tree(tree); } + self.has_preceding_item = default(); } } @@ -121,7 +125,7 @@ where Inner: TreeConsumer<'s> + GroupHierarchyConsumer<'s> // ============================== trait CompoundTokenBuilder<'s>: Sized { - fn start(token: Token<'s>) -> StartStep>; + fn start(token: Token<'s>, has_preceding_item: bool) -> StartStep>; fn step(self, token: Token<'s>) -> Step, Tree<'s>>; fn flush(self) -> Option>; } @@ -176,12 +180,19 @@ impl StartStep { } impl<'s> CompoundTokenBuilder<'s> for CompoundToken<'s> { - fn start(token: Token<'s>) -> StartStep> { + fn start(token: Token<'s>, has_preceding_item: bool) -> StartStep> { use CompoundToken::*; StartStep::Reject(token) - .or_else(|token| TextLiteralBuilder::start(token).map_state(TextLiteral)) - .or_else(|token| OperatorIdentifierBuilder::start(token).map_state(OperatorIdentifier)) - .or_else(|token| AutoscopeBuilder::start(token).map_state(Autoscope)) + .or_else(|token| { + TextLiteralBuilder::start(token, has_preceding_item).map_state(TextLiteral) + }) + .or_else(|token| { + OperatorIdentifierBuilder::start(token, has_preceding_item) + .map_state(OperatorIdentifier) + }) + .or_else(|token| { + AutoscopeBuilder::start(token, has_preceding_item).map_state(Autoscope) + }) } fn step(self, token: Token<'s>) -> Step, Tree<'s>> { @@ -210,17 +221,23 @@ impl<'s> CompoundTokenBuilder<'s> for CompoundToken<'s> { #[derive(Debug)] struct TextLiteralBuilder<'s> { - open: token::TextStart<'s>, - newline: Option>, - elements: Vec>, + open: token::TextStart<'s>, + newline: Option>, + elements: Vec>, + has_preceding_item: bool, } impl<'s> CompoundTokenBuilder<'s> for TextLiteralBuilder<'s> { - fn start(token: Token<'s>) -> StartStep> { + fn start(token: Token<'s>, has_preceding_item: bool) -> StartStep> { match token.variant { token::Variant::TextStart(variant) => { let token = token.with_variant(variant); - StartStep::Start(Self { open: token, newline: default(), elements: default() }) + StartStep::Start(Self { + open: token, + newline: default(), + elements: default(), + has_preceding_item, + }) } _ => StartStep::Reject(token), } @@ -260,18 +277,21 @@ impl<'s> CompoundTokenBuilder<'s> for TextLiteralBuilder<'s> { } fn flush(self) -> Option> { - let Self { open, newline, elements } = self; + let Self { open, newline, elements, has_preceding_item: _ } = self; Some(Tree::text_literal(Some(open), newline, elements, None)) } } impl<'s> TextLiteralBuilder<'s> { fn finish(self, close: token::TextEnd<'s>) -> Tree<'s> { - let Self { open, newline, elements } = self; + let Self { open, newline, elements, has_preceding_item } = self; if open.code.starts_with('#') { assert_eq!(newline, None); let doc = syntax::tree::DocComment { open, elements, newlines: default() }; - Tree::documented(doc, default()) + let tree = Tree::documented(doc, default()); + let error = + has_preceding_item.then_some(SyntaxError::DocumentationUnexpectedNonInitial); + maybe_with_error(tree, error) } else { let close = if close.code.is_empty() { None } else { Some(close) }; Tree::text_literal(Some(open), newline, elements, close) @@ -288,7 +308,7 @@ impl<'s> TextLiteralBuilder<'s> { struct OperatorIdentifierBuilder; impl<'s> CompoundTokenBuilder<'s> for OperatorIdentifierBuilder { - fn start(token: Token<'s>) -> StartStep> { + fn start(token: Token<'s>, _has_preceding_item: bool) -> StartStep> { match token.variant { token::Variant::DotOperator(_) => StartStep::RejectButStart(Self, token), _ => StartStep::Reject(token), @@ -322,7 +342,7 @@ struct AutoscopeBuilder<'s> { } impl<'s> CompoundTokenBuilder<'s> for AutoscopeBuilder<'s> { - fn start(token: Token<'s>) -> StartStep> { + fn start(token: Token<'s>, _has_preceding_item: bool) -> StartStep> { match token.variant { token::Variant::AutoscopeOperator(variant) => { let operator = token.with_variant(variant);