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

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