Package com.foundationdb.qp.operator

Source Code of com.foundationdb.qp.operator.Update_Default$Execution

/**
* Copyright (C) 2009-2013 FoundationDB, LLC
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU Affero General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
* GNU Affero General Public License for more details.
*
* You should have received a copy of the GNU Affero General Public License
* along with this program.  If not, see <http://www.gnu.org/licenses/>.
*/

package com.foundationdb.qp.operator;

import com.foundationdb.qp.exec.UpdatePlannable;
import com.foundationdb.qp.exec.UpdateResult;
import com.foundationdb.qp.row.Row;
import com.foundationdb.server.explain.*;
import com.foundationdb.server.explain.std.DUIOperatorExplainer;
import com.foundationdb.util.ArgumentValidation;
import com.foundationdb.util.Strings;
import com.foundationdb.util.tap.InOutTap;
import com.foundationdb.util.tap.Tap;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import java.util.Collections;
import java.util.List;

/**

<h1>Overview</h1>

Provides row update functionality.

<h1>Arguments</h1>

<ul>

<li><b>PhysicalOperator inputOperator:</b> Provides rows to be updated

<li><b>UpdateFunction updateFunction:</b> specifies which rows are to be updated, and how

</ul>

<h1>Behavior</h1>

For each row from the input operator's cursor, the UpdateOperator
invokes <i>updateFunction.rowIsSelected</i> to determine if the row
should be updated. If so, it invokes <i>updateFunction.evaluate</i> to
get the new version of the row. It then performs the update in an
unspecified way (in practice, this is currently done via
<i>StoreAdapater.updateRow</i>, which is implemented by
<i>PersistitAdapater.updateRow</i>, which invokes
<i>PersistitStore.updateRow</i>).

The result of this update is an <i>UpdaateResult</i> instance which summarizes how many rows were updated and how long the operation took.

<h1>Output</h1>

N/A (this is an UpdatePlannable, not an Operator).

<h1>Assumptions</h1>

Selected rows must have a <i>RowType</i> such
that <i>rowType.hasTable() == true</i>.

<h1>Performance</h1>

Updating rows may be slow, especially since indexes are also
updated. There are several random-access reads and writes involved,
which depend on the indexes defined for that row type.

There are potentially ways to optimize this, if we can
push <i>WHERE</i> clauses down; this would mean we could update some
indexes as a batch operation, rather than one at a time. This would
require changes to the API, and is not currently a priority.

<h1>Memory Requirements</h1>

Each <i>UpdateFunction.evaluate</i> method may generate a
new <i>Row</i>.

*/

/* Deprecated */
class Update_Default implements UpdatePlannable {

    // Object interface

    @Override
    public String toString() {
        return String.format("%s(%s -> %s)", getName(), inputOperator, updateFunction);
    }

    // constructor

    public Update_Default(Operator inputOperator, UpdateFunction updateFunction) {
        ArgumentValidation.notNull("update lambda", updateFunction);
       
        this.inputOperator = inputOperator;
        this.updateFunction = updateFunction;
    }

    // UpdatePlannable interface

    @Override
    public UpdateResult run(QueryContext context, QueryBindings bindings) {
        QueryBindingsCursor bindingsCursor = new SingletonQueryBindingsCursor(bindings);
        return new Execution(context, inputOperator.cursor(context, bindingsCursor)).run();
    }

    // Plannable interface

    @Override
    public String getName()
    {
        return getClass().getSimpleName();
    }

    @Override
    public List<Operator> getInputOperators() {
        return Collections.singletonList(inputOperator);
    }

    @Override
    public String describePlan()
    {
        return describePlan(inputOperator);
    }

    @Override
    public String describePlan(Operator inputOperator) {
        return inputOperator + Strings.nl() + this;
    }

    // Object state

    private final Operator inputOperator;
    private final UpdateFunction updateFunction;
    private static final InOutTap UPDATE_TAP = Tap.createTimer("operator: Update_Default");
    private static final Logger LOG = LoggerFactory.getLogger(Update_Default.class);

    @Override
    public CompoundExplainer getExplainer(ExplainContext context)
    {
        Attributes atts = new Attributes();
        if (context.hasExtraInfo(this))
            atts.putAll(context.getExtraInfo(this).get());
        atts.put(Label.EXTRA_TAG, PrimitiveExplainer.getInstance(updateFunction.toString()));
        CompoundExplainer ex = new DUIOperatorExplainer(getName(), atts, inputOperator, context);
        return ex;
    }

    // Inner classes

    private class Execution extends ExecutionBase
    {
        public UpdateResult run()
        {
            int seen = 0, modified = 0;
            if (TAP_NEXT_ENABLED) {
                UPDATE_TAP.in();
            }
            try {
                QueryBindings bindings = input.openTopLevel();
                Row oldRow;
                while ((oldRow = input.next()) != null) {
                    checkQueryCancelation();
                    ++seen;
                    if (updateFunction.rowIsSelected(oldRow)) {
                        Row newRow = updateFunction.evaluate(oldRow, context, bindings);
                        adapter().updateRow(oldRow, newRow);
                        if (LOG_EXECUTION) {
                            LOG.debug("Update_Default: update {} to {}", oldRow, newRow);
                        }
                        ++modified;
                    }
                }
            } finally {
                if (input != null) {
                    input.close();
                }
                if (TAP_NEXT_ENABLED) {
                    UPDATE_TAP.out();
                }
            }
            return new StandardUpdateResult(seen, modified);
        }

        public Execution(QueryContext queryContext, Cursor input)
        {
            super(queryContext);
            this.input = input;
        }

        private final Cursor input;
    }
}
TOP

Related Classes of com.foundationdb.qp.operator.Update_Default$Execution

TOP
Copyright © 2018 www.massapi.com. All rights reserved.
All source code are property of their respective owners. Java is a trademark of Sun Microsystems, Inc and owned by ORACLE Inc. Contact coftware#gmail.com.