Add security for follow

This commit is contained in:
2019-08-31 00:14:05 +02:00
parent 52dfaaf814
commit cb91c50e58
8 changed files with 75 additions and 21 deletions

View File

@@ -109,7 +109,8 @@ fun Application.module(env: Env = PROD) {
ConstitutionVoter(), ConstitutionVoter(),
CitizenVoter(), CitizenVoter(),
CommentVoter(), CommentVoter(),
VoteVoter() VoteVoter(),
FollowVoter()
) )
} }

View File

@@ -9,8 +9,8 @@ class Citizen(
var name: Name?, var name: Name?,
var birthday: DateTime?, var birthday: DateTime?,
var userId: UUID? = null, var userId: UUID? = null,
var voteanonymous: Boolean? = null, var voteAnonymous: Boolean = true,
var followanonymous: Boolean? = null, var followAnonymous: Boolean = true,
var user: User? var user: User?
) : UuidEntity(id), ) : UuidEntity(id),
EntityCreatedAt by EntityCreatedAtImp(), EntityCreatedAt by EntityCreatedAtImp(),

View File

@@ -14,7 +14,7 @@ open class Vote <T: UuidEntity>(override var requester: Requester): RepositoryI<
fun vote(vote: VoteEntity<T>) { fun vote(vote: VoteEntity<T>) {
val reference = vote.target::class.simpleName!!.toLowerCase() val reference = vote.target::class.simpleName!!.toLowerCase()
val author = vote.createdBy ?: error("vote must be contain an author") val author = vote.createdBy ?: error("vote must be contain an author")
val anonymous = author.voteanonymous val anonymous = author.voteAnonymous
requester requester
.getFunction("vote") .getFunction("vote")
.sendQuery( .sendQuery(

View File

@@ -2,6 +2,8 @@ package fr.dcproject.routes
import fr.dcproject.citizen import fr.dcproject.citizen
import fr.dcproject.entity.Citizen import fr.dcproject.entity.Citizen
import fr.dcproject.security.voter.FollowVoter.Action.*
import fr.dcproject.security.voter.assertCan
import io.ktor.application.call import io.ktor.application.call
import io.ktor.http.HttpStatusCode import io.ktor.http.HttpStatusCode
import io.ktor.locations.* import io.ktor.locations.*
@@ -21,24 +23,21 @@ object FollowArticlePaths {
fun Route.followArticle(repo: FollowArticleRepository) { fun Route.followArticle(repo: FollowArticleRepository) {
post<FollowArticlePaths.ArticleFollowRequest> { post<FollowArticlePaths.ArticleFollowRequest> {
val follow = FollowEntity(target = it.article, createdBy = this.citizen) val follow = FollowEntity(target = it.article, createdBy = this.citizen)
// TODO create voter assertCan(CREATE, follow)
// assertCan(FollowVoter.Action.CREATE, follow)
repo.follow(follow) repo.follow(follow)
call.respond(HttpStatusCode.Created) call.respond(HttpStatusCode.Created)
} }
delete<FollowArticlePaths.ArticleFollowRequest> { delete<FollowArticlePaths.ArticleFollowRequest> {
val follow = FollowEntity(target = it.article, createdBy = this.citizen) val follow = FollowEntity(target = it.article, createdBy = this.citizen)
// TODO create voter assertCan(DELETE, follow)
// assertCan(FollowVoter.Action.DELETE, follow)
repo.unfollow(follow) repo.unfollow(follow)
call.respond(HttpStatusCode.NoContent) call.respond(HttpStatusCode.NoContent)
} }
get<FollowArticlePaths.CitizenFollowArticleRequest> { get<FollowArticlePaths.CitizenFollowArticleRequest> {
val follows = repo.findByCitizen(it.citizen) val follows = repo.findByCitizen(it.citizen)
// TODO add security assertCan(VIEW, follows.result)
// assertCan(FollowVoter.Action.VIEW, follows)
call.respond(follows) call.respond(follows)
} }
} }

View File

@@ -2,6 +2,8 @@ package fr.dcproject.routes
import fr.dcproject.citizen import fr.dcproject.citizen
import fr.dcproject.entity.Citizen import fr.dcproject.entity.Citizen
import fr.dcproject.security.voter.FollowVoter.Action.*
import fr.dcproject.security.voter.assertCan
import io.ktor.application.call import io.ktor.application.call
import io.ktor.http.HttpStatusCode import io.ktor.http.HttpStatusCode
import io.ktor.locations.* import io.ktor.locations.*
@@ -21,24 +23,21 @@ object FollowConstitutionPaths {
fun Route.followConstitution(repo: FollowConstitutionRepository) { fun Route.followConstitution(repo: FollowConstitutionRepository) {
post<FollowConstitutionPaths.ConstitutionFollowRequest> { post<FollowConstitutionPaths.ConstitutionFollowRequest> {
val follow = FollowEntity(target = it.constitution, createdBy = this.citizen) val follow = FollowEntity(target = it.constitution, createdBy = this.citizen)
// TODO create voter assertCan(CREATE, follow)
// assertCan(FollowVoter.Action.CREATE, follow)
repo.follow(follow) repo.follow(follow)
call.respond(HttpStatusCode.Created) call.respond(HttpStatusCode.Created)
} }
delete<FollowConstitutionPaths.ConstitutionFollowRequest> { delete<FollowConstitutionPaths.ConstitutionFollowRequest> {
val follow = FollowEntity(target = it.constitution, createdBy = this.citizen) val follow = FollowEntity(target = it.constitution, createdBy = this.citizen)
// TODO create voter assertCan(DELETE, follow)
// assertCan(FollowVoter.Action.DELETE, follow)
repo.unfollow(follow) repo.unfollow(follow)
call.respond(HttpStatusCode.NoContent) call.respond(HttpStatusCode.NoContent)
} }
get<FollowConstitutionPaths.CitizenFollowConstitutionRequest> { get<FollowConstitutionPaths.CitizenFollowConstitutionRequest> {
val follows = repo.findByCitizen(it.citizen) val follows = repo.findByCitizen(it.citizen)
// TODO create voter assertCan(VIEW, follows.result)
// assertCan(FollowVoter.Action.VIEW, follows)
call.respond(follows) call.respond(follows)
} }
} }

View File

@@ -0,0 +1,53 @@
package fr.dcproject.security.voter
import io.ktor.application.ApplicationCall
import fr.dcproject.entity.Follow as FollowEntity
import fr.dcproject.entity.User as UserEntity
class FollowVoter: Voter {
enum class Action: ActionI {
CREATE,
DELETE,
VIEW
}
override fun supports(action: ActionI, call: ApplicationCall, subject: Any?): Boolean {
return (action is Action) &&
(subject is List<*> || subject is FollowEntity<*>?)
}
override fun vote(action: ActionI, call: ApplicationCall, subject: Any?): Vote {
val user = call.user
if (action == Action.CREATE) {
return if (user != null) Vote.GRANTED
else Vote.DENIED
}
if (action == Action.DELETE) {
return if (user != null) Vote.GRANTED
else Vote.DENIED
}
if (action == Action.VIEW) {
if (subject is FollowEntity<*>) {
return voteView(user, subject)
}
if (subject is List<*>) {
subject.forEach {
if (it !is FollowEntity<*> || voteView(user, it) == Vote.DENIED) {
return Vote.DENIED
}
}
return Vote.GRANTED
}
return Vote.DENIED
}
return Vote.ABSTAIN
}
private fun voteView(user: UserEntity?, subject: FollowEntity<*>): Vote {
return if ((user != null && subject.createdBy?.user?.id == user.id) || subject.createdBy?.followAnonymous == false) Vote.GRANTED
else Vote.DENIED
}
}

View File

@@ -78,7 +78,9 @@ class KtorServerAuthSteps: En, KoinTest {
id = UUID.fromString(id), id = UUID.fromString(id),
name = Citizen.Name(firstName, lastName), name = Citizen.Name(firstName, lastName),
birthday = DateTime.now(), birthday = DateTime.now(),
user = user user = user,
followAnonymous = false,
voteAnonymous = false
) )
try { try {

View File

@@ -7,8 +7,8 @@ Feature: follow Article and Constitution
Then the response status code should be 201 Then the response status code should be 201
Scenario: The route for get follows of articles must response a 200 and return objects Scenario: The route for get follows of articles must response a 200 and return objects
Given I have citizen John Doe with id "64b7b379-2298-43ec-b428-ba134930cabd" Given I have citizen John Smith with id "e3c0b08c-11be-418e-95e0-8596b4402feb"
When I send a GET request to "/citizens/64b7b379-2298-43ec-b428-ba134930cabd/follows/articles" When I send a GET request to "/citizens/e3c0b08c-11be-418e-95e0-8596b4402feb/follows/articles"
Then the response status code should be 200 Then the response status code should be 200
And the response should contain object: And the response should contain object:
| current_page | 1 | | current_page | 1 |
@@ -26,8 +26,8 @@ Feature: follow Article and Constitution
Then the response status code should be 201 Then the response status code should be 201
Scenario: The route for get follows of constitutions must response a 200 and return objects Scenario: The route for get follows of constitutions must response a 200 and return objects
Given I have citizen John Doe with id "64b7b379-2298-43ec-b428-ba134930cabd" Given I have citizen John Smith with id "e3c0b08c-11be-418e-95e0-8596b4402feb"
When I send a GET request to "/citizens/64b7b379-2298-43ec-b428-ba134930cabd/follows/constitutions" When I send a GET request to "/citizens/e3c0b08c-11be-418e-95e0-8596b4402feb/follows/constitutions"
Then the response status code should be 200 Then the response status code should be 200
And the response should contain object: And the response should contain object:
| current_page | 1 | | current_page | 1 |