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

harness.sql.autoSchema.PartialState.scala Maven / Gradle / Ivy

package harness.sql.autoSchema

import cats.data.EitherNel
import cats.syntax.either.*
import cats.syntax.parallel.*
import harness.sql.{Col, TableSchema}

final case class PartialState private (
    schemas: Map[SchemaRef, PartialState.Schema],
)
object PartialState {

  def apply(schemas: Map[SchemaRef, PartialState.Schema]): PartialState =
    if (schemas.contains(SchemaRef.Public)) new PartialState(schemas)
    else new PartialState(schemas.updated(SchemaRef.Public, PartialState.Schema(Map.empty)))

  def fromTables(tables: Tables): PartialState = {
    def fromTable(table: TableSchema.AnySchema): (TableRef, PartialState.Table) =
      (
        TableRef(SchemaRef(table.tableSchema), table.tableName),
        PartialState.Table(
          table.colChunk.map { col => (col.colName, PartialState.Column(col.colType, KeyType(col.keyType), col.nullable)) }.toMap,
        ),
      )

    PartialState(
      tables.tableSchemas
        .map(fromTable)
        .groupMap(_._1.schemaRef) { case (k, v) => (k.tableName, v) }
        .map { case (k, v) => (k, PartialState.Schema(v.toMap)) },
    )
  }

  def fromDbState(state: DbState): PartialState =
    PartialState(
      state.schemas.map { case (schemaRef, schema) =>
        schemaRef ->
          Schema(
            schema.tables.map { case (tableName, table) =>
              tableName ->
                Table(
                  table.columns.map { case (colName, col) =>
                    colName ->
                      Column(
                        col.colType,
                        col.keyType,
                        col.nullable,
                      )
                  },
                )
            },
          )
      },
    )

  final case class Schema(
      tables: Map[String, Table],
  )

  final case class Table(
      columns: Map[String, Column],
  )

  final case class Column(
      colType: Col.ColType,
      keyType: KeyType,
      nullable: Boolean,
  )

  // =====|  |=====

  object validate {

    def apply(state: PartialState, exp: PartialState): EitherNel[String, Unit] =
      (state.schemas.keySet | exp.schemas.keySet).toList.sortBy(_.schemaName).parTraverse_ { schemaName =>
        (state.schemas.get(schemaName), exp.schemas.get(schemaName)) match {
          case (Some(state), Some(exp)) => validateSchema(schemaName, state, exp)
          case (None, Some(_))          => s"Schema $schemaName is not in the correct state. Expected: exists, State: DNE.".leftNel
          case (Some(_), None)          => s"Schema $schemaName is not in the correct state. Expected: DNE, State: exists.".leftNel
          case (None, None)             => s"Internal defect (validateState): $state, $exp".leftNel
        }
      }

    private def validateSchema(ref: SchemaRef, state: PartialState.Schema, exp: PartialState.Schema): EitherNel[String, Unit] =
      (state.tables.keySet | exp.tables.keySet).toList.sorted.parTraverse_ { tableName =>
        (state.tables.get(tableName), exp.tables.get(tableName)) match {
          case (Some(state), Some(exp)) => validateTable(TableRef(ref, tableName), state, exp)
          case (None, Some(_))          => s"Table $ref.$tableName is not in the correct state. Expected: exists, State: DNE.".leftNel
          case (Some(_), None)          => s"Table $ref.$tableName is not in the correct state. Expected: DNE, State: exists.".leftNel
          case (None, None)             => s"Internal defect (validateSchema): $ref, $state, $exp".leftNel
        }
      }

    private def validateTable(ref: TableRef, state: PartialState.Table, exp: PartialState.Table): EitherNel[String, Unit] =
      (state.columns.keySet | exp.columns.keySet).toList.sorted.parTraverse_ { colName =>
        (state.columns.get(colName), exp.columns.get(colName)) match {
          case (Some(state), Some(exp)) => validateColumn(ColRef(ref.schemaRef, ref.tableName, colName), state, exp)
          case (None, Some(_))          => s"Column ${ref.schemaRef}.${ref.tableName} ($colName) is not in the correct state. Expected: exists, State: DNE.".leftNel
          case (Some(_), None)          => s"Column ${ref.schemaRef}.${ref.tableName} ($colName) is not in the correct state. Expected: DNE, State: exists.".leftNel
          case (None, None)             => s"Internal defect (validateTable): $ref, $state, $exp".leftNel
        }
      }

    private def validateColumn(ref: ColRef, state: PartialState.Column, exp: PartialState.Column): EitherNel[String, Unit] = {
      def compare[A](field: String, f: PartialState.Column => A): EitherNel[String, Unit] = {
        val stateValue = f(state)
        val expValue = f(exp)
        if (stateValue == expValue) ().asRight
        else s"Column ${ref.colName} in table ${ref.schemaRef}.${ref.tableName} is not in the correct state (field: $field). Expected: $expValue, State: $stateValue.".leftNel
      }

      (
        compare("colType", _.colType),
        compare("keyType", _.keyType),
        compare("nullable", _.nullable),
      ).parTupled.map(_ => ())
    }

  }

  object diff {

    def apply(state: PartialState, exp: PartialState, allowDrops: Boolean): EitherNel[String, List[MigrationStep.InMemory.Auto]] =
      (state.schemas.keySet | exp.schemas.keySet).toList
        .sortBy(_.schemaName)
        .parTraverse { schemaRef =>
          (state.schemas.get(schemaRef), exp.schemas.get(schemaRef)) match {
            case (Some(state), Some(exp)) => diffSchema(schemaRef, state, exp, allowDrops)
            case (Some(state), None) =>
              schemaRef match {
                case SchemaRef.Public =>
                  diffSchema(schemaRef, state, PartialState.Schema(Map.empty), allowDrops)
                case schemaRef: SchemaRef.Custom =>
                  if (allowDrops)
                    diffSchema(schemaRef, state, PartialState.Schema(Map.empty), allowDrops).map { MigrationStep.DropSchema(schemaRef) :: _ }
                  else s"Auto-drop disabled, unable to drop schema: $schemaRef".leftNel
              }
            case (None, Some(exp)) =>
              diffSchema(schemaRef, PartialState.Schema(Map.empty), exp, allowDrops).map { steps =>
                schemaRef match {
                  case SchemaRef.Public            => steps
                  case schemaRef: SchemaRef.Custom => MigrationStep.CreateSchema(schemaRef) :: steps
                }
              }
            case (None, None) => s"Internal defect (diffState): $schemaRef, $state, $exp".leftNel
          }
        }
        .map(_.flatten.sorted)

    private def diffSchema(ref: SchemaRef, state: PartialState.Schema, exp: PartialState.Schema, allowDrops: Boolean): EitherNel[String, List[MigrationStep.InMemory.Auto]] =
      (state.tables.keySet | exp.tables.keySet).toList.sorted
        .parTraverse { tableName =>
          val tableRef = TableRef(ref, tableName)
          (state.tables.get(tableName), exp.tables.get(tableName)) match {
            case (Some(state), Some(exp)) => diffTable(tableRef, state, exp, allowDrops)
            case (Some(state), None) =>
              if (allowDrops) diffTable(tableRef, state, PartialState.Table(Map.empty), allowDrops).map(MigrationStep.DropTable(tableRef) :: _)
              else s"Auto-drop disabled, unable to drop table: ${tableRef.schemaRef}.${tableRef.tableName}".leftNel
            case (None, Some(exp)) => diffTable(tableRef, PartialState.Table(Map.empty), exp, allowDrops).map(MigrationStep.CreateTable(tableRef) :: _)
            case (None, None)      => s"Internal defect (diffSchema): $tableRef, $state, $exp".leftNel
          }
        }
        .map(_.flatten)

    private def diffTable(ref: TableRef, state: PartialState.Table, exp: PartialState.Table, allowDrops: Boolean): EitherNel[String, List[MigrationStep.InMemory.Auto]] =
      (state.columns.keySet | exp.columns.keySet).toList.sorted
        .parTraverse { colName =>
          val colRef = ColRef(ref.schemaRef, ref.tableName, colName)
          (state.columns.get(colName), exp.columns.get(colName)) match {
            case (Some(state), Some(exp)) => diffColumn(colRef, state, exp)
            case (Some(state), None) =>
              if (allowDrops) (MigrationStep.DropCol(colRef, state.keyType) :: Nil).asRight
              else s"Auto-drop disabled, unable to drop col: ${colRef.colName} in table ${colRef.schemaRef}.${colRef.tableName}".leftNel
            case (None, Some(exp)) => (MigrationStep.CreateCol(colRef, exp.colType, exp.keyType, exp.nullable) :: Nil).asRight
            case (None, None)      => s"Internal defect (diffTable): $colRef, $state, $exp".leftNel
          }
        }
        .map(_.flatten)

    private def diffColumn(ref: ColRef, state: PartialState.Column, exp: PartialState.Column): EitherNel[String, List[MigrationStep.InMemory.Auto]] = {
      def compare[A](field: String, f: PartialState.Column => A): EitherNel[String, Unit] = {
        val stateValue = f(state)
        val expValue = f(exp)
        if (stateValue == expValue) ().asRight
        else s"Column ${ref.colName} in table ${ref.schemaRef}.${ref.tableName} is not in the correct state (field: $field). Expected: $expValue, State: $stateValue.".leftNel
      }

      (
        compare("colType", _.colType),
        compare("keyType", _.keyType),
      ).parTupled.map { _ =>
        (state.nullable, exp.nullable) match {
          case (true, true) | (false, false) => Nil
          case (true, false)                 => MigrationStep.SetColNotNullable(ref) :: Nil
          case (false, true)                 => MigrationStep.SetColNullable(ref) :: Nil
        }
      }
    }

  }

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy