Skip to content

Commit

Permalink
Browse files Browse the repository at this point in the history
…llerina-dev-tools into open-api-support
  • Loading branch information
KavinduZoysa committed Oct 2, 2024
2 parents 228a7ff + e7d03a1 commit 3264e9e
Show file tree
Hide file tree
Showing 72 changed files with 4,133 additions and 1,338 deletions.
Original file line number Diff line number Diff line change
Expand Up @@ -116,10 +116,9 @@ private void setAvailableNodesForIteratingBlock(NonTerminalNode node, SemanticMo
setDefaultNodes();
setStopNode(node);
this.rootBuilder.stepIn(Category.Name.CONTROL)
.stepIn(Category.Name.ITERATION)
.node(NodeKind.BREAK)
.node(NodeKind.CONTINUE)
.stepOut();
.node(NodeKind.BREAK)
.node(NodeKind.CONTINUE)
.stepOut();
}

private void setDefaultNodes() {
Expand All @@ -140,25 +139,21 @@ private void setDefaultNodes() {
.node(function)
.stepOut()
.stepIn(Category.Name.CONTROL)
.stepIn(Category.Name.BRANCH)
.node(NodeKind.IF)
.node(NodeKind.SWITCH)
.stepOut()
.stepIn(Category.Name.ITERATION)
.node(NodeKind.WHILE)
.node(NodeKind.FOREACH)
.stepOut()
.stepIn(Category.Name.TERMINATION)
.node(NodeKind.RETURN)
.stepOut()
.node(NodeKind.IF)
.node(NodeKind.MATCH)
.node(NodeKind.WHILE)
.node(NodeKind.FOREACH)
.node(NodeKind.RETURN)
.stepOut()
.stepIn(Category.Name.DATA)
.node(NodeKind.JSON_PAYLOAD)
.node(NodeKind.XML_PAYLOAD)
.node(NodeKind.BINARY_DATA)
.node(NodeKind.DATA_MAPPER)
.stepOut()
.stepIn(Category.Name.ERROR_HANDLING)
.node(NodeKind.ERROR_HANDLER)
.node(NodeKind.FAIL)
.node(NodeKind.PANIC)
.stepOut()
.stepIn(Category.Name.CONCURRENCY)
Expand All @@ -173,9 +168,8 @@ private void setStopNode(NonTerminalNode node) {
while (parent != null) {
if (isStopNodeAvailable(parent)) {
this.rootBuilder.stepIn(Category.Name.CONTROL)
.stepIn(Category.Name.TERMINATION)
.node(NodeKind.STOP)
.stepOut();
.node(NodeKind.STOP)
.stepOut();
}
parent = parent.parent();
}
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -34,6 +34,7 @@
import io.ballerina.compiler.syntax.tree.AssignmentStatementNode;
import io.ballerina.compiler.syntax.tree.BlockStatementNode;
import io.ballerina.compiler.syntax.tree.BreakStatementNode;
import io.ballerina.compiler.syntax.tree.ByteArrayLiteralNode;
import io.ballerina.compiler.syntax.tree.CheckExpressionNode;
import io.ballerina.compiler.syntax.tree.ClientResourceAccessActionNode;
import io.ballerina.compiler.syntax.tree.CommentNode;
Expand Down Expand Up @@ -94,6 +95,7 @@
import io.ballerina.flowmodelgenerator.core.model.NodeKind;
import io.ballerina.flowmodelgenerator.core.model.Property;
import io.ballerina.flowmodelgenerator.core.model.node.Assign;
import io.ballerina.flowmodelgenerator.core.model.node.BinaryData;
import io.ballerina.flowmodelgenerator.core.model.node.DataMapper;
import io.ballerina.flowmodelgenerator.core.model.node.Fail;
import io.ballerina.flowmodelgenerator.core.model.node.If;
Expand Down Expand Up @@ -396,6 +398,14 @@ public void visit(TemplateExpressionNode templateExpressionNode) {
}
}

@Override
public void visit(ByteArrayLiteralNode byteArrayLiteralNode) {
startNode(NodeKind.BINARY_DATA)
.metadata()
.stepOut()
.properties().expression(byteArrayLiteralNode);
}

@Override
public void visit(VariableDeclarationNode variableDeclarationNode) {
Optional<ExpressionNode> initializer = variableDeclarationNode.initializer();
Expand All @@ -422,6 +432,8 @@ public void visit(VariableDeclarationNode variableDeclarationNode) {
nodeBuilder.properties().payload(variableDeclarationNode.typedBindingPattern(), "xml");
} else if (nodeBuilder instanceof JsonPayload) {
nodeBuilder.properties().payload(variableDeclarationNode.typedBindingPattern(), "json");
} else if (nodeBuilder instanceof BinaryData) {
nodeBuilder.properties().payload(variableDeclarationNode.typedBindingPattern(), "byte[]");
} else {
nodeBuilder.properties().dataVariable(variableDeclarationNode.typedBindingPattern());
}
Expand Down Expand Up @@ -459,7 +471,8 @@ public void visit(AssignmentStatementNode assignmentStatementNode) {
.variable(assignmentStatementNode.varRef());
}

if (nodeBuilder instanceof XmlPayload || nodeBuilder instanceof JsonPayload) {
if (nodeBuilder instanceof XmlPayload || nodeBuilder instanceof JsonPayload
|| nodeBuilder instanceof BinaryData) {
nodeBuilder.properties().variable(assignmentStatementNode.varRef());
}
endNode(assignmentStatementNode);
Expand Down Expand Up @@ -672,7 +685,7 @@ public void visit(RetryStatementNode retryStatementNode) {

@Override
public void visit(MatchStatementNode matchStatementNode) {
startNode(NodeKind.SWITCH)
startNode(NodeKind.MATCH)
.properties().condition(matchStatementNode.condition());

NodeList<MatchClauseNode> matchClauseNodes = matchStatementNode.matchClauses();
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,136 @@
/*
* Copyright (c) 2024, WSO2 LLC. (http://www.wso2.com)
*
* WSO2 LLC. licenses this file to you under the Apache License,
* Version 2.0 (the "License"); you may not use this file except
* in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing,
* software distributed under the License is distributed on an
* "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
* KIND, either express or implied. See the License for the
* specific language governing permissions and limitations
* under the License.
*/

package io.ballerina.flowmodelgenerator.core;

import com.google.gson.Gson;
import com.google.gson.JsonArray;
import io.ballerina.compiler.api.SemanticModel;
import io.ballerina.compiler.api.symbols.Qualifier;
import io.ballerina.compiler.api.symbols.SymbolKind;
import io.ballerina.compiler.api.symbols.VariableSymbol;
import io.ballerina.compiler.syntax.tree.ModulePartNode;
import io.ballerina.compiler.syntax.tree.NonTerminalNode;
import io.ballerina.compiler.syntax.tree.SyntaxKind;
import io.ballerina.projects.Document;
import io.ballerina.tools.text.LinePosition;
import io.ballerina.tools.text.LineRange;
import io.ballerina.tools.text.TextRange;
import org.ballerinalang.diagramutil.connector.models.connector.Type;
import org.ballerinalang.langserver.common.utils.PositionUtil;

import java.util.ArrayList;
import java.util.Arrays;
import java.util.Collections;
import java.util.List;
import java.util.Optional;

/**
* Generates types of the visible variable types for the given cursor position.
*
* @since 1.4.0
*/
public class VisibleVariableTypesGenerator {

private final SemanticModel semanticModel;
private final Document document;
private final LinePosition position;
private final Gson gson;

public VisibleVariableTypesGenerator(SemanticModel semanticModel, Document document, LinePosition position) {
this.semanticModel = semanticModel;
this.document = document;
this.position = position;
this.gson = new Gson();
}

public JsonArray getVisibleVariableTypes() {
Optional<LineRange> functionLineRange = findFunctionLineRange();
List<Category.Variable> moduleVariables = new ArrayList<>();
List<Category.Variable> configurableVariables = new ArrayList<>();
List<Category.Variable> localVariables = new ArrayList<>();
List<Category> categories = Arrays.asList(
new Category(Category.MODULE_CATEGORY, moduleVariables),
new Category(Category.CONFIGURABLE_CATEGORY, configurableVariables),
new Category(Category.LOCAL_CATEGORY, localVariables)
);

semanticModel.visibleSymbols(document, position).stream()
.filter(symbol -> symbol.kind() == SymbolKind.VARIABLE)
.map(symbol -> (VariableSymbol) symbol)
.forEach(variableSymbol -> {
String name = variableSymbol.getName().orElse("");
Type type = Type.fromSemanticSymbol(variableSymbol);

if (variableSymbol.qualifiers().contains(Qualifier.CONFIGURABLE)) {
configurableVariables.add(new Category.Variable(name, type));
} else if (functionLineRange.isPresent() &&
isInFunctionRange(variableSymbol, functionLineRange.get())) {
localVariables.add(new Category.Variable(name, type));
} else {
moduleVariables.add(new Category.Variable(name, type));
}
});

categories.forEach(category -> Collections.sort(category.types()));
return gson.toJsonTree(categories).getAsJsonArray();
}

private boolean isInFunctionRange(VariableSymbol variableSymbol, LineRange functionLineRange) {
return variableSymbol.getLocation().isPresent() &&
PositionUtil.isWithinLineRange(variableSymbol.getLocation().get().lineRange(), functionLineRange);
}

private Optional<LineRange> findFunctionLineRange() {
ModulePartNode rootNode = document.syntaxTree().rootNode();
NonTerminalNode parent =
rootNode.findNode(TextRange.from(document.textDocument().textPositionFrom(position), 0));
while (parent != null && notDefinitionKind(parent.kind())) {
parent = parent.parent();
}
return parent == null ? Optional.empty() : Optional.of(parent.lineRange());
}

private static boolean notDefinitionKind(SyntaxKind kind) {
return kind != SyntaxKind.FUNCTION_DEFINITION && kind != SyntaxKind.RESOURCE_ACCESSOR_DEFINITION;
}

/**
* Represents a category of variables.
*
* @param name the name of the category
* @param types the list of variables in the category
* @since 1.4.0
*/
private record Category(String name, List<Variable> types) {

public static final String MODULE_CATEGORY = "Module Variables";
public static final String CONFIGURABLE_CATEGORY = "Configurable Variables";
public static final String LOCAL_CATEGORY = "Local Variables";

public record Variable(String name, Type type) implements Comparable<Variable> {

@Override
public int compareTo(Variable o) {
return this.name.compareTo(o.name);
}
}

}

}
Original file line number Diff line number Diff line change
Expand Up @@ -36,6 +36,7 @@
import io.ballerina.flowmodelgenerator.core.CommonUtils;
import io.ballerina.flowmodelgenerator.core.model.node.ActionCall;
import io.ballerina.flowmodelgenerator.core.model.node.Assign;
import io.ballerina.flowmodelgenerator.core.model.node.BinaryData;
import io.ballerina.flowmodelgenerator.core.model.node.Break;
import io.ballerina.flowmodelgenerator.core.model.node.Comment;
import io.ballerina.flowmodelgenerator.core.model.node.Continue;
Expand All @@ -49,12 +50,12 @@
import io.ballerina.flowmodelgenerator.core.model.node.If;
import io.ballerina.flowmodelgenerator.core.model.node.JsonPayload;
import io.ballerina.flowmodelgenerator.core.model.node.Lock;
import io.ballerina.flowmodelgenerator.core.model.node.Match;
import io.ballerina.flowmodelgenerator.core.model.node.NewConnection;
import io.ballerina.flowmodelgenerator.core.model.node.Panic;
import io.ballerina.flowmodelgenerator.core.model.node.Return;
import io.ballerina.flowmodelgenerator.core.model.node.Start;
import io.ballerina.flowmodelgenerator.core.model.node.Stop;
import io.ballerina.flowmodelgenerator.core.model.node.Switch;
import io.ballerina.flowmodelgenerator.core.model.node.Transaction;
import io.ballerina.flowmodelgenerator.core.model.node.While;
import io.ballerina.flowmodelgenerator.core.model.node.XmlPayload;
Expand Down Expand Up @@ -126,13 +127,14 @@ public abstract class NodeBuilder {
put(NodeKind.FAIL, Fail::new);
put(NodeKind.XML_PAYLOAD, XmlPayload::new);
put(NodeKind.JSON_PAYLOAD, JsonPayload::new);
put(NodeKind.BINARY_DATA, BinaryData::new);
put(NodeKind.STOP, Stop::new);
put(NodeKind.FUNCTION_CALL, FunctionCall::new);
put(NodeKind.FOREACH, Foreach::new);
put(NodeKind.DATA_MAPPER, DataMapper::new);
put(NodeKind.ASSIGN, Assign::new);
put(NodeKind.COMMENT, Comment::new);
put(NodeKind.SWITCH, Switch::new);
put(NodeKind.MATCH, Match::new);
}};

public static NodeBuilder getNodeFromKind(NodeKind kind) {
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -27,11 +27,12 @@ public enum NodeKind {
ASSIGN,
XML_PAYLOAD,
JSON_PAYLOAD,
BINARY_DATA,
STOP,
FOREACH,
DATA_MAPPER,
COMMENT,
SWITCH,
MATCH,
FUNCTION,

// Branches
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,65 @@
/*
* Copyright (c) 2024, WSO2 LLC. (http://www.wso2.com)
*
* WSO2 LLC. licenses this file to you under the Apache License,
* Version 2.0 (the "License"); you may not use this file except
* in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing,
* software distributed under the License is distributed on an
* "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
* KIND, either express or implied. See the License for the
* specific language governing permissions and limitations
* under the License.
*/

package io.ballerina.flowmodelgenerator.core.model.node;

import io.ballerina.flowmodelgenerator.core.model.NodeBuilder;
import io.ballerina.flowmodelgenerator.core.model.NodeKind;
import io.ballerina.flowmodelgenerator.core.model.Property;
import io.ballerina.flowmodelgenerator.core.model.SourceBuilder;
import org.eclipse.lsp4j.TextEdit;

import java.nio.file.Path;
import java.util.List;
import java.util.Map;
import java.util.Optional;

/**
* Represents the properties of Binary Data node.
*
* @since 1.4.0
*/
public class BinaryData extends NodeBuilder {

public static final String LABEL = "Binary Data";
public static final String DESCRIPTION = LABEL;
public static final String BINARY_DATA_DOC = "Create new Binary Data";
private static final String DUMMY_BINARY_DATA_EXPR = "base64 `abcd`";

@Override
public void setConcreteConstData() {
metadata().label(LABEL).description(DESCRIPTION);
codedata().node(NodeKind.BINARY_DATA);
}

@Override
public void setConcreteTemplateData(TemplateContext context) {
properties().payload(null, "byte[]")
.expression(DUMMY_BINARY_DATA_EXPR, BINARY_DATA_DOC);
}

@Override
public Map<Path, List<TextEdit>> toSource(SourceBuilder sourceBuilder) {
sourceBuilder.newVariable();

Optional<Property> exprProperty = sourceBuilder.flowNode.getProperty(Property.EXPRESSION_KEY);
exprProperty.ifPresent(value -> sourceBuilder.token().expression(value).endOfStatement());

return sourceBuilder.textEdit(false).build();
}
}
Loading

0 comments on commit 3264e9e

Please sign in to comment.