✨ Use new template renderer
[Sone.git] / src / main / kotlin / net / pterodactylus / sone / web / pages / UploadImagePage.kt
1 package net.pterodactylus.sone.web.pages
2
3 import freenet.support.api.Bucket
4 import net.pterodactylus.sone.data.Sone
5 import net.pterodactylus.sone.main.*
6 import net.pterodactylus.sone.text.TextFilter
7 import net.pterodactylus.sone.utils.emptyToNull
8 import net.pterodactylus.sone.utils.headers
9 import net.pterodactylus.sone.utils.isPOST
10 import net.pterodactylus.sone.utils.parameters
11 import net.pterodactylus.sone.utils.use
12 import net.pterodactylus.sone.web.WebInterface
13 import net.pterodactylus.sone.web.page.*
14 import net.pterodactylus.util.template.Template
15 import net.pterodactylus.util.template.TemplateContext
16 import java.awt.image.BufferedImage
17 import java.io.ByteArrayInputStream
18 import java.io.ByteArrayOutputStream
19 import javax.imageio.ImageIO
20 import javax.inject.Inject
21
22 /**
23  * Page implementation that lets the user upload an image.
24  */
25 class UploadImagePage @Inject constructor(template: Template, webInterface: WebInterface, loaders: Loaders, templateRenderer: TemplateRenderer):
26                 LoggedInPage("uploadImage.html", template, "Page.UploadImage.Title", webInterface, loaders, templateRenderer) {
27
28         override fun handleRequest(soneRequest: SoneRequest, currentSone: Sone, templateContext: TemplateContext) {
29                 if (soneRequest.isPOST) {
30                         val parentAlbum = soneRequest.parameters["parent"]!!.let(soneRequest.core::getAlbum) ?: throw RedirectException("noPermission.html")
31                         if (parentAlbum.sone != currentSone) {
32                                 throw RedirectException("noPermission.html")
33                         }
34                         val title = soneRequest.parameters["title", 200].emptyToNull ?: throw RedirectException("emptyImageTitle.html")
35
36                         val uploadedFile = soneRequest.httpRequest.getUploadedFile("image")
37                         val bytes = uploadedFile.data.use { it.toByteArray() }
38                         val bufferedImage = bytes.toImage()
39                         if (bufferedImage == null) {
40                                 templateContext["messages"] = soneRequest.l10n.getString("Page.UploadImage.Error.InvalidImage")
41                                 return
42                         }
43
44                         val temporaryImage = soneRequest.core.createTemporaryImage(bytes.mimeType, bytes)
45                         soneRequest.core.createImage(currentSone, parentAlbum, temporaryImage).modify().apply {
46                                 setWidth(bufferedImage.width)
47                                 setHeight(bufferedImage.height)
48                                 setTitle(title)
49                                 setDescription(TextFilter.filter(soneRequest.headers["Host"], soneRequest.parameters["description", 4000]))
50                         }.update()
51                         throw RedirectException("imageBrowser.html?album=${parentAlbum.id}")
52                 }
53         }
54
55         private fun Bucket.toByteArray(): ByteArray = ByteArrayOutputStream(size().toInt()).use { outputStream ->
56                 inputStream.copyTo(outputStream)
57                 outputStream.toByteArray()
58         }
59
60         private fun ByteArray.toImage(): BufferedImage? = ByteArrayInputStream(this).use {
61                 ImageIO.read(it)
62         }
63
64         private val ByteArray.mimeType get() = ByteArrayInputStream(this).use {
65                 ImageIO.createImageInputStream(it).use {
66                         ImageIO.getImageReaders(it).asSequence()
67                                         .firstOrNull()?.originatingProvider?.mimeTypes?.firstOrNull()
68                                         ?: UNKNOWN_MIME_TYPE
69                 }
70         }
71
72 }
73
74 private const val UNKNOWN_MIME_TYPE = "application/octet-stream"