use of org.ballerinalang.plugins.idea.psi.StructDefinitionNode in project ballerina by ballerina-lang.
the class FieldReference method resolve.
@Nullable
@Override
public PsiElement resolve() {
// Get the current element.
IdentifierPSINode identifier = getElement();
// Get the parent element.
PsiElement parent = identifier.getParent();
PsiElement prevSibling;
// resolve multi level structs. Eg: user.name.<caret>
if (parent instanceof StatementNode) {
// Get the previous element.
PsiElement prevVisibleLeaf = PsiTreeUtil.prevVisibleLeaf(parent);
if (prevVisibleLeaf == null) {
return null;
}
// Previous leaf element should be "." if the references are correctly defined.
if (!".".equals(prevVisibleLeaf.getText())) {
return null;
}
// Get the prevSibling. This is used to resolve the current field.
prevSibling = PsiTreeUtil.prevVisibleLeaf(prevVisibleLeaf);
if (prevSibling == null) {
return null;
}
} else if (parent instanceof NameReferenceNode) {
prevSibling = PsiTreeUtil.prevVisibleLeaf(parent);
if (prevSibling != null && ".".equals(prevSibling.getText())) {
prevSibling = PsiTreeUtil.prevVisibleLeaf(prevSibling);
}
} else {
PsiElement parentPrevSibling = parent.getPrevSibling();
if (parentPrevSibling != null) {
if (parentPrevSibling instanceof VariableReferenceNode) {
PsiElement[] children = parentPrevSibling.getChildren();
if (children.length <= 0) {
return null;
}
PsiElement firstChild = children[0].getFirstChild();
if (firstChild == null) {
return null;
}
prevSibling = firstChild;
} else {
return null;
}
} else {
// If the current statement is correctly resolved, that means all the fields are identified properly.
// Get the prevSibling. This is used to resolve the current field.
prevSibling = PsiTreeUtil.prevVisibleLeaf(parent);
}
}
// If the prevSibling is null, we return from this method because we cannot resolve the element.
if (prevSibling == null) {
return null;
}
// We get the reference at end. This is because struct field access can be multiple levels deep.
// Eg: user.name.first - If the current node is 'first', then the previous node will be 'user.name'. If
// we get the reference at the beginning, we we get the reference for 'user'. But we want to resolve the
// 'name' field first. That is why we get the reference at the end.
PsiReference variableReference = prevSibling.findReferenceAt(prevSibling.getTextLength());
if (variableReference == null) {
return null;
}
// Resolve the reference. The resolved element can be an identifier of either a struct of a field
// depending on the current node.
// Eg: user.name.firstName - If the current node is 'name', resolved element will be a struct definition. if
// the current element is 'firstName', then the resolved element will be a field definition.
PsiElement resolvedElement = variableReference.resolve();
if (resolvedElement == null || !(resolvedElement instanceof IdentifierPSINode)) {
return null;
}
// Get the parent of the resolved element.
PsiElement resolvedElementParent = resolvedElement.getParent();
StructDefinitionNode structDefinitionNode = null;
// Resolve the corresponding resolvedElementParent to get the struct definition.
if (resolvedElementParent instanceof VariableDefinitionNode || resolvedElementParent instanceof CodeBlockParameterNode || resolvedElementParent instanceof ParameterNode) {
// Resolve the Type of the VariableDefinitionNode to get the corresponding struct.
// Eg: User user = {}
// In here, "User" is resolved and struct identifier is returned.
structDefinitionNode = BallerinaPsiImplUtil.resolveStructFromDefinitionNode(resolvedElementParent);
} else if (resolvedElementParent instanceof FieldDefinitionNode) {
// If the resolvedElementParent is of type FieldDefinitionNode, that means we need to resolve the type of
// the field to get the struct definition.
// Eg: user.name.firstName - In here, if we want to resolve the 'firstName' we will get the 'Name name;'
// field. So we need to resolve the type of the field which is 'Name'. Then we will get the Name struct.
// Then we need to get the 'firstName' field from that.
structDefinitionNode = BallerinaPsiImplUtil.resolveTypeNodeStruct((resolvedElementParent));
} else if (resolvedElementParent instanceof NameReferenceNode) {
structDefinitionNode = BallerinaPsiImplUtil.findStructDefinition((IdentifierPSINode) resolvedElement);
} else if (resolvedElementParent instanceof EnumDefinitionNode) {
return ((EnumDefinitionNode) resolvedElementParent).resolve(identifier);
}
if (structDefinitionNode == null) {
return null;
}
// Resolve the field and return the resolved element.
return structDefinitionNode.resolve(identifier);
}
use of org.ballerinalang.plugins.idea.psi.StructDefinitionNode in project ballerina by ballerina-lang.
the class BallerinaStructureViewElement method getChildren.
@NotNull
@Override
public TreeElement[] getChildren() {
// The element can be one of BallerinaFile, ConnectorDefinitionNode instance.
if (element instanceof BallerinaFile) {
List<TreeElement> treeElements = new ArrayList<>();
// Add services.
Collection<ServiceDefinitionNode> services = PsiTreeUtil.findChildrenOfType(element, ServiceDefinitionNode.class);
for (PsiElement service : services) {
// In here, instead of using the service, we use service.getParent(). This is done because we
// want to show resources under a service node. This is how the sub nodes can be added.
treeElements.add(new BallerinaStructureViewElement(service));
}
// Add functions.
Collection<FunctionDefinitionNode> functions = PsiTreeUtil.findChildrenOfType(element, FunctionDefinitionNode.class);
for (PsiElement function : functions) {
treeElements.add(new BallerinaStructureViewElement(function));
}
// Add connectors.
Collection<ConnectorDefinitionNode> connectors = PsiTreeUtil.findChildrenOfType(element, ConnectorDefinitionNode.class);
for (PsiElement connector : connectors) {
// In here, instead of using the connector, we use connector.getParent(). This is done because we
// want to show actions under a connector node. This is how the sub nodes can be added.
treeElements.add(new BallerinaStructureViewElement(connector));
}
// Add annotations.
Collection<AnnotationDefinitionNode> annotations = PsiTreeUtil.findChildrenOfType(element, AnnotationDefinitionNode.class);
for (PsiElement annotation : annotations) {
treeElements.add(new BallerinaStructureViewElement(annotation));
}
// Add structs
Collection<StructDefinitionNode> structs = PsiTreeUtil.findChildrenOfType(element, StructDefinitionNode.class);
for (PsiElement struct : structs) {
treeElements.add(new BallerinaStructureViewElement(struct));
}
// Convert the list to an array and return.
return treeElements.toArray(new TreeElement[treeElements.size()]);
} else if (element instanceof ConnectorDefinitionNode) {
// If the element is a ConnectorDefinitionNode instance, we get all actions.
List<TreeElement> treeElements = new ArrayList<>();
// Add actions.
Collection<ActionDefinitionNode> actions = PsiTreeUtil.findChildrenOfType(element, ActionDefinitionNode.class);
for (PsiElement action : actions) {
treeElements.add(new BallerinaStructureViewElement(action));
}
// Convert the list to an array and return.
return treeElements.toArray(new TreeElement[treeElements.size()]);
} else if (element instanceof ServiceDefinitionNode) {
// If the element is a ServiceDefinitionNode instance, we get all resources.
List<TreeElement> treeElements = new ArrayList<>();
// Add resources.
Collection<ResourceDefinitionNode> resources = PsiTreeUtil.findChildrenOfType(element, ResourceDefinitionNode.class);
for (PsiElement resource : resources) {
treeElements.add(new BallerinaStructureViewElement(resource));
}
// Convert the list to an array and return.
return treeElements.toArray(new TreeElement[treeElements.size()]);
} else if (element instanceof FunctionDefinitionNode || element instanceof ResourceDefinitionNode || element instanceof ActionDefinitionNode) {
// If the element is a FunctionDefinitionNode instance, we get all workers.
List<TreeElement> treeElements = new ArrayList<>();
// Add workers.
Collection<WorkerDeclarationNode> workers = PsiTreeUtil.findChildrenOfType(element, WorkerDeclarationNode.class);
for (PsiElement worker : workers) {
treeElements.add(new BallerinaStructureViewElement(worker));
}
// Convert the list to an array and return.
return treeElements.toArray(new TreeElement[treeElements.size()]);
}
// If the element type other than what we check above, return an empty array.
return new TreeElement[0];
}
use of org.ballerinalang.plugins.idea.psi.StructDefinitionNode in project ballerina by ballerina-lang.
the class BallerinaParserDefinition method createElement.
@NotNull
public PsiElement createElement(ASTNode node) {
IElementType elementType = node.getElementType();
if (elementType instanceof TokenIElementType) {
return new ANTLRPsiNode(node);
}
if (!(elementType instanceof RuleIElementType)) {
return new ANTLRPsiNode(node);
}
RuleIElementType ruleElType = (RuleIElementType) elementType;
switch(ruleElType.getRuleIndex()) {
case BallerinaParser.RULE_functionDefinition:
return new FunctionDefinitionNode(node);
case BallerinaParser.RULE_callableUnitBody:
return new CallableUnitBodyNode(node);
case BallerinaParser.RULE_nameReference:
return new NameReferenceNode(node);
case BallerinaParser.RULE_variableReference:
return new VariableReferenceNode(node);
case BallerinaParser.RULE_variableDefinitionStatement:
return new VariableDefinitionNode(node);
case BallerinaParser.RULE_parameter:
return new ParameterNode(node);
case BallerinaParser.RULE_resourceDefinition:
return new ResourceDefinitionNode(node);
case BallerinaParser.RULE_packageName:
return new PackageNameNode(node);
case BallerinaParser.RULE_fullyQualifiedPackageName:
return new FullyQualifiedPackageNameNode(node);
case BallerinaParser.RULE_expressionList:
return new ExpressionListNode(node);
case BallerinaParser.RULE_expression:
return new ExpressionNode(node);
case BallerinaParser.RULE_functionInvocation:
return new FunctionInvocationNode(node);
case BallerinaParser.RULE_compilationUnit:
return new CompilationUnitNode(node);
case BallerinaParser.RULE_packageDeclaration:
return new PackageDeclarationNode(node);
case BallerinaParser.RULE_annotationAttachment:
return new AnnotationAttachmentNode(node);
case BallerinaParser.RULE_serviceBody:
return new ServiceBodyNode(node);
case BallerinaParser.RULE_importDeclaration:
return new ImportDeclarationNode(node);
case BallerinaParser.RULE_statement:
return new StatementNode(node);
case BallerinaParser.RULE_typeName:
return new TypeNameNode(node);
case BallerinaParser.RULE_constantDefinition:
return new ConstantDefinitionNode(node);
case BallerinaParser.RULE_structDefinition:
return new StructDefinitionNode(node);
case BallerinaParser.RULE_simpleLiteral:
return new SimpleLiteralNode(node);
case BallerinaParser.RULE_packageAlias:
return new AliasNode(node);
case BallerinaParser.RULE_parameterList:
return new ParameterListNode(node);
case BallerinaParser.RULE_fieldDefinition:
return new FieldDefinitionNode(node);
case BallerinaParser.RULE_parameterTypeNameList:
return new ParameterTypeNameList(node);
case BallerinaParser.RULE_parameterTypeName:
return new ConnectorInitNode(node);
case BallerinaParser.RULE_serviceDefinition:
return new ServiceDefinitionNode(node);
case BallerinaParser.RULE_valueTypeName:
return new ValueTypeNameNode(node);
case BallerinaParser.RULE_annotationDefinition:
return new AnnotationAttributeValueNode(node);
case BallerinaParser.RULE_structBody:
return new StructBodyNode(node);
case BallerinaParser.RULE_returnParameter:
return new ReturnParameterNode(node);
case BallerinaParser.RULE_attachmentPoint:
return new AttachmentPointNode(node);
case BallerinaParser.RULE_definition:
return new DefinitionNode(node);
case BallerinaParser.RULE_ifElseStatement:
return new IfElseStatementNode(node);
case BallerinaParser.RULE_assignmentStatement:
return new AssignmentStatementNode(node);
case BallerinaParser.RULE_variableReferenceList:
return new VariableReferenceListNode(node);
case BallerinaParser.RULE_recordLiteral:
return new RecordLiteralNode(node);
case BallerinaParser.RULE_globalVariableDefinition:
return new GlobalVariableDefinitionNode(node);
case BallerinaParser.RULE_recordKeyValue:
return new RecordKeyValueNode(node);
case BallerinaParser.RULE_forkJoinStatement:
return new ForkJoinStatementNode(node);
case BallerinaParser.RULE_returnStatement:
return new ReturnStatementNode(node);
case BallerinaParser.RULE_throwStatement:
return new ThrowStatementNode(node);
case BallerinaParser.RULE_transformerDefinition:
return new TransformerDefinitionNode(node);
case BallerinaParser.RULE_workerReply:
return new WorkerReplyNode(node);
case BallerinaParser.RULE_triggerWorker:
return new TriggerWorkerNode(node);
case BallerinaParser.RULE_workerDeclaration:
return new WorkerDeclarationNode(node);
case BallerinaParser.RULE_workerBody:
return new WorkerBodyNode(node);
case BallerinaParser.RULE_joinConditions:
return new JoinConditionNode(node);
case BallerinaParser.RULE_field:
return new FieldNode(node);
case BallerinaParser.RULE_recordKey:
return new RecordKeyNode(node);
case BallerinaParser.RULE_recordValue:
return new RecordValueNode(node);
case BallerinaParser.RULE_functionReference:
return new FunctionReferenceNode(node);
case BallerinaParser.RULE_codeBlockBody:
return new CodeBlockBodyNode(node);
case BallerinaParser.RULE_nonEmptyCodeBlockBody:
return new NonEmptyCodeBlockBodyNode(node);
case BallerinaParser.RULE_tryCatchStatement:
return new TryCatchStatementNode(node);
case BallerinaParser.RULE_catchClause:
return new CatchClauseNode(node);
case BallerinaParser.RULE_codeBlockParameter:
return new CodeBlockParameterNode(node);
case BallerinaParser.RULE_foreachStatement:
return new ForEachStatementNode(node);
case BallerinaParser.RULE_joinClause:
return new JoinClauseNode(node);
case BallerinaParser.RULE_timeoutClause:
return new TimeoutClauseNode(node);
case BallerinaParser.RULE_xmlAttrib:
return new XmlAttribNode(node);
case BallerinaParser.RULE_namespaceDeclaration:
return new NamespaceDeclarationNode(node);
case BallerinaParser.RULE_stringTemplateLiteral:
return new StringTemplateLiteralNode(node);
case BallerinaParser.RULE_stringTemplateContent:
return new StringTemplateContentNode(node);
case BallerinaParser.RULE_typeConversion:
return new TypeConversionNode(node);
case BallerinaParser.RULE_xmlContent:
return new XmlContentNode(node);
case BallerinaParser.RULE_invocation:
return new InvocationNode(node);
case BallerinaParser.RULE_enumDefinition:
return new EnumDefinitionNode(node);
case BallerinaParser.RULE_enumField:
return new EnumFieldNode(node);
case BallerinaParser.RULE_builtInReferenceTypeName:
return new BuiltInReferenceTypeNameNode(node);
case BallerinaParser.RULE_referenceTypeName:
return new ReferenceTypeNameNode(node);
case BallerinaParser.RULE_functionTypeName:
return new FunctionTypeNameNode(node);
case BallerinaParser.RULE_lambdaFunction:
return new LambdaFunctionNode(node);
case BallerinaParser.RULE_endpointDeclaration:
return new EndpointDeclarationNode(node);
case BallerinaParser.RULE_anonStructTypeName:
return new AnonStructTypeNameNode(node);
case BallerinaParser.RULE_userDefineTypeName:
return new UserDefinedTypeName(node);
case BallerinaParser.RULE_anyIdentifierName:
return new AnyIdentifierNameNode(node);
case BallerinaParser.RULE_documentationAttachment:
return new DocumentationAttachmentNode(node);
case BallerinaParser.RULE_documentationTemplateInlineCode:
return new DocumentationTemplateInlineCodeNode(node);
case BallerinaParser.RULE_singleBackTickDocInlineCode:
return new SingleBackTickDocInlineCodeNode(node);
case BallerinaParser.RULE_doubleBackTickDocInlineCode:
return new DoubleBackTickInlineCodeNode(node);
case BallerinaParser.RULE_tripleBackTickDocInlineCode:
return new TripleBackTickInlineCodeNode(node);
case BallerinaParser.RULE_deprecatedAttachment:
return new DeprecatedAttachmentNode(node);
case BallerinaParser.RULE_deprecatedText:
return new DeprecatedTextNode(node);
case BallerinaParser.RULE_singleBackTickDeprecatedInlineCode:
return new SingleBackTickDeprecatedInlineCodeNode(node);
case BallerinaParser.RULE_doubleBackTickDeprecatedInlineCode:
return new DoubleBackTickDeprecatedInlineCodeNode(node);
case BallerinaParser.RULE_tripleBackTickDeprecatedInlineCode:
return new TripleBackTickDeprecatedInlineCodeNode(node);
case BallerinaParser.RULE_documentationTemplateAttributeDescription:
return new DocumentationTemplateAttributeDescriptionNode(node);
case BallerinaParser.RULE_formalParameterList:
return new FormalParameterListNode(node);
case BallerinaParser.RULE_integerLiteral:
return new IntegerLiteralNode(node);
case BallerinaParser.RULE_matchPatternClause:
return new MatchPatternClauseNode(node);
default:
return new ANTLRPsiNode(node);
}
}
use of org.ballerinalang.plugins.idea.psi.StructDefinitionNode in project ballerina by ballerina-lang.
the class BallerinaPsiImplUtil method getErrorStruct.
/**
* Returns corresponding error struct for type casts and type conversions.
*
* @param assignmentStatementNode {@link AssignmentStatementNode} which we are currently checking
* @param referenceNode resolved variable reference node
* @param isTypeCast indicates whether the {@link AssignmentStatementNode} contains a type cast
* @param isTypeConversion indicates whether the {@link AssignmentStatementNode} contains a type conversion
* @return {@code TypeCastError} if the provided assignment statement is a type cast operation.
* {@code TypeConversionError} if the provided assignment statement is a type conversion operation.
*/
@Nullable
public static StructDefinitionNode getErrorStruct(@NotNull AssignmentStatementNode assignmentStatementNode, @NotNull IdentifierPSINode referenceNode, boolean isTypeCast, boolean isTypeConversion) {
List<IdentifierPSINode> variablesFromVarAssignment = BallerinaPsiImplUtil.getVariablesFromVarAssignment(assignmentStatementNode);
// might be unnecessary and identified at the semantic analyzer.
if (variablesFromVarAssignment.size() < 2) {
return null;
}
// Get the second value from the assignment statement.
IdentifierPSINode errorVariable = variablesFromVarAssignment.get(1);
// This should be the provided reference node. Since the error is returned as the second variable.
if (!errorVariable.equals(referenceNode)) {
return null;
}
// Now we need to find the error.bal file which contains the definition of these error structs.
Project project = referenceNode.getProject();
VirtualFile errorFile = BallerinaPsiImplUtil.findFileInSDK(project, referenceNode, "/ballerina/builtin/core/source.bal");
if (errorFile == null) {
return null;
}
// Find the file.
PsiFile psiFile = PsiManager.getInstance(project).findFile(errorFile);
if (psiFile == null) {
return null;
}
// Get all struct definitions in the file.
Collection<StructDefinitionNode> structDefinitionNodes = PsiTreeUtil.findChildrenOfType(psiFile, StructDefinitionNode.class);
// Iterate through all struct definitions.
for (StructDefinitionNode definitionNode : structDefinitionNodes) {
IdentifierPSINode nameNode = PsiTreeUtil.getChildOfType(definitionNode, IdentifierPSINode.class);
if (nameNode != null) {
// Check and return the matching definition.
if (isTypeCast && "TypeCastError".equals(nameNode.getText())) {
return definitionNode;
}
if (isTypeConversion && "TypeConversionError".equals(nameNode.getText())) {
return definitionNode;
}
}
}
return null;
}
use of org.ballerinalang.plugins.idea.psi.StructDefinitionNode in project ballerina by ballerina-lang.
the class BallerinaPsiImplUtil method getAttachedStruct.
@Nullable
public static StructDefinitionNode getAttachedStruct(@NotNull IdentifierPSINode function) {
CodeBlockParameterNode codeBlockParameterNode = PsiTreeUtil.getChildOfType(function.getParent(), CodeBlockParameterNode.class);
if (codeBlockParameterNode == null) {
return null;
}
TypeNameNode typeNameNode = PsiTreeUtil.findChildOfType(codeBlockParameterNode, TypeNameNode.class);
if (typeNameNode == null) {
return null;
}
PsiReference reference = typeNameNode.findReferenceAt(typeNameNode.getTextLength());
if (reference == null) {
return null;
}
PsiElement resolvedElement = reference.resolve();
if (resolvedElement == null) {
return null;
}
PsiElement parent = resolvedElement.getParent();
if (!(parent instanceof StructDefinitionNode)) {
return null;
}
return (StructDefinitionNode) parent;
}
Aggregations