*/
class GetNotificationsAjaxPageTest : JsonPageTest("getNotifications.ajax", requiresLogin = false, needsFormPassword = false, pageSupplier = ::GetNotificationsAjaxPage) {
- private val notifications = listOf(
+ private val testNotifications = listOf(
createNotification("n1", 2000, "t1", 5000, true),
createNotification("n2", 1000, "t2", 6000, false),
createNotification("n3", 3000, "t3", 7000, true)
@Test
fun `notification hash is calculated correctly`() {
- notifications.forEach { addNotification(it) }
+ testNotifications.forEach { addNotification(it) }
assertThat(json.isSuccess, equalTo(true))
- assertThat(json["notificationHash"].asInt(), equalTo(listOf(1, 0, 2).map(notifications::get).hashCode()))
+ assertThat(json["notificationHash"].asInt(), equalTo(listOf(1, 0, 2).map(testNotifications::get).hashCode()))
}
@Test
@Test
fun `notifications are rendered correctly`() {
- notifications.forEach { addNotification(it) }
+ testNotifications.forEach { addNotification(it) }
assertThat(json.isSuccess, equalTo(true))
assertThat(json["notifications"].toList().map { node -> listOf("id", "text", "createdTime", "lastUpdatedTime", "dismissable").map { it to node.get(it).asText() }.toMap() }, containsInAnyOrder(
mapOf("id" to "n1", "createdTime" to "2000", "lastUpdatedTime" to "5000", "dismissable" to "true", "text" to "t1"),
whenever(templateContext).thenReturn(TemplateContext())
whenever(render(any(), any())).then { it.get<Writer>(1).write("t4") }
}
- notifications.forEach { addNotification(it) }
+ testNotifications.forEach { addNotification(it) }
addNotification(templateNotification)
assertThat(json.isSuccess, equalTo(true))
assertThat(json["notifications"].last()["text"].asText(), equalTo("t4"))
private val timeTextConverter = mock<TimeTextConverter>()
private val l10nFilter = mock<L10nFilter>()
override val page: JsonPage by lazy { GetTimesAjaxPage(webInterface, timeTextConverter, l10nFilter, getTimeZone("UTC")) }
- private val posts = listOf(createPost(1), createPost(2))
- private val replies = listOf(createReply(1), createReply(2))
+ private val testPosts = listOf(createPost(1), createPost(2))
+ private val testReplies = listOf(createReply(1), createReply(2))
private fun createPost(index: Int): Post {
return mock<Post>().apply {
@Test
fun `request with single post parameter responds with post times and empty reply times`() {
- addPost(posts[0])
+ addPost(testPosts[0])
addRequestParameter("posts", "post1")
assertThat(json.isSuccess, equalTo(true))
assertThat(json["postTimes"].fields().asSequence().map { it.key to it.value }.toList(), containsInAnyOrder<Pair<String, JsonNode>>(
@Test
fun `request with single reply parameter responds with reply times and empty post times`() {
- addReply(replies[0])
+ addReply(testReplies[0])
addRequestParameter("replies", "reply1")
assertThat(json.isSuccess, equalTo(true))
assertThat(json["postTimes"].toList(), emptyIterable())
@Test
fun `request with multiple post parameter responds with post times and empty reply times`() {
- addPost(posts[0])
- addPost(posts[1])
+ addPost(testPosts[0])
+ addPost(testPosts[1])
addRequestParameter("posts", "post1,post2,post3")
assertThat(json.isSuccess, equalTo(true))
assertThat(json["postTimes"].fields().asSequence().map { it.key to it.value }.toList(), containsInAnyOrder<Pair<String, JsonNode>>(
@Test
fun `request with multiple reply parameters responds with reply times and empty post times`() {
- addReply(replies[0])
- addReply(replies[1])
+ addReply(testReplies[0])
+ addReply(testReplies[1])
addRequestParameter("replies", "reply1,reply2,reply3")
assertThat(json.isSuccess, equalTo(true))
assertThat(json["postTimes"].toList(), emptyIterable())
package net.pterodactylus.sone.web.ajax
-import com.google.common.eventbus.EventBus
-import freenet.clients.http.ToadletContext
-import freenet.l10n.BaseL10n
-import freenet.support.SimpleReadOnlyArrayBucket
-import freenet.support.api.HTTPRequest
-import net.pterodactylus.sone.core.Core
-import net.pterodactylus.sone.core.ElementLoader
-import net.pterodactylus.sone.core.LinkedElement
-import net.pterodactylus.sone.core.Preferences
-import net.pterodactylus.sone.core.UpdateChecker
-import net.pterodactylus.sone.data.Album
-import net.pterodactylus.sone.data.Image
-import net.pterodactylus.sone.data.Post
-import net.pterodactylus.sone.data.PostReply
-import net.pterodactylus.sone.data.Profile
-import net.pterodactylus.sone.data.Sone
-import net.pterodactylus.sone.data.Sone.SoneStatus
-import net.pterodactylus.sone.data.Sone.SoneStatus.idle
-import net.pterodactylus.sone.data.SoneOptions.DefaultSoneOptions
-import net.pterodactylus.sone.test.deepMock
-import net.pterodactylus.sone.test.get
import net.pterodactylus.sone.test.mock
-import net.pterodactylus.sone.test.whenever
-import net.pterodactylus.sone.utils.asOptional
import net.pterodactylus.sone.web.WebInterface
-import net.pterodactylus.sone.web.page.FreenetRequest
-import net.pterodactylus.util.notify.Notification
-import net.pterodactylus.util.template.TemplateContextFactory
-import net.pterodactylus.util.web.Method.GET
import org.hamcrest.MatcherAssert.assertThat
import org.hamcrest.Matchers.equalTo
-import org.junit.Before
import org.junit.Test
-import org.mockito.ArgumentMatchers.any
-import org.mockito.ArgumentMatchers.anyBoolean
-import org.mockito.ArgumentMatchers.anyInt
-import org.mockito.ArgumentMatchers.anyString
-import org.mockito.ArgumentMatchers.eq
-import org.mockito.ArgumentMatchers.isNull
-import java.util.NoSuchElementException
-import javax.naming.SizeLimitExceededException
/**
* Base class for tests for any [JsonPage] implementations.
private val expectedPath: String,
private val requiresLogin: Boolean = true,
private val needsFormPassword: Boolean = true,
- pageSupplier: (WebInterface) -> JsonPage = { _ -> mock<JsonPage>() }) {
+ pageSupplier: (WebInterface) -> JsonPage = { mock() }): TestObjects() {
- protected val webInterface = mock<WebInterface>()
- protected val l10n = mock<BaseL10n>()
- protected val core = mock<Core>()
- protected val eventBus = mock<EventBus>()
- protected val preferences = Preferences(eventBus)
- protected val updateChecker = mock<UpdateChecker>()
- protected val elementLoader = mock<ElementLoader>()
protected open val page: JsonPage by lazy { pageSupplier(webInterface) }
- protected val json by lazy { page.createJsonObject(freenetRequest) }
-
- protected val toadletContext = mock<ToadletContext>()
- protected val freenetRequest = mock<FreenetRequest>()
- protected val httpRequest = mock<HTTPRequest>()
- protected val currentSone = deepMock<Sone>()
- protected val profile = Profile(currentSone)
-
- private val requestHeaders = mutableMapOf<String, String>()
- private val requestParameters = mutableMapOf<String, String>()
- private val requestParts = mutableMapOf<String, String>()
- private val localSones = mutableMapOf<String, Sone>()
- private val remoteSones = mutableMapOf<String, Sone>()
- private val posts = mutableMapOf<String, Post>()
- private val postLikes = mutableMapOf<Post, Set<Sone>>()
- private val newPosts = mutableMapOf<String, Post>()
- private val replies = mutableMapOf<String, PostReply>()
- private val replyLikes = mutableMapOf<PostReply, Set<Sone>>()
- private val newReplies = mutableMapOf<String, PostReply>()
- private val linkedElements = mutableMapOf<String, LinkedElement>()
- private val notifications = mutableMapOf<String, Notification>()
- private val albums = mutableMapOf<String, Album>()
- private val images = mutableMapOf<String, Image>()
- private val translations = mutableMapOf<String, String>()
-
- @Before
- fun setupWebInterface() {
- whenever(webInterface.templateContextFactory).thenReturn(TemplateContextFactory())
- whenever(webInterface.getCurrentSone(eq(toadletContext), anyBoolean())).thenReturn(currentSone)
- whenever(webInterface.getCurrentSoneCreatingSession(toadletContext)).thenReturn(currentSone)
- whenever(webInterface.getCurrentSoneWithoutCreatingSession(toadletContext)).thenReturn(currentSone)
- whenever(webInterface.core).thenReturn(core)
- whenever(webInterface.getNotifications(currentSone)).thenAnswer { notifications.values }
- whenever(webInterface.getNotification(anyString())).then { notifications[it[0]].asOptional() }
- whenever(webInterface.getNewPosts(currentSone)).thenAnswer { newPosts.values }
- whenever(webInterface.getNewReplies(currentSone)).thenAnswer { newReplies.values }
- whenever(webInterface.l10n).thenReturn(l10n)
- }
-
- @Before
- fun setupTranslations() {
- whenever(l10n.getString(anyString())).then { translations[it[0]] }
- }
-
- @Before
- fun setupCore() {
- whenever(core.preferences).thenReturn(preferences)
- whenever(core.updateChecker).thenReturn(updateChecker)
- whenever(core.getSone(anyString())).thenAnswer { (localSones + remoteSones)[it.getArgument(0)].asOptional() }
- whenever(core.getLocalSone(anyString())).thenAnswer { localSones[it[0]] }
- whenever(core.getPost(anyString())).thenAnswer { (posts + newPosts)[it[0]].asOptional() }
- whenever(core.getLikes(any<Post>())).then { postLikes[it[0]] ?: emptySet<Sone>() }
- whenever(core.getLikes(any<PostReply>())).then { replyLikes[it[0]] ?: emptySet<Sone>() }
- whenever(core.getPostReply(anyString())).then { replies[it[0]].asOptional() }
- whenever(core.getAlbum(anyString())).then { albums[it[0]] }
- whenever(core.getImage(anyString())).then { images[it[0]] }
- whenever(core.getImage(anyString(), anyBoolean())).then { images[it[0]] }
- }
-
- @Before
- fun setupElementLoader() {
- whenever(elementLoader.loadElement(anyString())).thenAnswer {
- linkedElements[it.getArgument(0)] ?: LinkedElement(it.getArgument(0), loading = true)
- }
- }
-
- @Before
- fun setupCurrentSone() {
- whenever(currentSone.options).thenReturn(DefaultSoneOptions())
- currentSone.mock("soneId", "Sone_Id", true, 1000, idle)
- }
-
- @Before
- fun setupFreenetRequest() {
- whenever(freenetRequest.toadletContext).thenReturn(toadletContext)
- whenever(freenetRequest.method).thenReturn(GET)
- whenever(freenetRequest.httpRequest).thenReturn(httpRequest)
- }
-
- @Before
- fun setupHttpRequest() {
- whenever(httpRequest.method).thenReturn("GET")
- whenever(httpRequest.getHeader(anyString())).thenAnswer { requestHeaders[it.get<String>(0).toLowerCase()] }
- whenever(httpRequest.getParam(anyString())).thenAnswer { requestParameters[it.getArgument(0)] ?: "" }
- whenever(httpRequest.getParam(anyString(), anyString())).thenAnswer { requestParameters[it.getArgument(0)] ?: it.getArgument(1) }
- whenever(httpRequest.getParam(anyString(), isNull())).thenAnswer { requestParameters[it.getArgument(0)] }
- whenever(httpRequest.getPart(anyString())).thenAnswer { requestParts[it.getArgument(0)]?.let { SimpleReadOnlyArrayBucket(it.toByteArray()) } }
- whenever(httpRequest.getPartAsBytesFailsafe(anyString(), anyInt())).thenAnswer { requestParts[it.getArgument(0)]?.toByteArray()?.copyOf(it.getArgument(1)) ?: ByteArray(0) }
- whenever(httpRequest.getPartAsBytesThrowing(anyString(), anyInt())).thenAnswer { invocation -> requestParts[invocation.getArgument(0)]?.let { it.toByteArray().let { if (it.size > invocation.getArgument<Int>(1)) throw SizeLimitExceededException() else it } } ?: throw NoSuchElementException() }
- whenever(httpRequest.getPartAsStringFailsafe(anyString(), anyInt())).thenAnswer { requestParts[it.getArgument(0)]?.substring(0, it.getArgument(1)) ?: "" }
- whenever(httpRequest.getPartAsStringThrowing(anyString(), anyInt())).thenAnswer { invocation -> requestParts[invocation.getArgument(0)]?.let { if (it.length > invocation.getArgument<Int>(1)) throw SizeLimitExceededException() else it } ?: throw NoSuchElementException() }
- whenever(httpRequest.getIntPart(anyString(), anyInt())).thenAnswer { invocation -> requestParts[invocation.getArgument(0)]?.toIntOrNull() ?: invocation.getArgument(1) }
- whenever(httpRequest.isPartSet(anyString())).thenAnswer { it.getArgument(0) in requestParts }
- }
-
- @Before
- fun setupProfile() {
- whenever(currentSone.profile).thenReturn(profile)
+ protected val json by lazy {
+ page.createJsonObject(freenetRequest)
}
protected val JsonReturnObject.error get() = if (this is JsonErrorReturnObject) this.error else null
- protected fun Sone.mock(id: String, name: String, local: Boolean = false, time: Long, status: SoneStatus = idle) = apply {
- whenever(this.id).thenReturn(id)
- whenever(this.name).thenReturn(name)
- whenever(isLocal).thenReturn(local)
- whenever(this.time).thenReturn(time)
- whenever(this.status).thenReturn(status)
- }
-
- protected fun unsetCurrentSone() {
- whenever(webInterface.getCurrentSone(eq(toadletContext), anyBoolean())).thenReturn(null)
- whenever(webInterface.getCurrentSoneWithoutCreatingSession(toadletContext)).thenReturn(null)
- whenever(webInterface.getCurrentSoneCreatingSession(toadletContext)).thenReturn(null)
- }
-
- protected fun addRequestHeader(key: String, value: String) {
- requestHeaders += key.toLowerCase() to value
- }
-
- protected fun addRequestParameter(key: String, value: String) {
- requestParameters += key to value
- }
-
- protected fun addRequestPart(key: String, value: String) {
- requestParts += key to value
- }
-
- protected fun addNotification(notification: Notification, notificationId: String? = null) {
- notifications[notificationId ?: notification.id] = notification
- }
-
- protected fun addSone(sone: Sone, soneId: String? = null) {
- remoteSones += (soneId ?: sone.id) to sone
- }
-
- protected fun addLocalSone(sone: Sone, id: String? = null) {
- localSones[id ?: sone.id] = sone
- }
-
- protected fun addPost(post: Post, id: String? = null) {
- posts[id ?: post.id] = post
- }
-
- protected fun addLikes(post: Post, vararg sones: Sone) {
- postLikes[post] = setOf(*sones)
- }
-
- protected fun addLikes(reply: PostReply, vararg sones: Sone) {
- replyLikes[reply] = setOf(*sones)
- }
-
- protected fun addNewPost(id: String, soneId: String, time: Long, recipientId: String? = null) =
- mock<Post>().apply {
- whenever(this.id).thenReturn(id)
- val sone = mock<Sone>().apply { whenever(this.id).thenReturn(soneId) }
- whenever(this.sone).thenReturn(sone)
- whenever(this.time).thenReturn(time)
- whenever(this.recipientId).thenReturn(recipientId.asOptional())
- }.also { newPosts[id] = it }
-
- protected fun addReply(reply: PostReply, id: String? = null) {
- replies[id ?: reply.id] = reply
- }
-
- protected fun addNewReply(id: String, soneId: String, postId: String, postSoneId: String) {
- newReplies[id] = mock<PostReply>().apply {
- whenever(this.id).thenReturn(id)
- val sone = mock<Sone>().apply { whenever(this.id).thenReturn(soneId) }
- whenever(this.sone).thenReturn(sone)
- val postSone = mock<Sone>().apply { whenever(this.id).thenReturn(postSoneId) }
- val post = mock<Post>().apply {
- whenever(this.sone).thenReturn(postSone)
- }
- whenever(this.post).thenReturn(post.asOptional())
- whenever(this.postId).thenReturn(postId)
- }
- }
-
- protected fun addLinkedElement(link: String, loading: Boolean, failed: Boolean) {
- linkedElements[link] = LinkedElement(link, failed, loading)
- }
-
- protected fun addAlbum(album: Album, albumId: String? = null) {
- albums[albumId ?: album.id] = album
- }
-
- protected fun addImage(image: Image, imageId: String? = null) {
- images[imageId ?: image.id] = image
- }
-
- protected fun addTranslation(key: String, value: String) {
- translations[key] = value
- }
-
@Test
fun `page returns correct path`() {
assertThat(page.path, equalTo(expectedPath))
--- /dev/null
+package net.pterodactylus.sone.web.ajax
+
+import com.google.common.eventbus.EventBus
+import freenet.clients.http.ToadletContext
+import freenet.l10n.BaseL10n
+import freenet.support.SimpleReadOnlyArrayBucket
+import freenet.support.api.HTTPRequest
+import net.pterodactylus.sone.core.Core
+import net.pterodactylus.sone.core.ElementLoader
+import net.pterodactylus.sone.core.LinkedElement
+import net.pterodactylus.sone.core.Preferences
+import net.pterodactylus.sone.core.UpdateChecker
+import net.pterodactylus.sone.data.Album
+import net.pterodactylus.sone.data.Image
+import net.pterodactylus.sone.data.Post
+import net.pterodactylus.sone.data.PostReply
+import net.pterodactylus.sone.data.Profile
+import net.pterodactylus.sone.data.Sone
+import net.pterodactylus.sone.data.Sone.SoneStatus
+import net.pterodactylus.sone.data.Sone.SoneStatus.idle
+import net.pterodactylus.sone.data.SoneOptions.DefaultSoneOptions
+import net.pterodactylus.sone.test.deepMock
+import net.pterodactylus.sone.test.get
+import net.pterodactylus.sone.test.mock
+import net.pterodactylus.sone.test.whenever
+import net.pterodactylus.sone.utils.asOptional
+import net.pterodactylus.sone.web.WebInterface
+import net.pterodactylus.sone.web.page.FreenetRequest
+import net.pterodactylus.util.notify.Notification
+import net.pterodactylus.util.template.TemplateContextFactory
+import net.pterodactylus.util.web.Method.GET
+import net.pterodactylus.util.web.Method.POST
+import org.mockito.ArgumentMatchers
+import java.util.NoSuchElementException
+import javax.naming.SizeLimitExceededException
+
+/**
+ * Base class for tests that supplies commonly used objects.
+ */
+open class TestObjects {
+
+ val webInterface = mock<WebInterface>()
+ var formPassword = "form-password"
+ val l10n = mock<BaseL10n>()
+ val core = mock<Core>()
+ val eventBus = mock<EventBus>()
+ val preferences = Preferences(eventBus)
+ val updateChecker = mock<UpdateChecker>()
+ val elementLoader = mock<ElementLoader>()
+
+ val toadletContext = mock<ToadletContext>()
+ val freenetRequest = mock<FreenetRequest>()
+ val httpRequest = mock<HTTPRequest>()
+ val currentSone = deepMock<Sone>()
+ val profile = Profile(currentSone)
+
+ val requestHeaders = mutableMapOf<String, String>()
+ val requestParameters = mutableMapOf<String, String>()
+ val requestParts = mutableMapOf<String, String>()
+ val localSones = mutableMapOf<String, Sone>()
+ val remoteSones = mutableMapOf<String, Sone>()
+ val posts = mutableMapOf<String, Post>()
+ val postLikes = mutableMapOf<Post, Set<Sone>>()
+ val newPosts = mutableMapOf<String, Post>()
+ val replies = mutableMapOf<String, PostReply>()
+ val replyLikes = mutableMapOf<PostReply, Set<Sone>>()
+ val newReplies = mutableMapOf<String, PostReply>()
+ val linkedElements = mutableMapOf<String, LinkedElement>()
+ val notifications = mutableMapOf<String, Notification>()
+ val albums = mutableMapOf<String, Album>()
+ val images = mutableMapOf<String, Image>()
+ val translations = mutableMapOf<String, String>()
+
+ init {
+ whenever(webInterface.templateContextFactory).thenReturn(TemplateContextFactory())
+ whenever(webInterface.getCurrentSone(ArgumentMatchers.eq(toadletContext), ArgumentMatchers.anyBoolean())).thenReturn(currentSone)
+ whenever(webInterface.getCurrentSoneCreatingSession(toadletContext)).thenReturn(currentSone)
+ whenever(webInterface.getCurrentSoneWithoutCreatingSession(toadletContext)).thenReturn(currentSone)
+ whenever(webInterface.core).thenReturn(core)
+ whenever(webInterface.formPassword).then { formPassword }
+ whenever(webInterface.getNotifications(currentSone)).thenAnswer { notifications.values }
+ whenever(webInterface.getNotification(ArgumentMatchers.anyString())).then { notifications[it[0]].asOptional() }
+ whenever(webInterface.getNewPosts(currentSone)).thenAnswer { newPosts.values }
+ whenever(webInterface.getNewReplies(currentSone)).thenAnswer { newReplies.values }
+ whenever(webInterface.l10n).thenReturn(l10n)
+
+ whenever(l10n.getString(ArgumentMatchers.anyString())).then { translations[it[0]] }
+
+ whenever(core.preferences).thenReturn(preferences)
+ whenever(core.updateChecker).thenReturn(updateChecker)
+ whenever(core.getSone(ArgumentMatchers.anyString())).thenAnswer { (localSones + remoteSones)[it.getArgument(0)].asOptional() }
+ whenever(core.getLocalSone(ArgumentMatchers.anyString())).thenAnswer { localSones[it[0]] }
+ whenever(core.getPost(ArgumentMatchers.anyString())).thenAnswer { (posts + newPosts)[it[0]].asOptional() }
+ whenever(core.getLikes(ArgumentMatchers.any<Post>())).then { postLikes[it[0]] ?: emptySet<Sone>() }
+ whenever(core.getLikes(ArgumentMatchers.any<PostReply>())).then { replyLikes[it[0]] ?: emptySet<Sone>() }
+ whenever(core.getPostReply(ArgumentMatchers.anyString())).then { replies[it[0]].asOptional() }
+ whenever(core.getAlbum(ArgumentMatchers.anyString())).then { albums[it[0]] }
+ whenever(core.getImage(ArgumentMatchers.anyString())).then { images[it[0]] }
+ whenever(core.getImage(ArgumentMatchers.anyString(), ArgumentMatchers.anyBoolean())).then { images[it[0]] }
+
+ whenever(elementLoader.loadElement(ArgumentMatchers.anyString())).thenAnswer {
+ linkedElements[it.getArgument(0)] ?: LinkedElement(it.getArgument(0), loading = true)
+ }
+
+ whenever(currentSone.options).thenReturn(DefaultSoneOptions())
+ currentSone.mock("soneId", "Sone_Id", true, 1000, idle)
+
+ whenever(freenetRequest.toadletContext).thenReturn(toadletContext)
+ whenever(freenetRequest.method).thenReturn(GET)
+ whenever(freenetRequest.httpRequest).thenReturn(httpRequest)
+
+ whenever(httpRequest.method).thenReturn("GET")
+ whenever(httpRequest.getHeader(ArgumentMatchers.anyString())).thenAnswer { requestHeaders[it.get<String>(0).toLowerCase()] }
+ whenever(httpRequest.getParam(ArgumentMatchers.anyString())).thenAnswer { requestParameters[it.getArgument(0)] ?: "" }
+ whenever(httpRequest.getParam(ArgumentMatchers.anyString(), ArgumentMatchers.anyString())).thenAnswer { requestParameters[it.getArgument(0)] ?: it.getArgument(1) }
+ whenever(httpRequest.getParam(ArgumentMatchers.anyString(), ArgumentMatchers.isNull())).thenAnswer { requestParameters[it.getArgument(0)] }
+ whenever(httpRequest.getPart(ArgumentMatchers.anyString())).thenAnswer { requestParts[it.getArgument(0)]?.let { SimpleReadOnlyArrayBucket(it.toByteArray()) } }
+ whenever(httpRequest.getPartAsBytesFailsafe(ArgumentMatchers.anyString(), ArgumentMatchers.anyInt())).thenAnswer { requestParts[it.getArgument(0)]?.toByteArray()?.copyOf(it.getArgument(1)) ?: ByteArray(0) }
+ whenever(httpRequest.getPartAsBytesThrowing(ArgumentMatchers.anyString(), ArgumentMatchers.anyInt())).thenAnswer { invocation -> requestParts[invocation.getArgument(0)]?.let { it.toByteArray().let { if (it.size > invocation.getArgument<Int>(1)) throw SizeLimitExceededException() else it } } ?: throw NoSuchElementException() }
+ whenever(httpRequest.getPartAsStringFailsafe(ArgumentMatchers.anyString(), ArgumentMatchers.anyInt())).thenAnswer { requestParts[it.getArgument(0)]?.substring(0, it.getArgument(1)) ?: "" }
+ whenever(httpRequest.getPartAsStringThrowing(ArgumentMatchers.anyString(), ArgumentMatchers.anyInt())).thenAnswer { invocation -> requestParts[invocation.getArgument(0)]?.let { if (it.length > invocation.getArgument<Int>(1)) throw SizeLimitExceededException() else it } ?: throw NoSuchElementException() }
+ whenever(httpRequest.getIntPart(ArgumentMatchers.anyString(), ArgumentMatchers.anyInt())).thenAnswer { invocation -> requestParts[invocation.getArgument(0)]?.toIntOrNull() ?: invocation.getArgument(1) }
+ whenever(httpRequest.isPartSet(ArgumentMatchers.anyString())).thenAnswer { it.getArgument(0) in requestParts }
+
+ whenever(currentSone.profile).thenReturn(profile)
+ }
+
+ protected fun Sone.mock(id: String, name: String, local: Boolean = false, time: Long, status: SoneStatus = idle) = apply {
+ whenever(this.id).thenReturn(id)
+ whenever(this.name).thenReturn(name)
+ whenever(isLocal).thenReturn(local)
+ whenever(this.time).thenReturn(time)
+ whenever(this.status).thenReturn(status)
+ }
+
+ protected fun unsetCurrentSone() {
+ whenever(webInterface.getCurrentSone(ArgumentMatchers.eq(toadletContext), ArgumentMatchers.anyBoolean())).thenReturn(null)
+ whenever(webInterface.getCurrentSoneWithoutCreatingSession(toadletContext)).thenReturn(null)
+ whenever(webInterface.getCurrentSoneCreatingSession(toadletContext)).thenReturn(null)
+ }
+
+ protected fun postRequest() {
+ whenever(freenetRequest.method).thenReturn(POST)
+ whenever(httpRequest.method).thenReturn("POST")
+ }
+
+ protected fun addRequestHeader(key: String, value: String) {
+ requestHeaders += key.toLowerCase() to value
+ }
+
+ protected fun addRequestParameter(key: String, value: String) {
+ requestParameters += key to value
+ }
+
+ protected fun addRequestPart(key: String, value: String) {
+ requestParts += key to value
+ }
+
+ protected fun addNotification(notification: Notification, notificationId: String? = null) {
+ notifications[notificationId ?: notification.id] = notification
+ }
+
+ protected fun addSone(sone: Sone, soneId: String? = null) {
+ remoteSones += (soneId ?: sone.id) to sone
+ }
+
+ protected fun addLocalSone(sone: Sone, id: String? = null) {
+ localSones[id ?: sone.id] = sone
+ }
+
+ protected fun addPost(post: Post, id: String? = null) {
+ posts[id ?: post.id] = post
+ }
+
+ protected fun addLikes(post: Post, vararg sones: Sone) {
+ postLikes[post] = setOf(*sones)
+ }
+
+ protected fun addLikes(reply: PostReply, vararg sones: Sone) {
+ replyLikes[reply] = setOf(*sones)
+ }
+
+ protected fun addNewPost(id: String, soneId: String, time: Long, recipientId: String? = null) =
+ mock<Post>().apply {
+ whenever(this.id).thenReturn(id)
+ val sone = mock<Sone>().apply { whenever(this.id).thenReturn(soneId) }
+ whenever(this.sone).thenReturn(sone)
+ whenever(this.time).thenReturn(time)
+ whenever(this.recipientId).thenReturn(recipientId.asOptional())
+ }.also { newPosts[id] = it }
+
+ protected fun addReply(reply: PostReply, id: String? = null) {
+ replies[id ?: reply.id] = reply
+ }
+
+ protected fun addNewReply(id: String, soneId: String, postId: String, postSoneId: String) {
+ newReplies[id] = mock<PostReply>().apply {
+ whenever(this.id).thenReturn(id)
+ val sone = mock<Sone>().apply { whenever(this.id).thenReturn(soneId) }
+ whenever(this.sone).thenReturn(sone)
+ val postSone = mock<Sone>().apply { whenever(this.id).thenReturn(postSoneId) }
+ val post = mock<Post>().apply {
+ whenever(this.sone).thenReturn(postSone)
+ }
+ whenever(this.post).thenReturn(post.asOptional())
+ whenever(this.postId).thenReturn(postId)
+ }
+ }
+
+ protected fun addLinkedElement(link: String, loading: Boolean, failed: Boolean) {
+ linkedElements[link] = LinkedElement(link, failed, loading)
+ }
+
+ protected fun addAlbum(album: Album, albumId: String? = null) {
+ albums[albumId ?: album.id] = album
+ }
+
+ protected fun addImage(image: Image, imageId: String? = null) {
+ images[imageId ?: image.id] = image
+ }
+
+ protected fun addTranslation(key: String, value: String) {
+ translations[key] = value
+ }
+
+}