li.strolch.migrations.Migrations Maven / Gradle / Ivy
/*
* Copyright 2015 Robert von Burg
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package li.strolch.migrations;
import java.io.File;
import java.io.FileFilter;
import java.text.MessageFormat;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.Map.Entry;
import java.util.ResourceBundle;
import java.util.Set;
import java.util.SortedSet;
import java.util.TreeSet;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import li.strolch.agent.api.ComponentContainer;
import li.strolch.agent.api.StrolchAgent;
import li.strolch.handler.operationslog.LogMessage;
import li.strolch.handler.operationslog.LogSeverity;
import li.strolch.handler.operationslog.OperationsLog;
import li.strolch.model.Locator;
import li.strolch.privilege.model.Certificate;
import li.strolch.utils.Version;
import li.strolch.utils.collections.MapOfLists;
import li.strolch.utils.dbc.DBC;
public class Migrations {
private static final Logger logger = LoggerFactory.getLogger(Migrations.class);
private ComponentContainer container;
private Set realmNames;
private boolean verbose;
private Map> dataMigrations;
private Map> codeMigrations;
private MapOfLists migrationsRan;
public Migrations(ComponentContainer container, Set realmNames, boolean verbose) {
this.container = container;
this.realmNames = realmNames;
this.verbose = verbose;
}
public void setVerbose(boolean verbose) {
this.verbose = verbose;
}
public boolean isVerbose() {
return verbose;
}
public MapOfLists getMigrationsRan() {
return this.migrationsRan;
}
public void parseMigrations(File migrationsPath) {
DBC.PRE.assertTrue("If migrations path is not a directory!", migrationsPath.isDirectory());
// data migrations
this.dataMigrations = loadDataMigrations(this.realmNames, migrationsPath);
// code migrations
this.codeMigrations = loadCodeMigrations(this.realmNames, migrationsPath);
// log found migrations
if (this.verbose)
logDetectedMigrations(this.realmNames, this.dataMigrations, this.codeMigrations);
}
public void runMigrations(Certificate certificate, Map currentVersions) {
MapOfLists migrationsRan = new MapOfLists<>();
for (Entry entry : currentVersions.entrySet()) {
String realm = entry.getKey();
MigrationVersion currentVersion = entry.getValue();
if (this.verbose)
logger.info("[" + realm + "] Performing all migrations after " + currentVersion);
Version nextPossibleCodeVersion = currentVersion.getCodeVersion().add(0, 0, 1);
Version nextPossibleDataVersion = currentVersion.getDataVersion().add(0, 0, 1);
CodeMigration currentCodeMigration = new CodeMigration(realm, nextPossibleCodeVersion, null);
DataMigration currentDataMigration = new DataMigration(realm, nextPossibleDataVersion, null);
SortedSet dataMigrations = this.dataMigrations.get(realm);
if (dataMigrations != null && !dataMigrations.isEmpty()) {
for (DataMigration migration : dataMigrations.tailSet(currentDataMigration)) {
String msg = "[{0}] Running data migration {1}";
logger.info(MessageFormat.format(msg, realm, migration.getVersion()));
migration.migrate(container, certificate);
migrationsRan.addElement(realm, migration.getVersion());
}
}
SortedSet codeMigrations = this.codeMigrations.get(realm);
if (codeMigrations != null && !codeMigrations.isEmpty()) {
for (CodeMigration migration : codeMigrations.tailSet(currentCodeMigration)) {
String msg = "[{0}] Running code migration {1} {2}";
logger.info(
MessageFormat.format(msg, realm, migration.getVersion(), migration.getClass().getName()));
migration.migrate(container, certificate);
migrationsRan.addElement(realm, migration.getVersion());
}
}
}
if (migrationsRan.isEmpty()) {
if (this.verbose)
logger.info("There were no migrations required!");
} else {
logger.info("Migrated " + migrationsRan.size() + " realms!");
addOperationLogs(migrationsRan);
}
this.migrationsRan = migrationsRan;
}
private void addOperationLogs(MapOfLists migrationsRan) {
if (this.container.hasComponent(OperationsLog.class)) {
OperationsLog operationsLog = this.container.getComponent(OperationsLog.class);
MigrationsHandler migrationsHandler = this.container.getComponent(MigrationsHandler.class);
Locator locator = migrationsHandler.getLocator();
for (String realm : migrationsRan.keySet()) {
List list = migrationsRan.getList(realm);
for (Version version : list) {
LogMessage logMessage = new LogMessage(realm, locator.append(StrolchAgent.getUniqueId()),
LogSeverity.INFO, ResourceBundle.getBundle("strolch-service"),
"execution.handler.migrations.version").value("version", version.toString());
operationsLog.addMessage(logMessage);
}
}
}
}
/**
* @param cert
* @param codeMigrationsByRealm
*/
public void runCodeMigrations(Certificate cert,
Map currentVersions,
MapOfLists codeMigrationsByRealm) {
MapOfLists migrationsRan = new MapOfLists<>();
for (String realm : codeMigrationsByRealm.keySet()) {
// ignore if no such realm
if (!this.realmNames.contains(realm))
continue;
MigrationVersion currentVersion = currentVersions.get(realm);
List listOfMigrations = codeMigrationsByRealm.getList(realm);
SortedSet migrations = new TreeSet<>((o1, o2) -> o1.getVersion().compareTo(o2.getVersion()));
migrations.addAll(listOfMigrations);
Version nextVersion = currentVersion.getCodeVersion().add(0, 0, 1);
CodeMigration nextMigration = new CodeMigration(realm, nextVersion);
SortedSet migrationsToRun = migrations.tailSet(nextMigration);
for (CodeMigration migration : migrationsToRun) {
DBC.INTERIM.assertEquals("Realms do not match!", realm, migration.getRealm());
Version migrateVersion = migration.getVersion();
boolean isLaterMigration = migrateVersion.compareTo(currentVersion.getCodeVersion()) > 0;
DBC.INTERIM.assertTrue(
"Current version " + currentVersion.getCodeVersion() + " is not before next " + migrateVersion,
isLaterMigration);
String msg = "[{0}] Running code migration {1} {2}";
logger.info(MessageFormat.format(msg, realm, migrateVersion, migration.getClass().getName()));
migration.migrate(this.container, cert);
migrationsRan.addElement(realm, migration.getVersion());
}
}
if (migrationsRan.isEmpty()) {
if (this.verbose)
logger.info("There were no migrations required!");
} else {
logger.info(
"Performed " + migrationsRan.size() + " migrations on " + migrationsRan.sizeKeys() + " realms.");
addOperationLogs(migrationsRan);
}
this.migrationsRan = migrationsRan;
}
private static void logDetectedMigrations(Set realmNames,
Map> allDataMigrations,
Map> allCodeMigrations) {
for (String realm : realmNames) {
SortedSet codeMigrations = allCodeMigrations.get(realm);
if (codeMigrations == null || codeMigrations.isEmpty()) {
logger.info("[" + realm + "] Found no code migrations.");
} else {
logger.info("[" + realm + "] Found " + codeMigrations.size() + " code migrations");
for (CodeMigration codeMigration : codeMigrations) {
logger.info("[" + realm + "] " + codeMigration.getVersion().toString());
}
}
SortedSet dataMigrations = allDataMigrations.get(realm);
if (dataMigrations == null || dataMigrations.isEmpty()) {
logger.info("[" + realm + "] Found no data migrations.");
} else {
logger.info("[" + realm + "] Found " + dataMigrations.size() + " data migrations");
for (DataMigration dataMigration : dataMigrations) {
logger.info("[" + realm + "] " + dataMigration.getVersion().toString());
}
}
}
}
private static Map> loadDataMigrations(Set realmNames,
File migrationsPath) {
Map> migrationsByRealm = new HashMap<>();
File dataDir = new File(migrationsPath, "data");
if (dataDir.exists()) {
DBC.PRE.assertTrue("migrations/data must be a directory!", dataDir.isDirectory());
// only list directories where name is a realmName
File[] realmMigrations = dataDir.listFiles((FileFilter) path -> realmNames.contains(path.getName()));
for (File realmMigration : realmMigrations) {
String realm = realmMigration.getName();
SortedSet migrations = new TreeSet<>(
(o1, o2) -> o1.getVersion().compareTo(o2.getVersion()));
migrationsByRealm.put(realm, migrations);
File[] migrationFiles = realmMigration
.listFiles((FileFilter) pathname -> pathname.getName().endsWith(".xml"));
for (File file : migrationFiles) {
String name = file.getName();
Version version = Version.valueOf(name.substring(0, name.length() - 4));
migrations.add(new DataMigration(realm, version, file));
}
}
}
return migrationsByRealm;
}
private static Map> loadCodeMigrations(Set realmNames,
File migrationsPath) {
Map> migrationsByRealm = new HashMap<>(); //new TreeSet<>((o1, o2) -> o1.getVersion().compareTo(o2.getVersion()));
File codeDir = new File(migrationsPath, "code");
if (codeDir.exists()) {
DBC.PRE.assertTrue("migrations/code must be a directory!", codeDir.isDirectory());
File[] realmMigrations = codeDir.listFiles((FileFilter) path -> realmNames.contains(path.getName()));
for (File realmMigration : realmMigrations) {
String realm = realmMigration.getName();
SortedSet migrations = new TreeSet<>(
(o1, o2) -> o1.getVersion().compareTo(o2.getVersion()));
migrationsByRealm.put(realm, migrations);
File[] migrationFiles = realmMigration
.listFiles((FileFilter) pathname -> pathname.getName().endsWith(".xml"));
for (File file : migrationFiles) {
String name = file.getName();
Version version = Version.valueOf(name.substring(0, name.length() - 4));
migrations.add(new CodeMigration(realm, version, file));
}
}
}
return migrationsByRealm;
}
public MapOfLists getMigrationsToRun(Map currentVersions) {
MapOfLists migrationsToRun = new MapOfLists<>();
for (Entry entry : currentVersions.entrySet()) {
String realm = entry.getKey();
Version nextPossibleCodeVersion = entry.getValue().getCodeVersion().add(0, 0, 1);
Version nextPossibleDataVersion = entry.getValue().getDataVersion().add(0, 0, 1);
CodeMigration currentCodeMigration = new CodeMigration(realm, nextPossibleCodeVersion, null);
DataMigration currentDataMigration = new DataMigration(realm, nextPossibleDataVersion, null);
SortedSet allCodeMigrations = this.codeMigrations.get(realm);
if (allCodeMigrations != null) {
SortedSet codeMigrations = allCodeMigrations.tailSet(currentCodeMigration);
for (CodeMigration codeMigration : codeMigrations) {
if (!migrationsToRun.containsElement(realm, codeMigration.getVersion()))
migrationsToRun.addElement(realm, codeMigration.getVersion());
}
}
SortedSet allDataMigrations = this.dataMigrations.get(realm);
if (allDataMigrations != null) {
SortedSet dataMigrations = allDataMigrations.tailSet(currentDataMigration);
for (DataMigration dataMigration : dataMigrations) {
if (!migrationsToRun.containsElement(realm, dataMigration.getVersion()))
migrationsToRun.addElement(realm, dataMigration.getVersion());
}
}
}
return migrationsToRun;
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy