Skip to content

Commit

Permalink
fix(interactive): Fix Bugs of Special Characters as Label Name (#4366)
Browse files Browse the repository at this point in the history
allow special characters and reserved words as the label names, use
back-ticks to escape, i.e.:

```
Match (n:`@person`) Return n;

# contains is a reserved word for operator `contains`
Match (n1)-[m:`contains`]->(n2) Return m;
```

Fixes #4347 
Fixes #4317
  • Loading branch information
shirly121 authored Dec 25, 2024
1 parent 936371c commit c38dd7d
Show file tree
Hide file tree
Showing 7 changed files with 95 additions and 8 deletions.
Original file line number Diff line number Diff line change
Expand Up @@ -279,7 +279,7 @@ public ExprVisitorResult visitOC_ParenthesizedExpression(

@Override
public ExprVisitorResult visitOC_Variable(CypherGSParser.OC_VariableContext ctx) {
String aliasName = ctx.getText();
String aliasName = Utils.getAliasName(ctx);
return new ExprVisitorResult(builder.variable(aliasName));
}

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -100,7 +100,7 @@ public GraphBuilder visitOC_Cypher(CypherGSParser.OC_CypherContext ctx) {
@Override
public GraphBuilder visitOC_Unwind(CypherGSParser.OC_UnwindContext ctx) {
RexNode expr = expressionVisitor.visitOC_Expression(ctx.oC_Expression()).getExpr();
String alias = ctx.oC_Variable() == null ? null : ctx.oC_Variable().getText();
String alias = Utils.getAliasName(ctx.oC_Variable());
return builder.unfold(expr, alias);
}

Expand Down Expand Up @@ -339,7 +339,8 @@ private boolean isGroupPattern(
for (CypherGSParser.OC_ProjectionItemContext itemCtx :
ctx.oC_ProjectionItems().oC_ProjectionItem()) {
ExprVisitorResult item = expressionVisitor.visitOC_Expression(itemCtx.oC_Expression());
String alias = (itemCtx.AS() == null) ? null : itemCtx.oC_Variable().getText();
String alias =
(itemCtx.AS() == null) ? null : Utils.getAliasName(itemCtx.oC_Variable());
if (item.getAggCalls().isEmpty()) {
keyExprs.add(item.getExpr());
keyAliases.add(alias);
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -100,6 +100,15 @@ public Object visitTerminal(TerminalNode node) {
}
}

@Override
public Object visitOC_SymbolicName(CypherGSParser.OC_SymbolicNameContext ctx) {
if (ctx.EscapedSymbolicName() != null) {
// escape the back-ticks
return ctx.getText().replace("`", "");
}
return ctx.getText();
}

/**
* Remove single/double quotes around String literal
*
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -24,20 +24,21 @@
import org.apache.calcite.rex.RexCall;
import org.apache.calcite.rex.RexNode;
import org.apache.calcite.sql.SqlKind;
import org.checkerframework.checker.nullness.qual.Nullable;

import java.util.ArrayList;
import java.util.List;

public abstract class Utils extends com.alibaba.graphscope.common.antlr4.Utils {
public static SourceConfig sourceConfig(CypherGSParser.OC_NodePatternContext ctx) {
String alias = (ctx.oC_Variable() != null) ? ctx.oC_Variable().getText() : null;
String alias = getAliasName(ctx.oC_Variable());
LabelConfig config = labelConfig(ctx.oC_NodeLabels());
// source
return new SourceConfig(GraphOpt.Source.VERTEX, config, alias);
}

public static GetVConfig getVConfig(CypherGSParser.OC_NodePatternContext ctx) {
String alias = (ctx.oC_Variable() != null) ? ctx.oC_Variable().getText() : null;
String alias = getAliasName(ctx.oC_Variable());
LabelConfig config = labelConfig(ctx.oC_NodeLabels());
// getV
return new GetVConfig(getVOpt(ctx), config, alias);
Expand All @@ -50,7 +51,7 @@ public static ExpandConfig expandConfig(CypherGSParser.OC_RelationshipPatternCon

public static ExpandConfig expandConfig(
CypherGSParser.OC_RelationshipDetailContext ctx, GraphOpt.Expand opt) {
String alias = (ctx.oC_Variable() != null) ? ctx.oC_Variable().getText() : null;
String alias = getAliasName(ctx.oC_Variable());
LabelConfig config = labelConfig(ctx.oC_RelationshipTypes());
return new ExpandConfig(opt, config, alias);
}
Expand All @@ -63,7 +64,7 @@ public static LabelConfig labelConfig(CypherGSParser.OC_NodeLabelsContext ctx) {
config = new LabelConfig(false);
for (CypherGSParser.OC_LabelNameContext ctx1 : ctx.oC_LabelName()) {
if (ctx1 == null) continue;
config.addLabel(ctx1.getText());
config.addLabel(getLabelName(ctx1));
}
}
return config;
Expand All @@ -77,7 +78,7 @@ public static LabelConfig labelConfig(CypherGSParser.OC_RelationshipTypesContext
config = new LabelConfig(false);
for (CypherGSParser.OC_RelTypeNameContext ctx1 : ctx.oC_RelTypeName()) {
if (ctx1 == null) continue;
config.addLabel(ctx1.getText());
config.addLabel(getLabelName(ctx1));
}
}
return config;
Expand Down Expand Up @@ -133,4 +134,16 @@ public static final List<RexNode> propertyFilters(
}
return filters;
}

public static @Nullable String getAliasName(CypherGSParser.OC_VariableContext ctx) {
return ctx == null ? null : (String) LiteralVisitor.INSTANCE.visit(ctx);
}

public static String getLabelName(CypherGSParser.OC_LabelNameContext ctx) {
return (String) LiteralVisitor.INSTANCE.visit(ctx);
}

public static String getLabelName(CypherGSParser.OC_RelTypeNameContext ctx) {
return (String) LiteralVisitor.INSTANCE.visit(ctx);
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -713,4 +713,22 @@ public void optional_shortest_path_test() {
+ " alias=[p1], opt=[VERTEX], uniqueKeyFilters=[=(_.id, ?0)])",
after.explain().trim());
}

@Test
public void special_label_name_test() {
GraphBuilder builder =
com.alibaba.graphscope.common.ir.Utils.mockGraphBuilder(optimizer, irMeta);
RelNode node =
Utils.eval("Match (n:`@person`)-[e:`contains`]->(n2) Return n", builder).build();
RelNode after = optimizer.optimize(node, new GraphIOProcessor(builder, irMeta));
Assert.assertEquals(
"GraphLogicalProject(n=[n], isAppend=[false])\n"
+ " GraphLogicalGetV(tableConfig=[{isAll=false, tables=[@person]}],"
+ " alias=[n2], opt=[END])\n"
+ " GraphLogicalExpand(tableConfig=[{isAll=false, tables=[contains]}],"
+ " alias=[e], startAlias=[n], opt=[OUT])\n"
+ " GraphLogicalSource(tableConfig=[{isAll=false, tables=[@person]}],"
+ " alias=[n], opt=[VERTEX])",
after.explain().trim());
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -100,6 +100,26 @@ schema:
date32:
primary_keys:
- id
- type_name: "@person"
type_id: 2
x_csr_params:
max_vertex_num: 100
properties:
- property_id: 0
property_name: id
property_type:
primitive_type: DT_SIGNED_INT64
- property_id: 1
property_name: name
property_type:
string:
long_text:
- property_id: 2
property_name: age
property_type:
primitive_type: DT_SIGNED_INT32
primary_keys:
- id
edge_types:
- type_name: knows
type_id: 0
Expand All @@ -123,3 +143,14 @@ schema:
property_name: weight
property_type:
primitive_type: DT_DOUBLE
- type_name: contains
type_id: 2
vertex_type_pair_relations:
- source_vertex: "@person"
destination_vertex: "@person"
relation: MANY_TO_MANY
properties:
- property_id: 0
property_name: weight
property_type:
primitive_type: DT_DOUBLE
Original file line number Diff line number Diff line change
Expand Up @@ -11,6 +11,11 @@
"type_id": 1,
"type_name": "software",
"count": 3
},
{
"type_id": 2,
"type_name": "@person",
"count": 3
}
],
"edge_type_statistics": [
Expand All @@ -35,6 +40,16 @@
"count": 4
}
]
},{
"type_id": 2,
"type_name": "contains",
"vertex_type_pair_statistics": [
{
"source_vertex": "@person",
"destination_vertex": "@person",
"count": 2
}
]
}
]
}

0 comments on commit c38dd7d

Please sign in to comment.