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

org.neo4j.dbms.database.SystemGraphComponent Maven / Gradle / Ivy

/*
 * Copyright (c) "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.dbms.database;

import org.neo4j.function.ThrowingConsumer;
import org.neo4j.graphdb.GraphDatabaseService;
import org.neo4j.graphdb.Label;
import org.neo4j.graphdb.Node;
import org.neo4j.graphdb.ResourceIterator;
import org.neo4j.graphdb.Transaction;
import org.neo4j.internal.kernel.api.security.SecurityContext;
import org.neo4j.kernel.api.KernelTransaction;
import org.neo4j.kernel.impl.coreapi.TransactionImpl;

/**
 * The system database is used to store information of interest to many parts of the DBMS. Each sub-graph can be thought of as being of primary interest to (or
 * managed by) a particular component of the DBMS. For example, the security sub-graph composed of User, Role, Privilege and various other related nodes and
 * relationships is primarily initialized, upgraded and read by the SecurityModule.
 * 

* This interface defines an external API to each component whereby external commands can ask about the current status of the component's sub-graph within the * system database, and also trigger upgrades (or migrations) of the sub-graph from an older version to the current version. *

* It is up to each component to define the rules for which older versions would be concurrently supported within any running DBMS. However, being supported * also means it is possible to upgrade to the current version. This means that if the detect method return REQUIRES_UPGRADE, then it * should be possible to call the upgradeToCurrent method and achieve the upgrade. */ public interface SystemGraphComponent { Label VERSION_LABEL = Label.label( "Version" ); /** * This string should be a unique identifier for the component. It will be used in two places: *

    *
  1. A key in a map of components for managing upgrades of the entire DBMS
  2. *
  3. A property key for the Version node used to store versioning information for this component
  4. *
*/ String componentName(); /** * Return the current status of this component. This involves reading the current contents of the system database and detecting whether the sub-graph * managed by this component is missing, old or up-to-date. Older sub-graphs could be supported for upgrade, or unsupported (in which case the server cannot * be started). */ Status detect( Transaction tx ); /** * If the component-specific sub-graph of the system database is not initialized yet (empty), this method should populate it with the default contents for * the current version of the component. * * @throws Exception on any possible error raised by the initialization process */ void initializeSystemGraph( GraphDatabaseService system, boolean firstInitialization ) throws Exception; /** * If the component-specific sub-graph of the system database is an older, but still supported, version, this method should upgrade it to the latest * supported version. * * @throws Exception on any possible error raised by the upgrade process */ void upgradeToCurrent( GraphDatabaseService system ) throws Exception; static void executeWithFullAccess( GraphDatabaseService system, ThrowingConsumer consumer ) throws Exception { try ( TransactionImpl tx = (TransactionImpl) system.beginTx(); KernelTransaction.Revertable ignore = tx.kernelTransaction().overrideWith( SecurityContext.AUTH_DISABLED ) ) { consumer.accept( tx ); tx.commit(); } } default Integer getVersion( Transaction tx, String componentVersionProperty ) { return getVersionNumber( tx, componentVersionProperty ); } /** * Get the version number of a component from the system graph. * * @param tx an open transaction * @param componentVersionProperty name of the property describing the version for the component * @return The version of the component or null if there is no stored information. */ static Integer getVersionNumber( Transaction tx, String componentVersionProperty ) { Integer result = null; try ( ResourceIterator nodes = tx.findNodes( VERSION_LABEL ) ) { if ( nodes.hasNext() ) { Node versionNode = nodes.next(); result = (Integer) versionNode.getProperty( componentVersionProperty, null ); } } return result; } enum Status { UNINITIALIZED( "No sub-graph detected for this component", "requires initialization" ), CURRENT( "The sub-graph is already at the current version", "nothing to do" ), REQUIRES_UPGRADE( "The sub-graph is supported, but is an older version and requires upgrade", "CALL dbms.upgrade()" ), UNSUPPORTED_BUT_CAN_UPGRADE( "The sub-graph is unsupported, this component cannot function", "Restart Neo4j in single-instance mode to upgrade this component at startup" ), UNSUPPORTED( "The sub-graph is unsupported because it is too old, this component cannot function", "Downgrade Neo4j and then upgrade this component before upgrading Neo4j again" ), UNSUPPORTED_FUTURE( "The sub-graph is unsupported because it is a newer version, this component cannot function", "Upgrade Neo4j" ); private final String description; private final String resolution; Status( String description, String resolution ) { this.description = description; this.resolution = resolution; } /** * Combine status's to present an overall status for the entire DBMS. Primarily we care about two cases: *
    *
  • If any component is unsupported, the entire DBMS is unsupported
  • *
  • If any component requires an upgrade, the entire DBMS requires an upgrade
  • *
* Initialization is handled through a different code path and so does not require handling here. */ public Status with( Status other ) { Status[] precedence = new Status[]{UNSUPPORTED_FUTURE, UNSUPPORTED, UNSUPPORTED_BUT_CAN_UPGRADE, REQUIRES_UPGRADE, UNINITIALIZED, CURRENT}; for ( Status status : precedence ) { if ( other == status || this == status ) { return status; } } return this; } public String description() { return description; } public String resolution() { return resolution; } } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy