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

net.nmoncho.helenus.zio.ZCqlSession.scala Maven / Gradle / Ivy

/*
 * Copyright (c) 2021 the original author or authors
 *
 * Permission is hereby granted, free of charge, to any person obtaining a copy of
 * this software and associated documentation files (the "Software"), to deal in
 * the Software without restriction, including without limitation the rights to
 * use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of
 * the Software, and to permit persons to whom the Software is furnished to do so,
 * subject to the following conditions:
 *
 * The above copyright notice and this permission notice shall be included in all
 * copies or substantial portions of the Software.
 *
 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS
 * FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR
 * COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER
 * IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
 * CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
 */

package net.nmoncho.helenus.zio

import java.net.InetSocketAddress

import scala.util.Try

import com.datastax.oss.driver.api.core._
import com.datastax.oss.driver.api.core.cql._
import com.datastax.oss.driver.api.core.servererrors.SyntaxError
import com.datastax.oss.driver.internal.core.config.typesafe.DefaultDriverConfigLoader
import com.typesafe.config.ConfigFactory
import net.nmoncho.helenus.ScalaBoundStatement
import net.nmoncho.helenus.api.RowMapper
import zio._

object ZCqlSession {

  /** Creates and opens a [[ZCqlSession]] against a Cassandra Ring
    *
    * This method pickups the configuration from a Typesafe Config, as the
    * underlying Datastax/Apache Java Driver uses that for configuration.
    *
    * Custom configuration can be done by extending the configuration defined in
    * the path [[DefaultDriverConfigLoader.DEFAULT_ROOT_PATH]], and then using
    * `configPath` to point to that extension.
    *
    * @param configPath what Typesafe Config path to use for configuring the session, defaults to [[DefaultDriverConfigLoader.DEFAULT_ROOT_PATH]]
    * @return Scoped [[ZCqlSession]] instance
    */
  def open(
      configPath: String = DefaultDriverConfigLoader.DEFAULT_ROOT_PATH
  ): ZIO[Scope, CassandraException, ZCqlSession] =
    open(
      ZIO
        .attempt(ConfigFactory.load())
        .flatMap(config => ZDefaultCqlSession(config, configPath))
    )

  /** Creates and opens a [[ZCqlSession]] against a Cassandra Ring
    *
    * Unlike the other `open` method, this one allows you to define how to create the Cassandra session.
    *
    * See the available `apply` methods on [[ZCqlSession]] companion object.
    *
    * @param sessionTask tasks creates
    * @return Scoped [[ZCqlSession]] instance
    */
  def open(
      sessionTask: Task[ZCqlSession]
  ): ZIO[Scope, CassandraException, ZCqlSession] =
    (for {
      _ <- ZIO.logInfo("Opening Cassandra Session...")
      session <- sessionTask
      _ <- ZIO.logInfo("Cassandra session is ready")
    } yield session)
      .mapError(new SessionOpenException("Failed to open Cassandra session", _))
      .withFinalizer(close)

  /** Closes a Cassandra Session.
    *
    * To be used as finalizer
    *
    * @param session session to close
    */
  private def close(session: ZCqlSession): ZIO[Any, Nothing, Unit] =
    (for {
      _ <- ZIO.logInfo("Closing Cassandra session")
      _ <- session.close()
      _ <- ZIO.logInfo("Cassandra session closed")
    } yield ())
      // TODO add stack trace to logging
      .catchAll(t => ZIO.logError("Failed trying to close cassandra session:\n" + t.getMessage))
}

object ZDefaultCqlSession {

  /** Simple Cassandra Session creation method
    *
    * @param host host of one contact point
    * @param port port of one contact point
    * @param datacenter datatcenter of one contact point, defaults to "datacenter1"
    */
  def apply(host: String, port: Int, datacenter: String = "datacenter1"): Task[ZDefaultCqlSession] =
    apply(
      CqlSession
        .builder()
        .addContactPoint(new InetSocketAddress(host, port))
        .withLocalDatacenter(datacenter)
    )

  /** Cassandra Session creation method using reference configuration
    *
    * For more information see https://docs.datastax.com/en/developer/java-driver/latest/manual/core/configuration/reference/index.html
    *
    * @param config typesafe Config with Cassandra configuration
    * @param configPath path to configuration
    */
  def apply(config: com.typesafe.config.Config, configPath: String): Task[ZDefaultCqlSession] =
    apply(
      new CqlSessionBuilder()
        .withConfigLoader(new DefaultDriverConfigLoader(() => config.getConfig(configPath)))
    )

  /** Cassandra Session creation method using a [[CqlSessionBuilder]]
    *
    * @param builder builder with desired configuration
    */
  def apply(builder: => CqlSessionBuilder): Task[ZDefaultCqlSession] =
    ZIO.fromCompletionStage(builder.buildAsync()).map(new ZDefaultCqlSession(_))
}

trait ZCqlSession {

  /** Closes this session
    */
  def close(): IO[CassandraException, Unit]

  /** Prepares a query synchronously
    *
    * @param query query to prepare
    * @return [[PreparedStatement]]
    */
  def prepare(query: String): IO[CassandraException, PreparedStatement]

  /** Prepares a query asynchronously
    *
    * @param query query to prepare
    * @return [[PreparedStatement]]
    */
  def prepareAsync(query: String): IO[CassandraException, PreparedStatement]

  /** Executes a simple (unprepared) statement synchronously
    *
    * @param statement statement to executed
    * @return [[ResultSet]] from execution
    */
  def execute(statement: String): IO[CassandraException, ResultSet]

  /** Executes a [[ScalaBoundStatement]] synchronously
    *
    * @param stmt statement to execute
    * @tparam Out result type of every row
    * @return [[PagingIterable]] containing rows
    */
  def execute[Out: RowMapper](
      stmt: ScalaBoundStatement[Out]
  ): IO[CassandraException, PagingIterable[Try[Out]]]

  /** Executes a simple (unprepared) statement asynchronously
    *
    * @param statement statement to executed
    * @return [[AsyncResultSet]] from execution
    */
  def executeAsync(statement: String): IO[CassandraException, AsyncResultSet]

  /** Executes a [[ScalaBoundStatement]] asynchronously
    *
    * @param stmt statement to execute
    * @tparam Out result type of every row
    * @return [[MappedAsyncPagingIterable]] containing rows
    */
  def executeAsync[Out: RowMapper](
      stmt: ScalaBoundStatement[Out]
  ): IO[CassandraException, MappedAsyncPagingIterable[Try[Out]]]

  /** Executes a [[BoundStatement]] asynchronously
    *
    * @param bs statement to execute
    * @return [[AsyncResultSet]] from execution
    */
  def executeAsyncFromJava(bs: BoundStatement): IO[CassandraException, AsyncResultSet]
}

class ZDefaultCqlSession(private val session: CqlSession) extends ZCqlSession {

  override def close(): IO[CassandraException, Unit] =
    ZIO
      .fromCompletionStage(session.closeAsync())
      .map(_ => ())
      .mapError(
        new SessionClosingException("Something went wrong while closing the session", _)
      )

  override def prepare(query: String): IO[CassandraException, PreparedStatement] =
    ZIO
      .attempt(session.prepare(query))
      .mapError {
        case syntaxError: SyntaxError =>
          new InvalidStatementException(s"Invalid query syntax [$query]", syntaxError)

        case npe: NullPointerException if query == null =>
          new InvalidStatementException("Query cannot be null", npe)

        case ex =>
          new GenericCassandraException(s"Something went wrong while preparing query [$query]", ex)
      }

  override def prepareAsync(query: String): IO[CassandraException, PreparedStatement] =
    ZIO
      .fromCompletionStage(session.prepareAsync(query))
      .mapError {
        case syntaxError: SyntaxError =>
          new InvalidStatementException(s"Invalid query syntax [$query]", syntaxError)

        case npe: NullPointerException if query == null =>
          new InvalidStatementException("Query cannot be null", npe)

        case ex =>
          new GenericCassandraException(s"Something went wrong while preparing query [$query]", ex)
      }

  override def execute(statement: String): IO[CassandraException, ResultSet] =
    ZIO
      .attempt(session.execute(statement))
      .mapError(ex =>
        new StatementExecutionException(
          "Something went wrong while trying to execute a statement",
          ex
        )
      )

  override def execute[Out: RowMapper](
      statement: ScalaBoundStatement[Out]
  ): IO[CassandraException, PagingIterable[Try[Out]]] = {
    val mapper = implicitly[RowMapper[Out]]

    ZIO
      .attempt(session.execute(statement))
      .map(rs => rs.map[Try[Out]]((row: Row) => Try(mapper(row))))
      .mapError(ex =>
        new StatementExecutionException(
          "Something went wrong while trying to execute a statement",
          ex
        )
      )
  }

  override def executeAsync(statement: String): IO[CassandraException, AsyncResultSet] =
    ZIO
      .fromCompletionStage(session.executeAsync(statement))
      .mapError(ex =>
        new StatementExecutionException(
          "Something went wrong while trying to execute async a statement",
          ex
        )
      )

  override def executeAsync[Out: RowMapper](
      statement: ScalaBoundStatement[Out]
  ): IO[CassandraException, MappedAsyncPagingIterable[Try[Out]]] = {
    val mapper = implicitly[RowMapper[Out]]

    ZIO
      .fromCompletionStage(session.executeAsync(statement))
      .map(rs => rs.map[Try[Out]]((row: Row) => Try(mapper(row))))
      .mapError(ex =>
        new StatementExecutionException(
          "Something went wrong while trying to execute async a statement",
          ex
        )
      )
  }

  override def executeAsyncFromJava(bs: BoundStatement): IO[CassandraException, AsyncResultSet] =
    ZIO
      .fromCompletionStage(session.executeAsync(bs))
      .mapError(ex =>
        new StatementExecutionException(
          "Something went wrong while trying to execute a statement",
          ex
        )
      )
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy