From 21f0f4a0b8b79a2a5ebe1ddd6487e38ef27dac34 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Ra=C3=BAl=20Raja=20Mart=C3=ADnez?= Date: Tue, 31 Oct 2023 09:31:50 +0100 Subject: [PATCH 1/4] Allow overriding React agent steps (#515) --- .../xef/reasoning/tools/ReActAgent.kt | 24 ++++++++++++++----- 1 file changed, 18 insertions(+), 6 deletions(-) diff --git a/reasoning/src/commonMain/kotlin/com/xebia/functional/xef/reasoning/tools/ReActAgent.kt b/reasoning/src/commonMain/kotlin/com/xebia/functional/xef/reasoning/tools/ReActAgent.kt index 25088ebda..1f8577308 100644 --- a/reasoning/src/commonMain/kotlin/com/xebia/functional/xef/reasoning/tools/ReActAgent.kt +++ b/reasoning/src/commonMain/kotlin/com/xebia/functional/xef/reasoning/tools/ReActAgent.kt @@ -19,7 +19,19 @@ class ReActAgent( private val scope: Conversation, private val tools: List, private val maxIterations: Int = 10, - private val configuration: PromptConfiguration = PromptConfiguration(temperature = 0.0) + private val configuration: PromptConfiguration = PromptConfiguration(temperature = 0.0), + private val critique: suspend ReActAgent.(String, Finish) -> Critique = + { input: String, finish: Finish -> + critiqueCall(input, finish) + }, + private val decide: suspend ReActAgent.(String, Int, ThoughtObservation) -> Decide = + { input: String, iterations: Int, thought: ThoughtObservation -> + decideCall(input, iterations, thought) + }, + private val finish: suspend ReActAgent.(String) -> Finish = { input: String -> + finishCall(input) + }, + private val runTool: suspend ReActAgent.() -> RunTool = { runToolCall() }, ) : Conversation by scope { sealed class Result { @@ -32,7 +44,7 @@ class ReActAgent( data class Finish(val result: String) : Result() } - private data class ThoughtObservation(val thought: String, val observation: String) + data class ThoughtObservation(val thought: String, val observation: String) @Serializable enum class NextStep { @@ -161,11 +173,11 @@ class ReActAgent( if (currentIteration > maxIterations) { emit(Result.MaxIterationsReached("🤷‍ Max iterations reached")) } else { - val decide = decideCall(prompt = prompt, thought = thought, iterations = currentIteration) + val decide = decide(prompt, currentIteration, thought) emit(Result.Log("🤖 I decided : ${decide.thought}")) when (decide.nextStep) { NextStep.RunTool -> { - val runTool = runToolCall() + val runTool = runTool() val tool = tools.find { it.name.equals(runTool.tool, ignoreCase = true) } if (tool == null) { emit(Result.Log("🤖 I don't know how to use the tool ${runTool.tool}")) @@ -187,8 +199,8 @@ class ReActAgent( } } NextStep.Finish -> { - val result = finishCall(prompt = prompt) - val critique = critiqueCall(prompt = prompt, finish = result) + val result = finish(prompt) + val critique = critique(prompt, result) emit(Result.Log("🤖 After critiquing the answer I decided : ${critique.thought}")) when (critique.outcome) { CompleteAnswerForUserRequest -> emit(Result.Finish(result.result)) From 009b8d9745886504ee7ff94c1348b38da144bc45 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Fede=20Fern=C3=A1ndez?= <720923+fedefernandez@users.noreply.github.com> Date: Tue, 31 Oct 2023 17:53:57 +0100 Subject: [PATCH 2/4] Allow custom mappings for different models (#513) --- .../xef/server/http/client/ModelUriAdapter.kt | 79 +++++++++++++++++++ .../http/client/ModelUriAdapterBuilder.kt | 17 ++++ .../xef/server/http/client/OpenAIPathType.kt | 15 ++++ .../xef/server/http/routes/AIRoutes.kt | 7 +- 4 files changed, 115 insertions(+), 3 deletions(-) create mode 100644 server/src/main/kotlin/com/xebia/functional/xef/server/http/client/ModelUriAdapter.kt create mode 100644 server/src/main/kotlin/com/xebia/functional/xef/server/http/client/ModelUriAdapterBuilder.kt create mode 100644 server/src/main/kotlin/com/xebia/functional/xef/server/http/client/OpenAIPathType.kt diff --git a/server/src/main/kotlin/com/xebia/functional/xef/server/http/client/ModelUriAdapter.kt b/server/src/main/kotlin/com/xebia/functional/xef/server/http/client/ModelUriAdapter.kt new file mode 100644 index 000000000..628560be9 --- /dev/null +++ b/server/src/main/kotlin/com/xebia/functional/xef/server/http/client/ModelUriAdapter.kt @@ -0,0 +1,79 @@ +package com.xebia.functional.xef.server.http.client + +import io.github.oshai.kotlinlogging.KotlinLogging +import io.ktor.client.* +import io.ktor.client.plugins.* +import io.ktor.client.request.* +import io.ktor.http.* +import io.ktor.http.content.* +import io.ktor.util.* +import io.ktor.util.pipeline.* +import kotlinx.serialization.json.Json +import kotlinx.serialization.json.JsonObject +import kotlinx.serialization.json.jsonPrimitive + +class ModelUriAdapter +internal constructor(private val urlMap: Map>) { + + val logger = KotlinLogging.logger {} + + fun isDefined(path: OpenAIPathType): Boolean = urlMap.containsKey(path) + + fun findPath(path: OpenAIPathType, model: String): String? = urlMap[path]?.get(model) + + companion object : HttpClientPlugin { + + override val key: AttributeKey = AttributeKey("ModelAuthAdapter") + + override fun prepare(block: ModelUriAdapterBuilder.() -> Unit): ModelUriAdapter = + ModelUriAdapterBuilder().apply(block).build() + + override fun install(plugin: ModelUriAdapter, scope: HttpClient) { + installModelAuthAdapter(plugin, scope) + } + + private fun readModelFromRequest(originalRequest: OutgoingContent.ByteArrayContent?): String? { + val requestBody = originalRequest?.bytes()?.toString(Charsets.UTF_8) + val json = requestBody?.let { Json.decodeFromString(it) } + return json?.get("model")?.jsonPrimitive?.content + } + + private fun installModelAuthAdapter(plugin: ModelUriAdapter, scope: HttpClient) { + val adaptAuthRequestPhase = PipelinePhase("ModelAuthAdaptRequest") + scope.sendPipeline.insertPhaseAfter(HttpSendPipeline.State, adaptAuthRequestPhase) + scope.sendPipeline.intercept(adaptAuthRequestPhase) { content -> + val originalPath = OpenAIPathType.from(context.url.encodedPath) ?: return@intercept + if (plugin.isDefined(originalPath)) { + val originalRequest = content as? OutgoingContent.ByteArrayContent + if (originalRequest == null) { + plugin.logger.warn { + """ + |Can't adapt the model auth. + |The body type is: ${content::class}, with Content-Type: ${context.contentType()}. + | + |If you expect serialized body, please check that you have installed the corresponding + |plugin(like `ContentNegotiation`) and set `Content-Type` header.""" + .trimMargin() + } + return@intercept + } + val model = readModelFromRequest(originalRequest) + val newURL = model?.let { plugin.findPath(originalPath, it) } + if (newURL == null) { + plugin.logger.info { + "Model auth didn't found a new url for path $originalPath and model $model" + } + } else { + val baseBuilder = URLBuilder(newURL).build() + context.url.set( + scheme = baseBuilder.protocol.name, + host = baseBuilder.host, + port = baseBuilder.port, + path = baseBuilder.encodedPath + ) + } + } + } + } + } +} diff --git a/server/src/main/kotlin/com/xebia/functional/xef/server/http/client/ModelUriAdapterBuilder.kt b/server/src/main/kotlin/com/xebia/functional/xef/server/http/client/ModelUriAdapterBuilder.kt new file mode 100644 index 000000000..a2bda8039 --- /dev/null +++ b/server/src/main/kotlin/com/xebia/functional/xef/server/http/client/ModelUriAdapterBuilder.kt @@ -0,0 +1,17 @@ +package com.xebia.functional.xef.server.http.client + +class ModelUriAdapterBuilder { + + private var pathMap: Map> = LinkedHashMap() + + fun setPathMap(pathMap: Map>) { + this.pathMap = pathMap + } + + fun addToPath(path: OpenAIPathType, vararg modelUriPaths: Pair) { + val newPathTypeMap = mapOf(*modelUriPaths.map { Pair(it.first, it.second) }.toTypedArray()) + this.pathMap += mapOf(path to newPathTypeMap) + } + + internal fun build(): ModelUriAdapter = ModelUriAdapter(pathMap) +} diff --git a/server/src/main/kotlin/com/xebia/functional/xef/server/http/client/OpenAIPathType.kt b/server/src/main/kotlin/com/xebia/functional/xef/server/http/client/OpenAIPathType.kt new file mode 100644 index 000000000..786fc2b80 --- /dev/null +++ b/server/src/main/kotlin/com/xebia/functional/xef/server/http/client/OpenAIPathType.kt @@ -0,0 +1,15 @@ +package com.xebia.functional.xef.server.http.client + +enum class OpenAIPathType(val value: String) { + CHAT("/v1/chat/completions"), + EMBEDDINGS("/v1/embeddings"), + FINE_TUNING("/v1/fine_tuning/jobs"), + FILES("/v1/files"), + IMAGES("/v1/images/generations"), + MODELS("/v1/models"), + MODERATION("/v1/moderations"); + + companion object { + fun from(v: String): OpenAIPathType? = entries.find { it.value == v } + } +} diff --git a/server/src/main/kotlin/com/xebia/functional/xef/server/http/routes/AIRoutes.kt b/server/src/main/kotlin/com/xebia/functional/xef/server/http/routes/AIRoutes.kt index 99bb38f43..407c12430 100644 --- a/server/src/main/kotlin/com/xebia/functional/xef/server/http/routes/AIRoutes.kt +++ b/server/src/main/kotlin/com/xebia/functional/xef/server/http/routes/AIRoutes.kt @@ -75,7 +75,7 @@ private suspend fun HttpClient.makeRequest( method = HttpMethod.Post setBody(body) } - call.response.headers.copyFrom(response.headers) + call.response.headers.copyFrom(response.headers, "Content-Length") call.respond(response.status, response.readBytes()) } @@ -91,17 +91,18 @@ private suspend fun HttpClient.makeStreaming( setBody(body) } .execute { httpResponse -> - call.response.headers.copyFrom(httpResponse.headers) + call.response.headers.copyFrom(httpResponse.headers, "Content-Length") call.respondOutputStream { httpResponse.bodyAsChannel().copyTo(this@respondOutputStream) } } } -private fun ResponseHeaders.copyFrom(headers: Headers) = +private fun ResponseHeaders.copyFrom(headers: Headers, vararg filterOut: String) = headers .entries() .filter { (key, _) -> !HttpHeaders.isUnsafe(key) } // setting unsafe headers results in exception + .filterNot { (key, _) -> filterOut.any { it.equals(key, true) } } .forEach { (key, values) -> values.forEach { value -> this.appendIfAbsent(key, value) } } internal fun HeadersBuilder.copyFrom(headers: Headers) = From 1ddcfb1e1993004e1a1417c7b1d923f192e94a67 Mon Sep 17 00:00:00 2001 From: Lawrence Lavigne Date: Wed, 1 Nov 2023 06:47:57 -0700 Subject: [PATCH 3/4] Scala examples refactor (#514) --- README.md | 4 +- docs/intro/scala.md | 123 +++++++----------- examples/scala/build.gradle.kts | 2 +- .../xef/examples/scala/Simple.scala | 10 ++ .../scala/context/serpapi/Simple.scala | 45 +++++++ .../scala/context/serpapi/UserQueries.scala | 20 +++ .../examples/scala/images/HybridCity.scala | 10 ++ .../scala/iteration/AnimalStory.scala | 20 +++ .../examples/scala/iteration/ChessGame.scala | 52 ++++++++ .../scala/serialization/Annotated.scala | 18 +++ .../examples/scala/serialization/Simple.scala | 37 ++++++ .../conversation/contexts/BreakingNews.scala | 22 ---- .../contexts/DivergentTasks.scala | 17 --- .../scala/conversation/contexts/Markets.scala | 24 ---- .../conversation/contexts/PDFDocument.scala | 25 ---- .../scala/conversation/contexts/Weather.scala | 15 --- .../conversation/conversations/Animal.scala | 26 ---- .../xef/scala/conversation/fields/Book.scala | 22 ---- .../scala/conversation/image/Population.scala | 19 --- .../conversation/serialization/ASCIIArt.scala | 13 -- .../conversation/serialization/ChessAI.scala | 63 --------- .../conversation/serialization/Movie.scala | 13 -- .../conversation/serialization/Recipe.scala | 13 -- gradle.properties | 2 +- scala/README.md | 9 +- .../conversation/KotlinXSerializers.scala | 38 ------ .../xef/scala/conversation/package.scala | 83 +++++------- .../serialization/KotlinXSerializers.scala | 18 +++ .../SerialDescriptor.scala | 30 ++--- .../SerialDescriptorInstances.scala | 2 +- .../SerialDescriptorSpec.scala | 2 +- 31 files changed, 334 insertions(+), 463 deletions(-) create mode 100644 examples/scala/src/main/scala/com/xebia/functional/xef/examples/scala/Simple.scala create mode 100644 examples/scala/src/main/scala/com/xebia/functional/xef/examples/scala/context/serpapi/Simple.scala create mode 100644 examples/scala/src/main/scala/com/xebia/functional/xef/examples/scala/context/serpapi/UserQueries.scala create mode 100644 examples/scala/src/main/scala/com/xebia/functional/xef/examples/scala/images/HybridCity.scala create mode 100644 examples/scala/src/main/scala/com/xebia/functional/xef/examples/scala/iteration/AnimalStory.scala create mode 100644 examples/scala/src/main/scala/com/xebia/functional/xef/examples/scala/iteration/ChessGame.scala create mode 100644 examples/scala/src/main/scala/com/xebia/functional/xef/examples/scala/serialization/Annotated.scala create mode 100644 examples/scala/src/main/scala/com/xebia/functional/xef/examples/scala/serialization/Simple.scala delete mode 100644 examples/scala/src/main/scala/com/xebia/functional/xef/scala/conversation/contexts/BreakingNews.scala delete mode 100644 examples/scala/src/main/scala/com/xebia/functional/xef/scala/conversation/contexts/DivergentTasks.scala delete mode 100644 examples/scala/src/main/scala/com/xebia/functional/xef/scala/conversation/contexts/Markets.scala delete mode 100644 examples/scala/src/main/scala/com/xebia/functional/xef/scala/conversation/contexts/PDFDocument.scala delete mode 100644 examples/scala/src/main/scala/com/xebia/functional/xef/scala/conversation/contexts/Weather.scala delete mode 100644 examples/scala/src/main/scala/com/xebia/functional/xef/scala/conversation/conversations/Animal.scala delete mode 100644 examples/scala/src/main/scala/com/xebia/functional/xef/scala/conversation/fields/Book.scala delete mode 100644 examples/scala/src/main/scala/com/xebia/functional/xef/scala/conversation/image/Population.scala delete mode 100644 examples/scala/src/main/scala/com/xebia/functional/xef/scala/conversation/serialization/ASCIIArt.scala delete mode 100644 examples/scala/src/main/scala/com/xebia/functional/xef/scala/conversation/serialization/ChessAI.scala delete mode 100644 examples/scala/src/main/scala/com/xebia/functional/xef/scala/conversation/serialization/Movie.scala delete mode 100644 examples/scala/src/main/scala/com/xebia/functional/xef/scala/conversation/serialization/Recipe.scala delete mode 100644 scala/src/main/scala/com/xebia/functional/xef/scala/conversation/KotlinXSerializers.scala create mode 100644 scala/src/main/scala/com/xebia/functional/xef/scala/serialization/KotlinXSerializers.scala rename scala/src/main/scala/com/xebia/functional/xef/scala/{conversation => serialization}/SerialDescriptor.scala (94%) rename scala/src/main/scala/com/xebia/functional/xef/scala/{conversation => serialization}/SerialDescriptorInstances.scala (98%) rename scala/src/test/scala/com/xebia/functional/xef/scala/{conversation => serialization}/SerialDescriptorSpec.scala (98%) diff --git a/README.md b/README.md index fc9f1c63b..bbbb5af8a 100644 --- a/README.md +++ b/README.md @@ -110,7 +110,7 @@ in your build, if you haven't done it before. ## 📖 Quick Introduction -In this small introduction we look at the main features of xef, including the `ai` function. +In this small introduction we look at the main features of xef, including the `conversation` function. - [Kotlin logo Kotlin version](https://github.com/xebia-functional/xef/blob/main/docs/intro/kotlin.md) - [Scala logo Scala version](https://github.com/xebia-functional/xef/blob/main/docs/intro/scala.md) @@ -121,5 +121,5 @@ In this small introduction we look at the main features of xef, including the `a You can also have a look at the examples to have a feeling of how using the library looks like. - [Kotlin logo Examples in Kotlin](https://github.com/xebia-functional/xef/tree/main/examples/kotlin/src/main/kotlin/com/xebia/functional/xef/conversation) -- [Scala logo Examples in Scala](https://github.com/xebia-functional/xef/tree/main/examples/scala/src/main/scala/com/xebia/functional/xef/scala/conversation) +- [Scala logo Examples in Scala](https://github.com/xebia-functional/xef/tree/main/examples/scala/src/main/scala/com/xebia/functional/xef/examples/scala) - [Java logo Examples in Java](https://github.com/xebia-functional/xef/tree/main/examples/java/src/main/java/com/xebia/functional/xef/java/auto) diff --git a/docs/intro/scala.md b/docs/intro/scala.md index ca47d3c19..dd5be1a56 100644 --- a/docs/intro/scala.md +++ b/docs/intro/scala.md @@ -1,51 +1,19 @@ # Quick introduction to xef.ai (Scala version) -After adding the library to your project -(see the [main README](https://github.com/xebia-functional/xef/blob/main/README.md) for instructions), -you get access to the `ai` function, which is your gate to the modern AI world. +After adding the library to your project(see the +[main README](https://github.com/xebia-functional/xef/blob/main/README.md) for instructions), +you get access to the `conversation` function, which is your port of entry to the modern AI world. Inside of it, you can _prompt_ for information, which means posing the question to an LLM -(Large Language Model). The easiest way is to just get the information back as a string. +(Large Language Model). The easiest way is to just get the information back as a string or list of strings. -```scala -import com.xebia.functional.xef.scala.auto.* +```scala 3 +import com.xebia.functional.xef.scala.conversation.* -@main def runBook: Unit = ai { - val topic: String = "functional programming" - promptMessage(s"Give me a selection of books about $topic") -}.getOrElse(ex => println(ex.getMessage)) -``` - -In the example above we _execute_ the `ai` block with `getOrElse`, so in case an exception -is thrown (for example, if your API key is not correct), we are handing the error by printing -the reason of the error. - -In the next examples we'll write functions that rely on `ai`'s DSL functionality, -but without actually extracting the values yet using `getOrThrow` or `getOrElse`. -We'll eventually call this functions from an `ai` block as we've shown above, and -this allows us to build larger pipelines, and only extract the final result at the end. - -This can be done by either using a context parameters or function _using_ `AIScope`. -Let's compare the two: - -```scala -def book(topic: String)(using scope: AIScope): List[String] = - promptMessage(s"Give me a selection of books about $topic") - -def book(topic: String): AIScope ?=> List[String] = - promptMessage(s"Give me a selection of books about $topic") -``` - -Using the type alias `AI`, defined in `com.xebia.functional.xef.scala.auto` as: - -```scala -type AI[A] = AIScope ?=> A -``` - -book function can be written in this way: - -```scala -def book(topic: String): AI[List[String]] = - promptMessage(s"Give me a selection of books about $topic") +def books(topic: String): Unit = conversation: + val topBook: String = promptMessage(s"Give me the top-selling book about $topic") + println(topBook) + val selectedBooks: List[String] = promptMessages(s"Give me a selection of books about $topic") + println(selectedBooks.mkString("\n")) ``` ## Additional setup @@ -54,7 +22,7 @@ If the code above fails, you may need to perform some additional setup. ### OpenAI -By default, the `ai` block connects to [OpenAI](https://platform.openai.com/). +By default, the `conversation` block connects to [OpenAI](https://platform.openai.com/). To use their services you should provide the corresponding API key in the `OPENAI_TOKEN` environment variable, and have enough credits. @@ -102,28 +70,29 @@ strings we obtain. Fortunately, you can also ask xef.ai to give you back the inf using a _custom type_. The library takes care of instructing the LLM on building such a structure, and deserialize the result back for you. -```scala -import com.xebia.functional.xef.scala.auto.* +This can be done by declaring a case class that `derives SerialDescriptor, Decoder`: + +```scala 3 +import com.xebia.functional.xef.scala.conversation.* +import com.xebia.functional.xef.scala.serialization.* import io.circe.Decoder -import io.circe.parser.decode -private final case class Book(name: String, author: String, summary: String) derives SerialDescriptor, Decoder +case class Book(name: String, author: String, pages: Int) derives SerialDescriptor, Decoder +``` -def summarizeBook(title: String, author: String)(using scope: AIScope): Book = - prompt(s"$title by $author summary.") +The `conversation` block can then be written in this way: -@main def runBook: Unit = - ai { - val toKillAMockingBird = summarizeBook("To Kill a Mockingbird", "Harper Lee") - println(s"${toKillAMockingBird.name} by ${toKillAMockingBird.author} summary:\n ${toKillAMockingBird.summary}") - }.getOrElse(ex => println(ex.getMessage)) +```scala 3 +def bookExample(topic: String): Unit = conversation: + val Book(title, author, pages) = prompt[Book](s"Give me the best-selling book about $topic") + println(s"The book $title is by $author and has $pages pages.") ``` -xef.ai for Scala uses xef.ai core, which it's based on Kotlin. Hence, the core +xef.ai for Scala uses xef.ai core, which is based on the Kotlin implementation. Hence, the core reuses [Kotlin's common serialization](https://kotlinlang.org/docs/serialization.html), and Scala uses [circe](https://github.com/circe/circe) to derive the required serializable instance. -The LLM is usually able to detect which kind of information should -go on each field based on its name (like `title` and `author` above). +The LLM is usually able to detect which kind of information should go in each field based on its name +(like `title` and `author` above). ## Context @@ -133,37 +102,45 @@ often want to supplement the LLM with more data: - Transient information referring to the current moment, like the current weather, or the trends in the stock market in the past 10 days. - Non-public information, for example for summarizing a piece of text you're creating - within you organization. + within your organization. These additional pieces of information are called the _context_ in xef.ai, and are attached to every question to the LLM. Although you can add arbitrary strings to the context at any point, the most common mode of usage is using an _agent_ to consult an external service, -and make its response part of the context. One such agent is `search`, which uses a web -search service to enrich that context. +and make its response part of the context. One such agent is `search`, which uses the +[Google Search API (SerpApi)](https://serpapi.com/) to enrich that context. + +(Note that a SerpApi token may be required to run this example.) + +```scala 3 +import com.xebia.functional.xef.conversation.llm.openai.* +import com.xebia.functional.xef.reasoning.serpapi.* +import com.xebia.functional.xef.scala.conversation.* +import com.xebia.functional.xef.scala.serialization.* +import io.circe.Decoder -```scala -import com.xebia.functional.xef.scala.agents.DefaultSearch -import com.xebia.functional.xef.scala.auto.* +val openAI: OpenAI = OpenAI.FromEnvironment -private def getQuestionAnswer(question: String)(using scope: AIScope): List[String] = - contextScope(DefaultSearch.search("Weather in Cádiz, Spain")) { - promptMessage(question) - } +def setContext(query: String)(using conversation: ScalaConversation): Unit = + addContext(Search(openAI.DEFAULT_CHAT, conversation, 3).search(query).get) -@main def runWeather: Unit = ai { +@main def runWeather(): Unit = conversation: + setContext("Weather in Cádiz, Spain") val question = "Knowing this forecast, what clothes do you recommend I should wear if I live in Cádiz?" - println(getQuestionAnswer(question).mkString("\n")) -}.getOrElse(ex => println(ex.getMessage)) + val answer = promptMessage(question) + println(answer) ``` > **Note** > The underlying mechanism of the context is a _vector store_, a data structure which > saves a set of strings, and is able to find those similar to another given one. -> By default xef.ai uses an _in-memory_ vector store, since it provides maximum +> By default xef.ai uses an _in-memory_ vector store, since this provides maximum > compatibility across platforms. However, if you foresee your context growing above > the hundreds of elements, you may consider switching to another alternative, like > Lucene or PostgreSQL. ## Examples -Check out the [examples folder](https://github.com/xebia-functional/xef/blob/main/examples/scala/auto) for a complete list of different use cases. +Check out the +[examples folder](https://github.com/xebia-functional/xef/blob/main/examples/scala/src/main/scala/com/xebia/functional/xef/examples) +for a complete list of different use cases. diff --git a/examples/scala/build.gradle.kts b/examples/scala/build.gradle.kts index 3ce0f1895..a88198b95 100644 --- a/examples/scala/build.gradle.kts +++ b/examples/scala/build.gradle.kts @@ -16,7 +16,7 @@ java { dependencies { implementation(projects.xefCore) implementation(projects.xefScala) - implementation(projects.xefReasoning) + implementation(projects.xefReasoning) implementation(projects.xefOpenai) implementation(libs.circe.parser) implementation(libs.scala.lang) diff --git a/examples/scala/src/main/scala/com/xebia/functional/xef/examples/scala/Simple.scala b/examples/scala/src/main/scala/com/xebia/functional/xef/examples/scala/Simple.scala new file mode 100644 index 000000000..c9f4b17d1 --- /dev/null +++ b/examples/scala/src/main/scala/com/xebia/functional/xef/examples/scala/Simple.scala @@ -0,0 +1,10 @@ +package com.xebia.functional.xef.examples.scala + +import com.xebia.functional.xef.scala.conversation.* + +@main def runBooks(): Unit = conversation: + val topic = "functional programming" + val topBook: String = promptMessage(s"Give me the top-selling book about $topic") + println(topBook) + val selectedBooks: List[String] = promptMessages(s"Give me a selection of books about $topic") + println(selectedBooks.mkString("\n")) diff --git a/examples/scala/src/main/scala/com/xebia/functional/xef/examples/scala/context/serpapi/Simple.scala b/examples/scala/src/main/scala/com/xebia/functional/xef/examples/scala/context/serpapi/Simple.scala new file mode 100644 index 000000000..fbe40da7e --- /dev/null +++ b/examples/scala/src/main/scala/com/xebia/functional/xef/examples/scala/context/serpapi/Simple.scala @@ -0,0 +1,45 @@ +package com.xebia.functional.xef.examples.scala.context.serpapi + +import com.xebia.functional.xef.conversation.llm.openai.* +import com.xebia.functional.xef.reasoning.serpapi.* +import com.xebia.functional.xef.scala.conversation.* +import com.xebia.functional.xef.scala.serialization.* +import io.circe.Decoder + +import java.text.SimpleDateFormat +import java.util.Date + +val openAI: OpenAI = OpenAI.FromEnvironment + +val sdf = SimpleDateFormat("dd/M/yyyy") +def currentDate: String = sdf.format(new Date) + +def setContext(query: String)(using conversation: ScalaConversation): Unit = + addContext(Search(openAI.DEFAULT_CHAT, conversation, 3).search(query).get) + +case class BreakingNews(summary: String) derives SerialDescriptor, Decoder + +case class MarketNews(news: String, risingStockSymbols: List[String], fallingStockSymbols: List[String]) derives SerialDescriptor, Decoder + +case class Estimate(number: Long) derives SerialDescriptor, Decoder + +@main def runWeather(): Unit = conversation: + setContext("Weather in Cádiz, Spain") + val question = "Knowing this forecast, what clothes do you recommend I should wear if I live in Cádiz?" + val answer = promptMessage(question) + println(answer) + +@main def runBreakingNews(): Unit = conversation: + setContext(s"$currentDate COVID News") + val BreakingNews(summary) = prompt[BreakingNews](s"Write a summary of about 300 words given the provided context.") + println(summary) + +@main def runMarketNews(): Unit = conversation: + setContext(s"$currentDate Stock market results, rising stocks, falling stocks") + val news = prompt[MarketNews]("Write a short summary of the stock market results given the provided context.") + println(news) + +@main def runFermiEstimate(): Unit = conversation: + setContext("Estimate the number of medical needles in the world") + val Estimate(needlesInWorld) = prompt[Estimate]("Answer the question with an integer number given the provided context.") + println(s"Needles in world: $needlesInWorld") diff --git a/examples/scala/src/main/scala/com/xebia/functional/xef/examples/scala/context/serpapi/UserQueries.scala b/examples/scala/src/main/scala/com/xebia/functional/xef/examples/scala/context/serpapi/UserQueries.scala new file mode 100644 index 000000000..711db19da --- /dev/null +++ b/examples/scala/src/main/scala/com/xebia/functional/xef/examples/scala/context/serpapi/UserQueries.scala @@ -0,0 +1,20 @@ +package com.xebia.functional.xef.examples.scala.context.serpapi + +import com.xebia.functional.xef.reasoning.pdf.PDF +import com.xebia.functional.xef.scala.conversation.* +import com.xebia.functional.xef.scala.serialization.* +import io.circe.Decoder + +import scala.io.StdIn.readLine + +case class AIResponse(answer: String) derives SerialDescriptor, Decoder + +val PdfUrl = "https://people.cs.ksu.edu/~schmidt/705a/Scala/Programming-in-Scala.pdf" + +@main def runUserQueries(): Unit = conversation: + val pdf = PDF(openAI.DEFAULT_CHAT, openAI.DEFAULT_SERIALIZATION, summon[ScalaConversation]) + addContext(Array(pdf.readPDFFromUrl.readPDFFromUrl(PdfUrl).get)) + while (true) + println("Enter your question: ") + val AIResponse(answer) = prompt[AIResponse](readLine()) + println(s"$answer\n---\n") diff --git a/examples/scala/src/main/scala/com/xebia/functional/xef/examples/scala/images/HybridCity.scala b/examples/scala/src/main/scala/com/xebia/functional/xef/examples/scala/images/HybridCity.scala new file mode 100644 index 000000000..1c9a3b4a9 --- /dev/null +++ b/examples/scala/src/main/scala/com/xebia/functional/xef/examples/scala/images/HybridCity.scala @@ -0,0 +1,10 @@ +package com.xebia.functional.xef.examples.scala.images + +import com.xebia.functional.xef.prompt.Prompt +import com.xebia.functional.xef.scala.conversation.* +import com.xebia.functional.xef.scala.serialization.* +import io.circe.Decoder + +@main def runHybridCity(): Unit = conversation: + val imageUrls = images(Prompt("A hybrid city of Cádiz, Spain and Seattle, US.")) + println(imageUrls) diff --git a/examples/scala/src/main/scala/com/xebia/functional/xef/examples/scala/iteration/AnimalStory.scala b/examples/scala/src/main/scala/com/xebia/functional/xef/examples/scala/iteration/AnimalStory.scala new file mode 100644 index 000000000..19e6d433e --- /dev/null +++ b/examples/scala/src/main/scala/com/xebia/functional/xef/examples/scala/iteration/AnimalStory.scala @@ -0,0 +1,20 @@ +package com.xebia.functional.xef.examples.scala.iteration + +import com.xebia.functional.xef.prompt.JvmPromptBuilder +import com.xebia.functional.xef.scala.conversation.* +import com.xebia.functional.xef.scala.serialization.* +import io.circe.Decoder + +case class Animal(name: String, habitat: String, diet: String) derives SerialDescriptor, Decoder +case class Invention(name: String, inventor: String, year: Int, purpose: String) derives SerialDescriptor, Decoder + +@main def runAnimalStory(): Unit = conversation: + val animal = prompt[Animal]("A unique animal species") + val invention = prompt[Invention]("A groundbreaking invention from the 20th century.") + println(s"Animal: $animal") + println(s"Invention: $invention") + val builder = new JvmPromptBuilder() + .addSystemMessage("You are a writer for a science fiction magazine.") + .addUserMessage("Write a short story of 200 words that involves the animal and the invention.") + val story = promptMessage(builder.build) + println(story) diff --git a/examples/scala/src/main/scala/com/xebia/functional/xef/examples/scala/iteration/ChessGame.scala b/examples/scala/src/main/scala/com/xebia/functional/xef/examples/scala/iteration/ChessGame.scala new file mode 100644 index 000000000..42472e1a0 --- /dev/null +++ b/examples/scala/src/main/scala/com/xebia/functional/xef/examples/scala/iteration/ChessGame.scala @@ -0,0 +1,52 @@ +package com.xebia.functional.xef.examples.scala.iteration + +import com.xebia.functional.xef.prompt.Prompt +import com.xebia.functional.xef.scala.conversation.* +import com.xebia.functional.xef.scala.serialization.* +import io.circe.Decoder + +import scala.annotation.tailrec + +case class ChessMove(player: String, move: String) derives SerialDescriptor, Decoder +case class ChessBoard(board: String) derives SerialDescriptor, Decoder +case class GameState(ended: Boolean = false, winner: Option[String] = None) derives SerialDescriptor, Decoder + +@tailrec +private def chessGame(moves: List[ChessMove] = Nil, gameState: GameState = new GameState)(using ScalaConversation): (String, ChessMove) = + if !gameState.ended then + val currentPlayer = if moves.size % 2 == 0 then "Player 1 (White)" else "Player 2 (Black)" + val previousMoves = moves.map(m => m.player + ":" + m.move).mkString(", ") + val movePrompt = moves match { + case Nil => s""" + |$currentPlayer, you are playing chess and it's your turn. + |Make your first move: + """.stripMargin + case l => s""" + |$currentPlayer, you are playing chess and it's your turn. + |Here are the previous moves: $previousMoves + |Make your next move: + """.stripMargin + } + println(movePrompt) + val move = prompt[ChessMove](movePrompt) + println(s"Move is: $move") + val boardPrompt = + s""" + |Given the following chess moves: $previousMoves, + |generate a chess board on a table with appropriate emoji representations for each move and piece. + |Add a brief description of the move and its implications. + """.stripMargin + val chessBoard = prompt[ChessBoard](boardPrompt) + println(s"Current board:\n${chessBoard.board}") + val gameStatePrompt = + s""" + |Given the following chess moves: ${moves.mkString(", ")}, + |has the game ended (win, draw, or stalemate)? + """.stripMargin + val gameState = prompt[GameState](gameStatePrompt) + chessGame(moves :+ move, gameState) + else (gameState.winner.getOrElse("Something went wrong"), moves.last) + +@main def runChessGame(): Unit = conversation: + val (winner, fMove) = chessGame() + println(s"Game over. Final move: $fMove, Winner: $winner") diff --git a/examples/scala/src/main/scala/com/xebia/functional/xef/examples/scala/serialization/Annotated.scala b/examples/scala/src/main/scala/com/xebia/functional/xef/examples/scala/serialization/Annotated.scala new file mode 100644 index 000000000..168599987 --- /dev/null +++ b/examples/scala/src/main/scala/com/xebia/functional/xef/examples/scala/serialization/Annotated.scala @@ -0,0 +1,18 @@ +package com.xebia.functional.xef.examples.scala.serialization + +import com.xebia.functional.xef.prompt.Prompt +import com.xebia.functional.xef.scala.conversation.* +import com.xebia.functional.xef.scala.serialization.* +import io.circe.Decoder + +@Description("A book") +case class AnnotatedBook( + @Description("The name of the book") name: String, + @Description("The author of the book") author: String, + @Description("A 50 word paragraph with a summary of this book") summary: String +) derives SerialDescriptor, + Decoder + +@main def runAnnotatedBook(): Unit = conversation: + val book = prompt[AnnotatedBook]("To Kill a Mockingbird") + println(book) diff --git a/examples/scala/src/main/scala/com/xebia/functional/xef/examples/scala/serialization/Simple.scala b/examples/scala/src/main/scala/com/xebia/functional/xef/examples/scala/serialization/Simple.scala new file mode 100644 index 000000000..2652fd8a3 --- /dev/null +++ b/examples/scala/src/main/scala/com/xebia/functional/xef/examples/scala/serialization/Simple.scala @@ -0,0 +1,37 @@ +package com.xebia.functional.xef.examples.scala.serialization + +import com.xebia.functional.xef.scala.conversation.* +import com.xebia.functional.xef.scala.serialization.* +import io.circe.Decoder + +case class AsciiArt(art: String) derives SerialDescriptor, Decoder + +case class Book(title: String, author: String, pages: Int) derives SerialDescriptor, Decoder + +case class City(population: Int, description: String) derives SerialDescriptor, Decoder + +case class Movie(title: String, genre: String, director: String) derives SerialDescriptor, Decoder + +case class Recipe(name: String, ingredients: List[String]) derives SerialDescriptor, Decoder + +@main def runAsciiArt(): Unit = conversation: + val AsciiArt(art) = prompt[AsciiArt]("ASCII art of a cat dancing") + println(art) + +@main def runBook(): Unit = conversation: + val topic = "functional programming" + val Book(title, author, pages) = prompt[Book](s"Give me the best-selling book about $topic") + println(s"The book $title is by $author and has $pages pages.") + +@main def runMovie(): Unit = conversation: + val Movie(title, genre, director) = prompt[Movie]("Inception movie genre and director.") + println(s"The movie $title is a $genre film directed by $director.") + +@main def runCities(): Unit = conversation: + val cadiz = prompt[City]("Cádiz, Spain") + val seattle = prompt[City]("Seattle, WA") + println(s"The population of Cádiz is ${cadiz.population} and the population of Seattle is ${seattle.population}.") + +@main def runRecipe(): Unit = conversation: + val Recipe(name, ingredients) = prompt[Recipe]("Recipe for chocolate chip cookies.") + println(s"The recipe for $name is $ingredients.") diff --git a/examples/scala/src/main/scala/com/xebia/functional/xef/scala/conversation/contexts/BreakingNews.scala b/examples/scala/src/main/scala/com/xebia/functional/xef/scala/conversation/contexts/BreakingNews.scala deleted file mode 100644 index 10316ae6d..000000000 --- a/examples/scala/src/main/scala/com/xebia/functional/xef/scala/conversation/contexts/BreakingNews.scala +++ /dev/null @@ -1,22 +0,0 @@ -package com.xebia.functional.xef.scala.conversation.contexts - -import com.xebia.functional.xef.conversation.llm.openai.OpenAI -import com.xebia.functional.xef.prompt.Prompt -import com.xebia.functional.xef.reasoning.serpapi.Search -import com.xebia.functional.xef.scala.conversation.* -import io.circe.Decoder - -import java.text.SimpleDateFormat -import java.util.Date - -private final case class BreakingNewsAboutCovid(summary: String) derives SerialDescriptor, Decoder - -@main def runBreakingNews: Unit = - conversation { - val sdf = SimpleDateFormat("dd/M/yyyy") - val currentDate = sdf.format(Date()) - val search = Search(OpenAI.FromEnvironment.DEFAULT_CHAT, summon[ScalaConversation], 3) - addContext(search.search(s"$currentDate Covid News").get()) - val news = prompt[BreakingNewsAboutCovid](Prompt(s"Write a paragraph of about 300 words about: $currentDate Covid News")) - println(news.summary) - } diff --git a/examples/scala/src/main/scala/com/xebia/functional/xef/scala/conversation/contexts/DivergentTasks.scala b/examples/scala/src/main/scala/com/xebia/functional/xef/scala/conversation/contexts/DivergentTasks.scala deleted file mode 100644 index bd842f932..000000000 --- a/examples/scala/src/main/scala/com/xebia/functional/xef/scala/conversation/contexts/DivergentTasks.scala +++ /dev/null @@ -1,17 +0,0 @@ -package com.xebia.functional.xef.scala.conversation.contexts - -import com.xebia.functional.xef.conversation.llm.openai.OpenAI -import com.xebia.functional.xef.prompt.Prompt -import com.xebia.functional.xef.reasoning.serpapi.Search -import com.xebia.functional.xef.scala.conversation.* -import io.circe.Decoder - -private final case class NumberOfMedicalNeedlesInWorld(numberOfNeedles: Long) derives SerialDescriptor, Decoder - -@main def runDivergentTasks: Unit = - conversation { - val search = Search(OpenAI.FromEnvironment.DEFAULT_CHAT, summon[ScalaConversation], 3) - addContext(search.search("Estimate amount of medical needles in the world").get()) - val needlesInWorld = prompt[NumberOfMedicalNeedlesInWorld](Prompt("Provide the number of medical needles in the world as an integer number")) - println(s"Needles in world: ${needlesInWorld.numberOfNeedles}") - } diff --git a/examples/scala/src/main/scala/com/xebia/functional/xef/scala/conversation/contexts/Markets.scala b/examples/scala/src/main/scala/com/xebia/functional/xef/scala/conversation/contexts/Markets.scala deleted file mode 100644 index 3eb8a0814..000000000 --- a/examples/scala/src/main/scala/com/xebia/functional/xef/scala/conversation/contexts/Markets.scala +++ /dev/null @@ -1,24 +0,0 @@ -package com.xebia.functional.xef.scala.conversation.contexts - -import com.xebia.functional.xef.conversation.llm.openai.OpenAI -import com.xebia.functional.xef.prompt.Prompt -import com.xebia.functional.xef.reasoning.serpapi.Search -import com.xebia.functional.xef.scala.conversation.* -import io.circe.Decoder - -import java.text.SimpleDateFormat -import java.util.Date - -private final case class MarketNews(news: String, raisingStockSymbols: List[String], decreasingStockSymbols: List[String]) - derives SerialDescriptor, - Decoder - -@main def runMarkets: Unit = - conversation { - val sdf = SimpleDateFormat("dd/M/yyyy") - val currentDate = sdf.format(Date()) - val search = Search(OpenAI.FromEnvironment.DEFAULT_CHAT, summon[ScalaConversation], 3) - addContext(search.search(s"$currentDate Stock market results, raising stocks, decreasing stocks").get()) - val news = prompt[MarketNews](Prompt("Write a short summary of the stock market results given the provided context.")) - println(news) - } diff --git a/examples/scala/src/main/scala/com/xebia/functional/xef/scala/conversation/contexts/PDFDocument.scala b/examples/scala/src/main/scala/com/xebia/functional/xef/scala/conversation/contexts/PDFDocument.scala deleted file mode 100644 index 2aa1d5bf8..000000000 --- a/examples/scala/src/main/scala/com/xebia/functional/xef/scala/conversation/contexts/PDFDocument.scala +++ /dev/null @@ -1,25 +0,0 @@ -package com.xebia.functional.xef.scala.conversation.contexts - -import com.xebia.functional.xef.conversation.llm.openai.OpenAI -import com.xebia.functional.xef.prompt.Prompt -import com.xebia.functional.xef.reasoning.pdf.PDF -import com.xebia.functional.xef.scala.conversation.* -import io.circe.Decoder - -import scala.io.StdIn.readLine - -private final case class AIResponse(answer: String) derives SerialDescriptor, Decoder - -val pdfUrl = "https://people.cs.ksu.edu/~schmidt/705a/Scala/Programming-in-Scala.pdf" - -@main def runPDFDocument: Unit = - conversation { - val pdf = PDF(OpenAI.FromEnvironment.DEFAULT_CHAT, OpenAI.FromEnvironment.DEFAULT_SERIALIZATION, summon[ScalaConversation]) - addContext(Array(pdf.readPDFFromUrl.readPDFFromUrl(pdfUrl).get())) - while (true) { - println("Enter your question: ") - val line = scala.io.StdIn.readLine() - val response = prompt[AIResponse](Prompt(line)) - println(s"${response.answer}\n---\n") - } - } diff --git a/examples/scala/src/main/scala/com/xebia/functional/xef/scala/conversation/contexts/Weather.scala b/examples/scala/src/main/scala/com/xebia/functional/xef/scala/conversation/contexts/Weather.scala deleted file mode 100644 index 453700db7..000000000 --- a/examples/scala/src/main/scala/com/xebia/functional/xef/scala/conversation/contexts/Weather.scala +++ /dev/null @@ -1,15 +0,0 @@ -package com.xebia.functional.xef.scala.conversation - -import com.xebia.functional.xef.scala.conversation.* -import com.xebia.functional.xef.conversation.llm.openai.OpenAI -import com.xebia.functional.xef.reasoning.serpapi.Search -import com.xebia.functional.xef.prompt.Prompt - -private def getQuestionAnswer(question: Prompt)(using conversation: ScalaConversation): String = - val search: Search = Search(OpenAI.FromEnvironment.DEFAULT_CHAT, conversation, 3) - addContext(search.search("Weather in Cádiz, Spain").get()) - promptMessage(question) - -@main def runWeather: Unit = - val question = Prompt("Knowing this forecast, what clothes do you recommend I should wear if I live in Cádiz?") - println(conversation(getQuestionAnswer(question)).mkString("\n")) diff --git a/examples/scala/src/main/scala/com/xebia/functional/xef/scala/conversation/conversations/Animal.scala b/examples/scala/src/main/scala/com/xebia/functional/xef/scala/conversation/conversations/Animal.scala deleted file mode 100644 index 4f3405bb2..000000000 --- a/examples/scala/src/main/scala/com/xebia/functional/xef/scala/conversation/conversations/Animal.scala +++ /dev/null @@ -1,26 +0,0 @@ -package com.xebia.functional.xef.scala.conversation.conversations - -import com.xebia.functional.xef.prompt.{JvmPromptBuilder, Prompt} -import com.xebia.functional.xef.scala.conversation.* -import io.circe.Decoder - -private final case class Animal(name: String, habitat: String, diet: String) derives SerialDescriptor, Decoder - -private final case class Invention(name: String, inventor: String, year: Int, purpose: String) derives SerialDescriptor, Decoder - -@main def runAnimal: Unit = - conversation { - val animal: Animal = prompt(Prompt("A unique animal species")) - val invention: Invention = prompt(Prompt("A groundbreaking invention from the 20th century.")) - - println(s"Animal: $animal") - println(s"Invention: $invention") - - val builder = new JvmPromptBuilder() - .addSystemMessage("You are a writer for a science fiction magazine.") - .addUserMessage("Write a short story of 200 words that involves the animal and the invention") - - val story = promptMessage(builder.build()) - - println(story) - } diff --git a/examples/scala/src/main/scala/com/xebia/functional/xef/scala/conversation/fields/Book.scala b/examples/scala/src/main/scala/com/xebia/functional/xef/scala/conversation/fields/Book.scala deleted file mode 100644 index 15f278c5f..000000000 --- a/examples/scala/src/main/scala/com/xebia/functional/xef/scala/conversation/fields/Book.scala +++ /dev/null @@ -1,22 +0,0 @@ -package com.xebia.functional.xef.scala.conversation.fields - -import com.xebia.functional.xef.prompt.Prompt -import com.xebia.functional.xef.scala.conversation.* -import io.circe.Decoder - -@Description("A book") -case class Book( - @Description("the name of the book") name: String, - @Description("the author of the book") author: String, - @Description("A 50 word paragraph with a summary of this book") summary: String -) derives SerialDescriptor, - Decoder - -def summarizeBook(title: String, author: String)(using conversation: ScalaConversation): Book = - prompt(Prompt(s"$title by $author summary.")) - -@main def runBook: Unit = - conversation { - val toKillAMockingBird = summarizeBook("To Kill a Mockingbird", "Harper Lee") - println(s"${toKillAMockingBird.name} by ${toKillAMockingBird.author} summary:\n ${toKillAMockingBird.summary}") - } diff --git a/examples/scala/src/main/scala/com/xebia/functional/xef/scala/conversation/image/Population.scala b/examples/scala/src/main/scala/com/xebia/functional/xef/scala/conversation/image/Population.scala deleted file mode 100644 index 62dc2b1c4..000000000 --- a/examples/scala/src/main/scala/com/xebia/functional/xef/scala/conversation/image/Population.scala +++ /dev/null @@ -1,19 +0,0 @@ -package com.xebia.functional.xef.scala.conversation.image - -import com.xebia.functional.xef.llm.models.images.ImagesGenerationResponse -import com.xebia.functional.xef.prompt.Prompt -import com.xebia.functional.xef.scala.conversation.* -import io.circe.Decoder - -private final case class Population(size: Int, description: String) derives SerialDescriptor, Decoder - -private final case class Image(description: String, url: String) derives SerialDescriptor, Decoder - -@main def runPopulation: Unit = - conversation { - val cadiz: Population = prompt(Prompt("Population of Cádiz, Spain.")) - val seattle: Population = prompt(Prompt("Population of Seattle, WA.")) - val imgs: ImagesGenerationResponse = images(Prompt("A hybrid city of Cádiz, Spain and Seattle, US.")) - println(imgs) - println(s"The population of Cádiz is ${cadiz.size} and the population of Seattle is ${seattle.size}") - } diff --git a/examples/scala/src/main/scala/com/xebia/functional/xef/scala/conversation/serialization/ASCIIArt.scala b/examples/scala/src/main/scala/com/xebia/functional/xef/scala/conversation/serialization/ASCIIArt.scala deleted file mode 100644 index ea963b93c..000000000 --- a/examples/scala/src/main/scala/com/xebia/functional/xef/scala/conversation/serialization/ASCIIArt.scala +++ /dev/null @@ -1,13 +0,0 @@ -package com.xebia.functional.xef.scala.conversation.serialization - -import com.xebia.functional.xef.prompt.Prompt -import com.xebia.functional.xef.scala.conversation.* -import io.circe.Decoder - -private final case class ASCIIArt(art: String) derives SerialDescriptor, Decoder - -@main def runASCIIArt: Unit = - lazy val asciiArt = conversation { - prompt[ASCIIArt](Prompt("ASCII art of a cat dancing")) - } - println(asciiArt.art) diff --git a/examples/scala/src/main/scala/com/xebia/functional/xef/scala/conversation/serialization/ChessAI.scala b/examples/scala/src/main/scala/com/xebia/functional/xef/scala/conversation/serialization/ChessAI.scala deleted file mode 100644 index d4dde7858..000000000 --- a/examples/scala/src/main/scala/com/xebia/functional/xef/scala/conversation/serialization/ChessAI.scala +++ /dev/null @@ -1,63 +0,0 @@ -package com.xebia.functional.xef.scala.conversation.serialization - -import com.xebia.functional.xef.prompt.Prompt -import com.xebia.functional.xef.scala.conversation.* -import io.circe.Decoder - -import scala.annotation.tailrec - -private final case class ChessMove(player: String, move: String) derives SerialDescriptor, Decoder - -private final case class ChessBoard(board: String) derives SerialDescriptor, Decoder - -private final case class GameState(ended: Boolean, winner: Option[String]) derives SerialDescriptor, Decoder - -@tailrec -private def chessGame(moves: List[ChessMove], gameState: GameState)(using conversation: ScalaConversation): (String, ChessMove) = - if !gameState.ended then - val currentPlayer = if moves.size % 2 == 0 then "Player 1 (White)" else "Player 2 (Black)" - - val previousMoves = moves.map(m => m.player + ":" + m.move).mkString(", ") - - val movePrompt = moves match { - case Nil => s""" - |$currentPlayer, you are playing chess and it's your turn. - |These are no previous chess board moves. - |Make your first move: - """.stripMargin - case l => s""" - |$currentPlayer, you are playing chess and it's your turn. - |These are the previous chess board moves: $previousMoves - |Make your next move: - """.stripMargin - } - println(movePrompt) - val move: ChessMove = prompt(Prompt(movePrompt)) - println(s"Move is: $move") - - val boardPrompt = - s""" - |Given the following chess moves: $previousMoves, - |generate a chess board on a table with appropriate emoji representations for each move and piece. - |Add a brief description of the move and it's implications - """.stripMargin - - val chessBoard: ChessBoard = prompt(Prompt(boardPrompt)) - println(s"Current board:\n${chessBoard.board}") - - val gameStatePrompt = - Prompt(s""" - |Given the following chess moves: ${moves.mkString(", ")}, - |has the game ended (win, draw, or stalemate)? - """.stripMargin) - - val gameState: GameState = prompt(gameStatePrompt) - - chessGame(moves :+ move, gameState) - else (gameState.winner.getOrElse("Something went wrong"), moves.last) - -@main def runChess: Unit = - conversation { - val (winner, fMove) = chessGame(Nil, GameState(false, None)) - println(s"Game over. Final move: $fMove, Winner: $winner") - } diff --git a/examples/scala/src/main/scala/com/xebia/functional/xef/scala/conversation/serialization/Movie.scala b/examples/scala/src/main/scala/com/xebia/functional/xef/scala/conversation/serialization/Movie.scala deleted file mode 100644 index 6eb070589..000000000 --- a/examples/scala/src/main/scala/com/xebia/functional/xef/scala/conversation/serialization/Movie.scala +++ /dev/null @@ -1,13 +0,0 @@ -package com.xebia.functional.xef.scala.conversation.serialization - -import com.xebia.functional.xef.prompt.Prompt -import com.xebia.functional.xef.scala.conversation.* -import io.circe.Decoder - -private final case class Movie(title: String, genre: String, director: String) derives SerialDescriptor, Decoder - -@main def runMovie: Unit = - conversation { - val movie = prompt[Movie](Prompt("Inception movie genre and director.")) - println(s"The movie ${movie.title} is a ${movie.genre} film directed by ${movie.director}.") - } diff --git a/examples/scala/src/main/scala/com/xebia/functional/xef/scala/conversation/serialization/Recipe.scala b/examples/scala/src/main/scala/com/xebia/functional/xef/scala/conversation/serialization/Recipe.scala deleted file mode 100644 index f548c49f2..000000000 --- a/examples/scala/src/main/scala/com/xebia/functional/xef/scala/conversation/serialization/Recipe.scala +++ /dev/null @@ -1,13 +0,0 @@ -package com.xebia.functional.xef.scala.conversation.serialization - -import com.xebia.functional.xef.prompt.Prompt -import com.xebia.functional.xef.scala.conversation.* -import io.circe.Decoder - -private final case class Recipe(name: String, ingredients: List[String]) derives SerialDescriptor, Decoder - -@main def runRecipe: Unit = - conversation { - val recipe: Recipe = prompt(Prompt("Recipe for chocolate chip cookies.")) - println(s"The recipe for ${recipe.name} is ${recipe.ingredients}") - } diff --git a/gradle.properties b/gradle.properties index 9400b3a33..a0979c287 100644 --- a/gradle.properties +++ b/gradle.properties @@ -22,4 +22,4 @@ systemProp.org.gradle.unsafe.kotlin.assignment=true # Workaround to disable Dokka setup from Arrow Gradle Config dokkaEnabled=false -scalaVersion=3.3.0 +scalaVersion=3.3.1 diff --git a/scala/README.md b/scala/README.md index 611295003..c04cd2a66 100644 --- a/scala/README.md +++ b/scala/README.md @@ -9,7 +9,8 @@ Build the project locally, from the project root: ## Scalafmt The Scala module uses the [spotless](https://github.com/diffplug/spotless/tree/main/plugin-gradle#scala) plugin. -Therefore, the previous command (`./gradlew build`) will fail in case there is any formatting issue. To apply format, you can run the following command: +Therefore, the previous command (`./gradlew build`) will fail if there are any formatting issues. +To apply formatting, you can run the following command: ```bash ./gradlew spotlessApply @@ -23,6 +24,6 @@ Check out some use case at the [Scala examples](../examples/scala) folder. How to run the examples (from IntelliJ IDEA): -* Set Java version 19 -* Set VM options: "--enable-preview" -* Set Env variable: "OPENAI_TOKEN=xxx" +* Set Java version 20 +* Set VM options: `--enable-preview` +* Set Env variable: `OPENAI_TOKEN=xxx` diff --git a/scala/src/main/scala/com/xebia/functional/xef/scala/conversation/KotlinXSerializers.scala b/scala/src/main/scala/com/xebia/functional/xef/scala/conversation/KotlinXSerializers.scala deleted file mode 100644 index cd57424c2..000000000 --- a/scala/src/main/scala/com/xebia/functional/xef/scala/conversation/KotlinXSerializers.scala +++ /dev/null @@ -1,38 +0,0 @@ -package com.xebia.functional.xef.scala.conversation - -import kotlinx.serialization.KSerializer -import kotlinx.serialization.builtins.BuiltinSerializersKt -import kotlinx.serialization.builtins.BuiltinSerializersKt.serializer - -import java.lang - -object KotlinXSerializers: - val int: KSerializer[Integer] = - serializer(kotlin.jvm.internal.IntCompanionObject.INSTANCE) - - val string: KSerializer[String] = - serializer(kotlin.jvm.internal.StringCompanionObject.INSTANCE) - - val boolean: KSerializer[lang.Boolean] = - serializer(kotlin.jvm.internal.BooleanCompanionObject.INSTANCE) - - val double: KSerializer[lang.Double] = - serializer(kotlin.jvm.internal.DoubleCompanionObject.INSTANCE) - - val float: KSerializer[lang.Float] = - serializer(kotlin.jvm.internal.FloatCompanionObject.INSTANCE) - - val long: KSerializer[lang.Long] = - serializer(kotlin.jvm.internal.LongCompanionObject.INSTANCE) - - val short: KSerializer[lang.Short] = - serializer(kotlin.jvm.internal.ShortCompanionObject.INSTANCE) - - val byte: KSerializer[lang.Byte] = - serializer(kotlin.jvm.internal.ByteCompanionObject.INSTANCE) - - val char: KSerializer[Character] = - serializer(kotlin.jvm.internal.CharCompanionObject.INSTANCE) - - val unit: KSerializer[kotlin.Unit] = - serializer(kotlin.Unit.INSTANCE) diff --git a/scala/src/main/scala/com/xebia/functional/xef/scala/conversation/package.scala b/scala/src/main/scala/com/xebia/functional/xef/scala/conversation/package.scala index 7f4d107c6..d65b83efa 100644 --- a/scala/src/main/scala/com/xebia/functional/xef/scala/conversation/package.scala +++ b/scala/src/main/scala/com/xebia/functional/xef/scala/conversation/package.scala @@ -1,84 +1,65 @@ package com.xebia.functional.xef.scala.conversation +import com.xebia.functional.xef.conversation.* import com.xebia.functional.xef.conversation.llm.openai.* -import com.xebia.functional.xef.prompt.Prompt -import com.xebia.functional.xef.conversation.{FromJson, JVMConversation} +import com.xebia.functional.xef.conversation.llm.openai.OpenAI.FromEnvironment.* import com.xebia.functional.xef.llm.* import com.xebia.functional.xef.llm.models.images.* -import com.xebia.functional.xef.store.{ConversationId, LocalVectorStore, VectorStore} -import com.xebia.functional.xef.metrics.{LogsMetric, Metric} +import com.xebia.functional.xef.metrics.* +import com.xebia.functional.xef.prompt.Prompt +import com.xebia.functional.xef.scala.serialization.* +import com.xebia.functional.xef.store.* import io.circe.Decoder import io.circe.parser.parse -import org.reactivestreams.{Subscriber, Subscription} +import org.reactivestreams.* -import java.util -import java.util.UUID +import java.util.UUID.* import java.util.concurrent.LinkedBlockingQueue import scala.jdk.CollectionConverters.* -class ScalaConversation(store: VectorStore, metric: Metric, conversationId: Option[ConversationId]) - extends JVMConversation(store, metric, conversationId.orNull) +class ScalaConversation(store: VectorStore, metric: Metric, conversationId: ConversationId) extends JVMConversation(store, metric, conversationId) def addContext(context: Array[String])(using conversation: ScalaConversation): Unit = conversation.addContextFromArray(context).join() -def prompt[A: Decoder: SerialDescriptor]( - prompt: Prompt, - chat: ChatWithFunctions = OpenAI.FromEnvironment.DEFAULT_SERIALIZATION -)(using - conversation: ScalaConversation -): A = - val fromJson = new FromJson[A] { - def fromJson(json: String): A = - parse(json).flatMap(Decoder[A].decodeJson(_)).fold(throw _, identity) - } +def prompt[A: Decoder: SerialDescriptor](prompt: Prompt, chat: ChatWithFunctions = DEFAULT_SERIALIZATION)(using conversation: ScalaConversation): A = conversation.prompt(chat, prompt, chat.chatFunction(SerialDescriptor[A].serialDescriptor), fromJson).join() -def promptMessage( - prompt: Prompt, - chat: Chat = OpenAI.FromEnvironment.DEFAULT_CHAT -)(using conversation: ScalaConversation): String = +def promptMessage(prompt: Prompt, chat: Chat = DEFAULT_CHAT)(using conversation: ScalaConversation): String = conversation.promptMessage(chat, prompt).join() -def promptMessages( - prompt: Prompt, - chat: Chat = OpenAI.FromEnvironment.DEFAULT_CHAT -)(using - conversation: ScalaConversation -): List[String] = +def promptMessages(prompt: Prompt, chat: Chat = DEFAULT_CHAT)(using conversation: ScalaConversation): List[String] = conversation.promptMessages(chat, prompt).join().asScala.toList -def promptStreaming( - prompt: Prompt, - chat: Chat = OpenAI.FromEnvironment.DEFAULT_CHAT -)(using - conversation: ScalaConversation -): LazyList[String] = +def promptStreaming(prompt: Prompt, chat: Chat = DEFAULT_CHAT)(using conversation: ScalaConversation): LazyList[String] = val publisher = conversation.promptStreamingToPublisher(chat, prompt) val queue = new LinkedBlockingQueue[String]() - publisher.subscribe(new Subscriber[String] { - // TODO change to fs2 or similar + publisher.subscribe(new Subscriber[String]: // TODO change to fs2 or similar def onSubscribe(s: Subscription): Unit = s.request(Long.MaxValue) - def onNext(t: String): Unit = queue.add(t); () - def onError(t: Throwable): Unit = throw t - def onComplete(): Unit = () - }) + ) LazyList.continually(queue.take) -def images( - prompt: Prompt, - images: Images = OpenAI.FromEnvironment.DEFAULT_IMAGES, - numberImages: Int = 1, - size: String = "1024x1024" -)(using +def prompt[A: Decoder: SerialDescriptor](message: String)(using ScalaConversation): A = + prompt(Prompt(message)) + +def promptMessage(message: String)(using ScalaConversation): String = + promptMessage(Prompt(message)) + +def promptMessages(message: String)(using ScalaConversation): List[String] = + promptMessages(Prompt(message)) + +def promptStreaming(message: String)(using ScalaConversation): LazyList[String] = + promptStreaming(Prompt(message)) + +def images(prompt: Prompt, images: Images = DEFAULT_IMAGES, numberImages: Int = 1, size: String = "1024x1024")(using conversation: ScalaConversation ): ImagesGenerationResponse = conversation.images(images, prompt, numberImages, size).join() -def conversation[A]( - block: ScalaConversation ?=> A, - conversationId: Option[ConversationId] = Some(ConversationId(UUID.randomUUID().toString)) -): A = block(using ScalaConversation(LocalVectorStore(OpenAI.FromEnvironment.DEFAULT_EMBEDDING), LogsMetric(), conversationId)) +def conversation[A](block: ScalaConversation ?=> A, id: ConversationId = ConversationId(randomUUID.toString)): A = + block(using ScalaConversation(LocalVectorStore(DEFAULT_EMBEDDING), LogsMetric(), id)) + +private def fromJson[A: Decoder]: FromJson[A] = json => parse(json).flatMap(Decoder[A].decodeJson(_)).fold(throw _, identity) diff --git a/scala/src/main/scala/com/xebia/functional/xef/scala/serialization/KotlinXSerializers.scala b/scala/src/main/scala/com/xebia/functional/xef/scala/serialization/KotlinXSerializers.scala new file mode 100644 index 000000000..30aa05e4a --- /dev/null +++ b/scala/src/main/scala/com/xebia/functional/xef/scala/serialization/KotlinXSerializers.scala @@ -0,0 +1,18 @@ +package com.xebia.functional.xef.scala.serialization + +import kotlin.jvm.internal.* +import kotlinx.serialization.KSerializer +import kotlinx.serialization.builtins.BuiltinSerializersKt +import kotlinx.serialization.builtins.BuiltinSerializersKt.serializer + +object KotlinXSerializers: + val int: KSerializer[java.lang.Integer] = serializer(IntCompanionObject.INSTANCE) + val string: KSerializer[String] = serializer(StringCompanionObject.INSTANCE) + val boolean: KSerializer[java.lang.Boolean] = serializer(BooleanCompanionObject.INSTANCE) + val double: KSerializer[java.lang.Double] = serializer(DoubleCompanionObject.INSTANCE) + val float: KSerializer[java.lang.Float] = serializer(FloatCompanionObject.INSTANCE) + val long: KSerializer[java.lang.Long] = serializer(LongCompanionObject.INSTANCE) + val short: KSerializer[java.lang.Short] = serializer(ShortCompanionObject.INSTANCE) + val byte: KSerializer[java.lang.Byte] = serializer(ByteCompanionObject.INSTANCE) + val char: KSerializer[Character] = serializer(CharCompanionObject.INSTANCE) + val unit: KSerializer[kotlin.Unit] = serializer(kotlin.Unit.INSTANCE) diff --git a/scala/src/main/scala/com/xebia/functional/xef/scala/conversation/SerialDescriptor.scala b/scala/src/main/scala/com/xebia/functional/xef/scala/serialization/SerialDescriptor.scala similarity index 94% rename from scala/src/main/scala/com/xebia/functional/xef/scala/conversation/SerialDescriptor.scala rename to scala/src/main/scala/com/xebia/functional/xef/scala/serialization/SerialDescriptor.scala index a7dc121df..9b574065a 100644 --- a/scala/src/main/scala/com/xebia/functional/xef/scala/conversation/SerialDescriptor.scala +++ b/scala/src/main/scala/com/xebia/functional/xef/scala/serialization/SerialDescriptor.scala @@ -1,20 +1,19 @@ -package com.xebia.functional.xef.scala.conversation +package com.xebia.functional.xef.scala.serialization import com.xebia.functional.xef.conversation.jvm.Description as JvmDescription -import kotlinx.serialization.descriptors.{SerialDescriptor as KtSerialDescriptor, SerialKind, StructureKind} import kotlinx.serialization.KSerializer +import kotlinx.serialization.descriptors.{SerialDescriptor as KtSerialDescriptor, SerialKind, StructureKind} import kotlinx.serialization.encoding.{Decoder as KtDecoder, Encoder as KtEncoder} -import scala.quoted.* import java.lang.annotation.Annotation import java.util -import scala.compiletime.{constValue, erasedValue, summonInline} +import scala.annotation.meta.* +import scala.compiletime.* import scala.deriving.* import scala.jdk.CollectionConverters.* +import scala.quoted.* import scala.reflect.ClassTag -import scala.annotation.meta._ - trait SerialDescriptor[A]: def serialDescriptor: KtSerialDescriptor def kserializer: KSerializer[A] = new KSerializer[A]: @@ -81,40 +80,33 @@ object SerialDescriptor extends SerialDescriptorInstances: // Does the element at the given index have a default value, or is it wrapped in `Option`, or is a union with `Null`? override def isElementOptional(index: Int): Boolean = false - def serialDescriptor = serialDescriptorImpl + def serialDescriptor: KtSerialDescriptor = serialDescriptorImpl inline def getStaticAnnotations[A]: List[Any] = ${ getAnnotationsImpl[A] } -def getAnnotationsImpl[A: Type](using Quotes): Expr[List[Any]] = { +def getAnnotationsImpl[A: Type](using Quotes): Expr[List[Any]] = import quotes.reflect.* val annotations = TypeRepr.of[A].typeSymbol.annotations.map(_.asExpr) Expr.ofList(annotations) -} inline def getFieldAnnotationsMap[A]: Map[String, List[Any]] = ${ getFieldAnnotationsMapImpl[A] } -def getFieldAnnotationsMapImpl[A: Type](using Quotes): Expr[Map[String, List[Any]]] = { - import quotes.reflect._ - +def getFieldAnnotationsMapImpl[A: Type](using Quotes): Expr[Map[String, List[Any]]] = + import quotes.reflect.* // Extract the fields from the type val fields = TypeRepr.of[A].typeSymbol.primaryConstructor.paramSymss.flatten - // For each field, extract its name and annotations - val fieldAnnotationsMap: List[Expr[(String, List[Any])]] = fields.map { field => + val fieldAnnotationsMap: List[Expr[(String, List[Any])]] = fields.map: field => val fieldName = Expr(field.name) val annotations = Expr.ofList(field.annotations.map(_.asExpr)) '{ ($fieldName, $annotations) } - } - // Convert list of (String, List[Any]) tuples to a map - fieldAnnotationsMap match { + fieldAnnotationsMap match case Nil => '{ Map.empty[String, List[Any]] } case _ => val mapExpr = Expr.ofList(fieldAnnotationsMap) '{ $mapExpr.toMap } - } -} /** * A scala annotation that maps to the jvm description annotation diff --git a/scala/src/main/scala/com/xebia/functional/xef/scala/conversation/SerialDescriptorInstances.scala b/scala/src/main/scala/com/xebia/functional/xef/scala/serialization/SerialDescriptorInstances.scala similarity index 98% rename from scala/src/main/scala/com/xebia/functional/xef/scala/conversation/SerialDescriptorInstances.scala rename to scala/src/main/scala/com/xebia/functional/xef/scala/serialization/SerialDescriptorInstances.scala index c213c28e9..ee761f27c 100644 --- a/scala/src/main/scala/com/xebia/functional/xef/scala/conversation/SerialDescriptorInstances.scala +++ b/scala/src/main/scala/com/xebia/functional/xef/scala/serialization/SerialDescriptorInstances.scala @@ -1,4 +1,4 @@ -package com.xebia.functional.xef.scala.conversation +package com.xebia.functional.xef.scala.serialization import kotlin.jvm.internal.Reflection import kotlin.reflect.KClass diff --git a/scala/src/test/scala/com/xebia/functional/xef/scala/conversation/SerialDescriptorSpec.scala b/scala/src/test/scala/com/xebia/functional/xef/scala/serialization/SerialDescriptorSpec.scala similarity index 98% rename from scala/src/test/scala/com/xebia/functional/xef/scala/conversation/SerialDescriptorSpec.scala rename to scala/src/test/scala/com/xebia/functional/xef/scala/serialization/SerialDescriptorSpec.scala index 6141cc617..4c103b6e1 100644 --- a/scala/src/test/scala/com/xebia/functional/xef/scala/conversation/SerialDescriptorSpec.scala +++ b/scala/src/test/scala/com/xebia/functional/xef/scala/serialization/SerialDescriptorSpec.scala @@ -1,4 +1,4 @@ -package com.xebia.functional.xef.scala.conversation +package com.xebia.functional.xef.scala.serialization import cats.syntax.either.* import kotlinx.serialization.builtins.BuiltinSerializersKt From 0e42da20a8d978acfb798fd992cf75a302d62c0c Mon Sep 17 00:00:00 2001 From: Lawrence Lavigne Date: Thu, 2 Nov 2023 11:13:19 -0700 Subject: [PATCH 4/4] Upgrade Gradle to 8.4 & Scala JVM to 21; clean up build scripts (#516) * Upgrade Gradle to 8.4 & Scala JVM to 21; clean up build scripts * Replaced all remaining tabs with spaces --- .../workflows/publish-development-version.yml | 2 +- .github/workflows/publish.yml | 2 +- .run/build.run.xml | 24 ++ .run/spotless.run.xml | 24 ++ .run/test.run.xml | 24 ++ .run/wrapper.run.xml | 24 ++ README.md | 8 +- build.gradle.kts | 38 +- .../kotlin/JavaPublishingConventionsPlugin.kt | 6 +- buildSrc/src/main/kotlin/Predef.kt | 16 +- .../ScalaPublishingConventionsPlugin.kt | 4 +- config/detekt/detekt.yml | 2 +- core/build.gradle.kts | 328 ++++++++---------- detekt-rules/build.gradle.kts | 13 +- .../example/detekt/JvmInlineAnnotationTest.kt | 10 +- ...ClassConstructorWithValueParametersTest.kt | 81 +++-- docs/intro/scala.md | 8 +- examples/java/build.gradle.kts | 6 - examples/kotlin/build.gradle.kts | 70 ++-- examples/scala/.scalafmt.conf | 5 +- examples/scala/build.gradle.kts | 38 +- filesystem/build.gradle.kts | 7 - gpt4all-kotlin/README.MD | 6 +- gpt4all-kotlin/build.gradle.kts | 32 +- gradle/wrapper/gradle-wrapper.jar | Bin 61608 -> 63721 bytes gradle/wrapper/gradle-wrapper.properties | 3 +- gradlew | 29 +- integrations/gcp/README.MD | 30 +- integrations/gcp/build.gradle.kts | 174 ++++------ integrations/lucene/build.gradle.kts | 1 - integrations/mlflow/build.gradle.kts | 28 +- integrations/opentelemetry/build.gradle.kts | 6 +- integrations/pdf/build.gradle.kts | 1 - integrations/postgresql/build.gradle.kts | 6 +- integrations/sql/build.gradle.kts | 13 +- java/README.md | 6 +- java/build.gradle.kts | 8 +- kotlin/build.gradle.kts | 11 +- models/gpt4all/get-models.sh | 1 - openai/build.gradle.kts | 49 +-- reasoning/build.gradle.kts | 48 +-- scala/.scalafmt.conf | 5 +- scala/README.md | 8 +- scala/build.gradle.kts | 22 +- server/README.md | 10 +- server/build.gradle.kts | 27 +- server/web/README.md | 9 +- settings.gradle.kts | 3 - tokenizer/build.gradle.kts | 6 - 49 files changed, 566 insertions(+), 716 deletions(-) create mode 100644 .run/build.run.xml create mode 100644 .run/spotless.run.xml create mode 100644 .run/test.run.xml create mode 100644 .run/wrapper.run.xml diff --git a/.github/workflows/publish-development-version.yml b/.github/workflows/publish-development-version.yml index da097bd62..d373c3827 100644 --- a/.github/workflows/publish-development-version.yml +++ b/.github/workflows/publish-development-version.yml @@ -29,7 +29,7 @@ jobs: uses: actions/setup-java@v3 with: distribution: 'zulu' - java-version: 11 + java-version: 20 - name: Assemble uses: gradle/gradle-build-action@v2 diff --git a/.github/workflows/publish.yml b/.github/workflows/publish.yml index d69aa9685..f6523d87c 100644 --- a/.github/workflows/publish.yml +++ b/.github/workflows/publish.yml @@ -27,7 +27,7 @@ jobs: uses: actions/setup-java@v3 with: distribution: 'zulu' - java-version: 11 + java-version: 20 - name: Assemble uses: gradle/gradle-build-action@v2 diff --git a/.run/build.run.xml b/.run/build.run.xml new file mode 100644 index 000000000..87e14e108 --- /dev/null +++ b/.run/build.run.xml @@ -0,0 +1,24 @@ + + + + + + + true + true + false + false + + + \ No newline at end of file diff --git a/.run/spotless.run.xml b/.run/spotless.run.xml new file mode 100644 index 000000000..74fba4040 --- /dev/null +++ b/.run/spotless.run.xml @@ -0,0 +1,24 @@ + + + + + + + true + true + false + false + + + \ No newline at end of file diff --git a/.run/test.run.xml b/.run/test.run.xml new file mode 100644 index 000000000..6904dabe8 --- /dev/null +++ b/.run/test.run.xml @@ -0,0 +1,24 @@ + + + + + + + true + true + false + false + + + \ No newline at end of file diff --git a/.run/wrapper.run.xml b/.run/wrapper.run.xml new file mode 100644 index 000000000..27b5fcbc6 --- /dev/null +++ b/.run/wrapper.run.xml @@ -0,0 +1,24 @@ + + + + + + + true + true + false + false + + + \ No newline at end of file diff --git a/README.md b/README.md index bbbb5af8a..dccf15acd 100644 --- a/README.md +++ b/README.md @@ -56,15 +56,13 @@ Libraries are published in Maven Central, under the `com.xebia` group. Libraries are published in Maven Central. You may need to add that repository explicitly in your build, if you haven't done it before. -```kotlin -repositories { - mavenCentral() -} +```groovy +repositories { mavenCentral() } ``` Then add the library in the usual way. -```kotlin +```groovy // In Gradle Kotlin dependencies { implementation("com.xebia:xef-kotlin:") diff --git a/build.gradle.kts b/build.gradle.kts index 21259956a..2b6f31483 100644 --- a/build.gradle.kts +++ b/build.gradle.kts @@ -1,18 +1,18 @@ @file:Suppress("DSL_SCOPE_VIOLATION") plugins { - base - alias(libs.plugins.kotlin.multiplatform) apply false - alias(libs.plugins.kotlinx.serialization) - alias(libs.plugins.spotless) - alias(libs.plugins.dokka) - alias(libs.plugins.arrow.gradle.nexus) - alias(libs.plugins.arrow.gradle.publish) apply false - alias(libs.plugins.semver.gradle) + base + alias(libs.plugins.kotlin.multiplatform) apply false + alias(libs.plugins.kotlinx.serialization) + alias(libs.plugins.spotless) + alias(libs.plugins.dokka) + alias(libs.plugins.arrow.gradle.nexus) + alias(libs.plugins.arrow.gradle.publish) apply false + alias(libs.plugins.semver.gradle) } allprojects { - group = property("project.group").toString() + group = property("project.group").toString() } fun isMultiplatformModule(project: Project): Boolean { @@ -22,10 +22,7 @@ fun isMultiplatformModule(project: Project): Boolean { val multiPlatformModules = project.subprojects.filter { isMultiplatformModule(it) }.map { it.name } -enum class ModuleType { - MULTIPLATFORM, - SINGLEPLATFORM -} +enum class ModuleType { SINGLEPLATFORM, MULTIPLATFORM } fun Project.configureBuildAndTestTask( taskName: String, @@ -33,7 +30,6 @@ fun Project.configureBuildAndTestTask( multiPlatformModules: List ) { val platform: String by extra - tasks.register(taskName) { doLast { val gradleCommand = getGradleCommand(platform) @@ -42,30 +38,26 @@ fun Project.configureBuildAndTestTask( } project.exec { when (moduleType) { + ModuleType.SINGLEPLATFORM -> { + commandLine(gradleCommand, "build", *buildExcludeOptions(multiPlatformModules)) + } ModuleType.MULTIPLATFORM -> { multiPlatformModules.forEach { module -> commandLine(gradleCommand, ":$module:${platform}Test") } } - ModuleType.SINGLEPLATFORM -> { - commandLine(gradleCommand, "build", *buildExcludeOptions(multiPlatformModules)) - } } } } } } -fun Project.buildExcludeOptions(modules: List): Array { +fun buildExcludeOptions(modules: List): Array { return modules.flatMap { listOf("-x", ":$it:build") }.toTypedArray() } fun getGradleCommand(platform: String): String { - return if (platform == "mingwX64") { - "gradlew.bat" - } else { - "./gradlew" - } + return if (platform == "mingwX64") "gradlew.bat" else "./gradlew" } configureBuildAndTestTask( diff --git a/buildSrc/src/main/kotlin/JavaPublishingConventionsPlugin.kt b/buildSrc/src/main/kotlin/JavaPublishingConventionsPlugin.kt index 1f9c7a29c..d01d6852d 100644 --- a/buildSrc/src/main/kotlin/JavaPublishingConventionsPlugin.kt +++ b/buildSrc/src/main/kotlin/JavaPublishingConventionsPlugin.kt @@ -1,17 +1,14 @@ import org.gradle.api.Plugin import org.gradle.api.Project -import org.gradle.api.plugins.BasePlugin -import org.gradle.api.plugins.BasePluginExtension import org.gradle.api.publish.PublishingExtension import org.gradle.api.publish.maven.MavenPublication -import org.gradle.api.tasks.TaskProvider -import org.gradle.api.tasks.bundling.Jar import org.gradle.kotlin.dsl.findByType import org.gradle.kotlin.dsl.get import org.gradle.kotlin.dsl.register import org.gradle.plugins.signing.SigningExtension class JavaPublishingConventionsPlugin : Plugin { + override fun apply(project: Project): Unit = project.run { val publishingExtension: PublishingExtension = extensions.findByType() @@ -35,7 +32,6 @@ class JavaPublishingConventionsPlugin : Plugin { val signingKeyId: String? = configValue("signing.keyId", "SIGNING_KEY_ID") val signingKey: String? = configValue("signing.key", "SIGNING_KEY") val signingPassphrase: String? = configValue("signing.passphrase", "SIGNING_KEY_PASSPHRASE") - isRequired = !isLocal useGpgCmd() useInMemoryPgpKeys(signingKeyId, signingKey, signingPassphrase) diff --git a/buildSrc/src/main/kotlin/Predef.kt b/buildSrc/src/main/kotlin/Predef.kt index d5efd8a20..4b03924a8 100644 --- a/buildSrc/src/main/kotlin/Predef.kt +++ b/buildSrc/src/main/kotlin/Predef.kt @@ -1,18 +1,13 @@ import org.gradle.api.Project import org.gradle.api.publish.maven.MavenPublication -internal fun Project.configValue(propertyName: String, environmentVariableName: String): String? { +internal fun Project.configValue(propertyName: String, envVarName: String): String? { val property: String? = project.properties[propertyName]?.toString() - val environmentVariable: String? = System.getenv(environmentVariableName) - - val configValue = property ?: environmentVariable - + val envVar: String? = System.getenv(envVarName) + val configValue = property ?: envVar return configValue.also { if (configValue.isNullOrBlank()) { - errorMessage( - "$propertyName Gradle property and " + - "$environmentVariableName environment variable are missing", - ) + errorMessage("$propertyName Gradle property and $envVarName environment variable are missing") } } } @@ -22,21 +17,18 @@ internal fun MavenPublication.pomConfiguration(project: Project) { name.set(project.properties["pom.name"]?.toString()) description.set(project.properties["pom.description"]?.toString()) url.set(project.properties["pom.url"]?.toString()) - licenses { license { name.set(project.properties["pom.license.name"]?.toString()) url.set(project.properties["pom.license.url"]?.toString()) } } - developers { developer { id.set(project.properties["pom.developer.id"].toString()) name.set(project.properties["pom.developer.name"].toString()) } } - scm { url.set(project.properties["pom.smc.url"].toString()) connection.set(project.properties["pom.smc.connection"].toString()) diff --git a/buildSrc/src/main/kotlin/ScalaPublishingConventionsPlugin.kt b/buildSrc/src/main/kotlin/ScalaPublishingConventionsPlugin.kt index 8f3b031cc..400817204 100644 --- a/buildSrc/src/main/kotlin/ScalaPublishingConventionsPlugin.kt +++ b/buildSrc/src/main/kotlin/ScalaPublishingConventionsPlugin.kt @@ -12,6 +12,7 @@ import org.gradle.kotlin.dsl.register import org.gradle.plugins.signing.SigningExtension class ScalaPublishingConventionsPlugin : Plugin { + override fun apply(project: Project): Unit = project.run { val scaladocJarTask: TaskProvider = tasks.register("scaladocJar") { group = BasePlugin.BUILD_GROUP @@ -37,11 +38,9 @@ class ScalaPublishingConventionsPlugin : Plugin { publications { register("maven") { val scala3Suffix = "_3" - artifactId = basePluginExtension.archivesName.get() + scala3Suffix from(components["java"]) artifact(scaladocJarTask) - pomConfiguration(project) } } @@ -52,7 +51,6 @@ class ScalaPublishingConventionsPlugin : Plugin { val signingKeyId: String? = configValue("signing.keyId", "SIGNING_KEY_ID") val signingKey: String? = configValue("signing.key", "SIGNING_KEY") val signingPassphrase: String? = configValue("signing.passphrase", "SIGNING_KEY_PASSPHRASE") - isRequired = !isLocal useGpgCmd() useInMemoryPgpKeys(signingKeyId, signingKey, signingPassphrase) diff --git a/config/detekt/detekt.yml b/config/detekt/detekt.yml index dec720c07..a1399cd00 100644 --- a/config/detekt/detekt.yml +++ b/config/detekt/detekt.yml @@ -42,7 +42,7 @@ console-reports: output-reports: active: false - exclude: +# exclude: # - 'TxtOutputReport' # - 'XmlOutputReport' # - 'HtmlOutputReport' diff --git a/core/build.gradle.kts b/core/build.gradle.kts index 5db9ed039..4db577d5c 100644 --- a/core/build.gradle.kts +++ b/core/build.gradle.kts @@ -3,32 +3,29 @@ import org.jetbrains.dokka.gradle.DokkaTask repositories { - mavenCentral() + mavenCentral() } plugins { - base - alias(libs.plugins.kotlin.multiplatform) - alias(libs.plugins.kotest.multiplatform) - alias(libs.plugins.kotlinx.serialization) - alias(libs.plugins.spotless) - alias(libs.plugins.dokka) - alias(libs.plugins.arrow.gradle.publish) - alias(libs.plugins.semver.gradle) - alias(libs.plugins.detekt) - //id("com.xebia.asfuture").version("0.0.1") + base + alias(libs.plugins.kotlin.multiplatform) + alias(libs.plugins.kotest.multiplatform) + alias(libs.plugins.kotlinx.serialization) + alias(libs.plugins.spotless) + alias(libs.plugins.dokka) + alias(libs.plugins.arrow.gradle.publish) + alias(libs.plugins.semver.gradle) + alias(libs.plugins.detekt) } dependencies { - detektPlugins(project(":detekt-rules")) + detektPlugins(project(":detekt-rules")) } java { - sourceCompatibility = JavaVersion.VERSION_11 - targetCompatibility = JavaVersion.VERSION_11 - toolchain { - languageVersion = JavaLanguageVersion.of(11) - } + sourceCompatibility = JavaVersion.VERSION_11 + targetCompatibility = JavaVersion.VERSION_11 + toolchain { languageVersion = JavaLanguageVersion.of(11) } } detekt { @@ -39,187 +36,166 @@ detekt { } kotlin { - jvm { - withJava() - compilations { - val integrationTest by compilations.creating { - // Create a test task to run the tests produced by this compilation: - tasks.register("integrationTest") { - description = "Run the integration tests" - group = "verification" - classpath = compileDependencyFiles + runtimeDependencyFiles + output.allOutputs - testClassesDirs = output.classesDirs - - testLogging { - events("passed") + jvm { + withJava() + compilations { + val integrationTest by compilations.creating { + // Create a test task to run the tests produced by this compilation: + tasks.register("integrationTest") { + description = "Run the integration tests" + group = "verification" + classpath = compileDependencyFiles + runtimeDependencyFiles + output.allOutputs + testClassesDirs = output.classesDirs + testLogging { events("passed") } + } } - } + val test by compilations.getting + integrationTest.associateWith(test) } - val test by compilations.getting - integrationTest.associateWith(test) - } - } - js(IR) { - browser() - nodejs() - } - - linuxX64() - macosX64() - macosArm64() - mingwX64() - - sourceSets { - all { - languageSettings.optIn("kotlin.ExperimentalStdlibApi") } - - val commonMain by getting { - dependencies { - api(libs.bundles.arrow) - api(libs.kotlinx.serialization.json) - api(libs.ktor.utils) - api(projects.xefTokenizer) - implementation(libs.bundles.ktor.client) - implementation(libs.klogging) - implementation(libs.uuid) + js(IR) { + browser() + nodejs() + } + linuxX64() + macosX64() + macosArm64() + mingwX64() + sourceSets { + all { + languageSettings.optIn("kotlin.ExperimentalStdlibApi") } - } - - val commonTest by getting { - dependencies { - implementation(libs.kotest.property) - implementation(libs.kotest.framework) - implementation(libs.kotest.assertions) + val commonMain by getting { + dependencies { + api(libs.bundles.arrow) + api(libs.kotlinx.serialization.json) + api(libs.ktor.utils) + api(projects.xefTokenizer) + implementation(libs.bundles.ktor.client) + implementation(libs.klogging) + implementation(libs.uuid) + } } - } - - val jvmMain by getting { - dependencies { - implementation(libs.ktor.http) - implementation(libs.logback) - implementation(libs.skrape) - implementation(libs.rss.reader) - api(libs.jackson) - api(libs.jackson.schema) - api(libs.jackson.schema.jakarta) - api(libs.jakarta.validation) - implementation(libs.kotlinx.coroutines.reactive) - api(libs.ktor.client.cio) + val commonTest by getting { + dependencies { + implementation(libs.kotest.property) + implementation(libs.kotest.framework) + implementation(libs.kotest.assertions) + } } - } - - val jsMain by getting { - dependencies { - api(libs.ktor.client.js) + val jvmMain by getting { + dependencies { + implementation(libs.ktor.http) + implementation(libs.logback) + implementation(libs.skrape) + implementation(libs.rss.reader) + api(libs.jackson) + api(libs.jackson.schema) + api(libs.jackson.schema.jakarta) + api(libs.jakarta.validation) + implementation(libs.kotlinx.coroutines.reactive) + api(libs.ktor.client.cio) + } } - } - - val jvmTest by getting { - dependencies { - implementation(libs.kotest.junit5) + val jsMain by getting { + dependencies { + api(libs.ktor.client.js) + } } - } - - val linuxX64Main by getting { - dependencies { - implementation(libs.ktor.client.cio) + val jvmTest by getting { + dependencies { + implementation(libs.kotest.junit5) + } } - } - val macosX64Main by getting { - dependencies { - implementation(libs.ktor.client.cio) + val linuxX64Main by getting { + dependencies { + implementation(libs.ktor.client.cio) + } } - } - val macosArm64Main by getting { - dependencies { - implementation(libs.ktor.client.cio) + val macosX64Main by getting { + dependencies { + implementation(libs.ktor.client.cio) + } } - } - val mingwX64Main by getting { - dependencies { - implementation(libs.ktor.client.winhttp) + val macosArm64Main by getting { + dependencies { + implementation(libs.ktor.client.cio) + } + } + val mingwX64Main by getting { + dependencies { + implementation(libs.ktor.client.winhttp) + } + } + val linuxX64Test by getting + val macosX64Test by getting + val macosArm64Test by getting + val mingwX64Test by getting + create("nativeMain") { + dependsOn(commonMain) + linuxX64Main.dependsOn(this) + macosX64Main.dependsOn(this) + macosArm64Main.dependsOn(this) + mingwX64Main.dependsOn(this) + } + create("nativeTest") { + dependsOn(commonTest) + linuxX64Test.dependsOn(this) + macosX64Test.dependsOn(this) + macosArm64Test.dependsOn(this) + mingwX64Test.dependsOn(this) } } - val linuxX64Test by getting - val macosX64Test by getting - val macosArm64Test by getting - val mingwX64Test by getting - - create("nativeMain") { - dependsOn(commonMain) - linuxX64Main.dependsOn(this) - macosX64Main.dependsOn(this) - macosArm64Main.dependsOn(this) - mingwX64Main.dependsOn(this) - } - - create("nativeTest") { - dependsOn(commonTest) - linuxX64Test.dependsOn(this) - macosX64Test.dependsOn(this) - macosArm64Test.dependsOn(this) - mingwX64Test.dependsOn(this) - } - } } spotless { - kotlin { - target("**/*.kt") - ktfmt().googleStyle().configure { - it.setRemoveUnusedImport(true) + kotlin { + target("**/*.kt") + ktfmt().googleStyle().configure { it.setRemoveUnusedImport(true) } } - } } tasks { - - withType().configureEach { - dependsOn(":detekt-rules:assemble") - autoCorrect = true - } - named("detektJvmMain") { - dependsOn(":detekt-rules:assemble") - getByName("build").dependsOn(this) - } - named("detekt") { - dependsOn(":detekt-rules:assemble") - getByName("build").dependsOn(this) - } - - withType().configureEach { - maxParallelForks = Runtime.getRuntime().availableProcessors() - useJUnitPlatform() - testLogging { - setExceptionFormat("full") - setEvents(listOf("passed", "skipped", "failed", "standardOut", "standardError")) - } - } - - withType().configureEach { - kotlin.sourceSets.forEach { kotlinSourceSet -> - dokkaSourceSets.named(kotlinSourceSet.name) { - perPackageOption { - matchingRegex.set(".*\\.internal.*") - suppress.set(true) + withType().configureEach { + dependsOn(":detekt-rules:assemble") + autoCorrect = true + } + named("detektJvmMain") { + dependsOn(":detekt-rules:assemble") + getByName("build").dependsOn(this) + } + named("detekt") { + dependsOn(":detekt-rules:assemble") + getByName("build").dependsOn(this) + } + withType().configureEach { + maxParallelForks = Runtime.getRuntime().availableProcessors() + useJUnitPlatform() + testLogging { + setExceptionFormat("full") + setEvents(listOf("passed", "skipped", "failed", "standardOut", "standardError")) } - skipDeprecated.set(true) - reportUndocumented.set(false) - val baseUrl: String = checkNotNull(project.properties["pom.smc.url"]?.toString()) - - kotlinSourceSet.kotlin.srcDirs.filter { it.exists() }.forEach { srcDir -> - sourceLink { - localDirectory.set(srcDir) - remoteUrl.set(uri("$baseUrl/blob/main/${srcDir.relativeTo(rootProject.rootDir)}").toURL()) - remoteLineSuffix.set("#L") - } + } + withType().configureEach { + kotlin.sourceSets.forEach { kotlinSourceSet -> + dokkaSourceSets.named(kotlinSourceSet.name) { + perPackageOption { + matchingRegex.set(".*\\.internal.*") + suppress.set(true) + } + skipDeprecated.set(true) + reportUndocumented.set(false) + val baseUrl = checkNotNull(project.properties["pom.smc.url"]?.toString()) + kotlinSourceSet.kotlin.srcDirs.filter { it.exists() }.forEach { srcDir -> + sourceLink { + localDirectory.set(srcDir) + remoteUrl.set(uri("$baseUrl/blob/main/${srcDir.relativeTo(rootProject.rootDir)}").toURL()) + remoteLineSuffix.set("#L") + } + } + } } - } } - } } -tasks.withType { - dependsOn(tasks.withType()) -} +tasks.withType { dependsOn(tasks.withType()) } diff --git a/detekt-rules/build.gradle.kts b/detekt-rules/build.gradle.kts index c8d0dadc3..3ac7db23d 100644 --- a/detekt-rules/build.gradle.kts +++ b/detekt-rules/build.gradle.kts @@ -5,7 +5,7 @@ repositories { mavenCentral() } plugins { `kotlin-dsl` base - alias(libs.plugins.spotless) + alias(libs.plugins.spotless) } spotless { @@ -15,24 +15,23 @@ spotless { } } - dependencies { - api(libs.detekt.api) - testImplementation(libs.detekt.test) + api(libs.detekt.api) + testImplementation(libs.detekt.test) testImplementation(libs.kotest.assertions) testImplementation("org.junit.jupiter:junit-jupiter:5.10.0") implementation(libs.klogging) } -tasks.withType() { +tasks.withType { duplicatesStrategy = DuplicatesStrategy.EXCLUDE } tasks.withType().configureEach { maxParallelForks = Runtime.getRuntime().availableProcessors() useJUnitPlatform() - systemProperty("junit.jupiter.testinstance.lifecycle.default", "per_class") - systemProperty("compile-snippet-tests", project.hasProperty("compile-test-snippets")) + systemProperty("junit.jupiter.testinstance.lifecycle.default", "per_class") + systemProperty("compile-snippet-tests", project.hasProperty("compile-test-snippets")) testLogging { setExceptionFormat("full") setEvents(listOf("passed", "skipped", "failed", "standardOut", "standardError")) diff --git a/detekt-rules/src/test/kotlin/org/example/detekt/JvmInlineAnnotationTest.kt b/detekt-rules/src/test/kotlin/org/example/detekt/JvmInlineAnnotationTest.kt index 509013791..ee3d225d0 100644 --- a/detekt-rules/src/test/kotlin/org/example/detekt/JvmInlineAnnotationTest.kt +++ b/detekt-rules/src/test/kotlin/org/example/detekt/JvmInlineAnnotationTest.kt @@ -13,8 +13,8 @@ internal class JvmInlineAnnotationTest(private val env: KotlinCoreEnvironment) { @Test fun `reports missing jvminline annotations`() { val code = """ - value class User(val id: String) - """ + value class User(val id: String) + """ val findings = JvmInlineAnnotation(Config.empty).compileAndLintWithContext(env, code) findings shouldHaveSize 1 } @@ -23,9 +23,9 @@ internal class JvmInlineAnnotationTest(private val env: KotlinCoreEnvironment) { fun `doesn't report annotated value classes`() { val code = """ - import kotlin.jvm.JvmInline - @JvmInline value class User(val id: String) - """ + import kotlin.jvm.JvmInline + @JvmInline value class User(val id: String) + """ val findings = JvmInlineAnnotation(Config.empty).compileAndLintWithContext(env, code) findings shouldHaveSize 0 } diff --git a/detekt-rules/src/test/kotlin/org/example/detekt/PublicDataClassConstructorWithValueParametersTest.kt b/detekt-rules/src/test/kotlin/org/example/detekt/PublicDataClassConstructorWithValueParametersTest.kt index fdc7a85c5..6d71d0b29 100644 --- a/detekt-rules/src/test/kotlin/org/example/detekt/PublicDataClassConstructorWithValueParametersTest.kt +++ b/detekt-rules/src/test/kotlin/org/example/detekt/PublicDataClassConstructorWithValueParametersTest.kt @@ -15,10 +15,10 @@ internal class PublicDataClassConstructorWithValueParametersTest( fun `reports data classes that have value class parameters without companion static constructor`() { val code = """ - import kotlin.jvm.JvmInline - @JvmInline value class Bar(val value: String) - data class Foo(val bar: Bar) - """ + import kotlin.jvm.JvmInline + @JvmInline value class Bar(val value: String) + data class Foo(val bar: Bar) + """ val findings = PublicDataClassConstructorWithValueParameters(Config.empty) .compileAndLintWithContext(env, code) @@ -30,18 +30,17 @@ internal class PublicDataClassConstructorWithValueParametersTest( fun `doesn't report data classes with value parameters with companion static constructor`() { val code = """ - import kotlin.jvm.JvmInline - import kotlin.jvm.JvmStatic - @JvmInline value class Bar(val value: String) - data class Foo(val bar: Bar) { - companion object { - @JvmStatic - fun apply(bar: Bar): Foo { - return Foo(bar) - } - } + import kotlin.jvm.JvmInline + import kotlin.jvm.JvmStatic + @JvmInline value class Bar(val value: String) + data class Foo(val bar: Bar) { + companion object { + @JvmStatic + fun apply(bar: Bar): Foo { + return Foo(bar) + } } - """ + }""" val findings = JvmInlineAnnotation(Config.empty).compileAndLintWithContext(env, code) findings shouldHaveSize 0 } @@ -49,15 +48,15 @@ internal class PublicDataClassConstructorWithValueParametersTest( @Test fun `reports data classes with value parameters with companion objects without static factory methods`() { val code = - """import kotlin.jvm.JvmInline - @JvmInline value class Bar(val value: String) - data class Foo(val bar: Bar){ - companion object { - val test: String = "test" - fun notAFactoryMethod(): String = test - } + """ + import kotlin.jvm.JvmInline + @JvmInline value class Bar(val value: String) + data class Foo(val bar: Bar){ + companion object { + val test: String = "test" + fun notAFactoryMethod(): String = test } - """ + }""" val findings = PublicDataClassConstructorWithValueParameters(Config.empty) .compileAndLintWithContext(env, code) @@ -68,16 +67,16 @@ internal class PublicDataClassConstructorWithValueParametersTest( @Test fun `reports data classes with value parameters with companion objects with static factory methods that are missing arguments of the targeted class, even if marked JvmStatic`() { val code = - """import kotlin.jvm.JvmInline - import kotlin.jvm.JvmStatic - @JvmInline value class Bar(val value: String) - data class Foo(val bar: Bar){ - companion object { - val test: Bar = Bar("test") - @JvmStatic fun applyMissingArguments():Foo = Foo(test) - } + """ + import kotlin.jvm.JvmInline + import kotlin.jvm.JvmStatic + @JvmInline value class Bar(val value: String) + data class Foo(val bar: Bar){ + companion object { + val test: Bar = Bar("test") + @JvmStatic fun applyMissingArguments():Foo = Foo(test) } - """ + }""" val findings = PublicDataClassConstructorWithValueParameters(Config.empty) .compileAndLintWithContext(env, code) @@ -88,16 +87,16 @@ internal class PublicDataClassConstructorWithValueParametersTest( @Test fun `does not report data classes with value parameters that have static factory methods in their companion objects that have all the arguments of the targeted class that are marked JvmStatic`() { val code = - """import kotlin.jvm.JvmStatic - import kotlin.jvm.JvmInline - @JvmInline value class Bar(val value: String) - data class Foo(val bar: Bar){ - companion object { - val test: String = "test" - @JvmStatic fun apply(bar: Bar):Foo = Foo(bar) - } + """ + import kotlin.jvm.JvmStatic + import kotlin.jvm.JvmInline + @JvmInline value class Bar(val value: String) + data class Foo(val bar: Bar) { + companion object { + val test: String = "test" + @JvmStatic fun apply(bar: Bar):Foo = Foo(bar) } - """ + }""" val findings = PublicDataClassConstructorWithValueParameters(Config.empty) .compileAndLintWithContext(env, code) diff --git a/docs/intro/scala.md b/docs/intro/scala.md index dd5be1a56..3ecb49a23 100644 --- a/docs/intro/scala.md +++ b/docs/intro/scala.md @@ -1,6 +1,6 @@ # Quick introduction to xef.ai (Scala version) -After adding the library to your project(see the +After adding the library to your project (see the [main README](https://github.com/xebia-functional/xef/blob/main/README.md) for instructions), you get access to the `conversation` function, which is your port of entry to the modern AI world. Inside of it, you can _prompt_ for information, which means posing the question to an LLM @@ -44,8 +44,8 @@ Set the environment variable `OPENAI_TOKEN=xxx` ### Project Loom The Scala module depends on project [Loom](https://openjdk.org/projects/loom/), -so you will need at least Java 20 to use the library. Furthermore, you need to pass -the `--enable-preview` flag. +so you will need at least Java 20 to use the library. Furthermore, if using Java 20 specifically, +you need to pass the `--enable-preview` flag.
SBT @@ -59,7 +59,7 @@ env OPENAI_TOKEN= sbt -J--enable-preview IntelliJ - Set the Java version to at least 20 -- Set VM options to `--enable-preview` +- If using Java 20 specifically, set VM options to `--enable-preview`
diff --git a/examples/java/build.gradle.kts b/examples/java/build.gradle.kts index 6acde3d30..aded6cd45 100644 --- a/examples/java/build.gradle.kts +++ b/examples/java/build.gradle.kts @@ -11,12 +11,6 @@ dependencies { implementation(projects.xefGpt4all) } -val ENABLE_PREVIEW = "--enable-preview" - -tasks.withType { - options.compilerArgs.add(ENABLE_PREVIEW) -} tasks.test { useJUnitPlatform() - jvmArgs(ENABLE_PREVIEW) } diff --git a/examples/kotlin/build.gradle.kts b/examples/kotlin/build.gradle.kts index 333c4cbf2..1addc2401 100644 --- a/examples/kotlin/build.gradle.kts +++ b/examples/kotlin/build.gradle.kts @@ -1,55 +1,47 @@ plugins { - id(libs.plugins.kotlin.jvm.get().pluginId) - id(libs.plugins.kotlinx.serialization.get().pluginId) - alias(libs.plugins.spotless) + id(libs.plugins.kotlin.jvm.get().pluginId) + id(libs.plugins.kotlinx.serialization.get().pluginId) + alias(libs.plugins.spotless) } -repositories { - mavenCentral() -} +repositories { mavenCentral() } java { - sourceCompatibility = JavaVersion.VERSION_11 - targetCompatibility = JavaVersion.VERSION_11 - toolchain { - languageVersion = JavaLanguageVersion.of(11) - } + sourceCompatibility = JavaVersion.VERSION_11 + targetCompatibility = JavaVersion.VERSION_11 + toolchain { languageVersion = JavaLanguageVersion.of(11) } } dependencies { - implementation(projects.xefKotlin) - implementation(projects.xefFilesystem) - implementation(projects.xefPdf) - implementation(projects.xefSql) - implementation(projects.xefTokenizer) - implementation(projects.xefGpt4all) - implementation(projects.xefGcp) - implementation(projects.xefOpenai) - implementation(projects.xefReasoning) - implementation(projects.xefOpentelemetry) - implementation(projects.xefMlflow) - implementation(libs.kotlinx.serialization.json) - implementation(libs.logback) - implementation(libs.klogging) - implementation(libs.bundles.arrow) - implementation(libs.okio) - implementation(libs.jdbc.mysql.connector) - api(libs.ktor.client) + implementation(projects.xefKotlin) + implementation(projects.xefFilesystem) + implementation(projects.xefPdf) + implementation(projects.xefSql) + implementation(projects.xefTokenizer) + implementation(projects.xefGpt4all) + implementation(projects.xefGcp) + implementation(projects.xefOpenai) + implementation(projects.xefReasoning) + implementation(projects.xefOpentelemetry) + implementation(projects.xefMlflow) + implementation(libs.kotlinx.serialization.json) + implementation(libs.logback) + implementation(libs.klogging) + implementation(libs.bundles.arrow) + implementation(libs.okio) + implementation(libs.jdbc.mysql.connector) + api(libs.ktor.client) } spotless { - kotlin { - target("**/*.kt") - ktfmt().googleStyle().configure { - it.setRemoveUnusedImport(true) + kotlin { + target("**/*.kt") + ktfmt().googleStyle().configure { it.setRemoveUnusedImport(true) } } - } } tasks.getByName("processResources") { - dependsOn(projects.xefGpt4all.dependencyProject.tasks.getByName("jvmProcessResources")) - from("${projects.xefGpt4all.dependencyProject.buildDir}/processedResources/jvm/main") - into("$buildDir/resources/main") + dependsOn(projects.xefGpt4all.dependencyProject.tasks.getByName("jvmProcessResources")) + from("${projects.xefGpt4all.dependencyProject.buildDir}/processedResources/jvm/main") + into("$buildDir/resources/main") } - - diff --git a/examples/scala/.scalafmt.conf b/examples/scala/.scalafmt.conf index f6540bcff..038d0ede0 100644 --- a/examples/scala/.scalafmt.conf +++ b/examples/scala/.scalafmt.conf @@ -1,6 +1,6 @@ # tune this file as appropriate to your style! see: https://olafurpg.github.io/scalafmt/#Configuration -version = "3.7.3" +version = "3.7.15" runner.dialect = "scala3" @@ -17,7 +17,6 @@ align { ifWhileOpenParen = false openParenCallSite = false openParenDefnSite = false - tokens = ["%", "%%"] } @@ -34,4 +33,4 @@ optIn { project.excludeFilters = [ "metals.sbt" -] \ No newline at end of file +] diff --git a/examples/scala/build.gradle.kts b/examples/scala/build.gradle.kts index a88198b95..615adf2a5 100644 --- a/examples/scala/build.gradle.kts +++ b/examples/scala/build.gradle.kts @@ -1,38 +1,30 @@ @file:Suppress("DSL_SCOPE_VIOLATION") plugins { - scala - alias(libs.plugins.spotless) + scala + alias(libs.plugins.spotless) } java { - sourceCompatibility = JavaVersion.VERSION_20 - targetCompatibility = JavaVersion.VERSION_20 - toolchain { - languageVersion = JavaLanguageVersion.of(20) - } + sourceCompatibility = JavaVersion.VERSION_21 + targetCompatibility = JavaVersion.VERSION_21 + toolchain { languageVersion = JavaLanguageVersion.of(21) } } dependencies { - implementation(projects.xefCore) - implementation(projects.xefScala) - implementation(projects.xefReasoning) - implementation(projects.xefOpenai) - implementation(libs.circe.parser) - implementation(libs.scala.lang) - implementation(libs.logback) + implementation(projects.xefCore) + implementation(projects.xefScala) + implementation(projects.xefReasoning) + implementation(projects.xefOpenai) + implementation(libs.circe.parser) + implementation(libs.scala.lang) + implementation(libs.logback) } -tasks.withType().configureEach { - useJUnit() -} +tasks.withType().configureEach { useJUnit() } tasks.withType { - scalaCompileOptions.additionalParameters = listOf("-Wunused:all", "-Wvalue-discard") + scalaCompileOptions.additionalParameters = listOf("-Wunused:all", "-Wvalue-discard") } -spotless { - scala { - scalafmt("3.7.3").configFile(".scalafmt.conf").scalaMajorVersion("2.13") - } -} +spotless { scala { scalafmt("3.7.15").configFile(".scalafmt.conf") } } diff --git a/filesystem/build.gradle.kts b/filesystem/build.gradle.kts index fdd5d6061..c2ced941d 100644 --- a/filesystem/build.gradle.kts +++ b/filesystem/build.gradle.kts @@ -21,11 +21,9 @@ repositories { mavenCentral() } kotlin { jvm() js(IR) { nodejs() } - linuxX64() macosX64() mingwX64() - sourceSets { val commonMain by getting { dependencies { @@ -34,9 +32,7 @@ kotlin { implementation(libs.klogging) } } - val jsMain by getting { dependencies { implementation(libs.okio.nodefilesystem) } } - commonTest { dependencies { implementation(libs.okio.fakefilesystem) @@ -45,13 +41,10 @@ kotlin { implementation(libs.kotest.assertions) } } - val jvmTest by getting { dependencies { implementation(libs.kotest.junit5) } } - val linuxX64Main by getting val macosX64Main by getting val mingwX64Main by getting - create("nativeMain") { dependsOn(commonMain) linuxX64Main.dependsOn(this) diff --git a/gpt4all-kotlin/README.MD b/gpt4all-kotlin/README.MD index 2d535b51b..8f937af29 100644 --- a/gpt4all-kotlin/README.MD +++ b/gpt4all-kotlin/README.MD @@ -16,9 +16,9 @@ It is also possible to use a GPT4All local installation by setting the `jna.libr In order to use the library, apart from the previous configuration, you must have downloaded some GPT4All model in your local, and pass both the model path and the model type: -``` +```kotlin GPT4All(Path.of("models/gpt4all/ggml-gpt4all-j-v1.3-groovy.bin"), GPT4AllModel.Type.GPTJ).use { gpt4All -> - val promptMessage = Message(Message.Role.USER, "Some prompt goes here") - gpt4All.chatCompletion(listOf(promptMessage)) + val promptMessage = Message(Message.Role.USER, "Some prompt goes here") + gpt4All.chatCompletion(listOf(promptMessage)) } ``` diff --git a/gpt4all-kotlin/build.gradle.kts b/gpt4all-kotlin/build.gradle.kts index 185db9d32..6a969980a 100644 --- a/gpt4all-kotlin/build.gradle.kts +++ b/gpt4all-kotlin/build.gradle.kts @@ -28,31 +28,23 @@ detekt { kotlin { jvm { compilations { - val integrationTest by - compilations.creating { - // Create a test task to run the tests produced by this compilation: - tasks.register("integrationTest") { - description = "Run the integration tests" - group = "verification" - classpath = - compileDependencyFiles + - runtimeDependencyFiles + - output.allOutputs - testClassesDirs = output.classesDirs - - testLogging { events("passed") } - } - } + val integrationTest by compilations.creating { + // Create a test task to run the tests produced by this compilation: + tasks.register("integrationTest") { + description = "Run the integration tests" + group = "verification" + classpath = compileDependencyFiles + runtimeDependencyFiles + output.allOutputs + testClassesDirs = output.classesDirs + testLogging { events("passed") } + } + } val test by compilations.getting integrationTest.associateWith(test) } } - js(IR) { browser() } - sourceSets { val commonMain by getting { dependencies { implementation(projects.xefCore) } } - commonTest { dependencies { implementation(kotlin("test")) @@ -61,16 +53,13 @@ kotlin { implementation(libs.kotest.assertions) } } - val jvmMain by getting { dependencies { implementation(libs.gpt4all.java.bindings) implementation(libs.ai.djl.huggingface.tokenizers) } } - val jsMain by getting {} - val jvmTest by getting { dependencies { implementation(libs.kotest.junit5) } } } } @@ -96,6 +85,5 @@ tasks { setEvents(listOf("passed", "skipped", "failed", "standardOut", "standardError")) } } - withType { dependsOn(withType()) } } diff --git a/gradle/wrapper/gradle-wrapper.jar b/gradle/wrapper/gradle-wrapper.jar index ccebba7710deaf9f98673a68957ea02138b60d0a..7f93135c49b765f8051ef9d0a6055ff8e46073d8 100644 GIT binary patch delta 38440 zcmZ6SQ*<3%6s?oSw$a$O+t{{kpWwtNwr$(CZL_f&H8y*DImUnA_rrded+oL6w`Tc1 z#Mv!G{V@^ba0IDo(ixEwh?a02n}!DKIvnc^%1aoMq1+#kAhJLP_oX|jgEPsr;V`0k zL+b-2$5Bo(3c}Buz&-hcO_SO<6900dXYt$Ter6+Mad;kQkW8)Rc`!+|ia-jh^V%qoFb5JQ`

cw$=gQbRBDL9uPo|6z1d&0P$^g)iXXq+aG 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 CEjxVx delta 36364 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%0Dd@gG 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 Pg;r%T9-eUK&({9{Yi$yT diff --git a/gradle/wrapper/gradle-wrapper.properties b/gradle/wrapper/gradle-wrapper.properties index c30b486a8..3fa8f862f 100644 --- a/gradle/wrapper/gradle-wrapper.properties +++ b/gradle/wrapper/gradle-wrapper.properties @@ -1,6 +1,7 @@ distributionBase=GRADLE_USER_HOME distributionPath=wrapper/dists -distributionUrl=https\://services.gradle.org/distributions/gradle-8.3-bin.zip +distributionUrl=https\://services.gradle.org/distributions/gradle-8.4-bin.zip networkTimeout=10000 +validateDistributionUrl=true zipStoreBase=GRADLE_USER_HOME zipStorePath=wrapper/dists diff --git a/gradlew b/gradlew index 79a61d421..1aa94a426 100755 --- a/gradlew +++ b/gradlew @@ -83,10 +83,8 @@ done # This is normally unused # shellcheck disable=SC2034 APP_BASE_NAME=${0##*/} -APP_HOME=$( cd "${APP_HOME:-./}" && pwd -P ) || exit - -# Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script. -DEFAULT_JVM_OPTS='"-Xmx64m" "-Xms64m"' +# Discard cd standard output in case $CDPATH is set (https://github.com/gradle/gradle/issues/25036) +APP_HOME=$( cd "${APP_HOME:-./}" > /dev/null && pwd -P ) || exit # Use the maximum available, or set MAX_FD != -1 to use that value. MAX_FD=maximum @@ -133,10 +131,13 @@ location of your Java installation." fi else JAVACMD=java - which java >/dev/null 2>&1 || die "ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH. + if ! command -v java >/dev/null 2>&1 + then + die "ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH. Please set the JAVA_HOME variable in your environment to match the location of your Java installation." + fi fi # Increase the maximum file descriptors if we can. @@ -144,7 +145,7 @@ if ! "$cygwin" && ! "$darwin" && ! "$nonstop" ; then case $MAX_FD in #( max*) # In POSIX sh, ulimit -H is undefined. That's why the result is checked to see if it worked. - # shellcheck disable=SC3045 + # shellcheck disable=SC2039,SC3045 MAX_FD=$( ulimit -H -n ) || warn "Could not query maximum file descriptor limit" esac @@ -152,7 +153,7 @@ if ! "$cygwin" && ! "$darwin" && ! "$nonstop" ; then '' | soft) :;; #( *) # In POSIX sh, ulimit -n is undefined. That's why the result is checked to see if it worked. - # shellcheck disable=SC3045 + # shellcheck disable=SC2039,SC3045 ulimit -n "$MAX_FD" || warn "Could not set maximum file descriptor limit to $MAX_FD" esac @@ -197,11 +198,15 @@ if "$cygwin" || "$msys" ; then done fi -# Collect all arguments for the java command; -# * $DEFAULT_JVM_OPTS, $JAVA_OPTS, and $GRADLE_OPTS can contain fragments of -# shell script including quotes and variable substitutions, so put them in -# double quotes to make sure that they get re-expanded; and -# * put everything else in single quotes, so that it's not re-expanded. + +# Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script. +DEFAULT_JVM_OPTS='"-Xmx64m" "-Xms64m"' + +# Collect all arguments for the java command: +# * DEFAULT_JVM_OPTS, JAVA_OPTS, JAVA_OPTS, and optsEnvironmentVar are not allowed to contain shell fragments, +# and any embedded shellness will be escaped. +# * For example: A user cannot expect ${Hostname} to be expanded, as it is an environment variable and will be +# treated as '${Hostname}' itself on the command line. set -- \ "-Dorg.gradle.appname=$APP_BASE_NAME" \ diff --git a/integrations/gcp/README.MD b/integrations/gcp/README.MD index 72342413e..402044d0e 100644 --- a/integrations/gcp/README.MD +++ b/integrations/gcp/README.MD @@ -11,14 +11,14 @@ The most simple solution would be to copy an existing http module, and modify th But let's cover the different important pieces. First we need to set up the Gradle plugins to configure both _Kotlin Multiplatform_, and _KotlinX Serialization_ for content negotiation. -```kotlin -id("org.jetbrains.kotlin.multiplatform") version "1.9.0" +```groovy +id("org.jetbrains.kotlin.multiplatform") version "1.9.10" ``` In the Xef project we've already defined these dependencies in the [Version Catalog](), so you get a typed DSL inside Gradle to set this up with automatic versioning. -```kotlin +```groovy id(libs.plugins.kotlin.multiplatform.get().pluginId) ``` @@ -27,7 +27,7 @@ and set up the targets for the desired platforms. For Xef we set up following targets: -```kotlin +```groovy kotlin { jvm() js(IR) { @@ -56,14 +56,14 @@ but remember you only have access to _common_ Kotlin code. So now JDK specific p So now we've configured Kotlin, we should set up KotlinX Serialization, and it's compiler plugin. This is needed so we can send `JSON`, and different formats over the network. -```kotlin -id("org.jetbrains.kotlin.plugin.serialization") version "1.9.0" +```groovy +id("org.jetbrains.kotlin.plugin.serialization") version "1.9.10" ``` In the Xef project we've already defined these dependencies in the [Version Catalog](), so you get a typed DSL inside Gradle to set this up with automatic versioning. -```kotlin +```groovy id(libs.plugins.kotlinx.serialization.get().pluginId) ``` @@ -78,7 +78,7 @@ We'll start with setting up some _common_ dependencies, we do so again within `k First let's add a dependency on `xef-core`, such that we can implement the `Chat`, `ChatWithFunction`, etc. interfaces based on our integration. -```kotlin +```groovy kotlin { sourceSets { val commonMain by getting { @@ -92,7 +92,7 @@ kotlin { Finally, we also need to set up Ktor. For most HTTP integration we need 3 _common_ dependencies. -```kotlin +```groovy implementation("io.ktor:ktor-client-core:2.3.2") implementation("io.ktor:ktor-client-content-negotiation:2.3.2") implementation("io.ktor:ktor-serialization-kotlinx-json:2.3.2") @@ -100,7 +100,7 @@ implementation("io.ktor:ktor-serialization-kotlinx-json:2.3.2") There are bundles in the Xef project, so we can more easily depend on them in a typed way. -```kotlin +```groovy kotlin { sourceSets { val commonMain by getting { @@ -123,42 +123,36 @@ Almost all targets are relying on the `CIO` engine, which is the _Coroutines_ en There are plenty of other options we can choose, more info [here](https://ktor.io/docs/http-client-engines.html#minimal-version). -```kotlin +```groovy kotlin { sourceSets { ... - val jvmMain by getting { dependencies { implementation(libs.logback) api(libs.ktor.client.cio) } } - val jsMain by getting { dependencies { api(libs.ktor.client.js) } } - val linuxX64Main by getting { dependencies { api(libs.ktor.client.cio) } } - val macosX64Main by getting { dependencies { api(libs.ktor.client.cio) } } - val macosArm64Main by getting { dependencies { api(libs.ktor.client.cio) } } - val mingwX64Main by getting { dependencies { api(libs.ktor.client.winhttp) @@ -365,4 +359,4 @@ HttpClient { ## Implementing Core's Chat interface -TODO \ No newline at end of file +TODO diff --git a/integrations/gcp/build.gradle.kts b/integrations/gcp/build.gradle.kts index e3b4d7fde..3c199d1a9 100644 --- a/integrations/gcp/build.gradle.kts +++ b/integrations/gcp/build.gradle.kts @@ -1,16 +1,13 @@ plugins { - id(libs.plugins.kotlin.multiplatform.get().pluginId) - id(libs.plugins.kotlinx.serialization.get().pluginId) - alias(libs.plugins.spotless) - alias(libs.plugins.arrow.gradle.publish) - alias(libs.plugins.semver.gradle) - alias(libs.plugins.detekt) + id(libs.plugins.kotlin.multiplatform.get().pluginId) + id(libs.plugins.kotlinx.serialization.get().pluginId) + alias(libs.plugins.spotless) + alias(libs.plugins.arrow.gradle.publish) + alias(libs.plugins.semver.gradle) + alias(libs.plugins.detekt) } - -dependencies { - detektPlugins(project(":detekt-rules")) -} +dependencies { detektPlugins(project(":detekt-rules")) } detekt { toolVersion = "1.23.1" @@ -19,105 +16,86 @@ detekt { autoCorrect = true } - -repositories { - mavenCentral() -} +repositories { mavenCentral() } java { - sourceCompatibility = JavaVersion.VERSION_11 - targetCompatibility = JavaVersion.VERSION_11 - toolchain { - languageVersion = JavaLanguageVersion.of(11) - } + sourceCompatibility = JavaVersion.VERSION_11 + targetCompatibility = JavaVersion.VERSION_11 + toolchain { languageVersion = JavaLanguageVersion.of(11) } } kotlin { - jvm() - js(IR) { - browser() - nodejs() - } - - linuxX64() - macosX64() - macosArm64() - mingwX64() - - sourceSets { - val commonMain by getting { - dependencies { - api(projects.xefCore) - implementation(libs.bundles.ktor.client) - implementation(libs.uuid) - implementation(libs.kotlinx.datetime) - } + jvm() + js(IR) { + browser() + nodejs() } - - val jvmMain by getting { - dependencies { - implementation(libs.logback) - api(libs.ktor.client.cio) - } - } - - val jsMain by getting { - dependencies { - api(libs.ktor.client.js) - } + linuxX64() + macosX64() + macosArm64() + mingwX64() + sourceSets { + val commonMain by getting { + dependencies { + api(projects.xefCore) + implementation(libs.bundles.ktor.client) + implementation(libs.uuid) + implementation(libs.kotlinx.datetime) + } + } + val jvmMain by getting { + dependencies { + implementation(libs.logback) + api(libs.ktor.client.cio) + } + } + val jsMain by getting { + dependencies { + api(libs.ktor.client.js) + } + } + val linuxX64Main by getting { + dependencies { + api(libs.ktor.client.cio) + } + } + val macosX64Main by getting { + dependencies { + api(libs.ktor.client.cio) + } + } + val macosArm64Main by getting { + dependencies { + api(libs.ktor.client.cio) + } + } + val mingwX64Main by getting { + dependencies { + api(libs.ktor.client.winhttp) + } + } } - - val linuxX64Main by getting { - dependencies { - api(libs.ktor.client.cio) - } - } - - val macosX64Main by getting { - dependencies { - api(libs.ktor.client.cio) - } - } - - val macosArm64Main by getting { - dependencies { - api(libs.ktor.client.cio) - } - } - - val mingwX64Main by getting { - dependencies { - api(libs.ktor.client.winhttp) - } - } - } } spotless { - kotlin { - target("**/*.kt") - ktfmt().googleStyle().configure { - it.setRemoveUnusedImport(true) + kotlin { + target("**/*.kt") + ktfmt().googleStyle().configure { it.setRemoveUnusedImport(true) } } - } } -tasks{ - withType().configureEach { - dependsOn(":detekt-rules:assemble") - autoCorrect = true - } - named("detektJvmMain") { - dependsOn(":detekt-rules:assemble") - getByName("build").dependsOn(this) - } - named("detekt") { - dependsOn(":detekt-rules:assemble") - getByName("build").dependsOn(this) - } - withType { - dependsOn(withType()) - } - +tasks { + withType().configureEach { + dependsOn(":detekt-rules:assemble") + autoCorrect = true + } + named("detektJvmMain") { + dependsOn(":detekt-rules:assemble") + getByName("build").dependsOn(this) + } + named("detekt") { + dependsOn(":detekt-rules:assemble") + getByName("build").dependsOn(this) + } + withType { dependsOn(withType()) } } - diff --git a/integrations/lucene/build.gradle.kts b/integrations/lucene/build.gradle.kts index a6557f3c8..1a6f9b961 100644 --- a/integrations/lucene/build.gradle.kts +++ b/integrations/lucene/build.gradle.kts @@ -38,6 +38,5 @@ tasks { dependsOn(":detekt-rules:assemble") getByName("build").dependsOn(this) } - withType { dependsOn(withType()) } } diff --git a/integrations/mlflow/build.gradle.kts b/integrations/mlflow/build.gradle.kts index da8ed47ff..7aa655f29 100644 --- a/integrations/mlflow/build.gradle.kts +++ b/integrations/mlflow/build.gradle.kts @@ -7,10 +7,7 @@ plugins { alias(libs.plugins.detekt) } - -dependencies { - detektPlugins(project(":detekt-rules")) -} +dependencies { detektPlugins(project(":detekt-rules")) } detekt { toolVersion = "1.23.1" @@ -19,17 +16,12 @@ detekt { autoCorrect = true } - -repositories { - mavenCentral() -} +repositories { mavenCentral() } java { sourceCompatibility = JavaVersion.VERSION_11 targetCompatibility = JavaVersion.VERSION_11 - toolchain { - languageVersion = JavaLanguageVersion.of(11) - } + toolchain { languageVersion = JavaLanguageVersion.of(11) } } kotlin { @@ -38,12 +30,10 @@ kotlin { browser() nodejs() } - linuxX64() macosX64() macosArm64() mingwX64() - sourceSets { val commonMain by getting { dependencies { @@ -53,38 +43,32 @@ kotlin { implementation(libs.kotlinx.datetime) } } - val jvmMain by getting { dependencies { implementation(libs.logback) api(libs.ktor.client.cio) } } - val jsMain by getting { dependencies { api(libs.ktor.client.js) } } - val linuxX64Main by getting { dependencies { api(libs.ktor.client.cio) } } - val macosX64Main by getting { dependencies { api(libs.ktor.client.cio) } } - val macosArm64Main by getting { dependencies { api(libs.ktor.client.cio) } } - val mingwX64Main by getting { dependencies { api(libs.ktor.client.winhttp) @@ -96,9 +80,7 @@ kotlin { spotless { kotlin { target("**/*.kt") - ktfmt().googleStyle().configure { - it.setRemoveUnusedImport(true) - } + ktfmt().googleStyle().configure { it.setRemoveUnusedImport(true) } } } @@ -118,6 +100,4 @@ tasks{ withType { dependsOn(withType()) } - } - diff --git a/integrations/opentelemetry/build.gradle.kts b/integrations/opentelemetry/build.gradle.kts index 6437e05f0..aae9caa41 100644 --- a/integrations/opentelemetry/build.gradle.kts +++ b/integrations/opentelemetry/build.gradle.kts @@ -32,7 +32,6 @@ dependencies { implementation(libs.opentelemetry.semconv) implementation(libs.opentelemetry.extension.kotlin) implementation(libs.opentelemetry.exporter.otlp) - testImplementation(libs.junit.jupiter.api) testImplementation(libs.kotest.property) testImplementation(libs.kotest.framework) @@ -43,9 +42,7 @@ dependencies { spotless { kotlin { target("**/*.kt") - ktfmt().googleStyle().configure { - it.setRemoveUnusedImport(true) - } + ktfmt().googleStyle().configure { it.setRemoveUnusedImport(true) } } } @@ -58,6 +55,5 @@ tasks { dependsOn(":detekt-rules:assemble") getByName("build").dependsOn(this) } - withType { dependsOn(withType()) } } diff --git a/integrations/pdf/build.gradle.kts b/integrations/pdf/build.gradle.kts index 2e814686d..fd49a4a85 100644 --- a/integrations/pdf/build.gradle.kts +++ b/integrations/pdf/build.gradle.kts @@ -38,6 +38,5 @@ tasks { dependsOn(":detekt-rules:assemble") getByName("build").dependsOn(this) } - withType { dependsOn(withType()) } } diff --git a/integrations/postgresql/build.gradle.kts b/integrations/postgresql/build.gradle.kts index e30f8dce9..91759b10f 100644 --- a/integrations/postgresql/build.gradle.kts +++ b/integrations/postgresql/build.gradle.kts @@ -29,7 +29,6 @@ dependencies { implementation(libs.uuid) implementation(libs.hikari) implementation(libs.postgresql) - testImplementation(libs.junit.jupiter.api) testImplementation(libs.kotest.property) testImplementation(libs.kotest.framework) @@ -48,10 +47,7 @@ tasks { dependsOn(":detekt-rules:assemble") getByName("build").dependsOn(this) } - withType { dependsOn(withType()) } } -tasks.test{ - useJUnitPlatform() -} +tasks.test{ useJUnitPlatform() } diff --git a/integrations/sql/build.gradle.kts b/integrations/sql/build.gradle.kts index f66a674cc..128751c42 100644 --- a/integrations/sql/build.gradle.kts +++ b/integrations/sql/build.gradle.kts @@ -1,24 +1,18 @@ -import org.jetbrains.kotlin.gradle.tasks.KotlinCompile - plugins { id(libs.plugins.kotlin.jvm.get().pluginId) alias(libs.plugins.arrow.gradle.publish) alias(libs.plugins.semver.gradle) - alias(libs.plugins.detekt) + alias(libs.plugins.detekt) } dependencies { detektPlugins(project(":detekt-rules")) } -repositories { - mavenCentral() -} +repositories { mavenCentral() } java { sourceCompatibility = JavaVersion.VERSION_11 targetCompatibility = JavaVersion.VERSION_11 - toolchain { - languageVersion = JavaLanguageVersion.of(11) - } + toolchain { languageVersion = JavaLanguageVersion.of(11) } } detekt { @@ -43,6 +37,5 @@ tasks { dependsOn(":detekt-rules:assemble") getByName("build").dependsOn(this) } - withType { dependsOn(withType()) } } diff --git a/java/README.md b/java/README.md index 8c8f7c4ee..380f6937b 100644 --- a/java/README.md +++ b/java/README.md @@ -2,7 +2,7 @@ Build the project locally, from the project root: -```bash +```shell ./gradlew build ``` @@ -11,7 +11,7 @@ Build the project locally, from the project root: The Java module uses the [spotless](https://github.com/diffplug/spotless/tree/main/plugin-gradle#java) plugin. Therefore, the previous command (`./gradlew build`) will fail in case there is any formatting issue. To apply format, you can run the following command: -```bash +```shell ./gradlew spotlessApply ``` @@ -23,4 +23,4 @@ Check out some use case at the [Java examples](../examples/java) folder. How to run the examples (from IntelliJ IDEA): -* Set Env variable: "OPENAI_TOKEN=xxx" +* Set Env variable: `OPENAI_TOKEN=xxx` diff --git a/java/build.gradle.kts b/java/build.gradle.kts index 4d9a61920..99ba21ab2 100644 --- a/java/build.gradle.kts +++ b/java/build.gradle.kts @@ -27,10 +27,6 @@ java { withSourcesJar() } -tasks.withType().configureEach { - useJUnit() -} +tasks.withType().configureEach { useJUnit() } -tasks.withType { - dependsOn(tasks.withType()) -} +tasks.withType { dependsOn(tasks.withType()) } diff --git a/kotlin/build.gradle.kts b/kotlin/build.gradle.kts index 2c49aba70..0b0d2e196 100644 --- a/kotlin/build.gradle.kts +++ b/kotlin/build.gradle.kts @@ -35,12 +35,10 @@ kotlin { browser() nodejs() } - linuxX64() macosX64() macosArm64() mingwX64() - sourceSets { val commonMain by getting { dependencies { @@ -79,7 +77,6 @@ tasks { setEvents(listOf("passed", "skipped", "failed", "standardOut", "standardError")) } } - withType().configureEach { kotlin.sourceSets.forEach { kotlinSourceSet -> dokkaSourceSets.named(kotlinSourceSet.name) { @@ -89,15 +86,11 @@ tasks { } skipDeprecated.set(true) reportUndocumented.set(false) - val baseUrl: String = checkNotNull(project.properties["pom.smc.url"]?.toString()) - + val baseUrl = checkNotNull(project.properties["pom.smc.url"]?.toString()) kotlinSourceSet.kotlin.srcDirs.filter { it.exists() }.forEach { srcDir -> sourceLink { localDirectory.set(srcDir) - remoteUrl.set( - uri("$baseUrl/blob/main/${srcDir.relativeTo(rootProject.rootDir)}") - .toURL() - ) + remoteUrl.set(uri("$baseUrl/blob/main/${srcDir.relativeTo(rootProject.rootDir)}").toURL()) remoteLineSuffix.set("#L") } } diff --git a/models/gpt4all/get-models.sh b/models/gpt4all/get-models.sh index 459dec821..32afa866d 100755 --- a/models/gpt4all/get-models.sh +++ b/models/gpt4all/get-models.sh @@ -1,4 +1,3 @@ #!/usr/bin/env bash -#curl --url https://gpt4all.io/models/ggml-gpt4all-l13b-snoozy.bin --output ./ggml-gpt4all-l13b-snoozy.bin curl --url https://gpt4all.io/models/ggml-gpt4all-j-v1.3-groovy.bin --output ./ggml-gpt4all-j-v1.3-groovy.bin diff --git a/openai/build.gradle.kts b/openai/build.gradle.kts index 20bc75bbe..7e846e525 100644 --- a/openai/build.gradle.kts +++ b/openai/build.gradle.kts @@ -14,7 +14,6 @@ plugins { alias(libs.plugins.arrow.gradle.publish) alias(libs.plugins.semver.gradle) alias(libs.plugins.detekt) - // id("com.xebia.asfuture").version("0.0.1") } dependencies { detektPlugins(project(":detekt-rules")) } @@ -35,21 +34,16 @@ detekt { kotlin { jvm { compilations { - val integrationTest by - compilations.creating { - // Create a test task to run the tests produced by this compilation: - tasks.register("integrationTest") { - description = "Run the integration tests" - group = "verification" - classpath = - compileDependencyFiles + - runtimeDependencyFiles + - output.allOutputs - testClassesDirs = output.classesDirs - - testLogging { events("passed") } - } - } + val integrationTest by compilations.creating { + // Create a test task to run the tests produced by this compilation: + tasks.register("integrationTest") { + description = "Run the integration tests" + group = "verification" + classpath = compileDependencyFiles + runtimeDependencyFiles + output.allOutputs + testClassesDirs = output.classesDirs + testLogging { events("passed") } + } + } val test by compilations.getting integrationTest.associateWith(test) } @@ -58,15 +52,12 @@ kotlin { browser() nodejs() } - linuxX64() macosX64() macosArm64() mingwX64() - sourceSets { all { languageSettings.optIn("kotlin.ExperimentalStdlibApi") } - val commonMain by getting { dependencies { implementation(projects.xefCore) @@ -74,7 +65,6 @@ kotlin { implementation(libs.klogging) } } - val commonTest by getting { dependencies { implementation(libs.kotest.property) @@ -82,31 +72,22 @@ kotlin { implementation(libs.kotest.assertions) } } - val jvmMain by getting { dependencies { implementation(libs.logback) api(libs.ktor.client.cio) } } - val jvmTest by getting { dependencies { implementation(libs.kotest.junit5) } } - val jsMain by getting { dependencies { api(libs.ktor.client.js) } } - val linuxX64Main by getting { dependencies { api(libs.ktor.client.cio) } } - val macosX64Main by getting { dependencies { api(libs.ktor.client.cio) } } - val macosArm64Main by getting { dependencies { api(libs.ktor.client.cio) } } - val mingwX64Main by getting { dependencies { api(libs.ktor.client.winhttp) } } - val linuxX64Test by getting val macosX64Test by getting val macosArm64Test by getting val mingwX64Test by getting - create("nativeMain") { dependsOn(commonMain) linuxX64Main.dependsOn(this) @@ -114,7 +95,6 @@ kotlin { macosArm64Main.dependsOn(this) mingwX64Main.dependsOn(this) } - create("nativeTest") { dependsOn(commonTest) linuxX64Test.dependsOn(this) @@ -153,7 +133,6 @@ tasks { setEvents(listOf("passed", "skipped", "failed", "standardOut", "standardError")) } } - withType().configureEach { kotlin.sourceSets.forEach { kotlinSourceSet -> dokkaSourceSets.named(kotlinSourceSet.name) { @@ -163,15 +142,11 @@ tasks { } skipDeprecated.set(true) reportUndocumented.set(false) - val baseUrl: String = checkNotNull(project.properties["pom.smc.url"]?.toString()) - + val baseUrl = checkNotNull(project.properties["pom.smc.url"]?.toString()) kotlinSourceSet.kotlin.srcDirs.filter { it.exists() }.forEach { srcDir -> sourceLink { localDirectory.set(srcDir) - remoteUrl.set( - uri("$baseUrl/blob/main/${srcDir.relativeTo(rootProject.rootDir)}") - .toURL() - ) + remoteUrl.set(uri("$baseUrl/blob/main/${srcDir.relativeTo(rootProject.rootDir)}").toURL()) remoteLineSuffix.set("#L") } } diff --git a/reasoning/build.gradle.kts b/reasoning/build.gradle.kts index ce103b59e..84a10a426 100644 --- a/reasoning/build.gradle.kts +++ b/reasoning/build.gradle.kts @@ -14,7 +14,6 @@ plugins { alias(libs.plugins.arrow.gradle.publish) alias(libs.plugins.semver.gradle) alias(libs.plugins.detekt) - // id("com.xebia.asfuture").version("0.0.1") } java { @@ -35,21 +34,16 @@ detekt { kotlin { jvm { compilations { - val integrationTest by - compilations.creating { - // Create a test task to run the tests produced by this compilation: - tasks.register("integrationTest") { - description = "Run the integration tests" - group = "verification" - classpath = - compileDependencyFiles + - runtimeDependencyFiles + - output.allOutputs - testClassesDirs = output.classesDirs - - testLogging { events("passed") } - } - } + val integrationTest by compilations.creating { + // Create a test task to run the tests produced by this compilation: + tasks.register("integrationTest") { + description = "Run the integration tests" + group = "verification" + classpath = compileDependencyFiles + runtimeDependencyFiles + output.allOutputs + testClassesDirs = output.classesDirs + testLogging { events("passed") } + } + } val test by compilations.getting integrationTest.associateWith(test) } @@ -58,15 +52,12 @@ kotlin { browser() nodejs() } - linuxX64() macosX64() macosArm64() mingwX64() - sourceSets { all { languageSettings.optIn("kotlin.ExperimentalStdlibApi") } - val commonMain by getting { dependencies { implementation(projects.xefCore) @@ -76,7 +67,6 @@ kotlin { implementation(libs.bundles.ktor.client) } } - val commonTest by getting { dependencies { implementation(libs.kotest.property) @@ -84,7 +74,6 @@ kotlin { implementation(libs.kotest.assertions) } } - val jvmMain by getting { dependencies { implementation(libs.logback) @@ -93,19 +82,12 @@ kotlin { api(libs.ktor.client.cio) } } - val jsMain by getting { dependencies { api(libs.ktor.client.js) } } - val jvmTest by getting { dependencies { implementation(libs.kotest.junit5) } } - val linuxX64Main by getting { dependencies { api(libs.ktor.client.cio) } } - val macosX64Main by getting { dependencies { api(libs.ktor.client.cio) } } - val macosArm64Main by getting { dependencies { api(libs.ktor.client.cio) } } - val mingwX64Main by getting { dependencies { api(libs.ktor.client.winhttp) } } - create("nativeMain") { dependsOn(commonMain) linuxX64Main.dependsOn(this) @@ -135,7 +117,6 @@ tasks { dependsOn(":detekt-rules:assemble") getByName("build").dependsOn(this) } - withType().configureEach { maxParallelForks = Runtime.getRuntime().availableProcessors() useJUnitPlatform() @@ -144,7 +125,6 @@ tasks { setEvents(listOf("passed", "skipped", "failed", "standardOut", "standardError")) } } - withType().configureEach { kotlin.sourceSets.forEach { kotlinSourceSet -> dokkaSourceSets.named(kotlinSourceSet.name) { @@ -154,15 +134,11 @@ tasks { } skipDeprecated.set(true) reportUndocumented.set(false) - val baseUrl: String = checkNotNull(project.properties["pom.smc.url"]?.toString()) - + val baseUrl = checkNotNull(project.properties["pom.smc.url"]?.toString()) kotlinSourceSet.kotlin.srcDirs.filter { it.exists() }.forEach { srcDir -> sourceLink { localDirectory.set(srcDir) - remoteUrl.set( - uri("$baseUrl/blob/main/${srcDir.relativeTo(rootProject.rootDir)}") - .toURL() - ) + remoteUrl.set(uri("$baseUrl/blob/main/${srcDir.relativeTo(rootProject.rootDir)}").toURL()) remoteLineSuffix.set("#L") } } diff --git a/scala/.scalafmt.conf b/scala/.scalafmt.conf index f6540bcff..038d0ede0 100644 --- a/scala/.scalafmt.conf +++ b/scala/.scalafmt.conf @@ -1,6 +1,6 @@ # tune this file as appropriate to your style! see: https://olafurpg.github.io/scalafmt/#Configuration -version = "3.7.3" +version = "3.7.15" runner.dialect = "scala3" @@ -17,7 +17,6 @@ align { ifWhileOpenParen = false openParenCallSite = false openParenDefnSite = false - tokens = ["%", "%%"] } @@ -34,4 +33,4 @@ optIn { project.excludeFilters = [ "metals.sbt" -] \ No newline at end of file +] diff --git a/scala/README.md b/scala/README.md index c04cd2a66..e74c18873 100644 --- a/scala/README.md +++ b/scala/README.md @@ -2,7 +2,7 @@ Build the project locally, from the project root: -```bash +```shell ./gradlew build ``` @@ -12,7 +12,7 @@ The Scala module uses the [spotless](https://github.com/diffplug/spotless/tree/m Therefore, the previous command (`./gradlew build`) will fail if there are any formatting issues. To apply formatting, you can run the following command: -```bash +```shell ./gradlew spotlessApply ``` @@ -24,6 +24,6 @@ Check out some use case at the [Scala examples](../examples/scala) folder. How to run the examples (from IntelliJ IDEA): -* Set Java version 20 -* Set VM options: `--enable-preview` +* Set Java version 20 or above +* Set VM options: `--enable-preview` (if using Java 20 specifically) * Set Env variable: `OPENAI_TOKEN=xxx` diff --git a/scala/build.gradle.kts b/scala/build.gradle.kts index 16d80a64b..e17e0c4c4 100644 --- a/scala/build.gradle.kts +++ b/scala/build.gradle.kts @@ -12,11 +12,9 @@ plugins { dependencies { implementation(projects.xefCore) implementation(projects.xefOpenai) - implementation(libs.kotlinx.coroutines.reactive) - + implementation(libs.kotlinx.coroutines.reactive) // TODO split to separate Scala library implementation(projects.xefPdf) - implementation(libs.circe.parser) implementation(libs.circe) implementation(libs.scala.lang) @@ -25,24 +23,16 @@ dependencies { } java { - sourceCompatibility = JavaVersion.VERSION_20 - targetCompatibility = JavaVersion.VERSION_20 - toolchain { - languageVersion = JavaLanguageVersion.of(20) - } + sourceCompatibility = JavaVersion.VERSION_21 + targetCompatibility = JavaVersion.VERSION_21 + toolchain { languageVersion = JavaLanguageVersion.of(21) } withSourcesJar() } -tasks.withType().configureEach { - useJUnit() -} +tasks.withType().configureEach { useJUnit() } tasks.withType { scalaCompileOptions.additionalParameters = listOf("-Wunused:all", "-Wvalue-discard") } -spotless { - scala { - scalafmt("3.7.3").configFile(".scalafmt.conf").scalaMajorVersion("2.13") - } -} +spotless { scala { scalafmt("3.7.15").configFile(".scalafmt.conf") } } diff --git a/server/README.md b/server/README.md index a5328da75..0c311f895 100644 --- a/server/README.md +++ b/server/README.md @@ -8,16 +8,16 @@ In order to run the server, you need to run the following services: ### Docker -```bash - docker-compose -f docker/postgresql/docker-compose.yaml up +```shell +docker-compose -f docker/postgresql/docker-compose.yaml up ``` ### Server -```bash - ./gradlew server +```shell +./gradlew server ``` ### Web -Please, refer to the [web README](web/README.md) for more information. +Please refer to the [web README](web/README.md) for more information. diff --git a/server/build.gradle.kts b/server/build.gradle.kts index 6b7025df3..d7d360436 100644 --- a/server/build.gradle.kts +++ b/server/build.gradle.kts @@ -7,21 +7,15 @@ plugins { alias(libs.plugins.spotless) } -repositories { - mavenCentral() -} +repositories { mavenCentral() } java { sourceCompatibility = JavaVersion.VERSION_11 targetCompatibility = JavaVersion.VERSION_11 - toolchain { - languageVersion = JavaLanguageVersion.of(11) - } + toolchain { languageVersion = JavaLanguageVersion.of(11) } } -node { - nodeProjectDir.set(file("${project.projectDir}/web")) -} +node { nodeProjectDir.set(file("${project.projectDir}/web")) } dependencies { implementation(libs.exposed.core) @@ -58,7 +52,6 @@ dependencies { implementation(projects.xefCore) implementation(projects.xefLucene) implementation(projects.xefPostgresql) - testImplementation(libs.junit.jupiter.api) testImplementation(libs.junit.jupiter.engine) testImplementation(libs.kotest.property) @@ -72,9 +65,7 @@ dependencies { spotless { kotlin { target("**/*.kt") - ktfmt().googleStyle().configure { - it.setRemoveUnusedImport(true) - } + ktfmt().googleStyle().configure { it.setRemoveUnusedImport(true) } } } @@ -90,7 +81,7 @@ task("web-app") { description = "xef-server web application" classpath = sourceSets.main.get().runtimeClasspath mainClass.set("com.xebia.functional.xef.server.WebApp") - } +} task("server") { dependsOn("compileKotlin") @@ -100,10 +91,6 @@ task("server") { mainClass.set("com.xebia.functional.xef.server.Server") } -tasks.named("test") { - useJUnitPlatform() -} +tasks.named("test") { useJUnitPlatform() } -tasks.withType { - dependsOn(tasks.withType()) -} +tasks.withType { dependsOn(tasks.withType()) } diff --git a/server/web/README.md b/server/web/README.md index 55ef9366a..371e940cd 100644 --- a/server/web/README.md +++ b/server/web/README.md @@ -19,7 +19,7 @@ You need to have Node.js versions 14.18+ or 16+. But the current LTS is [Node.js As other general JS and Node.js based projects, dependencies are managed through `npm`, so the first thing needed to be done to run the project is to install its dependencies with: -```bash +```shell npm install ``` @@ -29,17 +29,17 @@ npm install Through vite, you can currently run the following commands in the project: - Start a dev server to work on the project: - ```bash + ```shell npm run dev ``` - Build the project for production: - ```bash + ```shell npm run build ``` - Locally preview a production build: - ```bash + ```shell npm run preview ``` @@ -100,4 +100,3 @@ Thorugh [ESLint](https://eslint.org/) and [Prettier](https://prettier.io/), you - Code concerning state and other contexts related functions are in `state`. - The MUI theme definition and configuration is set in the `styles` folder. You might not need to touch any of these settings. - Other utility and helper functions should be placed under `src/utils`. - diff --git a/settings.gradle.kts b/settings.gradle.kts index 6c8417956..6816a46bc 100644 --- a/settings.gradle.kts +++ b/settings.gradle.kts @@ -63,7 +63,6 @@ project(":xef-kotlin").projectDir = file("kotlin") include("xef-kotlin-examples") project(":xef-kotlin-examples").projectDir = file("examples/kotlin") - // // @@ -72,7 +71,6 @@ project(":xef-scala-examples").projectDir = file("examples/scala") include("xef-scala") project(":xef-scala").projectDir = file("scala") - // // @@ -100,4 +98,3 @@ project(":xef-server").projectDir = file("server") include("detekt-rules") project(":detekt-rules").projectDir = file("detekt-rules") // - diff --git a/tokenizer/build.gradle.kts b/tokenizer/build.gradle.kts index d0e72fe80..a5e71ff72 100644 --- a/tokenizer/build.gradle.kts +++ b/tokenizer/build.gradle.kts @@ -35,10 +35,8 @@ kotlin { macosX64() macosArm64() mingwX64() - sourceSets { val commonMain by getting - commonTest { dependencies { implementation(kotlin("test")) @@ -48,19 +46,15 @@ kotlin { implementation("com.goncalossilva:resources:0.3.2") } } - val jvmTest by getting { dependencies { implementation(libs.kotest.junit5) } } - js { nodejs { testTask { useMocha { timeout = "10000" } } } browser { testTask { useMocha { timeout = "10000" } } } } - val linuxX64Main by getting val macosX64Main by getting val macosArm64Main by getting val mingwX64Main by getting - create("nativeMain") { dependsOn(commonMain) linuxX64Main.dependsOn(this)