Added JokeResponse data class

This commit is contained in:
Erik C. Thauvin 2024-12-25 18:04:40 -08:00
parent 5ac8f910b5
commit b95e1541fb
Signed by: erik
GPG key ID: 776702A6A2DA330E
11 changed files with 88 additions and 29 deletions

5
.idea/misc.xml generated
View file

@ -1,9 +1,14 @@
<?xml version="1.0" encoding="UTF-8"?> <?xml version="1.0" encoding="UTF-8"?>
<project version="4"> <project version="4">
<component name="EntryPointsManager"> <component name="EntryPointsManager">
<entry_points version="2.0">
<entry_point TYPE="field" FQNAME="net.thauvin.erik.jokeapi.models.Category SPOOKY" />
</entry_points>
<pattern value="net.thauvin.erik.JokeApiBuild" method="jacoco" /> <pattern value="net.thauvin.erik.JokeApiBuild" method="jacoco" />
<pattern value="net.thauvin.erik.JokeApiBuild" method="detekt" /> <pattern value="net.thauvin.erik.JokeApiBuild" method="detekt" />
<pattern value="net.thauvin.erik.JokeApiBuild" method="detektBaseline" /> <pattern value="net.thauvin.erik.JokeApiBuild" method="detektBaseline" />
<pattern value="net.thauvin.erik.jokeapi.models.Category" />
<pattern value="net.thauvin.erik.jokeapi.models.Category" method="Category" />
</component> </component>
<component name="ProjectRootManager" version="2" languageLevel="JDK_17" project-jdk-name="17" project-jdk-type="JavaSDK"> <component name="ProjectRootManager" version="2" languageLevel="JDK_17" project-jdk-name="17" project-jdk-type="JavaSDK">
<output url="file://$PROJECT_DIR$/build" /> <output url="file://$PROJECT_DIR$/build" />

View file

@ -16,7 +16,7 @@ A simple library to retrieve jokes from [Sv443's JokeAPI](https://v2.jokeapi.dev
## Examples (TL;DR) ## Examples (TL;DR)
```kotlin ```kotlin
import net.thauvin.erik.jokeapi.getJoke import net.thauvin.erik.jokeapi.joke
val joke = joke() val joke = joke()
val safe = joke(safe = true) val safe = joke(safe = true)
@ -124,8 +124,9 @@ You can also retrieve one or more raw (unprocessed) jokes in all [supported form
For example for YAML: For example for YAML:
```kotlin ```kotlin
var joke = getRawJokes(format = Format.YAML, idRange = IdRange(22)) var jokes = getRawJokes(format = Format.YAML, idRange = IdRange(22))
println(joke) println(jokes.data)
```
``` ```
```yaml ```yaml
error: false error: false
@ -158,7 +159,7 @@ val lang = JokeApi.apiCall(
path = "french", path = "french",
params = mapOf(Parameter.FORMAT to Format.YAML.value) params = mapOf(Parameter.FORMAT to Format.YAML.value)
) )
println(lang) println(lang.data)
``` ```
```yaml ```yaml
error: false error: false

View file

@ -54,7 +54,7 @@ object JokeApi {
/** /**
* Makes a direct API call. * Makes a direct API call.
* *
* Sse the [JokeAPI Documentation](https://jokeapi.dev/#endpoints) for more details. * See the [JokeAPI Documentation](https://jokeapi.dev/#endpoints) for more details.
*/ */
@JvmStatic @JvmStatic
@JvmOverloads @JvmOverloads
@ -64,7 +64,7 @@ object JokeApi {
path: String = "", path: String = "",
params: Map<String, String> = emptyMap(), params: Map<String, String> = emptyMap(),
auth: String = "" auth: String = ""
): String { ): JokeResponse {
val urlBuilder = StringBuilder("$API_URL$endPoint") val urlBuilder = StringBuilder("$API_URL$endPoint")
if (path.isNotEmpty()) { if (path.isNotEmpty()) {
@ -98,7 +98,7 @@ object JokeApi {
*/ */
@JvmStatic @JvmStatic
@Throws(HttpErrorException::class) @Throws(HttpErrorException::class)
fun getRawJokes(config: JokeConfig): String { fun getRawJokes(config: JokeConfig): JokeResponse {
return rawJokes( return rawJokes(
categories = config.categories, categories = config.categories,
lang = config.lang, lang = config.lang,
@ -213,7 +213,7 @@ fun joke(
idRange = idRange, idRange = idRange,
safe = safe, safe = safe,
auth = auth auth = auth
) ).data
) )
if (json.getBoolean("error")) { if (json.getBoolean("error")) {
throw parseError(json) throw parseError(json)
@ -281,7 +281,7 @@ fun jokes(
amount = amount, amount = amount,
safe = safe, safe = safe,
auth = auth auth = auth
) ).data
) )
if (json.getBoolean("error")) { if (json.getBoolean("error")) {
throw parseError(json) throw parseError(json)
@ -333,6 +333,7 @@ fun jokes(
* At the moment, you will only receive one of these tokens temporarily if something breaks or if you are a business * At the moment, you will only receive one of these tokens temporarily if something breaks or if you are a business
* and need more than 120 requests per minute. * and need more than 120 requests per minute.
*/ */
@Throws(HttpErrorException::class)
fun rawJokes( fun rawJokes(
categories: Set<Category> = setOf(Category.ANY), categories: Set<Category> = setOf(Category.ANY),
lang: Language = Language.EN, lang: Language = Language.EN,
@ -344,7 +345,7 @@ fun rawJokes(
amount: Int = 1, amount: Int = 1,
safe: Boolean = false, safe: Boolean = false,
auth: String = "" auth: String = ""
): String { ): JokeResponse {
val params = mutableMapOf<String, String>() val params = mutableMapOf<String, String>()
// Categories // Categories

View file

@ -45,7 +45,7 @@ import java.util.logging.Level
/** /**
* Fetch a URL. * Fetch a URL.
*/ */
internal fun fetchUrl(url: String, auth: String = ""): String { internal fun fetchUrl(url: String, auth: String = ""): JokeResponse {
if (JokeApi.logger.isLoggable(Level.FINE)) { if (JokeApi.logger.isLoggable(Level.FINE)) {
JokeApi.logger.fine(url) JokeApi.logger.fine(url)
} }
@ -63,11 +63,10 @@ internal fun fetchUrl(url: String, auth: String = ""): String {
val body = stream.bufferedReader().use { it.readText() } val body = stream.bufferedReader().use { it.readText() }
if (body.isBlank()) { if (body.isBlank()) {
throw httpError(connection.responseCode) throw httpError(connection.responseCode)
} } else if (JokeApi.logger.isLoggable(Level.FINE)) {
if (JokeApi.logger.isLoggable(Level.FINE)) {
JokeApi.logger.fine(body) JokeApi.logger.fine(body)
} }
return body return JokeResponse(connection.responseCode, body)
} finally { } finally {
connection.disconnect() connection.disconnect()
} }

View file

@ -44,7 +44,6 @@ class HttpErrorException @JvmOverloads constructor(
cause: Throwable? = null cause: Throwable? = null
) : IOException(message, cause) { ) : IOException(message, cause) {
companion object { companion object {
@Suppress("ConstPropertyName")
private const val serialVersionUID = 1L private const val serialVersionUID = 1L
} }
} }

View file

@ -0,0 +1,39 @@
/*
* JokeResponse.kt
*
* Copyright 2022-2024 Erik C. Thauvin (erik@thauvin.net)
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions are met:
*
* Redistributions of source code must retain the above copyright notice, this
* list of conditions and the following disclaimer.
*
* Redistributions in binary form must reproduce the above copyright notice,
* this list of conditions and the following disclaimer in the documentation
* and/or other materials provided with the distribution.
*
* Neither the name of this project nor the names of its contributors may be
* used to endorse or promote products derived from this software without
* specific prior written permission.
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
* AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
* DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE
* FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
* DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
* SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
* CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
* OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
* OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*/
package net.thauvin.erik.jokeapi.models
/**
* The Joke API response.
*
* @property code The HTTP status code.
* @property data The response text.
*/
data class JokeResponse(val code: Int, val data: String)

View file

@ -34,6 +34,7 @@ package net.thauvin.erik.jokeapi.models
/** /**
* The available [URL Parameters](https://jokeapi.dev/#url-parameters). * The available [URL Parameters](https://jokeapi.dev/#url-parameters).
*/ */
@Suppress("unused")
object Parameter { object Parameter {
const val AMOUNT = "amount" const val AMOUNT = "amount"
const val CONTAINS = "contains" const val CONTAINS = "contains"

View file

@ -32,6 +32,7 @@
package net.thauvin.erik.jokeapi package net.thauvin.erik.jokeapi
import assertk.assertThat import assertk.assertThat
import assertk.assertions.isEqualTo
import assertk.assertions.isGreaterThan import assertk.assertions.isGreaterThan
import assertk.assertions.startsWith import assertk.assertions.startsWith
import net.thauvin.erik.jokeapi.JokeApi.apiCall import net.thauvin.erik.jokeapi.JokeApi.apiCall
@ -51,8 +52,9 @@ internal class ApiCallTest {
fun `Get Flags`() { fun `Get Flags`() {
// See https://v2.jokeapi.dev/#flags-endpoint // See https://v2.jokeapi.dev/#flags-endpoint
val response = apiCall(endPoint = "flags") val response = apiCall(endPoint = "flags")
val json = JSONObject(response) val json = JSONObject(response.data)
assertAll("Validate JSON", assertAll(
"Validate JSON",
{ assertFalse(json.getBoolean("error"), "apiCall(flags).error") }, { assertFalse(json.getBoolean("error"), "apiCall(flags).error") },
{ assertThat(json.getJSONArray("flags").length(), "apiCall(flags).flags").isGreaterThan(0) }, { assertThat(json.getJSONArray("flags").length(), "apiCall(flags).flags").isGreaterThan(0) },
{ assertThat(json.getLong("timestamp"), "apiCall(flags).timestamp").isGreaterThan(0) }) { assertThat(json.getLong("timestamp"), "apiCall(flags).timestamp").isGreaterThan(0) })
@ -65,14 +67,16 @@ internal class ApiCallTest {
endPoint = "langcode", path = "french", endPoint = "langcode", path = "french",
params = mapOf(Parameter.FORMAT to Format.YAML.value) params = mapOf(Parameter.FORMAT to Format.YAML.value)
) )
assertContains(lang, "code: \"fr\"", false, "apiCall(langcode, french, yaml)") assertThat(lang.code).isEqualTo(200)
assertContains(lang.data, "code: \"fr\"", false, "apiCall(langcode, french, yaml)")
} }
@Test @Test
fun `Get Ping Response`() { fun `Get Ping Response`() {
// See https://v2.jokeapi.dev/#ping-endpoint // See https://v2.jokeapi.dev/#ping-endpoint
val ping = apiCall(endPoint = "ping", params = mapOf(Parameter.FORMAT to Format.TXT.value)) val ping = apiCall(endPoint = "ping", params = mapOf(Parameter.FORMAT to Format.TXT.value))
assertThat(ping, "apiCall(ping, txt)").startsWith("Pong!") assertThat(ping.code).isEqualTo(200)
assertThat(ping.data).startsWith("Pong!")
} }
@Test @Test
@ -82,6 +86,7 @@ internal class ApiCallTest {
endPoint = "languages", endPoint = "languages",
params = mapOf(Parameter.FORMAT to Format.XML.value, Parameter.LANG to Language.FR.value) params = mapOf(Parameter.FORMAT to Format.XML.value, Parameter.LANG to Language.FR.value)
) )
assertThat(lang).startsWith("<?xml version='1.0'?>") assertThat(lang.code).isEqualTo(200)
assertThat(lang.data).startsWith("<?xml version='1.0'?>")
} }
} }

View file

@ -34,6 +34,7 @@ package net.thauvin.erik.jokeapi
import assertk.all import assertk.all
import assertk.assertThat import assertk.assertThat
import assertk.assertions.doesNotContain import assertk.assertions.doesNotContain
import assertk.assertions.isEqualTo
import assertk.assertions.isNotEmpty import assertk.assertions.isNotEmpty
import assertk.assertions.startsWith import assertk.assertions.startsWith
import net.thauvin.erik.jokeapi.models.Format import net.thauvin.erik.jokeapi.models.Format
@ -47,7 +48,8 @@ internal class GetRawJokesTest {
@Test @Test
fun `Get Raw Joke with TXT`() { fun `Get Raw Joke with TXT`() {
val response = rawJokes(format = Format.TXT) val response = rawJokes(format = Format.TXT)
assertThat(response, "rawJoke(txt)").all { assertThat(response.code).isEqualTo(200)
assertThat(response.data, "rawJoke(data)").all {
isNotEmpty() isNotEmpty()
doesNotContain("Error") doesNotContain("Error")
} }
@ -56,24 +58,28 @@ internal class GetRawJokesTest {
@Test @Test
fun `Get Raw Joke with XML`() { fun `Get Raw Joke with XML`() {
val response = rawJokes(format = Format.XML) val response = rawJokes(format = Format.XML)
assertThat(response, "rawJoke(xml)").startsWith("<?xml version='1.0'?>\n<data>\n <error>false</error>") assertThat(response.code).isEqualTo(200)
assertThat(response.data, "rawJoke(xml)").startsWith("<?xml version='1.0'?>\n<data>\n <error>false</error>")
} }
@Test @Test
fun `Get Raw Joke with YAML`() { fun `Get Raw Joke with YAML`() {
val response = rawJokes(format = Format.YAML) val response = rawJokes(format = Format.YAML)
assertThat(response, "rawJoke(yaml)").startsWith("error: false") assertThat(response.code).isEqualTo(200)
assertThat(response.data, "rawJoke(yaml)").startsWith("error: false")
} }
@Test @Test
fun `Get Raw Jokes`() { fun `Get Raw Jokes`() {
val response = rawJokes(amount = 2) val response = rawJokes(amount = 2)
assertContains(response, "\"amount\": 2", false, "rawJoke(2)") assertThat(response.code).isEqualTo(200)
assertContains(response.data, "\"amount\": 2", false, "rawJoke(2)")
} }
@Test @Test
fun `Get Raw Invalid Jokes`() { fun `Get Raw Invalid Jokes`() {
val response = rawJokes(contains = "foo", safe = true, amount = 2, idRange = IdRange(160, 161)) val response = rawJokes(contains = "foo", safe = true, amount = 2, idRange = IdRange(160, 161))
assertContains(response, "\"error\": true", false, "getRawJokes(foo)") assertThat(response.code).isEqualTo(400)
assertContains(response.data, "\"error\": true", false, "getRawJokes(foo)")
} }
} }

View file

@ -102,8 +102,9 @@ internal class JokeConfigTest {
amount(2) amount(2)
safe(true) safe(true)
}.build() }.build()
val joke = getRawJokes(config) val jokes = getRawJokes(config)
assertContains(joke, "----------------------------------------------", false, "config.amount(2)") assertThat(jokes.code).isEqualTo(200)
assertContains(jokes.data, "----------------------------------------------", false, "config.amount(2)")
} }
@Test @Test

View file

@ -33,6 +33,7 @@ package net.thauvin.erik.jokeapi
import assertk.assertThat import assertk.assertThat
import assertk.assertions.contains import assertk.assertions.contains
import assertk.assertions.isEqualTo
import org.json.JSONException import org.json.JSONException
import org.json.JSONObject import org.json.JSONObject
import org.junit.jupiter.api.Test import org.junit.jupiter.api.Test
@ -54,7 +55,8 @@ internal class JokeUtilTest {
@Test @Test
fun `Validate Authentication Header`() { fun `Validate Authentication Header`() {
val token = "AUTH-TOKEN" val token = "AUTH-TOKEN"
val body = fetchUrl("https://postman-echo.com/get", token) val response = fetchUrl("https://postman-echo.com/get", token)
assertThat(body, "body").contains("\"authentication\": \"$token\"") assertThat(response.code).isEqualTo(200)
assertThat(response.data, "body").contains("\"authentication\": \"$token\"")
} }
} }