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

raw.compiler.base.CompilerContext.scala Maven / Gradle / Ivy

/*
 * Copyright 2023 RAW Labs S.A.
 *
 * Use of this software is governed by the Business Source License
 * included in the file licenses/BSL.txt.
 *
 * As of the Change Date specified in that file, in accordance with
 * the Business Source License, use of this software will be governed
 * by the Apache License, Version 2.0, included in the file
 * licenses/APL.txt.
 */

package raw.compiler.base

import com.google.common.base.Stopwatch
import com.typesafe.scalalogging.StrictLogging
import raw.api.AuthenticatedUser
import raw.config.RawSettings
import raw.sources.SourceContext
import com.google.common.cache.{CacheBuilder, CacheLoader, LoadingCache}
import raw.inferrer.{InferrerException, InferrerProperties, InferrerService, InputFormatDescriptor}
import raw.utils._

import java.util.concurrent.{Executors, TimeUnit, TimeoutException}
import scala.concurrent.ExecutionException

object CompilerContext {
  private val INFERRER_THREAD_POOL_SIZE = "raw.compiler.inferrer.thread-pool-size"
  private val INFERRER_TIMEOUT = "raw.compiler.inferrer.timeout"
  private val INFERRER_CACHE_SIZE = "raw.compiler.inferrer.cache-size"
  private val INFERRER_EXPIRY = "raw.compiler.inferrer.expiry"
}

/**
 * Contains state that is shared between different programs.
 */
abstract class CompilerContext(
    val language: String,
    val user: AuthenticatedUser,
    val inferrer: InferrerService,
    val sourceContext: SourceContext
)(
    implicit val settings: RawSettings
) extends StrictLogging {

  import CompilerContext._

  private val inferrerTimeoutMillis = settings.getDuration(INFERRER_TIMEOUT).toMillis
  private val inferrerCacheSize = settings.getInt(INFERRER_CACHE_SIZE)
  private val inferrerExpirySeconds = settings.getDuration(INFERRER_EXPIRY).toSeconds

  private val inferrerThreadPoolSize = settings.getInt(INFERRER_THREAD_POOL_SIZE)
  private val inferrerThreadPool =
    Executors.newFixedThreadPool(inferrerThreadPoolSize, newThreadFactory("compiler-context-inferrer"))

  def infer(
      properties: InferrerProperties
  ): Either[String, InputFormatDescriptor] = {
    inferCache.get(properties)
  }

  private val inferCache: LoadingCache[InferrerProperties, Either[String, InputFormatDescriptor]] = CacheBuilder
    .newBuilder()
    .maximumSize(inferrerCacheSize)
    .expireAfterAccess(inferrerExpirySeconds, TimeUnit.SECONDS)
    .build(new CacheLoader[InferrerProperties, Either[String, InputFormatDescriptor]] {
      def load(properties: InferrerProperties): Either[String, InputFormatDescriptor] = {
        val inferrerFuture = inferrerThreadPool.submit(() => {
          try {
            Right(inferrer.infer(properties))
          } catch {
            case ex: InferrerException => Left(ex.getMessage)
          }
        })

        try {
          val start = Stopwatch.createStarted()
          val r = inferrerFuture.get(inferrerTimeoutMillis, TimeUnit.MILLISECONDS)
          logger.debug(s"Inferrer done in ${start.elapsed(TimeUnit.MILLISECONDS)} millis.")
          r
        } catch {
          // Note that in InterruptException or TimeoutException, we do not cancel the request.
          // That's because other independent programs can themselves be waiting on the 'get' and those can succeed
          // in under 'inferrerTimeoutMillis' because they started a bit later to do .get blocking call.
          // In any case, this shouldn't be too much of a problem because the LoadingCache ensure that a single request
          // is running for a single key.
          case _: TimeoutException => Left("inferrer took too long")
          case ex: ExecutionException => throw ex.getCause
        }
      }
    })

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy