raw.client.api.CompilerServiceProvider.scala Maven / Gradle / Ivy
The newest version!
/*
* 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.client.api
import raw.utils.RawSettings
import java.util.ServiceLoader
import scala.collection.JavaConverters._
import scala.collection.mutable
object CompilerServiceProvider {
private val instanceLock = new Object
private val instanceMap = new mutable.HashMap[(Set[String], Option[ClassLoader]), CompilerService]
def apply(language: String, maybeClassLoader: Option[ClassLoader] = None)(
implicit settings: RawSettings
): CompilerService = {
maybeClassLoader match {
case Some(cl) => apply(language, cl)
case None => apply(language)
}
}
def apply(language: String)(implicit settings: RawSettings): CompilerService = {
instanceLock.synchronized {
instanceMap.collectFirst { case ((l, None), i) if l.contains(language) => i } match {
case Some(instance) => instance
case None =>
val instance = build(language)
instanceMap.put((instance.language, None), instance)
instance
}
}
}
def apply(language: String, classLoader: ClassLoader)(
implicit settings: RawSettings
): CompilerService = {
instanceLock.synchronized {
instanceMap.collectFirst { case ((l, Some(cl)), i) if cl == classLoader && l.contains(language) => i } match {
case Some(instance) => instance
case None =>
val instance = build(language, Some(classLoader))
instanceMap.put((instance.language, Some(classLoader)), instance)
instance
}
}
}
// This method is only used by the test framework.
private[raw] def set(language: Set[String], instance: CompilerService): Unit = {
instanceLock.synchronized {
if (instance == null) {
// Stop and remove entries that match the `language`, regardless the class loader.
instanceMap.filterKeys(_._1 == language).foreach {
case (key, compiler) =>
compiler.stop()
instanceMap.remove(key)
}
} else {
instanceMap.put((language, None), instance)
}
}
}
private def build(language: String, maybeClassLoader: Option[ClassLoader] = None)(
implicit settings: RawSettings
): CompilerService = {
val services = maybeClassLoader match {
case Some(cl) => ServiceLoader.load(classOf[CompilerServiceBuilder], cl).asScala.toArray
case None => ServiceLoader.load(classOf[CompilerServiceBuilder]).asScala.toArray
}
if (services.isEmpty) {
throw new CompilerException("no compiler service available")
} else {
services.find(p => p.language.contains(language)) match {
case Some(builder) => builder.build(maybeClassLoader)
case None => throw new CompilerException(s"cannot find compiler service: $language")
}
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy