@@ -4,10 +4,14 @@ import assertions.assertSuccess
4
4
import database.entity.Area
5
5
import io.ktor.client.statement.bodyAsChannel
6
6
import io.ktor.client.statement.readRawBytes
7
+ import io.ktor.http.HttpHeaders
8
+ import io.ktor.http.etag
7
9
import io.ktor.http.isSuccess
10
+ import io.ktor.http.lastModified
8
11
import io.ktor.utils.io.readBuffer
9
12
import java.awt.image.BufferedImage
10
13
import java.io.File
14
+ import java.security.MessageDigest
11
15
import javax.imageio.ImageIO
12
16
import kotlin.test.Test
13
17
import kotlin.test.assertEquals
@@ -17,27 +21,32 @@ import kotlinx.io.copyTo
17
21
import server.DataProvider
18
22
import server.base.ApplicationTestBase
19
23
import server.base.StubApplicationTestBuilder
24
+ import server.response.FileSource
25
+ import server.response.FileUUID
26
+ import storage.FileType
27
+ import storage.HashUtils
28
+ import storage.MessageDigestAlgorithm
20
29
import storage.Storage
21
30
22
31
class TestFileDownloading : ApplicationTestBase () {
23
32
private suspend inline fun StubApplicationTestBuilder.provideImageFile (
24
33
imageFile : String = "/images/alcoi.jpg",
25
- block : (imageUUID: String ) -> Unit
34
+ block : (imageUUID: String , imageFile: File ) -> Unit
26
35
) {
27
36
val areaId = DataProvider .provideSampleArea(this , imageFile = imageFile)
28
37
29
- var image : String ? = null
38
+ var imageFile : File ? = null
30
39
31
40
get(" /area/$areaId " ).apply {
32
41
assertSuccess<Area > { data ->
33
42
assertNotNull(data)
34
- image = data.image.toRelativeString( Storage . ImagesDir )
43
+ imageFile = data.image
35
44
}
36
45
}
37
46
38
- assertNotNull(image )
47
+ assertNotNull(imageFile )
39
48
40
- block(image )
49
+ block(imageFile.toRelativeString( Storage . ImagesDir ), imageFile )
41
50
}
42
51
43
52
private fun downloadResized (
@@ -46,7 +55,7 @@ class TestFileDownloading : ApplicationTestBase() {
46
55
fetch : (BufferedImage ) -> Int ,
47
56
imageFile : String = "/images/alcoi.jpg"
48
57
) = test {
49
- provideImageFile(imageFile) { image ->
58
+ provideImageFile(imageFile) { image, _ ->
50
59
val tempFile = File .createTempFile(" eaic" , null )
51
60
val response = get(" /download/$image ?$argument =$value " )
52
61
assertTrue(
@@ -69,14 +78,29 @@ class TestFileDownloading : ApplicationTestBase() {
69
78
70
79
@Test
71
80
fun `test downloading files` () = test {
72
- provideImageFile { image ->
81
+ provideImageFile { image, imageFile ->
73
82
get(" /download/$image " ).apply {
74
- headers[" Content-Type" ].let { contentType ->
75
- assertEquals(
76
- " image/jpeg" ,
77
- contentType,
78
- " Content-Type header is not JPEG. Got: $contentType "
83
+ headers[HttpHeaders .ContentType ].let { contentType ->
84
+ assertEquals(" image/jpeg" , contentType, " Content-Type header is not JPEG. Got: $contentType " )
85
+ }
86
+ assertEquals(image, headers[HttpHeaders .FileUUID ], " File UUID header is not correct" )
87
+ assertEquals(
88
+ FileType .IMAGE .headerValue,
89
+ headers[HttpHeaders .FileSource ],
90
+ " File source header is not correct."
91
+ )
92
+ etag().let {
93
+ val hash = HashUtils .getCheckSumFromFile(
94
+ MessageDigest .getInstance(MessageDigestAlgorithm .SHA_256 ),
95
+ imageFile
79
96
)
97
+ assertEquals(" \" $hash \" " , it, " ETag header is not correct" )
98
+ }
99
+ lastModified()?.time.let {
100
+ // Value may be truncated to seconds and converted again to ms, so we need to truncate it
101
+ val fileLastModified = imageFile.lastModified() / 1000 * 1000
102
+ val headerLastModified = it?.div(1000 )?.times(1000 )
103
+ assertEquals(fileLastModified, headerLastModified, " Last-Modified header is not correct" )
80
104
}
81
105
readRawBytes()
82
106
}
0 commit comments