Compare commits
No commits in common. "1c77a99f54d19177653f67d76e171de48cdae0d8" and "1d5bfdd88fc37e9b81351745c656cf2fa1cb716c" have entirely different histories.
1c77a99f54
...
1d5bfdd88f
@ -1,15 +0,0 @@
|
|||||||
name: Gitea Actions Demo
|
|
||||||
run-name: ${{ gitea.actor }} is testing out Gitea Actions 🚀
|
|
||||||
on: [push]
|
|
||||||
|
|
||||||
jobs:
|
|
||||||
Explore-Gitea-Actions:
|
|
||||||
runs-on: ubuntu-latest
|
|
||||||
steps:
|
|
||||||
- uses: actions/checkout@v4
|
|
||||||
- uses: actions/setup-java@v4
|
|
||||||
with:
|
|
||||||
java-version: '17'
|
|
||||||
distribution: 'temurin'
|
|
||||||
- name: Run the Maven verify phase
|
|
||||||
run: mvn --batch-mode --update-snapshots verify
|
|
2
.idea/misc.xml
generated
2
.idea/misc.xml
generated
@ -40,7 +40,7 @@
|
|||||||
</list>
|
</list>
|
||||||
</option>
|
</option>
|
||||||
</component>
|
</component>
|
||||||
<component name="ProjectRootManager" version="2" languageLevel="JDK_22" default="true" project-jdk-name="openjdk-22" project-jdk-type="JavaSDK">
|
<component name="ProjectRootManager" version="2" languageLevel="JDK_21" default="true" project-jdk-name="openjdk-21" project-jdk-type="JavaSDK">
|
||||||
<output url="file://$PROJECT_DIR$/out" />
|
<output url="file://$PROJECT_DIR$/out" />
|
||||||
</component>
|
</component>
|
||||||
</project>
|
</project>
|
@ -1,21 +1,18 @@
|
|||||||
import ast.ASTNode;
|
import ast.ASTNode;
|
||||||
import org.antlr.v4.runtime.*;
|
import ast.ClassNode;
|
||||||
import org.antlr.v4.runtime.Token;
|
|
||||||
import ast.ProgramNode;
|
import ast.ProgramNode;
|
||||||
import bytecode.ByteCodeGenerator;
|
import bytecode.ByteCodeGenerator;
|
||||||
import org.antlr.v4.runtime.CharStream;
|
import org.antlr.v4.runtime.CharStream;
|
||||||
import org.antlr.v4.runtime.CharStreams;
|
import org.antlr.v4.runtime.CharStreams;
|
||||||
import org.antlr.v4.runtime.tree.ParseTree;
|
|
||||||
import org.antlr.v4.runtime.CommonTokenStream;
|
import org.antlr.v4.runtime.CommonTokenStream;
|
||||||
|
import org.antlr.v4.runtime.tree.ParseTree;
|
||||||
import parser.ASTBuilder;
|
import parser.ASTBuilder;
|
||||||
import parser.generated.SimpleJavaLexer;
|
import parser.generated.SimpleJavaLexer;
|
||||||
import parser.generated.SimpleJavaParser;
|
import parser.generated.SimpleJavaParser;
|
||||||
import semantic.SemanticAnalyzer;
|
import semantic.SemanticAnalyzer;
|
||||||
import bytecode.ByteCodeGenerator;
|
|
||||||
|
|
||||||
import java.io.IOException;
|
import java.io.IOException;
|
||||||
import java.nio.file.Paths;
|
import java.nio.file.Paths;
|
||||||
import java.util.List;
|
|
||||||
|
|
||||||
public class Main {
|
public class Main {
|
||||||
public static void main(String[] args) throws Exception {
|
public static void main(String[] args) throws Exception {
|
||||||
@ -33,103 +30,20 @@ public class Main {
|
|||||||
|
|
||||||
|
|
||||||
static void parseFile(CharStream codeCharStream) {
|
static void parseFile(CharStream codeCharStream) {
|
||||||
/* ------------------------- Scanner -> tokens ------------------------- */
|
|
||||||
SimpleJavaLexer lexer = new SimpleJavaLexer(codeCharStream);
|
SimpleJavaLexer lexer = new SimpleJavaLexer(codeCharStream);
|
||||||
CommonTokenStream tokenStream = new CommonTokenStream(lexer);
|
CommonTokenStream tokens = new CommonTokenStream(lexer);
|
||||||
|
SimpleJavaParser parser = new SimpleJavaParser(tokens);
|
||||||
|
|
||||||
// Printing the tokens
|
ParseTree tree = parser.program(); // parse the input
|
||||||
// tokenStream.fill();
|
|
||||||
// List<Token> tokens = tokenStream.getTokens();
|
|
||||||
// System.out.println("-------------------- Scanner -> tokens --------------------");
|
|
||||||
// for (Token token : tokens) {
|
|
||||||
// String tokenType = SimpleJavaLexer.VOCABULARY.getSymbolicName(token.getType());
|
|
||||||
// String tokenText = token.getText();
|
|
||||||
// // System.out.println("Token Type: " + tokenType + ", Token Text: " +
|
|
||||||
// // tokenText);
|
|
||||||
// System.out.println(tokenType + " " + tokenText);
|
|
||||||
// }
|
|
||||||
// System.out.println();
|
|
||||||
|
|
||||||
/* ------------------------- Parser -> Parsetree ------------------------- */
|
ASTBuilder builder = new ASTBuilder();
|
||||||
SimpleJavaParser parser = new SimpleJavaParser(tokenStream);
|
ProgramNode ast = (ProgramNode) builder.visit(tree); // build the AST
|
||||||
ParseTree parseTree = parser.program(); // parse the input
|
|
||||||
|
|
||||||
// Printing the parse tree
|
ProgramNode typedAst = (ProgramNode) SemanticAnalyzer.generateTast(ast);
|
||||||
// System.out.println("-------------------- Parser -> Parsetree --------------------");
|
|
||||||
// System.out.println(parseTree.toStringTree(parser));
|
|
||||||
// printTree(parseTree, parser, 0);
|
|
||||||
// System.out.println();
|
|
||||||
|
|
||||||
/* ------------------------- AST builder -> AST ------------------------- */
|
|
||||||
ASTBuilder astBuilder = new ASTBuilder();
|
|
||||||
ProgramNode abstractSyntaxTree = (ProgramNode) astBuilder.visit(parseTree);
|
|
||||||
|
|
||||||
// Printing the AST
|
|
||||||
// System.out.println("-------------------- AST builder -> AST --------------------");
|
|
||||||
// // System.out.println("AST: " + ast.toString());
|
|
||||||
// printAST(abstractSyntaxTree, 0);
|
|
||||||
// System.out.println();
|
|
||||||
|
|
||||||
/*
|
|
||||||
* ------------------------- Semantic Analyzer -> Tast -------------------------
|
|
||||||
*/
|
|
||||||
SemanticAnalyzer semanticAnalyzer = new SemanticAnalyzer();
|
|
||||||
ProgramNode typedAst = (ProgramNode) semanticAnalyzer.generateTast(abstractSyntaxTree);
|
|
||||||
|
|
||||||
// Printing the Tast
|
|
||||||
System.out.println("Tast generated");
|
|
||||||
|
|
||||||
/*
|
|
||||||
* ------------------------- Bytecode Generator -> Bytecode
|
|
||||||
* -------------------------
|
|
||||||
*/
|
|
||||||
ByteCodeGenerator byteCodeGenerator = new ByteCodeGenerator();
|
ByteCodeGenerator byteCodeGenerator = new ByteCodeGenerator();
|
||||||
//byteCodeGenerator.generateByteCode(abstractSyntaxTree);
|
if (typedAst != null)
|
||||||
byteCodeGenerator.visit(typedAst);
|
byteCodeGenerator.visit(typedAst);
|
||||||
System.out.println("Bytecode generated");
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* This method is used to print the parse tree in a structured format.
|
|
||||||
* It recursively traverses the tree and prints the rule names and text of the
|
|
||||||
* nodes.
|
|
||||||
*
|
|
||||||
* @param tree The parse tree to be printed.
|
|
||||||
* @param parser The parser used to parse the input. It's used to get the rule
|
|
||||||
* names.
|
|
||||||
* @param indent The current indentation level. It's used to format the output.
|
|
||||||
*/
|
|
||||||
public static void printTree(ParseTree tree, Parser parser, int indent) {
|
|
||||||
// Create an indentation string based on the current indentation level
|
|
||||||
String indentString = " ".repeat(indent * 2);
|
|
||||||
|
|
||||||
// If the tree node is an instance of RuleContext (i.e., it's an internal node),
|
|
||||||
// print the rule name
|
|
||||||
if (tree instanceof RuleContext) {
|
|
||||||
int ruleIndex = ((RuleContext) tree).getRuleIndex();
|
|
||||||
String ruleName = parser.getRuleNames()[ruleIndex];
|
|
||||||
System.out.println(indentString + ruleName);
|
|
||||||
} else {
|
|
||||||
// If the tree node is not an instance of RuleContext (i.e., it's a leaf node),
|
|
||||||
// print the text of the node
|
|
||||||
System.out.println(indentString + tree.getText());
|
|
||||||
}
|
|
||||||
|
|
||||||
// Recursively print the children of the current node, increasing the
|
|
||||||
// indentation level
|
|
||||||
for (int i = 0; i < tree.getChildCount(); i++) {
|
|
||||||
printTree(tree.getChild(i), parser, indent + 1);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public static void printAST(ASTNode node, int indent) {
|
|
||||||
String indentString = " ".repeat(indent * 2);
|
|
||||||
System.out.println(indentString + node.getClass().toString());
|
|
||||||
|
|
||||||
// for (ASTNode child : node.) {
|
|
||||||
// printAST(child, indent + 1);
|
|
||||||
// }
|
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
@ -1,12 +1,7 @@
|
|||||||
package ast;
|
package ast;
|
||||||
|
|
||||||
//import java.util.List;
|
|
||||||
|
|
||||||
public interface ASTNode {
|
public interface ASTNode {
|
||||||
/**
|
|
||||||
* Please implement this method to return a list of children of each node.
|
|
||||||
*/
|
|
||||||
// public List<ASTNode> getChildren();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
29
src/main/java/ast/VarNode.java
Normal file
29
src/main/java/ast/VarNode.java
Normal file
@ -0,0 +1,29 @@
|
|||||||
|
package ast;
|
||||||
|
|
||||||
|
import semantic.SemanticVisitor;
|
||||||
|
import typechecker.TypeCheckResult;
|
||||||
|
import visitor.Visitable;
|
||||||
|
|
||||||
|
public class VarNode implements ASTNode, Visitable {
|
||||||
|
|
||||||
|
private String identifier;
|
||||||
|
private String type;
|
||||||
|
|
||||||
|
public VarNode(String type, String identifier){
|
||||||
|
this.type = type;
|
||||||
|
this.identifier = identifier;
|
||||||
|
}
|
||||||
|
|
||||||
|
public String getType(){
|
||||||
|
return type;
|
||||||
|
}
|
||||||
|
|
||||||
|
public String getIdentifier(){
|
||||||
|
return identifier;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public TypeCheckResult accept(SemanticVisitor visitor) {
|
||||||
|
return visitor.analyze(this);
|
||||||
|
}
|
||||||
|
}
|
@ -7,9 +7,9 @@ import visitor.Visitable;
|
|||||||
public class BinaryExpressionNode implements ExpressionNode, Visitable {
|
public class BinaryExpressionNode implements ExpressionNode, Visitable {
|
||||||
public ExpressionNode left;
|
public ExpressionNode left;
|
||||||
public ExpressionNode right;
|
public ExpressionNode right;
|
||||||
public ExpresssionOperator operator; // Stores the operator as a string (e.g., "+", "-", "&&")
|
public String operator; // Stores the operator as a string (e.g., "+", "-", "&&")
|
||||||
|
|
||||||
public BinaryExpressionNode(ExpressionNode left, ExpressionNode right, ExpresssionOperator operator) {
|
public BinaryExpressionNode(ExpressionNode left, ExpressionNode right, String operator) {
|
||||||
this.left = left;
|
this.left = left;
|
||||||
this.right = right;
|
this.right = right;
|
||||||
this.operator = operator;
|
this.operator = operator;
|
||||||
|
@ -1,14 +0,0 @@
|
|||||||
package ast.expression;
|
|
||||||
|
|
||||||
public enum ExpresssionOperator {
|
|
||||||
DOT, // .
|
|
||||||
PLUS, // +
|
|
||||||
MINUS, // -
|
|
||||||
MULTIPLY, // *
|
|
||||||
DIVIDE, // /
|
|
||||||
NOT, // !
|
|
||||||
ASSIGNMENT, // =
|
|
||||||
EQUALS, // ==
|
|
||||||
UNEQUALS, // !=
|
|
||||||
ERROR //TODO: Remove This
|
|
||||||
}
|
|
@ -18,5 +18,3 @@ public class UnaryExpressionNode implements ExpressionNode, Visitable {
|
|||||||
return visitor.analyze(this);
|
return visitor.analyze(this);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@ -1,14 +1,17 @@
|
|||||||
package ast.statement;
|
package ast.statement;
|
||||||
|
|
||||||
import ast.expression.BinaryExpressionNode;
|
import ast.VarNode;
|
||||||
|
import ast.expression.ExpressionNode;
|
||||||
import semantic.SemanticVisitor;
|
import semantic.SemanticVisitor;
|
||||||
import typechecker.TypeCheckResult;
|
import typechecker.TypeCheckResult;
|
||||||
import visitor.Visitable;
|
import visitor.Visitable;
|
||||||
|
|
||||||
public class AssignmentStatementNode extends StatementNode implements Visitable {
|
public class AssignmentStatementNode extends StatementNode implements Visitable {
|
||||||
public BinaryExpressionNode expression;
|
public VarNode varNode;
|
||||||
|
public ExpressionNode expression;
|
||||||
|
|
||||||
public AssignmentStatementNode(BinaryExpressionNode expression) {
|
public AssignmentStatementNode(VarNode varNode, ExpressionNode expression) {
|
||||||
|
this.varNode = varNode;
|
||||||
this.expression = expression;
|
this.expression = expression;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Binary file not shown.
@ -3,7 +3,6 @@ package parser;
|
|||||||
import ast.*;
|
import ast.*;
|
||||||
import ast.expression.BinaryExpressionNode;
|
import ast.expression.BinaryExpressionNode;
|
||||||
import ast.expression.ExpressionNode;
|
import ast.expression.ExpressionNode;
|
||||||
import ast.expression.ExpresssionOperator;
|
|
||||||
import ast.expression.IdentifierExpressionNode;
|
import ast.expression.IdentifierExpressionNode;
|
||||||
import ast.expression.UnaryExpressionNode;
|
import ast.expression.UnaryExpressionNode;
|
||||||
import ast.member.FieldNode;
|
import ast.member.FieldNode;
|
||||||
@ -138,14 +137,14 @@ public class ASTBuilder extends SimpleJavaBaseVisitor<ASTNode> {
|
|||||||
|
|
||||||
@Override
|
@Override
|
||||||
public ASTNode visitAssignmentStatement(SimpleJavaParser.AssignmentStatementContext ctx) {
|
public ASTNode visitAssignmentStatement(SimpleJavaParser.AssignmentStatementContext ctx) {
|
||||||
|
VarNode varNode = (VarNode) visit(ctx.var());
|
||||||
BinaryExpressionNode expression = (BinaryExpressionNode) visit(ctx.expression());
|
ExpressionNode expression = (ExpressionNode) visit(ctx.expression());
|
||||||
return new AssignmentStatementNode(expression);
|
return new AssignmentStatementNode(varNode, expression);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public ASTNode visitVar(SimpleJavaParser.VarContext ctx) {
|
public ASTNode visitVar(SimpleJavaParser.VarContext ctx) {
|
||||||
return null;
|
return new VarNode("int", ctx.getText());
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
@ -181,7 +180,8 @@ public class ASTBuilder extends SimpleJavaBaseVisitor<ASTNode> {
|
|||||||
if (ctx.getChildCount() == 3 && ctx.getChild(1) instanceof TerminalNode) {
|
if (ctx.getChildCount() == 3 && ctx.getChild(1) instanceof TerminalNode) {
|
||||||
ExpressionNode left = (ExpressionNode) visit(ctx.expression(0));
|
ExpressionNode left = (ExpressionNode) visit(ctx.expression(0));
|
||||||
ExpressionNode right = (ExpressionNode) visit(ctx.expression(1));
|
ExpressionNode right = (ExpressionNode) visit(ctx.expression(1));
|
||||||
return new BinaryExpressionNode(left, right, ExpresssionOperator.ERROR);
|
String operator = ctx.getChild(1).getText();
|
||||||
|
return new BinaryExpressionNode(left, right, operator);
|
||||||
}
|
}
|
||||||
// Handle unary operations
|
// Handle unary operations
|
||||||
else if (ctx.getChildCount() == 2) {
|
else if (ctx.getChildCount() == 2) {
|
||||||
|
@ -22,12 +22,12 @@ import typechecker.TypeCheckResult;
|
|||||||
|
|
||||||
public class SemanticAnalyzer implements SemanticVisitor {
|
public class SemanticAnalyzer implements SemanticVisitor {
|
||||||
|
|
||||||
private static ArrayList<String> currentFields = new ArrayList<>();
|
private ArrayList<String> currentFields = new ArrayList<>();
|
||||||
|
|
||||||
public static ArrayList<Exception> errors = new ArrayList<>();
|
public static ArrayList<Exception> errors = new ArrayList<>();
|
||||||
|
|
||||||
private static Scope currentScope;
|
private Scope currentScope;
|
||||||
private static ClassNode currentClass;
|
private ClassNode currentClass;
|
||||||
|
|
||||||
public static ASTNode generateTast(ASTNode node) {
|
public static ASTNode generateTast(ASTNode node) {
|
||||||
SemanticAnalyzer semanticCheck = new SemanticAnalyzer();
|
SemanticAnalyzer semanticCheck = new SemanticAnalyzer();
|
||||||
@ -43,13 +43,6 @@ public class SemanticAnalyzer implements SemanticVisitor {
|
|||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
public static void clearAnalyzier(){
|
|
||||||
currentFields.clear();
|
|
||||||
errors.clear();
|
|
||||||
currentScope = null;
|
|
||||||
currentClass = null;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public TypeCheckResult analyze(ProgramNode node) {
|
public TypeCheckResult analyze(ProgramNode node) {
|
||||||
|
|
||||||
@ -145,17 +138,21 @@ public class SemanticAnalyzer implements SemanticVisitor {
|
|||||||
|
|
||||||
@Override
|
@Override
|
||||||
public TypeCheckResult analyze(AssignmentStatementNode assignmentStatementNode) {
|
public TypeCheckResult analyze(AssignmentStatementNode assignmentStatementNode) {
|
||||||
boolean valid = true;
|
if (assignmentStatementNode.expression instanceof LiteralNode literalNode) {
|
||||||
BinaryExpressionNode binaryExpressionNode = assignmentStatementNode.expression;
|
TypeCheckResult varResult = assignmentStatementNode.varNode.accept(this);
|
||||||
var result = binaryExpressionNode.accept(this);
|
TypeCheckResult expressionResult = assignmentStatementNode.expression.accept(this);
|
||||||
valid = valid && result.isValid();
|
}
|
||||||
return new TypeCheckResult(valid, null);
|
return new TypeCheckResult(true, null);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public TypeCheckResult analyze(VarNode toCheck) {
|
||||||
|
return new TypeCheckResult(true, null);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public TypeCheckResult analyze(BinaryExpressionNode toCheck) {
|
public TypeCheckResult analyze(BinaryExpressionNode toCheck) {
|
||||||
boolean valid = true;
|
return null;
|
||||||
return new TypeCheckResult(valid, null);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
@ -3,6 +3,7 @@ package semantic;
|
|||||||
|
|
||||||
import ast.ClassNode;
|
import ast.ClassNode;
|
||||||
import ast.ProgramNode;
|
import ast.ProgramNode;
|
||||||
|
import ast.VarNode;
|
||||||
import ast.expression.BinaryExpressionNode;
|
import ast.expression.BinaryExpressionNode;
|
||||||
import ast.expression.IdentifierExpressionNode;
|
import ast.expression.IdentifierExpressionNode;
|
||||||
import ast.expression.UnaryExpressionNode;
|
import ast.expression.UnaryExpressionNode;
|
||||||
@ -23,6 +24,8 @@ public interface SemanticVisitor {
|
|||||||
|
|
||||||
TypeCheckResult analyze(AssignmentStatementNode toCheck);
|
TypeCheckResult analyze(AssignmentStatementNode toCheck);
|
||||||
|
|
||||||
|
TypeCheckResult analyze(VarNode toCheck);
|
||||||
|
|
||||||
TypeCheckResult analyze(BinaryExpressionNode toCheck);
|
TypeCheckResult analyze(BinaryExpressionNode toCheck);
|
||||||
|
|
||||||
TypeCheckResult analyze(IdentifierExpressionNode toCheck);
|
TypeCheckResult analyze(IdentifierExpressionNode toCheck);
|
||||||
|
@ -2,17 +2,9 @@ public class Example {
|
|||||||
|
|
||||||
public int a;
|
public int a;
|
||||||
|
|
||||||
public static int testMethod(char x){
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
public class Test {
|
|
||||||
|
|
||||||
public static int testMethod(char x, int a){
|
public static int testMethod(char x, int a){
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
@ -1,5 +1,3 @@
|
|||||||
package resources;
|
|
||||||
|
|
||||||
public class AllFeaturesClassExample {
|
public class AllFeaturesClassExample {
|
||||||
int a;
|
int a;
|
||||||
boolean b;
|
boolean b;
|
Binary file not shown.
@ -1,4 +1,4 @@
|
|||||||
public class EmptyClassExample {
|
public class EmptyClassExample {
|
||||||
private class Inner {
|
private class Inner {
|
||||||
}
|
}
|
||||||
} // -o für outout
|
}
|
@ -1,72 +0,0 @@
|
|||||||
import static org.junit.jupiter.api.Assertions.assertNotNull;
|
|
||||||
import static org.junit.jupiter.api.Assertions.assertTrue;
|
|
||||||
|
|
||||||
import org.antlr.v4.runtime.CharStream;
|
|
||||||
import org.antlr.v4.runtime.CharStreams;
|
|
||||||
import org.junit.jupiter.api.Test;
|
|
||||||
|
|
||||||
import javax.tools.JavaCompiler;
|
|
||||||
import javax.tools.ToolProvider;
|
|
||||||
import java.io.File;
|
|
||||||
import java.io.IOException;
|
|
||||||
import java.nio.file.Paths;
|
|
||||||
import java.util.Arrays;
|
|
||||||
import java.util.List;
|
|
||||||
|
|
||||||
public class FailureTest {
|
|
||||||
private static final List<String> TEST_FILES = Arrays.asList(
|
|
||||||
"src/main/test/resources/failureTests/TestClass1.java",
|
|
||||||
"src/main/test/resources/failureTests/TestClass2.java",
|
|
||||||
"src/main/test/resources/failureTests/TestClass3.java",
|
|
||||||
"src/main/test/resources/failureTests/TestClass4.java",
|
|
||||||
"src/main/test/resources/failureTests/TestClass5.java",
|
|
||||||
"src/main/test/resources/failureTests/TestClass6.java",
|
|
||||||
"src/main/test/resources/failureTests/TestClass7.java",
|
|
||||||
"src/main/test/resources/failureTests/TestClass8.java",
|
|
||||||
"src/main/test/resources/failureTests/TestClass9.java",
|
|
||||||
"src/main/test/resources/failureTests/TestClass10.java",
|
|
||||||
"src/main/test/resources/failureTests/TestClass11.java"
|
|
||||||
);
|
|
||||||
|
|
||||||
/**
|
|
||||||
* This test method checks if invalid Java files fail to compile as expected.
|
|
||||||
* It uses the JavaCompiler from the ToolProvider to compile the files.
|
|
||||||
* The test passes if all the files fail to compile.
|
|
||||||
*/
|
|
||||||
@Test
|
|
||||||
public void invalidJavaFilesTest() {
|
|
||||||
// Get the system Java compiler
|
|
||||||
JavaCompiler compiler = ToolProvider.getSystemJavaCompiler();
|
|
||||||
|
|
||||||
// Assert that the compiler is available
|
|
||||||
assertNotNull(compiler, "Java Compiler is not available");
|
|
||||||
|
|
||||||
// Iterate over the test files
|
|
||||||
for (String fileName : TEST_FILES) {
|
|
||||||
// Create a File object for the current file
|
|
||||||
File file = new File(fileName);
|
|
||||||
|
|
||||||
// Try to compile the file and get the result
|
|
||||||
// The run method returns 0 if the compilation was successful, and non-zero otherwise
|
|
||||||
int result = compiler.run(null, null, null, file.getPath());
|
|
||||||
|
|
||||||
// Assert that the compilation failed (i.e., the result is non-zero)
|
|
||||||
assertTrue(result != 0, "Expected compilation failure for " + fileName);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// schmeißt John Fehler, wenn namen doppelt sind?
|
|
||||||
// Input: ParseTree mit genanntem Fehler
|
|
||||||
// Output: Fehlermeldung
|
|
||||||
@Test
|
|
||||||
void typedASTTest() throws IOException {
|
|
||||||
CharStream codeCharStream = null;
|
|
||||||
try {
|
|
||||||
codeCharStream = CharStreams.fromPath(Paths.get("src/main/test/resources/EmptyClassExample.java"));
|
|
||||||
Main.parsefile(codeCharStream);
|
|
||||||
} catch (IOException e) {
|
|
||||||
System.err.println("Error reading the file: " + e.getMessage());
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
@ -21,7 +21,7 @@ public class MainTest {
|
|||||||
void testEmptyClass() {
|
void testEmptyClass() {
|
||||||
CharStream codeCharStream = null;
|
CharStream codeCharStream = null;
|
||||||
try {
|
try {
|
||||||
codeCharStream = CharStreams.fromPath(Paths.get("src/main/test/resources/EmptyClassExample.java"));
|
codeCharStream = CharStreams.fromPath(Paths.get("src/main/test/java/EmptyClassExample.java"));
|
||||||
Main.parsefile(codeCharStream);
|
Main.parsefile(codeCharStream);
|
||||||
} catch (IOException e) {
|
} catch (IOException e) {
|
||||||
System.err.println("Error reading the file: " + e.getMessage());
|
System.err.println("Error reading the file: " + e.getMessage());
|
||||||
|
@ -1,5 +1,3 @@
|
|||||||
package resources;
|
|
||||||
|
|
||||||
public class MoreFeaturesClassExample {
|
public class MoreFeaturesClassExample {
|
||||||
int hallo;
|
int hallo;
|
||||||
private class Inner {
|
private class Inner {
|
@ -17,12 +17,6 @@
|
|||||||
## Scanner Output
|
## Scanner Output
|
||||||
### Beispiel 1: Empty Class
|
### Beispiel 1: Empty Class
|
||||||
|
|
||||||
Token Type; Token Text
|
|
||||||
Type gibts nur bei Terminalen, Text bei allen
|
|
||||||
|
|
||||||
[null "public", null "class", IDENTIFIER "Name", null "{", null "}", EOF "<EOF>"]
|
|
||||||
|
|
||||||
Bsp von Ihm mal:
|
|
||||||
[TokPublic,TokClass,TokIdentifier "Name",TokLeftBrace,TokRightBrace]
|
[TokPublic,TokClass,TokIdentifier "Name",TokLeftBrace,TokRightBrace]
|
||||||
|
|
||||||
### Beispiel 2: Filled Class
|
### Beispiel 2: Filled Class
|
@ -1,11 +1,5 @@
|
|||||||
public class Tester {
|
public class Tester {
|
||||||
public static void main(String[] args) {
|
public static void main(String[] args) {
|
||||||
new EmptyClassExample();
|
new EmptyClassExample();
|
||||||
// cp mitgeben
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
// java -jar pfadtocompiler.jar EmptyClass.java
|
|
||||||
//mit bash scipt ode rmakefile test automatisieren
|
|
||||||
//mvn package
|
|
||||||
// javac tester // tester compilen
|
|
||||||
// java tester // tester ausführen
|
|
@ -1,25 +0,0 @@
|
|||||||
package resources;
|
|
||||||
|
|
||||||
public class CombinedExample {
|
|
||||||
int number;
|
|
||||||
boolean flag;
|
|
||||||
char letter;
|
|
||||||
|
|
||||||
public CombinedExample(int number, boolean flag, char letter) {
|
|
||||||
this.number = number;
|
|
||||||
this.flag = flag;
|
|
||||||
this.letter = letter;
|
|
||||||
}
|
|
||||||
|
|
||||||
public void displayValues() {
|
|
||||||
System.out.println("Number: " + number);
|
|
||||||
System.out.println("Flag: " + flag);
|
|
||||||
System.out.println("Letter: " + letter);
|
|
||||||
}
|
|
||||||
|
|
||||||
public static void main(String[] args) {
|
|
||||||
CombinedExample obj = new CombinedExample(10, true, 'X');
|
|
||||||
obj.displayValues();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
@ -1,6 +0,0 @@
|
|||||||
// Syntax Error: Missing semicolon
|
|
||||||
public class TestClass1 {
|
|
||||||
public static void main(String[] args) {
|
|
||||||
System.out.println("Hello, World!") // Missing semicolon here
|
|
||||||
}
|
|
||||||
}
|
|
@ -1,10 +0,0 @@
|
|||||||
// Semantic Error: Non-static method called from static context
|
|
||||||
public class TestClass10 {
|
|
||||||
public static void main(String[] args) {
|
|
||||||
greet(); // Non-static method 'greet' cannot be referenced from a static context
|
|
||||||
}
|
|
||||||
|
|
||||||
public void greet() {
|
|
||||||
System.out.println("Hi!");
|
|
||||||
}
|
|
||||||
}
|
|
@ -1,10 +0,0 @@
|
|||||||
//Compile error: (4, 9) java: variable number is already defined in method main(java.lang.String[])
|
|
||||||
public class TestClass10 {
|
|
||||||
public static void main(String[] args) {
|
|
||||||
// Declare and initialize an integer variable named 'number' with the value 12
|
|
||||||
int number = 12;
|
|
||||||
|
|
||||||
// This line will cause a compile-time error because 'number' is already defined in this scope
|
|
||||||
int number =13;
|
|
||||||
}
|
|
||||||
}
|
|
@ -1,6 +0,0 @@
|
|||||||
// Syntax Error: Unclosed string literal
|
|
||||||
public class TestClass2 {
|
|
||||||
public static void main(String[] args) {
|
|
||||||
System.out.println("Hello, World!); // Unclosed string literal
|
|
||||||
}
|
|
||||||
}
|
|
@ -1,10 +0,0 @@
|
|||||||
// Syntax Error: Missing parentheses in method declaration
|
|
||||||
public class TestClass3 {
|
|
||||||
public static void main(String[] args) {
|
|
||||||
System.out.println("Hello, World!");
|
|
||||||
}
|
|
||||||
|
|
||||||
public void greet { // Missing parentheses
|
|
||||||
System.out.println("Hi!");
|
|
||||||
}
|
|
||||||
}
|
|
@ -1,4 +0,0 @@
|
|||||||
// Syntax Error: Missing class body
|
|
||||||
public class TestClass4 {-
|
|
||||||
// Missing class body
|
|
||||||
}
|
|
@ -1,7 +0,0 @@
|
|||||||
// Semantic Error: Variable used before declaration
|
|
||||||
public class TestClass5 {
|
|
||||||
public static void main(String[] args) {
|
|
||||||
System.out.println(number); // Variable 'number' used before declaration
|
|
||||||
int number = 10;
|
|
||||||
}
|
|
||||||
}
|
|
@ -1,10 +0,0 @@
|
|||||||
// Semantic Error: Method with wrong return type
|
|
||||||
public class TestClass6 {
|
|
||||||
public static void main(String[] args) {
|
|
||||||
System.out.println("Hello, World!");
|
|
||||||
}
|
|
||||||
|
|
||||||
public int greet() { // Method should return int, but no return statement is provided
|
|
||||||
System.out.println("Hi!");
|
|
||||||
}
|
|
||||||
}
|
|
@ -1,5 +0,0 @@
|
|||||||
// Syntax Error: Unmatched braces
|
|
||||||
public class TestClass7 {
|
|
||||||
public static void main(String[] args) {
|
|
||||||
System.out.println("Hello, World!");
|
|
||||||
// Missing closing brace for the class
|
|
@ -1,14 +0,0 @@
|
|||||||
// Semantic Error: Duplicate method definition
|
|
||||||
public class TestClass8 {
|
|
||||||
public static void main(String[] args) {
|
|
||||||
System.out.println("Hello, World!");
|
|
||||||
}
|
|
||||||
|
|
||||||
public void greet() {
|
|
||||||
System.out.println("Hi!");
|
|
||||||
}
|
|
||||||
|
|
||||||
public void greet() { // Duplicate method definition
|
|
||||||
System.out.println("Hello!");
|
|
||||||
}
|
|
||||||
}
|
|
@ -1,6 +0,0 @@
|
|||||||
// Syntax Error: Incompatible types
|
|
||||||
public class TestClass9 {
|
|
||||||
public static void main(String[] args) {
|
|
||||||
int number = "Hello"; // Incompatible types: String cannot be converted to int
|
|
||||||
}
|
|
||||||
}
|
|
@ -1,24 +0,0 @@
|
|||||||
package resources.featureTests;
|
|
||||||
|
|
||||||
public class BooleanOperations {
|
|
||||||
boolean flag;
|
|
||||||
|
|
||||||
public BooleanOperations(boolean flag) {
|
|
||||||
this.flag = flag;
|
|
||||||
}
|
|
||||||
|
|
||||||
public boolean isFlag() {
|
|
||||||
return flag;
|
|
||||||
}
|
|
||||||
|
|
||||||
public void toggleFlag() {
|
|
||||||
flag = !flag;
|
|
||||||
}
|
|
||||||
|
|
||||||
public static void main(String[] args) {
|
|
||||||
BooleanOperations obj = new BooleanOperations(true);
|
|
||||||
System.out.println(obj.isFlag());
|
|
||||||
obj.toggleFlag();
|
|
||||||
System.out.println(obj.isFlag());
|
|
||||||
}
|
|
||||||
}
|
|
@ -1,24 +0,0 @@
|
|||||||
package resources.featureTests;
|
|
||||||
|
|
||||||
public class CharManipulation {
|
|
||||||
char letter;
|
|
||||||
|
|
||||||
public CharManipulation(char letter) {
|
|
||||||
this.letter = letter;
|
|
||||||
}
|
|
||||||
|
|
||||||
public char getLetter() {
|
|
||||||
return letter;
|
|
||||||
}
|
|
||||||
|
|
||||||
public void setLetter(char letter) {
|
|
||||||
this.letter = letter;
|
|
||||||
}
|
|
||||||
|
|
||||||
public static void main(String[] args) {
|
|
||||||
CharManipulation obj = new CharManipulation('A');
|
|
||||||
System.out.println(obj.getLetter());
|
|
||||||
obj.setLetter('B');
|
|
||||||
System.out.println(obj.getLetter());
|
|
||||||
}
|
|
||||||
}
|
|
@ -1,30 +0,0 @@
|
|||||||
package resources.featureTests;
|
|
||||||
|
|
||||||
public class ConditionalStatements {
|
|
||||||
int number;
|
|
||||||
|
|
||||||
public ConditionalStatements(int number) {
|
|
||||||
this.number = number;
|
|
||||||
}
|
|
||||||
|
|
||||||
public void checkNumber() {
|
|
||||||
if (number > 0) {
|
|
||||||
System.out.println("The number is positive.");
|
|
||||||
} else if (number < 0) {
|
|
||||||
System.out.println("The number is negative.");
|
|
||||||
} else {
|
|
||||||
System.out.println("The number is zero.");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public static void main(String[] args) {
|
|
||||||
ConditionalStatements obj1 = new ConditionalStatements(5);
|
|
||||||
ConditionalStatements obj2 = new ConditionalStatements(-3);
|
|
||||||
ConditionalStatements obj3 = new ConditionalStatements(0);
|
|
||||||
|
|
||||||
obj1.checkNumber();
|
|
||||||
obj2.checkNumber();
|
|
||||||
obj3.checkNumber();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
@ -1,18 +0,0 @@
|
|||||||
package resources.featureTests;
|
|
||||||
|
|
||||||
public class LoopExamples {
|
|
||||||
public static void main(String[] args) {
|
|
||||||
// For loop example
|
|
||||||
for (int i = 0; i < 5; i++) {
|
|
||||||
System.out.println("For loop iteration: " + i);
|
|
||||||
}
|
|
||||||
|
|
||||||
// While loop example
|
|
||||||
int j = 0;
|
|
||||||
while (j < 5) {
|
|
||||||
System.out.println("While loop iteration: " + j);
|
|
||||||
j++;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
@ -1,18 +0,0 @@
|
|||||||
package resources.featureTests;
|
|
||||||
|
|
||||||
public class MethodOverloading {
|
|
||||||
public int add(int a, int b) {
|
|
||||||
return a + b;
|
|
||||||
}
|
|
||||||
|
|
||||||
public int add(int a, int b, int c) {
|
|
||||||
return a + b + c;
|
|
||||||
}
|
|
||||||
|
|
||||||
public static void main(String[] args) {
|
|
||||||
MethodOverloading obj = new MethodOverloading();
|
|
||||||
System.out.println("Sum of 2 and 3: " + obj.add(2, 3));
|
|
||||||
System.out.println("Sum of 1, 2, and 3: " + obj.add(1, 2, 3));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
@ -1,107 +0,0 @@
|
|||||||
package semantic;
|
|
||||||
|
|
||||||
|
|
||||||
import ast.*;
|
|
||||||
import ast.expression.BinaryExpressionNode;
|
|
||||||
import ast.expression.ExpressionNode;
|
|
||||||
import ast.expression.ExpresssionOperator;
|
|
||||||
import ast.expression.IdentifierExpressionNode;
|
|
||||||
import ast.member.FieldNode;
|
|
||||||
import ast.member.MemberNode;
|
|
||||||
import ast.member.MethodNode;
|
|
||||||
import ast.parameter.ParameterListNode;
|
|
||||||
import ast.parameter.ParameterNode;
|
|
||||||
import ast.statement.AssignmentStatementNode;
|
|
||||||
import ast.statement.StatementNode;
|
|
||||||
import ast.type.AccessTypeNode;
|
|
||||||
import ast.type.BaseTypeNode;
|
|
||||||
import ast.type.EnumAccessTypeNode;
|
|
||||||
import ast.type.EnumTypeNode;
|
|
||||||
import org.junit.jupiter.api.BeforeEach;
|
|
||||||
import org.junit.jupiter.api.Test;
|
|
||||||
import semantic.exeptions.AlreadyDeclearedException;
|
|
||||||
|
|
||||||
import java.util.ArrayList;
|
|
||||||
import java.util.List;
|
|
||||||
|
|
||||||
import static org.junit.jupiter.api.Assertions.assertEquals;
|
|
||||||
|
|
||||||
public class SemanticTest {
|
|
||||||
|
|
||||||
@BeforeEach
|
|
||||||
public void init() {
|
|
||||||
SemanticAnalyzer.clearAnalyzier();
|
|
||||||
}
|
|
||||||
|
|
||||||
@Test
|
|
||||||
public void alreadyDeclaredLocalFieldVar(){
|
|
||||||
|
|
||||||
ProgramNode programNode = new ProgramNode();
|
|
||||||
List<ClassNode> classList = new ArrayList<ClassNode>();
|
|
||||||
AccessTypeNode accessTypeNode = new AccessTypeNode(EnumAccessTypeNode.PUBLIC);
|
|
||||||
ClassNode classNode = new ClassNode(accessTypeNode, "testClass");
|
|
||||||
|
|
||||||
MemberNode memberNode1 = new FieldNode(accessTypeNode, new BaseTypeNode(EnumTypeNode.INT), "testVar");
|
|
||||||
classNode.members.add(memberNode1);
|
|
||||||
|
|
||||||
MemberNode memberNode2 = new FieldNode(accessTypeNode, new BaseTypeNode(EnumTypeNode.INT), "testVar");
|
|
||||||
classNode.members.add(memberNode2);
|
|
||||||
|
|
||||||
classList.add(classNode);
|
|
||||||
programNode.classes = classList;
|
|
||||||
|
|
||||||
ASTNode typedAst = SemanticAnalyzer.generateTast(programNode);
|
|
||||||
|
|
||||||
assertEquals(1, SemanticAnalyzer.errors.size());
|
|
||||||
assertEquals(true, SemanticAnalyzer.errors.get(0) instanceof AlreadyDeclearedException);
|
|
||||||
assertEquals(null, typedAst);
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
@Test
|
|
||||||
public void shouldWorkWithNoError(){
|
|
||||||
|
|
||||||
ProgramNode programNode = new ProgramNode();
|
|
||||||
List<ClassNode> classList = new ArrayList<ClassNode>();
|
|
||||||
AccessTypeNode accessTypeNode = new AccessTypeNode(EnumAccessTypeNode.PUBLIC);
|
|
||||||
ClassNode classNode = new ClassNode(accessTypeNode, "testClass");
|
|
||||||
|
|
||||||
MemberNode memberNode1 = new FieldNode(accessTypeNode, new BaseTypeNode(EnumTypeNode.INT), "testVar1");
|
|
||||||
classNode.members.add(memberNode1);
|
|
||||||
|
|
||||||
MemberNode memberNode2 = new FieldNode(accessTypeNode, new BaseTypeNode(EnumTypeNode.INT), "testVar2");
|
|
||||||
classNode.members.add(memberNode2);
|
|
||||||
|
|
||||||
List<ParameterNode> parameterNodeList = new ArrayList<ParameterNode>();
|
|
||||||
ParameterNode parameterNode1 = new ParameterNode(new BaseTypeNode(EnumTypeNode.INT), "param1");
|
|
||||||
parameterNodeList.add(parameterNode1);
|
|
||||||
ParameterListNode parameterListNode = new ParameterListNode(parameterNodeList);
|
|
||||||
|
|
||||||
List<StatementNode> statementNodeList = new ArrayList<StatementNode>();
|
|
||||||
|
|
||||||
ExpressionNode expressionNodeObjectVariableLeft = new IdentifierExpressionNode("this");
|
|
||||||
ExpressionNode expressionNodeObjectVariableRight = new IdentifierExpressionNode("objectVar");
|
|
||||||
|
|
||||||
ExpressionNode expressionNodeLeft = new BinaryExpressionNode(expressionNodeObjectVariableLeft, expressionNodeObjectVariableRight, ExpresssionOperator.DOT);
|
|
||||||
|
|
||||||
ExpressionNode expressionNodeRight = new LiteralNode(1);
|
|
||||||
|
|
||||||
BinaryExpressionNode expressionNode = new BinaryExpressionNode(expressionNodeLeft, expressionNodeRight, ExpresssionOperator.ASSIGNMENT);
|
|
||||||
|
|
||||||
StatementNode statementNode1 = new AssignmentStatementNode(expressionNode);
|
|
||||||
statementNodeList.add(statementNode1);
|
|
||||||
|
|
||||||
MemberNode memberNode3 = new MethodNode(accessTypeNode, new BaseTypeNode(EnumTypeNode.INT), "testVar2",parameterListNode, statementNodeList );
|
|
||||||
classNode.members.add(memberNode3);
|
|
||||||
|
|
||||||
classList.add(classNode);
|
|
||||||
programNode.classes = classList;
|
|
||||||
|
|
||||||
ASTNode typedAst = SemanticAnalyzer.generateTast(programNode);
|
|
||||||
|
|
||||||
assertEquals(0, SemanticAnalyzer.errors.size());
|
|
||||||
assertEquals(programNode, typedAst);
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
Loading…
x
Reference in New Issue
Block a user