All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.neo4j.kernel.impl.api.TransactionToApply Maven / Gradle / Ivy

/*
 * Copyright (c) 2018-2020 "Graph Foundation,"
 * Graph Foundation, Inc. [https://graphfoundation.org]
 *
 * This file is part of ONgDB.
 *
 * ONgDB is free software: you can redistribute it and/or modify
 * it under the terms of the GNU 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 General Public License for more details.
 *
 * You should have received a copy of the GNU General Public License
 * along with this program.  If not, see .
 */
/*
 * Copyright (c) 2002-2020 "Neo4j,"
 * Neo4j Sweden AB [http://neo4j.com]
 *
 * This file is part of Neo4j.
 *
 * Neo4j is free software: you can redistribute it and/or modify
 * it under the terms of the GNU 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 General Public License for more details.
 *
 * You should have received a copy of the GNU General Public License
 * along with this program.  If not, see .
 */
package org.neo4j.kernel.impl.api;

import java.io.IOException;
import java.util.function.LongConsumer;

import org.neo4j.helpers.collection.Visitor;
import org.neo4j.io.pagecache.tracing.cursor.context.EmptyVersionContext;
import org.neo4j.io.pagecache.tracing.cursor.context.VersionContext;
import org.neo4j.kernel.impl.transaction.TransactionRepresentation;
import org.neo4j.kernel.impl.transaction.log.Commitment;
import org.neo4j.kernel.impl.transaction.log.LogPosition;
import org.neo4j.kernel.impl.transaction.tracing.CommitEvent;
import org.neo4j.kernel.impl.util.HexPrinter;
import org.neo4j.storageengine.api.CommandsToApply;
import org.neo4j.storageengine.api.StorageCommand;
import org.neo4j.storageengine.api.TransactionApplicationMode;

import static org.neo4j.helpers.Format.date;

/**
 * A chain of transactions to apply. Transactions form a linked list, each pointing to the {@link #next()}
 * or {@code null}. This design chosen for less garbage and convenience, i.e. that we pass in a number of transactions
 * while also expecting some results for each and every one of those transactions back. The results are
 * written directly into each instance instead of creating another data structure which is then returned.
 * This is an internal class so even if it mixes arguments with results it's easier to work with,
 * requires less code... and less objects.
 *
 * State and methods are divided up into two parts, one part being the responsibility of the user to manage,
 * the other part up to the commit process to manage.
 *
 * The access pattern looks like:
 * 
    *
  1. === USER ===
  2. *
  3. Construct instances
  4. *
  5. Form the linked list using {@link #next(TransactionToApply)}
  6. *
  7. Pass into {@link TransactionCommitProcess#commit(TransactionToApply, CommitEvent, TransactionApplicationMode)}
  8. *
  9. === COMMIT PROCESS ===
  10. *
  11. Commit, where {@link #commitment(Commitment, long)} is called to store the {@link Commitment} and transaction id
  12. *
  13. Apply, where {@link #commitment()}, * {@link #transactionRepresentation()} and {@link #next()} are called
  14. *
  15. === USER ===
  16. *
  17. Data about the commit can now also be accessed using f.ex {@link #commitment()} or {@link #transactionId()}
  18. *
*/ public class TransactionToApply implements CommandsToApply, AutoCloseable { public static final long TRANSACTION_ID_NOT_SPECIFIED = 0; // These fields are provided by user private final TransactionRepresentation transactionRepresentation; private long transactionId; private final VersionContext versionContext; private TransactionToApply nextTransactionInBatch; // These fields are provided by commit process, storage engine, or recovery process private Commitment commitment; private LongConsumer closedCallback; private LogPosition logPosition; /** * Used when committing a transaction that hasn't already gotten a transaction id assigned. */ public TransactionToApply( TransactionRepresentation transactionRepresentation ) { this( transactionRepresentation, EmptyVersionContext.EMPTY ); } /** * Used when committing a transaction that hasn't already gotten a transaction id assigned. */ public TransactionToApply( TransactionRepresentation transactionRepresentation, VersionContext versionContext ) { this( transactionRepresentation, TRANSACTION_ID_NOT_SPECIFIED, versionContext ); } public TransactionToApply( TransactionRepresentation transactionRepresentation, long transactionId ) { this( transactionRepresentation, transactionId, EmptyVersionContext.EMPTY ); } public TransactionToApply( TransactionRepresentation transactionRepresentation, long transactionId, VersionContext versionContext ) { this.transactionRepresentation = transactionRepresentation; this.transactionId = transactionId; this.versionContext = versionContext; } // These methods are called by the user when building a batch public void next( TransactionToApply next ) { nextTransactionInBatch = next; } // These methods are called by the commit process public Commitment commitment() { return commitment; } @Override public long transactionId() { return transactionId; } @Override public boolean accept( Visitor visitor ) throws IOException { return transactionRepresentation.accept( visitor ); } public TransactionRepresentation transactionRepresentation() { return transactionRepresentation; } @Override public boolean requiresApplicationOrdering() { return commitment.hasExplicitIndexChanges(); } public void commitment( Commitment commitment, long transactionId ) { this.commitment = commitment; this.transactionId = transactionId; this.versionContext.initWrite( transactionId ); } public void logPosition( LogPosition position ) { this.logPosition = position; } @Override public TransactionToApply next() { return nextTransactionInBatch; } public void onClose( LongConsumer closedCallback ) { this.closedCallback = closedCallback; } @Override public void close() { if ( closedCallback != null ) { closedCallback.accept( transactionId ); } } @Override public String toString() { TransactionRepresentation tr = this.transactionRepresentation; return "Transaction #" + transactionId + (logPosition != null ? " at log position " + logPosition : " (no log position)") + " {started " + date( tr.getTimeStarted() ) + ", committed " + date( tr.getTimeCommitted() ) + ", with " + countCommands() + " commands in this transaction" + ", authored by " + tr.getAuthorId() + ", with master id " + tr.getMasterId() + ", lock session " + tr.getLockSessionId() + ", latest committed transaction id when started was " + tr.getLatestCommittedTxWhenStarted() + ", additional header bytes: " + HexPrinter.hex( tr.additionalHeader(), Integer.MAX_VALUE, "" ) + "}"; } private String countCommands() { class Counter implements Visitor { private int count; @Override public boolean visit( StorageCommand element ) { count++; return false; } } try { Counter counter = new Counter(); accept( counter ); return String.valueOf( counter.count ); } catch ( Throwable e ) { return "(unable to count: " + e.getMessage() + ")"; } } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy