Skip to content

Commit

Permalink
Merge branch 'main' into syntax-role-modifiers
Browse files Browse the repository at this point in the history
  • Loading branch information
jurgenvinju committed Nov 8, 2023
2 parents 06f10c5 + a57d215 commit b809bef
Show file tree
Hide file tree
Showing 24 changed files with 586 additions and 132 deletions.
41 changes: 9 additions & 32 deletions .github/workflows/build.yaml
Original file line number Diff line number Diff line change
Expand Up @@ -38,18 +38,11 @@ jobs:
with:
check_name: "Test Report - ${{ runner.os }}"

- name: Report Failure
if: failure()
uses: act10ns/slack@v1
with:
status: ${{ job.status }}
steps: ${{ toJson(steps) }}
env:
SLACK_WEBHOOK_URL: ${{ secrets.RASCAL_SLACK_WEBHOOK }}

builds:
if: ${{ !(github.ref == 'refs/heads/main' && contains(github.event.head_commit.message, '[maven-release-plugin]')) }}
needs: [test-linux]
permissions:
contents: write
runs-on: buildjet-4vcpu-ubuntu-2204
steps:
- uses: browser-actions/setup-chrome@latest
Expand Down Expand Up @@ -97,15 +90,6 @@ jobs:
tests (buildjet-4vcpu-ubuntu-2204-arm)
ttl: 15

- name: Prepare Draft Release
uses: softprops/action-gh-release@v1
if: startsWith(github.ref, 'refs/tags/')
with:
draft: true
files: ${{ steps.build-artifact.outputs.artifact-root-dir}}/**/*
env:
GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}

- name: Deploy
if: startsWith(github.ref, 'refs/tags/v')
uses: usethesource/releases-maven-action@v1
Expand All @@ -118,14 +102,15 @@ jobs:
ssh-username: ${{ secrets.RELEASE_SSH_USERNAME }}
ssh-private-key: ${{ secrets.RELEASE_SSH_PRIVATE_KEY }}

- name: Report Failure
if: failure()
uses: act10ns/slack@v1
- name: Prepare Draft Release
uses: softprops/action-gh-release@v1
if: startsWith(github.ref, 'refs/tags/')
with:
status: ${{ job.status }}
steps: ${{ toJson(steps) }}
draft: true
files: ${{ steps.build-artifact.outputs.artifact-root-dir}}/**/*
env:
SLACK_WEBHOOK_URL: ${{ secrets.RASCAL_SLACK_WEBHOOK }}
GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}


tests:
if: ${{ !(github.ref == 'refs/heads/main' && contains(github.event.head_commit.message, '[maven-release-plugin]')) }}
Expand Down Expand Up @@ -160,11 +145,3 @@ jobs:
with:
check_name: "Test Report - ${{ runner.os }}"

- name: Report Failure
if: failure()
uses: act10ns/slack@v1
with:
status: ${{ job.status }}
steps: ${{ toJson(steps) }}
env:
SLACK_WEBHOOK_URL: ${{ secrets.RASCAL_SLACK_WEBHOOK }}
15 changes: 5 additions & 10 deletions pom.xml
Original file line number Diff line number Diff line change
Expand Up @@ -3,7 +3,7 @@

<groupId>org.rascalmpl</groupId>
<artifactId>rascal</artifactId>
<version>0.34.0-RC2-SNAPSHOT</version>
<version>0.35.0-RC2-SNAPSHOT</version>
<packaging>jar</packaging>

<scm>
Expand Down Expand Up @@ -32,7 +32,7 @@
<exec.mainClass>org.rascalmpl.shell.RascalShell</exec.mainClass>
<rascal.test.memory>2</rascal.test.memory>
<maven.compiler.release>11</maven.compiler.release>
<rascal-maven.version>0.21.0</rascal-maven.version>
<rascal-maven.version>0.22.0-RC2</rascal-maven.version>
</properties>


Expand Down Expand Up @@ -219,6 +219,7 @@
<includes>
<include>**/org/rascalmpl/test/AllSuiteParallel.java</include>
<include>**/org/rascalmpl/test/value/AllTests.java</include>
<include>**/org/rascalmpl/*Test.java</include>
</includes>
</configuration>
</plugin>
Expand Down Expand Up @@ -361,10 +362,10 @@
<artifactId>junit</artifactId>
<version>4.13.1</version>
</dependency>
<dependency>
<dependency>
<groupId>io.usethesource</groupId>
<artifactId>vallang</artifactId>
<version>0.15.1</version>
<version>1.0.0-RC3</version>
</dependency>
<dependency>
<groupId>org.ow2.asm</groupId>
Expand Down Expand Up @@ -441,11 +442,5 @@
<artifactId>icu4j</artifactId>
<version>69.1</version>
</dependency>
<dependency>
<groupId>com.beust</groupId>
<artifactId>jcommander</artifactId>
<version>1.72</version>
<scope>provided</scope>
</dependency>
</dependencies>
</project>
Original file line number Diff line number Diff line change
Expand Up @@ -113,7 +113,7 @@ public boolean next() {

if (dynMatchType.isOpen()) {
// type parameter hygiene required (consider self-application of a function like `&T id(&T v) = v`)
dynMatchType = AbstractFunction.renameType(dynMatchType, new HashMap<>());
dynMatchType = AbstractFunction.renameType(dynMatchType, new HashMap<>(), ctx.getCurrentAST().getLocation());
}

if (!declaredType.match(dynMatchType, dynBindings)) {
Expand All @@ -134,7 +134,7 @@ public boolean next() {
Type dynType = subject.getValue().getType();
if (dynType.isOpen()) {
// type parameter hygiene required (consider self-application of a function like `&T id(&T v) = v`)
dynType = AbstractFunction.renameType(dynType, new HashMap<>());
dynType = AbstractFunction.renameType(dynType, new HashMap<>(), ctx.getCurrentAST().getLocation());
}

if (!dynType.isSubtypeOf(declaredType.instantiate(ctx.getCurrentEnvt().getDynamicTypeBindings()))) {
Expand Down
14 changes: 7 additions & 7 deletions src/org/rascalmpl/interpreter/result/AbstractFunction.java
Original file line number Diff line number Diff line change
Expand Up @@ -24,8 +24,6 @@
import java.util.List;
import java.util.Map;
import java.util.Map.Entry;
import java.util.UUID;

import org.rascalmpl.ast.AbstractAST;
import org.rascalmpl.ast.Expression;
import org.rascalmpl.ast.FunctionDeclaration;
Expand All @@ -41,10 +39,11 @@
import org.rascalmpl.uri.URIUtil;
import org.rascalmpl.values.RascalValueFactory;
import org.rascalmpl.values.functions.IFunction;

import io.usethesource.vallang.IConstructor;
import io.usethesource.vallang.IConstructor;
import io.usethesource.vallang.IExternalValue;
import io.usethesource.vallang.IListWriter;
import io.usethesource.vallang.ISourceLocation;
import io.usethesource.vallang.IValue;
import io.usethesource.vallang.IValueFactory;
import io.usethesource.vallang.IWithKeywordParameters;
Expand Down Expand Up @@ -347,8 +346,9 @@ protected Type bindTypeParameters(Type actualStaticTypes, IValue[] actuals, Type
if (actualStaticTypes.isOpen()) {
// we have to make the environment hygenic now, because the caller scope
// may have the same type variable names as the current scope
actualStaticTypes = renameType(actualStaticTypes, renamings);
actualStaticTypes = renameType(actualStaticTypes, renamings, env.getLocation());
}


Map<Type, Type> staticBindings = new HashMap<Type, Type>();

Expand Down Expand Up @@ -394,13 +394,13 @@ protected static Type unrenameType(Map<Type, Type> renamings, Type resultType) {
return resultType;
}

public static Type renameType(Type actualTypes, Map<Type, Type> renamings) {
public static Type renameType(Type actualTypes, Map<Type, Type> renamings, ISourceLocation uniquePrefix) {
actualTypes.match(TypeFactory.getInstance().voidType(), renamings);

// rename all the bound type parameters
for (Entry<Type,Type> entry : renamings.entrySet()) {
Type key = entry.getKey();
renamings.put(key, TypeFactory.getInstance().parameterType(key.getName() + ":" + UUID.randomUUID().toString(), key.getBound()));
renamings.put(key, TypeFactory.getInstance().parameterType(key.getName() + ":" + uniquePrefix, key.getBound()));
}
actualTypes = actualTypes.instantiate(renamings);
return actualTypes;
Expand Down
2 changes: 1 addition & 1 deletion src/org/rascalmpl/interpreter/result/NamedFunction.java
Original file line number Diff line number Diff line change
Expand Up @@ -164,7 +164,7 @@ protected void checkReturnTypeIsNotVoid(List<Expression> formals, IValue[] actua
Map<Type, Type> bindings = new HashMap<>();

for (int i = 0; i < actuals.length; i++) {
formals.get(i).typeOf(declarationEnvironment, getEval(), false).match(renameType(actuals[i].getType(), renamings), bindings);
formals.get(i).typeOf(declarationEnvironment, getEval(), false).match(renameType(actuals[i].getType(), renamings, ctx.getCurrentAST().getLocation()), bindings);
}

if (!getReturnType().isBottom() && getReturnType().instantiate(bindings).isBottom()) {
Expand Down
1 change: 0 additions & 1 deletion src/org/rascalmpl/interpreter/result/RascalFunction.java
Original file line number Diff line number Diff line change
Expand Up @@ -128,7 +128,6 @@ private List<Statement> cloneBody() {
return getAst().clone(body);
}


private String computeIndexedLabel(int pos, AbstractAST ast) {
return ast.accept(new NullASTVisitor<String>() {
@Override
Expand Down
47 changes: 34 additions & 13 deletions src/org/rascalmpl/library/ParseTree.rsc
Original file line number Diff line number Diff line change
Expand Up @@ -416,27 +416,48 @@ which leads to the prefix of the `src` fields of the resulting tree.
The parse function behaves differently depending of the given keyword parameters:
* `allowAmbiguity`: if true then no exception is thrown in case of ambiguity and a parse forest is returned. if false,
* the parser throws an exception during tree building and produces only the first ambiguous subtree in its message.
* if set to `false`, the parse constructs trees in linear time. if set to `true` the parser constructs trees in polynomial time.
the parser throws an exception during tree building and produces only the first ambiguous subtree in its message.
if set to `false`, the parse constructs trees in linear time. if set to `true` the parser constructs trees in polynomial time.
*
* `hasSideEffects`: if false then the parser is a lot faster when constructing trees, since it does not execute the parse _actions_ in an
* interpreted environment to make side effects (like a symbol table) and it can share more intermediate results as a result.
*
* `firstAmbiguity`: if true, then the parser returns the subforest for the first (left-most innermost) ambiguity instead of a parse tree for
* the entire input string. This is for grammar debugging purposes a much faster solution then waiting for an entire
* parse forest to be constructed in polynomial time.
interpreted environment to make side effects (like a symbol table) and it can share more intermediate results as a result.
}
@javaClass{org.rascalmpl.library.Prelude}
java &T (value input, loc origin) parser(type[&T] grammar, bool allowAmbiguity=false, bool hasSideEffects=false, bool firstAmbiguity=false, set[Tree(Tree)] filters={});
java &T (value input, loc origin) parser(type[&T] grammar, bool allowAmbiguity=false, bool hasSideEffects=false, set[Tree(Tree)] filters={});
@javaClass{org.rascalmpl.library.Prelude}
@synopsis{Generates a parser function that can be used to find the left-most deepest ambiguous sub-sentence.}
@benefits{
* Instead of trying to build a polynomially sized parse forest, this function only builds the smallest part of
the tree that exhibits ambiguity. This can be done very quickly, while the whole forest could take minutes to hours to construct.
* Use this function for ambiguity diagnostics and regression testing for ambiguity.
}
@pitfalls{
* The returned sub-tree usually has a different type than the parameter of the type[] symbol that was passed in.
The reason is that sub-trees typically have a different non-terminal than the start non-terminal of a grammar.
}
java Tree (value input, loc origin) firstAmbiguityFinder(type[Tree] grammar, bool hasSideEffects=false, set[Tree(Tree)] filters={});
@synopsis{Generates parsers from a grammar (reified type), where all non-terminals in the grammar can be used as start-symbol.}
@description{
This parser generator behaves the same as the `parser` function, but it produces parser functions which have an additional
nonterminal parameter. This can be used to select a specific non-terminal from the grammar to use as start-symbol for parsing.
}
@javaClass{org.rascalmpl.library.Prelude}
java &U (type[&U] nonterminal, value input, loc origin) parsers(type[&T] grammar, bool allowAmbiguity=false, bool hasSideEffects=false, bool firstAmbiguity=false, set[Tree(Tree)] filters={});
java &U (type[&U] nonterminal, value input, loc origin) parsers(type[&T] grammar, bool allowAmbiguity=false, bool hasSideEffects=false, set[Tree(Tree)] filters={});
@javaClass{org.rascalmpl.library.Prelude}
@synopsis{Generates a parser function that can be used to find the left-most deepest ambiguous sub-sentence.}
@benefits{
* Instead of trying to build a polynomially sized parse forest, this function only builds the smallest part of
the tree that exhibits ambiguity. This can be done very quickly, while the whole forest could take minutes to hours to construct.
* Use this function for ambiguity diagnostics and regression testing for ambiguity.
}
@pitfalls{
* The returned sub-tree usually has a different type than the parameter of the type[] symbol that was passed in.
The reason is that sub-trees typically have a different non-terminal than the start non-terminal of a grammar.
}
java Tree (type[Tree] nonterminal, value input, loc origin) firstAmbiguityFinders(type[Tree] grammar, bool hasSideEffects=false, set[Tree(Tree)] filters={});
@synopsis{Parse the input but instead of returning the entire tree, return the trees for the first ambiguous substring.}
@description{
Expand All @@ -447,10 +468,10 @@ the cost of constructing nested ambiguity clusters.
If the input sentence is not ambiguous after all, simply the entire tree is returned.
}
Tree firstAmbiguity(type[Tree] begin, str input)
= parser(begin, firstAmbiguity=true)(input, |unknown:///|);
= firstAmbiguityFinder(begin)(input, |unknown:///|);
Tree firstAmbiguity(type[Tree] begin, loc input)
= parser(begin, firstAmbiguity=true)(input, input);
= firstAmbiguityFinder(begin)(input, input);
@javaClass{org.rascalmpl.library.Prelude}
@synopsis{Generate a parser and store it in serialized form for later reuse.}
Expand Down Expand Up @@ -514,15 +535,15 @@ p(type(sort("E"), ()), "e+e", |src:///|);
* reifiying types (use of `#`) will trigger the loading of a parser generator anyway. You have to use
this notation for types to avoid that: `type(\start(sort("MySort")), ())` to avoid the computation for `#start[A]`
}
java &U (type[&U] nonterminal, value input, loc origin) loadParsers(loc savedParsers, bool allowAmbiguity=false, bool hasSideEffects=false, bool firstAmbiguity=false, set[Tree(Tree)] filters={});
java &U (type[&U] nonterminal, value input, loc origin) loadParsers(loc savedParsers, bool allowAmbiguity=false, bool hasSideEffects=false, set[Tree(Tree)] filters={});
@synopsis{Load a previously serialized parser, for a specific non-terminal, from disk for usage}
@description{
This loader behaves just like ((loadParsers)), except that the resulting parser function is already
bound to a specific non-terminal.
}
@javaClass{org.rascalmpl.library.Prelude}
java &U (value input, loc origin) loadParser(type[&U] nonterminal, loc savedParsers, bool allowAmbiguity=false, bool hasSideEffects=false, bool firstAmbiguity=false, set[Tree(Tree)] filters={});
java &U (value input, loc origin) loadParser(type[&U] nonterminal, loc savedParsers, bool allowAmbiguity=false, bool hasSideEffects=false, set[Tree(Tree)] filters={});
@synopsis{Yield the string of characters that form the leafs of the given parse tree.}
@description{
Expand Down
24 changes: 16 additions & 8 deletions src/org/rascalmpl/library/Prelude.java
Original file line number Diff line number Diff line change
Expand Up @@ -2343,12 +2343,20 @@ public INode arbNode() {

protected final TypeReifier tr;

public IFunction parser(IValue start, IBool allowAmbiguity, IBool hasSideEffects, IBool firstAmbiguity, ISet filters) {
return rascalValues.parser(start, allowAmbiguity, hasSideEffects, firstAmbiguity, filters);
public IFunction parser(IValue start, IBool allowAmbiguity, IBool hasSideEffects, ISet filters) {
return rascalValues.parser(start, allowAmbiguity, hasSideEffects, values.bool(false), filters);
}

public IFunction firstAmbiguityFinder(IValue start, IBool hasSideEffects, ISet filters) {
return rascalValues.parser(start, values.bool(true), hasSideEffects, values.bool(true), filters);
}

public IFunction parsers(IValue start, IBool allowAmbiguity, IBool hasSideEffects, IBool firstAmbiguity, ISet filters) {
return rascalValues.parsers(start, allowAmbiguity, hasSideEffects, firstAmbiguity, filters);
public IFunction parsers(IValue start, IBool allowAmbiguity, IBool hasSideEffects, ISet filters) {
return rascalValues.parsers(start, allowAmbiguity, hasSideEffects, values.bool(false), filters);
}

public IFunction firstAmbiguityFinders(IValue start, IBool hasSideEffects, ISet filters) {
return rascalValues.parsers(start, values.bool(true), hasSideEffects, values.bool(true), filters);
}

public void storeParsers(IValue start, ISourceLocation saveLocation) {
Expand All @@ -2363,18 +2371,18 @@ public void storeParsers(IValue start, ISourceLocation saveLocation) {
}
}

public IFunction loadParsers(ISourceLocation savedLocation, IBool allowAmbiguity, IBool hasSideEffects, IBool firstAmbiguity, ISet filters) {
public IFunction loadParsers(ISourceLocation savedLocation, IBool allowAmbiguity, IBool hasSideEffects, ISet filters) {
try {
return rascalValues.loadParsers(savedLocation, allowAmbiguity, hasSideEffects, firstAmbiguity, filters);
return rascalValues.loadParsers(savedLocation, allowAmbiguity, hasSideEffects, values.bool(false), filters);
}
catch (IOException | ClassNotFoundException e) {
throw RuntimeExceptionFactory.io(e.getMessage());
}
}

public IFunction loadParser(IValue grammar, ISourceLocation savedLocation, IBool allowAmbiguity, IBool hasSideEffects, IBool firstAmbiguity, ISet filters) {
public IFunction loadParser(IValue grammar, ISourceLocation savedLocation, IBool allowAmbiguity, IBool hasSideEffects, ISet filters) {
try {
return rascalValues.loadParser(grammar, savedLocation, allowAmbiguity, hasSideEffects, firstAmbiguity, filters);
return rascalValues.loadParser(grammar, savedLocation, allowAmbiguity, hasSideEffects, values.bool(false), filters);
}
catch (IOException | ClassNotFoundException e) {
throw RuntimeExceptionFactory.io(e.getMessage());
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -39,7 +39,7 @@ public set[Production] holes(Grammar object) {
return { regular(iter(\char-class([range(48,57)]))),
prod(label("$MetaHole",getTargetSymbol(nont)),
[ \char-class([range(0,0)]),
lit("<denormalize(nont)>"),lit(":"),iter(\char-class([range(48,57)])),
lit("<removeConditionals(denormalize(nont))>"),lit(":"),iter(\char-class([range(48,57)])),
\char-class([range(0,0)])
],{Attr::\tag("holeType"(nont))}) // TODO: added qualifier to help compiler
| Symbol nont <- object.rules, quotable(nont)
Expand All @@ -64,6 +64,10 @@ private Symbol denormalize(Symbol s) = visit (s) {
case \seq(ss) => seq([t | t <- ss, !(t is layouts)])
};

private Symbol removeConditionals(Symbol sym) = visit(sym) {
case conditional(s, _) => s
};

@synopsis{This is needed such that list variables can be repeatedly used as elements of the same list}
private Symbol getTargetSymbol(Symbol sym) {
switch(sym) {
Expand Down
Loading

0 comments on commit b809bef

Please sign in to comment.