blob: 08766b009be9f22997fccf5acbf7f62a9b6238e3 [file] [log] [blame]
/*
* 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.vxquery.compiler.rewriter.rules;
import java.io.DataInputStream;
import java.io.DataOutput;
import java.io.File;
import java.io.IOException;
import java.util.List;
import org.apache.commons.lang3.mutable.Mutable;
import org.apache.hyracks.algebricks.common.exceptions.AlgebricksException;
import org.apache.hyracks.algebricks.core.algebra.base.ILogicalExpression;
import org.apache.hyracks.algebricks.core.algebra.base.ILogicalOperator;
import org.apache.hyracks.algebricks.core.algebra.base.IOptimizationContext;
import org.apache.hyracks.algebricks.core.algebra.base.LogicalExpressionTag;
import org.apache.hyracks.algebricks.core.algebra.expressions.AbstractFunctionCallExpression;
import org.apache.hyracks.algebricks.core.algebra.expressions.ConstantExpression;
import org.apache.hyracks.algebricks.core.algebra.expressions.VariableReferenceExpression;
import org.apache.hyracks.algebricks.core.rewriter.base.IAlgebraicRewriteRule;
import org.apache.hyracks.data.std.primitive.UTF8StringPointable;
import org.apache.hyracks.data.std.util.ArrayBackedValueStorage;
import org.apache.hyracks.dataflow.common.comm.util.ByteBufferInputStream;
import org.apache.vxquery.compiler.algebricks.VXQueryConstantValue;
import org.apache.vxquery.compiler.rewriter.rules.util.ExpressionToolbox;
import org.apache.vxquery.compiler.rewriter.rules.util.OperatorToolbox;
import org.apache.vxquery.datamodel.accessors.TaggedValuePointable;
import org.apache.vxquery.datamodel.builders.atomic.StringValueBuilder;
import org.apache.vxquery.datamodel.values.ValueTag;
import org.apache.vxquery.functions.BuiltinFunctions;
import org.apache.vxquery.metadata.VXQueryMetadataProvider;
import org.apache.vxquery.types.BuiltinTypeRegistry;
import org.apache.vxquery.types.Quantifier;
import org.apache.vxquery.types.SequenceType;
/**
* The rule searches for where the function_doc1 function is in the plan in place of XQuery function.
* It replaces the string contained in the function with its absolute file path.
*
* <pre>
* Before
*
* plan__parent
* %OPERATOR( $v1 : fn:doc( \@string ) )
* plan__child
*
* Where xquery_function creates an atomic value.
*
* After
*
* plan__parent
* %OPERATOR( $v1 : fn:doc( \@absolute_file_path ) ) )
* plan__child
* </pre>
*
* @author shivanim
*/
public class ReplaceSourceMapInDocExpression implements IAlgebraicRewriteRule {
final ByteBufferInputStream bbis = new ByteBufferInputStream();
final DataInputStream di = new DataInputStream(bbis);
final UTF8StringPointable stringp = (UTF8StringPointable) UTF8StringPointable.FACTORY.createPointable();
final TaggedValuePointable tvp = (TaggedValuePointable) TaggedValuePointable.FACTORY.createPointable();
final ArrayBackedValueStorage abvs = new ArrayBackedValueStorage();
final DataOutput dOut = abvs.getDataOutput();
StringBuilder toStr = new StringBuilder();
String docArg = null;
@Override
public boolean rewritePre(Mutable<ILogicalOperator> opRef, IOptimizationContext context) throws AlgebricksException {
// TODO Auto-generated method stub
return false;
}
@Override
public boolean rewritePost(Mutable<ILogicalOperator> opRef, IOptimizationContext context)
throws AlgebricksException {
boolean modified = false;
List<Mutable<ILogicalExpression>> expressions = OperatorToolbox.getExpressions(opRef);
for (Mutable<ILogicalExpression> expression : expressions) {
Mutable<ILogicalExpression> docExpression = ExpressionToolbox.findFirstFunctionExpression(expression,
BuiltinFunctions.FN_DOC_1.getFunctionIdentifier());
if (docExpression != null) {
AbstractFunctionCallExpression absFnCall = (AbstractFunctionCallExpression) docExpression.getValue();
if (updateDocExpression(opRef, absFnCall.getArguments().get(0), context)) {
modified = true;
}
}
docExpression = ExpressionToolbox.findFirstFunctionExpression(expression,
BuiltinFunctions.JN_JSON_DOC_1.getFunctionIdentifier());
if (docExpression != null) {
AbstractFunctionCallExpression absFnCall = (AbstractFunctionCallExpression) docExpression.getValue();
if (updateDocExpression(opRef, absFnCall.getArguments().get(0), context)) {
modified = true;
}
}
}
return modified;
}
protected boolean updateDocExpression(Mutable<ILogicalOperator> opRef, Mutable<ILogicalExpression> funcExpression,
IOptimizationContext context) {
VXQueryConstantValue constantValue = null;
ConstantExpression constantExpression = null;
ILogicalExpression logicalExpression = (ILogicalExpression) funcExpression.getValue();
if (logicalExpression.getExpressionTag() == LogicalExpressionTag.CONSTANT) {
constantExpression = (ConstantExpression) logicalExpression;
constantValue = (VXQueryConstantValue) constantExpression.getValue();
} else if (logicalExpression.getExpressionTag() == LogicalExpressionTag.VARIABLE) {
VariableReferenceExpression varLogicalExpression = (VariableReferenceExpression) logicalExpression;
Mutable<ILogicalOperator> lop = OperatorToolbox.findProducerOf(opRef,
varLogicalExpression.getVariableReference());
ILogicalExpression variableLogicalExpression = (ILogicalExpression) OperatorToolbox.getExpressionOf(lop,
varLogicalExpression.getVariableReference()).getValue();
if (variableLogicalExpression.getExpressionTag() != LogicalExpressionTag.CONSTANT) {
return false;
}
constantExpression = (ConstantExpression) variableLogicalExpression;
constantValue = (VXQueryConstantValue) constantExpression.getValue();
} else {
return false;
}
if (constantValue.getType() != SequenceType.create(BuiltinTypeRegistry.XS_STRING, Quantifier.QUANT_ONE)) {
return false;
}
tvp.set(constantValue.getValue(), 0, constantValue.getValue().length);
tvp.getValue(stringp);
if (tvp.getTag() != ValueTag.XS_STRING_TAG) {
return false;
}
stringp.toString(toStr);
docArg = toStr.toString();
VXQueryMetadataProvider mdp = (VXQueryMetadataProvider) context.getMetadataProvider();
if (mdp.getSourceFileMap() == null) {
return false;
}
if (!mdp.getSourceFileMap().containsKey(docArg)) {
return false;
}
File file = mdp.getSourceFileMap().get(docArg);
StringValueBuilder svb = new StringValueBuilder();
try {
abvs.reset();
dOut.write(ValueTag.XS_STRING_TAG);
svb.write(file.getAbsolutePath(), dOut);
} catch (IOException e) {
throw new IllegalStateException(e);
}
VXQueryConstantValue vxqcv = new VXQueryConstantValue(SequenceType.create(BuiltinTypeRegistry.XS_STRING,
Quantifier.QUANT_ONE), abvs.getByteArray());
constantExpression.setValue(vxqcv);
return true;
}
}