forked from hexagontk/hexagon
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathBooksTest.kt
141 lines (117 loc) · 4.58 KB
/
BooksTest.kt
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
package com.hexagonkt.http.server.examples
import com.hexagonkt.helpers.require
import com.hexagonkt.http.client.Client
import com.hexagonkt.http.client.Response
import com.hexagonkt.http.client.ahc.AhcAdapter
import com.hexagonkt.http.server.Server
import com.hexagonkt.http.server.ServerPort
import org.junit.jupiter.api.AfterAll
import org.junit.jupiter.api.BeforeAll
import org.junit.jupiter.api.Test
import org.junit.jupiter.api.TestInstance
import org.junit.jupiter.api.TestInstance.Lifecycle.PER_CLASS
@TestInstance(PER_CLASS)
abstract class BooksTest(adapter: ServerPort) {
// books
data class Book(val author: String, val title: String)
private val books: MutableMap<Int, Book> = linkedMapOf(
100 to Book("Miguel de Cervantes", "Don Quixote"),
101 to Book("William Shakespeare", "Hamlet"),
102 to Book("Homer", "The Odyssey")
)
val server: Server = Server(adapter) {
post("/books") {
// Require fails if parameter does not exists
val author = queryParameters.require("author")
val title = queryParameters.require("title")
val id = (books.keys.max() ?: 0) + 1
books += id to Book(author, title)
send(201, id)
}
get("/books/{id}") {
val bookId = pathParameters.require("id").toInt()
val book = books[bookId]
if (book != null)
// ok() is a shortcut to send(200)
ok("Title: ${book.title}, Author: ${book.author}")
else
send(404, "Book not found")
}
put("/books/{id}") {
val bookId = pathParameters.require("id").toInt()
val book = books[bookId]
if (book != null) {
books += bookId to book.copy(
author = queryParameters["author"] ?: book.author,
title = queryParameters["title"] ?: book.title
)
ok("Book with id '$bookId' updated")
}
else {
send(404, "Book not found")
}
}
delete("/books/{id}") {
val bookId = pathParameters.require("id").toInt()
val book = books[bookId]
books -= bookId
if (book != null)
ok("Book with id '$bookId' deleted")
else
send(404, "Book not found")
}
// Matches path's requests with *any* HTTP method as a fallback (return 404 instead 405)
any("/books/{id}") { send(405) }
get("/books") { ok(books.keys.joinToString(" ", transform = Int::toString)) }
}
// books
private val client: Client by lazy {
Client(AhcAdapter(), "http://localhost:${server.runtimePort}")
}
@BeforeAll fun initialize() {
server.start()
}
@AfterAll fun shutdown() {
server.stop()
}
@Test fun `Create book returns 201 and new book ID`() {
val result = client.post("/books?author=Vladimir%20Nabokov&title=Lolita")
assert(Integer.valueOf(result.body) > 0)
assert(201 == result.status)
}
@Test fun `List books contains all books IDs`() {
val result = client.get("/books")
assertResponseContains(result, "100", "101")
}
@Test fun `Get book returns all book's fields`() {
val result = client.get("/books/101")
assertResponseContains(result, "William Shakespeare", "Hamlet")
}
@Test fun `Update book overrides existing book data`() {
val resultPut = client.put("/books/100?title=Don%20Quixote")
assertResponseContains(resultPut, "100", "updated")
val resultGet = client.get("/books/100")
assertResponseContains(resultGet, "Miguel de Cervantes", "Don Quixote")
}
@Test fun `Delete book returns the deleted record ID`() {
val result = client.delete("/books/102")
assertResponseContains(result, "102", "deleted")
}
@Test fun `Book not found returns a 404`() {
val result = client.get("/books/9999")
assertResponseContains(result, 404, "not found")
}
@Test fun `Invalid method returns 405`() {
val result = client.options("/books/9999")
assert(405 == result.status)
}
private fun assertResponseContains(response: Response?, status: Int, vararg content: String) {
assert(response?.status == status)
content.forEach {
assert(response?.body?.contains(it) ?: false)
}
}
private fun assertResponseContains(response: Response?, vararg content: String) {
assertResponseContains(response, 200, *content)
}
}