Added JokeResponse data class
This commit is contained in:
parent
5ac8f910b5
commit
b95e1541fb
11 changed files with 88 additions and 29 deletions
5
.idea/misc.xml
generated
5
.idea/misc.xml
generated
|
@ -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" />
|
||||||
|
|
|
@ -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
|
||||||
|
|
|
@ -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
|
||||||
|
|
|
@ -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()
|
||||||
}
|
}
|
||||||
|
|
|
@ -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
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -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)
|
|
@ -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"
|
||||||
|
|
|
@ -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'?>")
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -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)")
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -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
|
||||||
|
|
|
@ -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\"")
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue