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

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

/*
 * Copyright (c) "Neo4j"
 * Neo4j Sweden AB [https://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 static org.neo4j.configuration.GraphDatabaseSettings.SYSTEM_DATABASE_NAME;
import static org.neo4j.dbms.database.SystemGraphComponent.VERSION_LABEL;

import java.util.ArrayList;
import java.util.LinkedHashMap;
import java.util.List;
import java.util.Map;
import java.util.Objects;
import java.util.function.Consumer;
import org.neo4j.function.ThrowingConsumer;
import org.neo4j.graphdb.GraphDatabaseService;
import org.neo4j.graphdb.Node;
import org.neo4j.graphdb.ResourceIterator;
import org.neo4j.graphdb.Transaction;
import org.neo4j.internal.helpers.Exceptions;
import org.neo4j.util.Preconditions;

/**
 * Central collection for managing multiple versioned system graph initializers. There could be several components in the DBMS that each have a requirement on
 * the system database to contain a graph with a specific schema. Each component needs to maintain that schema and support multiple versions of that schema in
 * order to allow rolling upgrades to be possible where newer versions of Neo4j will be able to run on older versions of the system database.
 * 

* The core design is that each component is able to detect the version of their own sub-graph and from that decide if they can support it or not, and how to * upgrade from one version to another. */ public class SystemGraphComponents { private final Map componentMap; private SystemGraphComponents(Map componentMap) { this.componentMap = componentMap; } public void forEachThrowing(ThrowingConsumer process) throws Exception { for (SystemGraphComponent component : componentMap.values()) { process.accept(component); } } public void forEach(Consumer process) { componentMap.values().forEach(process); } public SystemGraphComponent.Status detect(Transaction tx) { return componentMap.values().stream() .map(c -> c.detect(tx)) .reduce(SystemGraphComponent.Status::with) .orElse(SystemGraphComponent.Status.CURRENT); } public SystemGraphComponent.Status detect(GraphDatabaseService system) { return componentMap.values().stream() .map(c -> c.detect(system)) .reduce(SystemGraphComponent.Status::with) .orElse(SystemGraphComponent.Status.CURRENT); } public void initializeSystemGraph(GraphDatabaseService system) { Preconditions.checkState( system.databaseName().equals(SYSTEM_DATABASE_NAME), "Cannot initialize system graph on database '" + system.databaseName() + "'"); boolean newlyCreated; try (Transaction tx = system.beginTx(); ResourceIterator nodes = tx.findNodes(VERSION_LABEL)) { newlyCreated = !nodes.hasNext(); } Exception failure = null; for (SystemGraphComponent component : componentMap.values()) { try { component.initializeSystemGraph(system, newlyCreated); } catch (Exception e) { failure = Exceptions.chain(failure, e); } } if (failure != null) { throw new IllegalStateException( "Failed to initialize system graph component: " + failure.getMessage(), failure); } } public void upgradeToCurrent(GraphDatabaseService system) throws Exception { Exception failure = null; for (SystemGraphComponent component : componentsToUpgrade(system)) { try { component.upgradeToCurrent(system); } catch (Exception e) { failure = Exceptions.chain(failure, e); } } if (failure != null) { throw new IllegalStateException("Failed to upgrade system graph:" + failure.getMessage(), failure); } } private List componentsToUpgrade(GraphDatabaseService system) throws Exception { List componentsToUpgrade = new ArrayList<>(); SystemGraphComponent.executeWithFullAccess(system, tx -> componentMap.values().stream() .filter(c -> { SystemGraphComponent.Status status = c.detect(tx); return status == SystemGraphComponent.Status.UNSUPPORTED_BUT_CAN_UPGRADE || status == SystemGraphComponent.Status.REQUIRES_UPGRADE || // New components are not currently initialised in cluster deployment when new binaries are // booted on top of an existing database. // This is a known shortcoming of the lifecycle and a state transfer from UNINITIALIZED to // CURRENT must be supported // as a workaround until it is fixed. status == SystemGraphComponent.Status.UNINITIALIZED; }) .forEach(componentsToUpgrade::add)); return componentsToUpgrade; } public abstract static class Builder { protected final Map componentMap = new LinkedHashMap<>(); public abstract void register(SystemGraphComponent component); public SystemGraphComponents build() { return new SystemGraphComponents(componentMap); } } public static class DefaultBuilder extends Builder { @Override public void register(SystemGraphComponent component) { if (componentMap.containsKey(component.componentName())) { throw new IllegalStateException("Duplicate component registration: " + component.componentName().name()); } componentMap.put(component.componentName(), component); } } @FunctionalInterface public interface UpgradeChecker { UpgradeChecker UPGRADE_ALWAYS_ALLOWED = () -> UpgradeCheckResult.UPGRADE_NECESSARY; UpgradeCheckResult upgradeCheck(); } public static final class UpgradeCheckResult { public static final UpgradeCheckResult UPGRADE_NOT_NECESSARY = new UpgradeCheckResult(false, null); public static final UpgradeCheckResult UPGRADE_NECESSARY = new UpgradeCheckResult(true, null); public static UpgradeCheckResult upgradeNotAllowed(String message) { Objects.requireNonNull(message); return new UpgradeCheckResult(false, message); } private final boolean upgradeNecessary; private final String whyUpgradeNotAllowed; private UpgradeCheckResult(boolean upgradeNecessary, String whyUpgradeNotAllowed) { this.upgradeNecessary = upgradeNecessary; this.whyUpgradeNotAllowed = whyUpgradeNotAllowed; } public boolean upgradeAllowed() { return whyUpgradeNotAllowed == null; } public boolean upgradeNecessary() { return upgradeNecessary; } public String whyUpgradeNotAllowed() { return whyUpgradeNotAllowed; } @Override public String toString() { return "UpgradeCheckResult{" + (upgradeAllowed() ? (upgradeNecessary ? "Upgrade necessary" : "Upgrade not necessary") : "Upgrade not allowed: " + whyUpgradeNotAllowed) + "}"; } @Override public boolean equals(Object o) { if (this == o) return true; if (o == null || getClass() != o.getClass()) return false; UpgradeCheckResult that = (UpgradeCheckResult) o; return upgradeNecessary == that.upgradeNecessary && Objects.equals(whyUpgradeNotAllowed, that.whyUpgradeNotAllowed); } @Override public int hashCode() { return Objects.hash(upgradeNecessary, whyUpgradeNotAllowed); } } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy