#42 Add tests for WorkgroupVoter
This commit is contained in:
@@ -6,7 +6,7 @@ import fr.postgresjson.entity.mutable.EntityDeletedAtImp
|
|||||||
import java.util.*
|
import java.util.*
|
||||||
|
|
||||||
class Workgroup(
|
class Workgroup(
|
||||||
id: UUID?,
|
id: UUID? = null,
|
||||||
name: String,
|
name: String,
|
||||||
description: String,
|
description: String,
|
||||||
logo: String? = null,
|
logo: String? = null,
|
||||||
@@ -28,7 +28,7 @@ class Workgroup(
|
|||||||
EntityUpdatedAt by EntityUpdatedAtImp()
|
EntityUpdatedAt by EntityUpdatedAtImp()
|
||||||
|
|
||||||
open class WorkgroupSimple<Z : CitizenRef>(
|
open class WorkgroupSimple<Z : CitizenRef>(
|
||||||
id: UUID?,
|
id: UUID? = null,
|
||||||
var name: String,
|
var name: String,
|
||||||
var description: String,
|
var description: String,
|
||||||
var logo: String? = null,
|
var logo: String? = null,
|
||||||
@@ -40,7 +40,7 @@ open class WorkgroupSimple<Z : CitizenRef>(
|
|||||||
EntityDeletedAt by EntityDeletedAtImp()
|
EntityDeletedAt by EntityDeletedAtImp()
|
||||||
|
|
||||||
open class WorkgroupRef(
|
open class WorkgroupRef(
|
||||||
id: UUID?
|
id: UUID? = null
|
||||||
) : UuidEntity(id ?: UUID.randomUUID()), WorkgroupI
|
) : UuidEntity(id ?: UUID.randomUUID()), WorkgroupI
|
||||||
|
|
||||||
interface WorkgroupWithAuthI<Z : CitizenWithUserI> : WorkgroupWithMembersI<Z>, EntityCreatedBy<Z>, EntityDeletedAt {
|
interface WorkgroupWithAuthI<Z : CitizenWithUserI> : WorkgroupWithMembersI<Z>, EntityCreatedBy<Z>, EntityDeletedAt {
|
||||||
|
|||||||
@@ -15,7 +15,7 @@ class WorkgroupVoter : Voter {
|
|||||||
|
|
||||||
override fun supports(action: ActionI, call: ApplicationCall, subject: Any?): Boolean {
|
override fun supports(action: ActionI, call: ApplicationCall, subject: Any?): Boolean {
|
||||||
return (action is Action)
|
return (action is Action)
|
||||||
.and(subject is List<*> || subject is WorkgroupI?)
|
.and(subject is WorkgroupI?)
|
||||||
}
|
}
|
||||||
|
|
||||||
override fun vote(action: ActionI, call: ApplicationCall, subject: Any?): Vote {
|
override fun vote(action: ActionI, call: ApplicationCall, subject: Any?): Vote {
|
||||||
@@ -31,14 +31,6 @@ class WorkgroupVoter : Voter {
|
|||||||
else if (subject.anonymous && user != null && subject.isMember(user)) Vote.GRANTED
|
else if (subject.anonymous && user != null && subject.isMember(user)) Vote.GRANTED
|
||||||
else Vote.DENIED
|
else Vote.DENIED
|
||||||
}
|
}
|
||||||
if (subject is List<*>) {
|
|
||||||
subject.forEach {
|
|
||||||
if (it !is WorkgroupWithAuthI<*> || it.isDeleted()) {
|
|
||||||
return Vote.DENIED
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return Vote.GRANTED
|
|
||||||
}
|
|
||||||
return Vote.DENIED
|
return Vote.DENIED
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -52,10 +44,9 @@ class WorkgroupVoter : Voter {
|
|||||||
}
|
}
|
||||||
|
|
||||||
return Vote.DENIED
|
return Vote.DENIED
|
||||||
}
|
} else if (subject !is WorkgroupWithAuthI<*> && (action == Action.DELETE || action == Action.UPDATE)) {
|
||||||
|
throw object :
|
||||||
if (action is Action) {
|
VoterException("Unable to define if your are granted, the subject must implement 'WorkgroupWithAuthI'") {}
|
||||||
return Vote.DENIED
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return Vote.ABSTAIN
|
return Vote.ABSTAIN
|
||||||
|
|||||||
@@ -0,0 +1,204 @@
|
|||||||
|
package fr.dcproject.security.voter
|
||||||
|
|
||||||
|
import fr.dcproject.entity.*
|
||||||
|
import io.ktor.application.ApplicationCall
|
||||||
|
import io.ktor.locations.KtorExperimentalLocationsAPI
|
||||||
|
import io.mockk.every
|
||||||
|
import io.mockk.mockk
|
||||||
|
import io.mockk.mockkStatic
|
||||||
|
import org.amshove.kluent.`should be`
|
||||||
|
import org.joda.time.DateTime
|
||||||
|
import org.junit.jupiter.api.Assertions
|
||||||
|
import org.junit.jupiter.api.Tag
|
||||||
|
import org.junit.jupiter.api.Test
|
||||||
|
import org.junit.jupiter.api.TestInstance
|
||||||
|
import fr.dcproject.entity.Workgroup as WorkgroupEntity
|
||||||
|
|
||||||
|
@KtorExperimentalLocationsAPI
|
||||||
|
@TestInstance(TestInstance.Lifecycle.PER_CLASS)
|
||||||
|
@Tag("voter")
|
||||||
|
internal class WorkgroupVoterTest {
|
||||||
|
private val tesla = CitizenBasic(
|
||||||
|
user = User(
|
||||||
|
username = "nicolas-tesla",
|
||||||
|
roles = listOf(UserI.Roles.ROLE_USER)
|
||||||
|
),
|
||||||
|
birthday = DateTime.now(),
|
||||||
|
email = "tesla@best.com",
|
||||||
|
name = CitizenI.Name("Nicolas", "Tesla"),
|
||||||
|
followAnonymous = false
|
||||||
|
)
|
||||||
|
|
||||||
|
private val einstein = CitizenBasic(
|
||||||
|
user = User(
|
||||||
|
username = "albert-einstein",
|
||||||
|
roles = listOf(UserI.Roles.ROLE_USER)
|
||||||
|
),
|
||||||
|
birthday = DateTime.now(),
|
||||||
|
email = "einstein@best.com",
|
||||||
|
name = CitizenI.Name("Albert", "Einstein"),
|
||||||
|
followAnonymous = true
|
||||||
|
)
|
||||||
|
|
||||||
|
private val article1 = Article(
|
||||||
|
content = "Hi",
|
||||||
|
createdBy = einstein,
|
||||||
|
description = "blablabla",
|
||||||
|
title = "Super article"
|
||||||
|
)
|
||||||
|
|
||||||
|
private val workgroupPublic = WorkgroupEntity(
|
||||||
|
createdBy = tesla,
|
||||||
|
description = "Super desc",
|
||||||
|
name = "super name",
|
||||||
|
owner = tesla,
|
||||||
|
anonymous = false
|
||||||
|
)
|
||||||
|
|
||||||
|
private val workgroupAnon = WorkgroupEntity(
|
||||||
|
createdBy = tesla,
|
||||||
|
description = "Super desc",
|
||||||
|
name = "super name",
|
||||||
|
owner = tesla,
|
||||||
|
anonymous = true
|
||||||
|
)
|
||||||
|
|
||||||
|
private val workgroupref = WorkgroupRef()
|
||||||
|
|
||||||
|
init {
|
||||||
|
mockkStatic("fr.dcproject.security.voter.VoterKt")
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
fun `support workgroup`() = WorkgroupVoter().run {
|
||||||
|
val p = object : ActionI {}
|
||||||
|
mockk<ApplicationCall> {
|
||||||
|
every { user } returns tesla.user
|
||||||
|
}.let {
|
||||||
|
supports(WorkgroupVoter.Action.VIEW, it, workgroupPublic) `should be` true
|
||||||
|
supports(WorkgroupVoter.Action.VIEW, it, article1) `should be` false
|
||||||
|
supports(p, it, workgroupPublic) `should be` false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
fun `can be view your workgroup`() = listOf(WorkgroupVoter()).run {
|
||||||
|
mockk<ApplicationCall> {
|
||||||
|
every { user } returns tesla.user
|
||||||
|
}.let {
|
||||||
|
can(WorkgroupVoter.Action.VIEW, it, workgroupPublic) `should be` true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
fun `can be view your workgroup if is not public`() = listOf(WorkgroupVoter()).run {
|
||||||
|
mockk<ApplicationCall> {
|
||||||
|
every { user } returns tesla.user
|
||||||
|
}.let {
|
||||||
|
can(WorkgroupVoter.Action.VIEW, it, workgroupAnon) `should be` true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
fun `can be view workgroup of other if is public`() = listOf(WorkgroupVoter()).run {
|
||||||
|
mockk<ApplicationCall> {
|
||||||
|
every { user } returns einstein.user
|
||||||
|
}.let {
|
||||||
|
can(WorkgroupVoter.Action.VIEW, it, workgroupPublic) `should be` true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
fun `can not be view workgroup of other if is not public`() = listOf(WorkgroupVoter()).run {
|
||||||
|
mockk<ApplicationCall> {
|
||||||
|
every { user } returns einstein.user
|
||||||
|
}.let {
|
||||||
|
can(WorkgroupVoter.Action.VIEW, it, workgroupAnon) `should be` false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
fun `can be not view the workgroup if is null`() = listOf(WorkgroupVoter()).run {
|
||||||
|
mockk<ApplicationCall> {
|
||||||
|
every { user } returns tesla.user
|
||||||
|
}.let {
|
||||||
|
can(WorkgroupVoter.Action.VIEW, it, null) `should be` false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
fun `can be view your workgroup list`() = listOf(WorkgroupVoter()).run {
|
||||||
|
mockk<ApplicationCall> {
|
||||||
|
every { user } returns tesla.user
|
||||||
|
}.let {
|
||||||
|
can(WorkgroupVoter.Action.VIEW, it, listOf(workgroupPublic)) `should be` true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
fun `can be create workgroup`() = listOf(WorkgroupVoter()).run {
|
||||||
|
mockk<ApplicationCall> {
|
||||||
|
every { user } returns tesla.user
|
||||||
|
}.let {
|
||||||
|
can(WorkgroupVoter.Action.CREATE, it, workgroupPublic) `should be` true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
fun `can not be create workgroup if not connected`() = listOf(WorkgroupVoter()).run {
|
||||||
|
mockk<ApplicationCall> {
|
||||||
|
every { user } returns null
|
||||||
|
}.let {
|
||||||
|
can(WorkgroupVoter.Action.CREATE, it, workgroupPublic) `should be` false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
fun `can be delete workgroup if owner`() = listOf(WorkgroupVoter()).run {
|
||||||
|
mockk<ApplicationCall> {
|
||||||
|
every { user } returns tesla.user
|
||||||
|
}.let {
|
||||||
|
can(WorkgroupVoter.Action.DELETE, it, workgroupPublic) `should be` true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
fun `can not be delete workgroup if not owner`() = listOf(WorkgroupVoter()).run {
|
||||||
|
mockk<ApplicationCall> {
|
||||||
|
every { user } returns einstein.user
|
||||||
|
}.let {
|
||||||
|
can(WorkgroupVoter.Action.DELETE, it, workgroupPublic) `should be` false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
fun `can be update workgroup if owner`() = listOf(WorkgroupVoter()).run {
|
||||||
|
mockk<ApplicationCall> {
|
||||||
|
every { user } returns tesla.user
|
||||||
|
}.let {
|
||||||
|
can(WorkgroupVoter.Action.UPDATE, it, workgroupPublic) `should be` true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
fun `can not be update workgroup if not owner`() = listOf(WorkgroupVoter()).run {
|
||||||
|
mockk<ApplicationCall> {
|
||||||
|
every { user } returns einstein.user
|
||||||
|
}.let {
|
||||||
|
can(WorkgroupVoter.Action.UPDATE, it, workgroupPublic) `should be` false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
fun `can not be update workgroup if workgroup has no user`() {
|
||||||
|
Assertions.assertThrows(VoterException::class.java) {
|
||||||
|
listOf(WorkgroupVoter()).run {
|
||||||
|
mockk<ApplicationCall> {
|
||||||
|
every { user } returns tesla.user
|
||||||
|
}.let {
|
||||||
|
can(WorkgroupVoter.Action.UPDATE, it, workgroupref)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
Reference in New Issue
Block a user