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);
}
}
}