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

org.scalarelational.versioning.VersioningSupport.scala Maven / Gradle / Ivy

package org.scalarelational.versioning

import org.scalarelational.extra.PersistentProperties
import org.scalarelational.util.Time
import pl.metastack.metarx._

trait VersioningSupport extends PersistentProperties {
  def version: Opt[Int] =
    withSession { implicit session =>
      persistence.intProperty("databaseVersion")
    }

  private var upgrades = Map.empty[Int, UpgradableVersion]

  def register(upgrade: UpgradableVersion): Unit = synchronized {
    upgrades += upgrade.version -> upgrade
  }

  /**
   * Upgrades the database to the latest version iteratively. This utilizes the
   * registered versions compared against the current version in the database
   * and iterates over all unapplied versions upgrading to the latest.
   */
  def upgrade(): Unit = synchronized {
    logger.info("Checking for Database Upgrades...")

    withSession { implicit session =>
      val latestVersion = upgrades.keys.toList match {
        case Nil => 0
        case keys => keys.max
      }

      val newDatabase = version.get.isEmpty && persistence.get("databaseVersion").isEmpty

      if (latestVersion == 0) {
        // Make sure the value is created in the database
        logger.info(s"New database created. Setting version to latest (version $latestVersion) without running upgrades.")
        version := latestVersion // New database created, we don't have to run upgrades
      } else {
        logger.info(s"Current Version: ${version.get}, Latest Version: $latestVersion")
        (version.get.getOrElse(0) until latestVersion).foreach { currentVersion =>
          transaction { implicit session =>
            val nextVersion = currentVersion + 1
            logger.info(s"Upgrading from version $currentVersion to $nextVersion...")
            val upgrade = upgrades.getOrElse(nextVersion,
              throw new RuntimeException(s"No version registered for $nextVersion."))
            if (newDatabase && !upgrade.runOnNewDatabase) {
              version := nextVersion
              logger.info(s"Skipping version $currentVersion to $nextVersion because it's a new database.")
            } else {
              val elapsed = Time.elapsed {
                upgrade.upgrade(session)
              }
              version := nextVersion
              logger.info(s"$nextVersion upgrade finished successfully in $elapsed seconds.")
            }
          }
        }
      }
    }

    upgrades = Map.empty  // Clear the map so upgrades can be released
  }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy