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

net.protocol.Execute.scala Maven / Gradle / Ivy

// Copyright (c) 2018-2021 by Rob Norris
// This software is licensed under the MIT License (MIT).
// For more information see LICENSE or https://opensource.org/licenses/MIT

package skunk.net.protocol

import cats.syntax.all._
import cats.MonadError
import skunk.~
import skunk.data.Completion
import skunk.exception.PostgresErrorException
import skunk.net.{ Protocol, MessageSocket }
import skunk.net.message.{ Execute => ExecuteMessage, _ }
import skunk.util.Typer
import natchez.Trace
import skunk.exception.CopyNotSupportedException
import skunk.exception.EmptyStatementException

trait Execute[F[_]] {
  def apply[A](portal: Protocol.CommandPortal[F, A]): F[Completion]
  def apply[A, B](portal: Protocol.QueryPortal[F, A, B], maxRows: Int, ty: Typer): F[List[B] ~ Boolean]
}

object Execute {

  def apply[F[_]: Exchange: MessageSocket: Trace](
    implicit ev: MonadError[F, Throwable]
  ): Execute[F] =
    new Unroll[F] with Execute[F] {

      override def apply[A](portal: Protocol.CommandPortal[F, A]): F[Completion] =
        exchange("execute") {
          for {
            _  <- send(ExecuteMessage(portal.id.value, 0))
            _  <- send(Flush)
            c  <- flatExpect {
              case CommandComplete(c) => send(Sync) *> expect { case ReadyForQuery(_) => c } // https://github.com/tpolecat/skunk/issues/210

              case EmptyQueryResponse =>
                send(Sync) *>
                expect { case ReadyForQuery(_) => } *>
                new EmptyStatementException(portal.preparedCommand.command).raiseError[F, Completion]

              case CopyOutResponse(_) =>
                receive.iterateUntil {
                  case CommandComplete(_) => true
                  case _                  => false
                } *>
                new CopyNotSupportedException(portal.preparedCommand.command).raiseError[F, Completion]

              case CopyInResponse(_) =>
                send(CopyFail) *>
                expect { case ErrorResponse(_) => } *>
                send(Sync) *>
                expect { case ReadyForQuery(_) => } *>
                new CopyNotSupportedException(portal.preparedCommand.command).raiseError[F, Completion]

              case ErrorResponse(info) =>
                for {
                  hi <- history(Int.MaxValue)
                  _  <- send(Sync)
                  _  <- expect { case ReadyForQuery(_) => }
                  a  <- new PostgresErrorException(
                          sql             = portal.preparedCommand.command.sql,
                          sqlOrigin       = Some(portal.preparedCommand.command.origin),
                          info            = info,
                          history         = hi,
                          arguments       = portal.preparedCommand.command.encoder.types.zip(portal.preparedCommand.command.encoder.encode(portal.arguments)),
                          argumentsOrigin = Some(portal.argumentsOrigin)
                        ).raiseError[F, Completion]
                } yield a
            }
          } yield c
        }

      override def apply[A, B](portal: Protocol.QueryPortal[F, A, B], maxRows: Int, ty: Typer): F[List[B] ~ Boolean] =
        exchange("execute") {
          for {
            _  <- Trace[F].put(
                    "max-rows"  -> maxRows,
                    "portal-id" -> portal.id.value
                  )
            _  <- send(ExecuteMessage(portal.id.value, maxRows))
            _  <- send(Flush)
            rs <- unroll(portal)
          } yield rs
        }

    }

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy