Skip to content

Commit

Permalink
[INLONG-10867][SDK] Transform support BETWEEN AND operator (#10871)
Browse files Browse the repository at this point in the history
  • Loading branch information
ying-hua authored Sep 9, 2024
1 parent 927e40a commit fa8f4c8
Show file tree
Hide file tree
Showing 2 changed files with 156 additions and 0 deletions.
Original file line number Diff line number Diff line change
@@ -0,0 +1,54 @@
/*
* Licensed to the Apache Software Foundation (ASF) under one or more
* contributor license agreements. See the NOTICE file distributed with
* this work for additional information regarding copyright ownership.
* The ASF 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 org.apache.inlong.sdk.transform.process.operator;

import org.apache.inlong.sdk.transform.decode.SourceData;
import org.apache.inlong.sdk.transform.process.Context;
import org.apache.inlong.sdk.transform.process.parser.ValueParser;

import net.sf.jsqlparser.expression.operators.relational.Between;
/**
* BetweenAndOperator
*
* This class implements the ExpressionOperator interface to handle the SQL BETWEEN AND operator.
* It checks if a left value is between start and end value.
*/
@TransformOperator(values = Between.class)
public class BetweenAndOperator implements ExpressionOperator {

private final ValueParser left;
private final ValueParser start;
private final ValueParser end;

public BetweenAndOperator(Between expr) {
this.left = OperatorTools.buildParser(expr.getLeftExpression());
this.start = OperatorTools.buildParser(expr.getBetweenExpressionStart());
this.end = OperatorTools.buildParser(expr.getBetweenExpressionEnd());
}

@SuppressWarnings("rawtypes")
@Override
public boolean check(SourceData sourceData, int rowIndex, Context context) {
Comparable leftValue = (Comparable) this.left.parse(sourceData, rowIndex, context);
Comparable startValue = (Comparable) this.start.parse(sourceData, rowIndex, context);
Comparable endValue = (Comparable) this.end.parse(sourceData, rowIndex, context);

return OperatorTools.compareValue(leftValue, startValue) >= 0 &&
OperatorTools.compareValue(leftValue, endValue) <= 0;
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,102 @@
/*
* Licensed to the Apache Software Foundation (ASF) under one or more
* contributor license agreements. See the NOTICE file distributed with
* this work for additional information regarding copyright ownership.
* The ASF 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 org.apache.inlong.sdk.transform.process;

import org.apache.inlong.sdk.transform.decode.SourceDecoderFactory;
import org.apache.inlong.sdk.transform.encode.SinkEncoderFactory;
import org.apache.inlong.sdk.transform.pojo.CsvSourceInfo;
import org.apache.inlong.sdk.transform.pojo.FieldInfo;
import org.apache.inlong.sdk.transform.pojo.KvSinkInfo;
import org.apache.inlong.sdk.transform.pojo.TransformConfig;
import org.apache.inlong.sdk.transform.process.converter.DoubleConverter;
import org.apache.inlong.sdk.transform.process.converter.LongConverter;
import org.apache.inlong.sdk.transform.process.converter.TypeConverter;

import org.junit.Assert;
import org.junit.Test;

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

public class TestBetweenAndOperator {

private static final List<FieldInfo> srcFields = new ArrayList<>();
private static final List<FieldInfo> dstFields = new ArrayList<>();
private static final CsvSourceInfo csvSource;
private static final KvSinkInfo kvSink;

static {
srcFields.add(new FieldInfo("numeric1", new DoubleConverter()));
srcFields.add(new FieldInfo("string2", TypeConverter.DefaultTypeConverter()));
srcFields.add(new FieldInfo("numeric3", new DoubleConverter()));
srcFields.add(new FieldInfo("numeric4", new LongConverter()));

FieldInfo field = new FieldInfo();
field.setName("result");
dstFields.add(field);
csvSource = new CsvSourceInfo("UTF-8", '|', '\\', srcFields);
kvSink = new KvSinkInfo("UTF-8", dstFields);
}

@Test
public void testBetweenAndOperator() throws Exception {
String transformSql = "select if(string2 between 3 and 5,1,0) from source";
TransformConfig config = new TransformConfig(transformSql);
// case1: '3a' between '3' and '5' -> 1
TransformProcessor<String, String> processor = TransformProcessor
.create(config, SourceDecoderFactory.createCsvDecoder(csvSource),
SinkEncoderFactory.createKvEncoder(kvSink));
List<String> output1 = processor.transform("3.14159265358979323846|3a|4|8");
Assert.assertEquals(1, output1.size());
Assert.assertEquals(output1.get(0), "result=1");
// case2: '4a' between '3' and '5' -> 1
List<String> output2 = processor.transform("3.14159265358979323846|4a|4|8");
Assert.assertEquals(1, output1.size());
Assert.assertEquals(output2.get(0), "result=1");
// case3: '6' between '3' and '5' -> 0
List<String> output3 = processor.transform("3.14159265358979323846|6|4|8");
Assert.assertEquals(1, output1.size());
Assert.assertEquals(output3.get(0), "result=0");
// case4: '3e2' between '3' and '5' -> 0
List<String> output4 = processor.transform("3.14159265358979323846|3e2|4|8");
Assert.assertEquals(1, output1.size());
Assert.assertEquals(output4.get(0), "result=0");

transformSql = "select if(numeric3 between 3 and 5,1,0) from source";
config = new TransformConfig(transformSql);
// case5: 4 between 3 and 5 -> 1
processor = TransformProcessor
.create(config, SourceDecoderFactory.createCsvDecoder(csvSource),
SinkEncoderFactory.createKvEncoder(kvSink));
List<String> output5 = processor.transform("3.14159265358979323846|4|4|8");
Assert.assertEquals(1, output1.size());
Assert.assertEquals(output5.get(0), "result=1");
// case6: 3 between 3 and 5 -> 1
List<String> output6 = processor.transform("3.14159265358979323846|4|3|8");
Assert.assertEquals(1, output1.size());
Assert.assertEquals(output6.get(0), "result=1");
// case7: 5 between 3 and 5 -> 1
List<String> output7 = processor.transform("3.14159265358979323846|4|5|8");
Assert.assertEquals(1, output1.size());
Assert.assertEquals(output7.get(0), "result=1");
// case8: 3e2 between 3 and 5 -> 0
List<String> output8 = processor.transform("3.14159265358979323846|4|3e2|8");
Assert.assertEquals(1, output1.size());
Assert.assertEquals(output8.get(0), "result=0");
}
}

0 comments on commit fa8f4c8

Please sign in to comment.