Test openapi schema of route /workgroup/*/members

This commit is contained in:
2021-03-24 19:01:12 +01:00
parent 66fa1ba840
commit 03f68213e8
6 changed files with 156 additions and 53 deletions

View File

@@ -7,6 +7,7 @@ import fr.dcproject.component.citizen.database.CitizenRef
import fr.dcproject.component.workgroup.WorkgroupAccessControl import fr.dcproject.component.workgroup.WorkgroupAccessControl
import fr.dcproject.component.workgroup.database.WorkgroupRepository import fr.dcproject.component.workgroup.database.WorkgroupRepository
import fr.dcproject.component.workgroup.database.WorkgroupWithMembersI import fr.dcproject.component.workgroup.database.WorkgroupWithMembersI
import fr.dcproject.component.workgroup.routes.toOutput
import io.ktor.application.ApplicationCall import io.ktor.application.ApplicationCall
import io.ktor.application.call import io.ktor.application.call
import io.ktor.http.HttpStatusCode import io.ktor.http.HttpStatusCode
@@ -22,8 +23,8 @@ import java.util.UUID
object AddMemberToWorkgroup { object AddMemberToWorkgroup {
@Location("/workgroups/{workgroupId}/members") @Location("/workgroups/{workgroupId}/members")
class WorkgroupsMembersRequest(val workgroupId: UUID) : KoinComponent { class WorkgroupsMembersRequest(val workgroupId: UUID) : KoinComponent {
class Input : MutableList<Input.Item> by mutableListOf() { class Input : MutableList<Input.Member> by mutableListOf() {
class Item(val citizen: CitizenRef, roles: List<String> = emptyList()) { class Member(val citizen: CitizenRef, roles: List<String> = emptyList()) {
val roles: List<WorkgroupWithMembersI.Member.Role> = roles.map { val roles: List<WorkgroupWithMembersI.Member.Role> = roles.map {
WorkgroupWithMembersI.Member.Role.valueOf(it) WorkgroupWithMembersI.Member.Role.valueOf(it)
} }
@@ -48,7 +49,10 @@ object AddMemberToWorkgroup {
ac.assert { canAddMembers(workgroup, citizenOrNull) } ac.assert { canAddMembers(workgroup, citizenOrNull) }
repo.addMembers(workgroup, members) repo.addMembers(workgroup, members)
}.let { members -> }.let { members ->
call.respond(HttpStatusCode.Created, members) call.respond(
HttpStatusCode.Created,
members.toOutput()
)
} }
} ?: call.respond(HttpStatusCode.NotFound) } ?: call.respond(HttpStatusCode.NotFound)
} }

View File

@@ -7,6 +7,7 @@ import fr.dcproject.component.citizen.database.CitizenRef
import fr.dcproject.component.workgroup.WorkgroupAccessControl import fr.dcproject.component.workgroup.WorkgroupAccessControl
import fr.dcproject.component.workgroup.database.WorkgroupRepository import fr.dcproject.component.workgroup.database.WorkgroupRepository
import fr.dcproject.component.workgroup.database.WorkgroupWithMembersI import fr.dcproject.component.workgroup.database.WorkgroupWithMembersI
import fr.dcproject.component.workgroup.routes.toOutput
import io.ktor.application.ApplicationCall import io.ktor.application.ApplicationCall
import io.ktor.application.call import io.ktor.application.call
import io.ktor.http.HttpStatusCode import io.ktor.http.HttpStatusCode
@@ -17,36 +18,33 @@ import io.ktor.response.respond
import io.ktor.routing.Route import io.ktor.routing.Route
import org.koin.core.KoinComponent import org.koin.core.KoinComponent
import java.util.UUID import java.util.UUID
import fr.dcproject.component.workgroup.routes.members.DeleteMembersOfWorkgroup.WorkgroupsMembersRequest.Input as Input
@KtorExperimentalLocationsAPI @KtorExperimentalLocationsAPI
object DeleteMembersOfWorkgroup { object DeleteMembersOfWorkgroup {
@Location("/workgroups/{workgroupId}/members") @Location("/workgroups/{workgroupId}/members")
class WorkgroupsMembersRequest(val workgroupId: UUID) : KoinComponent { class WorkgroupsMembersRequest(val workgroupId: UUID) : KoinComponent {
class Input : MutableList<Input.Item> by mutableListOf() { class Input : MutableList<Input.Member> by mutableListOf() {
class Item(val citizen: CitizenRef, roles: List<String> = emptyList()) { class Member(val citizen: CitizenRef)
val roles: List<WorkgroupWithMembersI.Member.Role> = roles.map {
WorkgroupWithMembersI.Member.Role.valueOf(it)
}
}
} }
} }
private suspend fun ApplicationCall.getMembersFromRequest(): List<WorkgroupWithMembersI.Member<CitizenRef>> = receiveOrBadRequest<WorkgroupsMembersRequest.Input>().map { private suspend fun ApplicationCall.getMembersFromRequest(): List<WorkgroupWithMembersI.Member<CitizenRef>> =
WorkgroupWithMembersI.Member( receiveOrBadRequest<Input>().map { WorkgroupWithMembersI.Member(it.citizen) }
citizen = it.citizen,
roles = it.roles
)
}
fun Route.deleteMemberOfWorkgroup(repo: WorkgroupRepository, ac: WorkgroupAccessControl) { fun Route.deleteMemberOfWorkgroup(repo: WorkgroupRepository, ac: WorkgroupAccessControl) {
/* Delete members of workgroup */ /* Delete members of workgroup */
delete<WorkgroupsMembersRequest> { delete<WorkgroupsMembersRequest> {
repo.findById(it.workgroupId)?.let { workgroup -> repo.findById(it.workgroupId)?.let { workgroup ->
call.getMembersFromRequest().let { members -> call.getMembersFromRequest()
ac.assert { canView(workgroup, citizenOrNull) } .let { members ->
ac.assert { canRemoveMembers(workgroup, citizenOrNull) }
repo.removeMembers(workgroup, members) repo.removeMembers(workgroup, members)
}.let { members -> }.let { members ->
call.respond(HttpStatusCode.OK, members) call.respond(
HttpStatusCode.OK,
members.toOutput()
)
} }
} ?: call.respond(HttpStatusCode.NotFound) } ?: call.respond(HttpStatusCode.NotFound)
} }

View File

@@ -7,6 +7,7 @@ import fr.dcproject.component.citizen.database.CitizenRef
import fr.dcproject.component.workgroup.WorkgroupAccessControl import fr.dcproject.component.workgroup.WorkgroupAccessControl
import fr.dcproject.component.workgroup.database.WorkgroupRepository import fr.dcproject.component.workgroup.database.WorkgroupRepository
import fr.dcproject.component.workgroup.database.WorkgroupWithMembersI import fr.dcproject.component.workgroup.database.WorkgroupWithMembersI
import fr.dcproject.component.workgroup.routes.toOutput
import io.ktor.application.ApplicationCall import io.ktor.application.ApplicationCall
import io.ktor.application.call import io.ktor.application.call
import io.ktor.http.HttpStatusCode import io.ktor.http.HttpStatusCode
@@ -46,7 +47,10 @@ object UpdateMemberOfWorkgroup {
ac.assert { canUpdateMembers(workgroup, citizenOrNull) } ac.assert { canUpdateMembers(workgroup, citizenOrNull) }
repo.updateMembers(workgroup, members) repo.updateMembers(workgroup, members)
}.let { members -> }.let { members ->
call.respond(HttpStatusCode.OK, members) call.respond(
HttpStatusCode.OK,
members.toOutput()
)
} }
} ?: call.respond(HttpStatusCode.NotFound) } ?: call.respond(HttpStatusCode.NotFound)
} }

View File

@@ -1,11 +1,13 @@
package fr.dcproject.component.workgroup.routes package fr.dcproject.component.workgroup.routes
import fr.dcproject.common.response.toOutput import fr.dcproject.common.response.toOutput
import fr.dcproject.component.citizen.database.CitizenCreator
import fr.dcproject.component.workgroup.database.WorkgroupForView import fr.dcproject.component.workgroup.database.WorkgroupForView
import fr.dcproject.component.workgroup.database.WorkgroupWithMembersI
import org.joda.time.DateTime import org.joda.time.DateTime
import java.util.UUID import java.util.UUID
fun WorkgroupForView<*>.toOutput(): Any = this.let { w -> fun WorkgroupForView<CitizenCreator>.toOutput(): Any = this.let { w ->
object { object {
val id: UUID = w.id val id: UUID = w.id
val name: String = w.name val name: String = w.name
@@ -14,20 +16,7 @@ fun WorkgroupForView<*>.toOutput(): Any = this.let { w ->
val anonymous: Boolean = w.anonymous val anonymous: Boolean = w.anonymous
val createdAt: DateTime = w.createdAt val createdAt: DateTime = w.createdAt
val createdBy: Any = w.createdBy.toOutput() val createdBy: Any = w.createdBy.toOutput()
val members: Any = w.members.map { m -> val members: Any = w.members.toOutput()
object {
val citizen: Any = object {
val id: UUID = m.citizen.id
val name: Any = m.citizen.name.let { n ->
object {
val firstName: String = n.firstName
val lastName: String = n.lastName
}
}
}
val roles: List<String> = m.roles.map { it.toString() }
}
}
} }
} }
@@ -40,3 +29,12 @@ fun WorkgroupForView<*>.toOutputListing(): Any = this.let { w ->
val createdAt: DateTime = w.createdAt val createdAt: DateTime = w.createdAt
} }
} }
fun List<WorkgroupWithMembersI.Member<CitizenCreator>>.toOutput(): Any {
return this.map { m ->
object {
val citizen: Any = m.citizen.toOutput()
val roles: List<String> = m.roles.map { it.toString() }
}
}
}

View File

@@ -1250,6 +1250,78 @@ paths:
description: Workgroup deleted description: Workgroup deleted
404: 404:
description: Workshop not exist or are already deleted description: Workshop not exist or are already deleted
/workgroups/{workgroup}/members:
parameters:
- $ref: '#/components/parameters/workgroup'
post:
summary: Add members to the workgroup
security:
- JWTAuth: []
tags:
- workgroup
requestBody:
content:
application/json:
schema:
$ref: '#/components/schemas/MembersRequest'
responses:
201:
description: the list of members
content:
application/json:
schema:
$ref: '#/components/schemas/MembersResponse'
put:
summary: Updates ALL members.
description: ⚠ PLEASE NOTE ⚠ This request removes all members who are not in request!
security:
- JWTAuth: []
tags:
- workgroup
requestBody:
content:
application/json:
schema:
$ref: '#/components/schemas/MembersRequest'
responses:
200:
description: the list of members
content:
application/json:
schema:
$ref: '#/components/schemas/MembersResponse'
400:
$ref: '#/components/responses/400'
delete:
summary: Delete members of workgroup
security:
- JWTAuth: []
tags:
- workgroup
requestBody:
content:
application/json:
schema:
description: members of workgroup
type: array
items:
type: object
properties:
citizen:
required:
- id
properties:
id:
$ref: '#/components/schemas/UUID'
responses:
200:
description: the list of members
content:
application/json:
schema:
$ref: '#/components/schemas/MembersResponse'
400:
$ref: '#/components/responses/400'
components: components:
parameters: parameters:
@@ -2058,7 +2130,6 @@ components:
logo: logo:
type: string type: string
nullable: true nullable: true
Workgroup: Workgroup:
description: Workgroup description: Workgroup
type: object type: object
@@ -2088,13 +2159,17 @@ components:
createdBy: createdBy:
$ref: '#/components/schemas/CitizenCreator' $ref: '#/components/schemas/CitizenCreator'
members: members:
$ref: '#/components/schemas/MembersResponse'
MembersRequest:
description: members of workgroup
type: array type: array
items: items:
type: object type: object
properties: properties:
citizen: citizen:
type: object required:
items: - id
properties: properties:
id: id:
$ref: '#/components/schemas/UUID' $ref: '#/components/schemas/UUID'
@@ -2102,6 +2177,31 @@ components:
type: array type: array
items: items:
type: string type: string
enum:
- MASTER
- MANAGER
- EDITOR
- REPORTER
example: MASTER
MembersResponse:
description: members of workgroup
type: array
items:
description: Member of workgroup
type: object
properties:
citizen:
$ref: '#/components/schemas/CitizenCreator'
roles:
type: array
items:
type: string
enum:
- MASTER
- MANAGER
- EDITOR
- REPORTER
example: MASTER
securitySchemes: securitySchemes:
JWTAuth: JWTAuth:

View File

@@ -204,8 +204,7 @@ class `Workgroup routes` : BaseTest() {
""" """
[ [
{ {
"citizen": {"id":"87909ba3-2069-431c-9924-219fd8411cf2"}, "citizen": {"id":"87909ba3-2069-431c-9924-219fd8411cf2"}
"roles": ["MASTER"]
} }
] ]
""" """