Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

refactor: Move GroupStatus update from responder to services (DEV-3292) #3195

Merged
merged 8 commits into from
Apr 19, 2024
Merged
Show file tree
Hide file tree
Changes from 7 commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Original file line number Diff line number Diff line change
Expand Up @@ -7,8 +7,6 @@ package org.knora.webapi.responders.admin

import zio._

import java.util.UUID

import dsp.errors._
import org.knora.webapi._
import org.knora.webapi.messages.admin.responder.usersmessages._
Expand Down Expand Up @@ -154,7 +152,7 @@ class GroupsResponderADMSpec extends CoreSpec {
group.selfjoin should equal(false)
}

"return 'NotFound' if a not-existing group IRI is submitted during update" in {
"return 'NotFoundException' if a not-existing group IRI is submitted during update" in {
val groupIri = "http://rdfh.ch/groups/0000/notexisting"
val exit = UnsafeZioRun.run(
groupRestService(
Expand Down Expand Up @@ -204,7 +202,7 @@ class GroupsResponderADMSpec extends CoreSpec {
)
}

"return 'BadRequest' if nothing would be changed during the update" in {
"return 'BadRequestException' if nothing would be changed during the update" in {
val exit = UnsafeZioRun.run(
groupRestService(
_.putGroup(
Expand Down Expand Up @@ -245,11 +243,11 @@ class GroupsResponderADMSpec extends CoreSpec {
group.members.size shouldBe 2

val statusChangeResponse = UnsafeZioRun.runOrThrow(
ZIO.serviceWithZIO[GroupsResponderADM](
_.updateGroupStatus(
groupRestService(
_.putGroupStatus(
GroupIri.unsafeFrom(imagesReviewerGroup.id),
GroupStatusUpdateRequest(GroupStatus.inactive),
UUID.randomUUID(),
rootUser,
),
),
)
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -28,7 +28,6 @@ import org.knora.webapi.messages.util.KnoraSystemInstances
import org.knora.webapi.responders.IriLocker
import org.knora.webapi.responders.IriService
import org.knora.webapi.responders.Responder
import org.knora.webapi.slice.admin.api.GroupsRequests.GroupStatusUpdateRequest
import org.knora.webapi.slice.admin.api.GroupsRequests.GroupUpdateRequest
import org.knora.webapi.slice.admin.domain.model
import org.knora.webapi.slice.admin.domain.model.Group
Expand Down Expand Up @@ -181,29 +180,6 @@ final case class GroupsResponderADM(
def groupMembersGetRequest(iri: GroupIri, user: User): Task[GroupMembersGetResponseADM] =
groupMembersGetADM(iri.value, user).map(GroupMembersGetResponseADM.apply)

/**
* Change group's status.
*
* @param groupIri the IRI of the group we want to change.
* @param request the change request.
* @param apiRequestID the unique request ID.
* @return a [[GroupGetResponseADM]].
*/
def updateGroupStatus(
groupIri: GroupIri,
request: GroupStatusUpdateRequest,
apiRequestID: UUID,
): Task[GroupGetResponseADM] = {
val task = for {
// update group status
updated <- updateGroupHelper(groupIri, GroupUpdateRequest(None, None, Some(request.status), None))

// remove all members from group if status is false
result <- removeGroupMembersIfNecessary(updated.group)
} yield result
IriLocker.runWithIriLock(apiRequestID, groupIri.value, task)
}

def deleteGroup(
iri: GroupIri,
apiRequestID: UUID,
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -75,9 +75,11 @@ final case class GroupRestService(

def putGroupStatus(iri: GroupIri, request: GroupStatusUpdateRequest, user: User): Task[GroupGetResponseADM] =
for {
_ <- auth.ensureSystemAdminOrProjectAdminOfGroup(user, iri)
uuid <- Random.nextUUID
internal <- responder.updateGroupStatus(iri, request, uuid)
_ <- auth.ensureSystemAdminOrProjectAdminOfGroup(user, iri)
groupToUpdate <- groupService
.findById(iri)
.someOrFail(NotFoundException(s"Group <${iri.value}> not found."))
internal <- groupService.updateGroupStatus(groupToUpdate, request.status).map(GroupGetResponseADM.apply)
external <- format.toExternalADM(internal)
} yield external

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -10,7 +10,7 @@ import zio.ZLayer
import org.knora.webapi.config.AppConfig
import org.knora.webapi.responders.IriService
import org.knora.webapi.slice.admin.domain.service.GroupService
import org.knora.webapi.slice.admin.domain.service.KnoraGroupService.KnoraGroupService
import org.knora.webapi.slice.admin.domain.service.KnoraGroupService
import org.knora.webapi.slice.admin.domain.service.KnoraProjectService
import org.knora.webapi.slice.admin.domain.service.PasswordService
import org.knora.webapi.slice.admin.domain.service.ProjectService
Expand All @@ -24,10 +24,23 @@ import org.knora.webapi.store.triplestore.api.TriplestoreService
object AdminDomainModule {

type Dependencies =
AppConfig & AdminRepoModule.Provided & CacheService & IriService & OntologyRepo & PredicateObjectMapper & TriplestoreService
AppConfig &
AdminRepoModule.Provided &
CacheService &
IriService &
OntologyRepo &
PredicateObjectMapper &
TriplestoreService

type Provided =
AdministrativePermissionService & GroupService & KnoraGroupService & KnoraProjectService & KnoraUserService & MaintenanceService & PasswordService & ProjectService
AdministrativePermissionService &
GroupService &
KnoraGroupService &
KnoraProjectService &
KnoraUserService &
MaintenanceService &
PasswordService &
ProjectService

val layer = ZLayer.makeSome[Dependencies, Provided](
AdministrativePermissionService.layer,
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -33,7 +33,7 @@ final case class GroupService(

private def toGroups(knoraGroups: Chunk[KnoraGroup]): Task[Chunk[Group]] = ZIO.foreach(knoraGroups)(toGroup)

private def toGroup(knoraGroup: KnoraGroup): Task[Group] =
private[service] def toGroup(knoraGroup: KnoraGroup): Task[Group] =
for {
project <- knoraGroup.belongsToProject.map(projectService.findById).getOrElse(ZIO.none)
} yield Group(
Expand All @@ -45,7 +45,7 @@ final case class GroupService(
selfjoin = knoraGroup.hasSelfJoinEnabled.value,
)

private def toKnoraGroup(group: Group): KnoraGroup =
private[service] def toKnoraGroup(group: Group): KnoraGroup =
KnoraGroup(
id = GroupIri.unsafeFrom(group.id),
groupName = GroupName.unsafeFrom(group.name),
Expand All @@ -60,6 +60,9 @@ final case class GroupService(

def updateGroup(groupToUpdate: Group, request: GroupUpdateRequest): Task[Group] =
knoraGroupService.updateGroup(toKnoraGroup(groupToUpdate), request).flatMap(toGroup)

def updateGroupStatus(groupToUpdate: Group, status: GroupStatus): Task[Group] =
knoraGroupService.updateGroupStatus(toKnoraGroup(groupToUpdate), status).flatMap(toGroup)
}

object GroupService {
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -10,17 +10,20 @@ import zio.Task
import zio.ZIO
import zio.ZLayer

import dsp.errors.BadRequestException
mpro7 marked this conversation as resolved.
Show resolved Hide resolved
import dsp.errors.DuplicateValueException
import org.knora.webapi.responders.IriService
import org.knora.webapi.slice.admin.api.GroupsRequests.GroupCreateRequest
import org.knora.webapi.slice.admin.api.GroupsRequests.GroupUpdateRequest
import org.knora.webapi.slice.admin.domain.model.GroupIri
import org.knora.webapi.slice.admin.domain.model.GroupName
import org.knora.webapi.slice.admin.domain.model.GroupStatus
import org.knora.webapi.slice.admin.domain.model.KnoraGroup
import org.knora.webapi.slice.admin.domain.model.KnoraProject

case class KnoraGroupService(
knoraGroupRepo: KnoraGroupRepo,
knoraUserService: KnoraUserService,
iriService: IriService,
) {

Expand Down Expand Up @@ -50,10 +53,7 @@ case class KnoraGroupService(

def updateGroup(groupToUpdate: KnoraGroup, request: GroupUpdateRequest): Task[KnoraGroup] =
for {
_ <- request.name match {
case Some(value) => ensureGroupNameIsUnique(value)
case None => ZIO.unit
}
_ <- ZIO.foreachDiscard(request.name)(ensureGroupNameIsUnique)

updatedGroup <-
knoraGroupRepo.save(
Expand All @@ -66,14 +66,22 @@ case class KnoraGroupService(
)
} yield updatedGroup

def updateGroupStatus(groupToUpdate: KnoraGroup, status: GroupStatus): Task[KnoraGroup] =
for {
group <- knoraGroupRepo.save(groupToUpdate.copy(status = status))
_ <- ZIO.unless(group.status.value)(knoraUserService.findByGroupMembership(group.id).flatMap { members =>
ZIO.foreachDiscard(members)(user =>
knoraUserService.removeUserFromKnoraGroup(user, group).mapError(BadRequestException.apply),
)
})
} yield group

private def ensureGroupNameIsUnique(name: GroupName) =
ZIO.whenZIO(knoraGroupRepo.existsByName(name)) {
ZIO.fail(DuplicateValueException(s"Group with name: '${name.value}' already exists."))
}
}

object KnoraGroupService {
object KnoraGroupService {
val layer = ZLayer.derive[KnoraGroupService]
}
val layer = ZLayer.derive[KnoraGroupService]
}
Original file line number Diff line number Diff line change
Expand Up @@ -50,6 +50,8 @@ trait KnoraUserRepo extends Repository[KnoraUser, UserIri] {

def findByProjectAdminMembership(projectIri: ProjectIri): Task[Chunk[KnoraUser]]

def findByGroupMembership(groupIri: GroupIri): Task[Chunk[KnoraUser]]

/**
* Saves a given user. Use the returned instance for further operations as the save operation might have changed the entity instance completely.
*
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -22,6 +22,7 @@ import org.knora.webapi.slice.admin.domain.model.FamilyName
import org.knora.webapi.slice.admin.domain.model.GivenName
import org.knora.webapi.slice.admin.domain.model.Group
import org.knora.webapi.slice.admin.domain.model.GroupIri
import org.knora.webapi.slice.admin.domain.model.KnoraGroup
import org.knora.webapi.slice.admin.domain.model.KnoraProject
import org.knora.webapi.slice.admin.domain.model.KnoraProject.ProjectIri
import org.knora.webapi.slice.admin.domain.model.KnoraUser
Expand Down Expand Up @@ -51,6 +52,9 @@ case class KnoraUserService(
def findByProjectAdminMembership(project: KnoraProject): Task[Chunk[KnoraUser]] =
userRepo.findByProjectAdminMembership(project.id)

def findByGroupMembership(groupIri: GroupIri): Task[Chunk[KnoraUser]] =
userRepo.findByGroupMembership(groupIri)

def updateSystemAdminStatus(knoraUser: KnoraUser, status: SystemAdmin): Task[KnoraUser] =
updateUser(knoraUser, UserChangeRequest(systemAdmin = Some(status)))

Expand Down Expand Up @@ -138,10 +142,20 @@ case class KnoraUserService(
def removeUserFromGroup(user: KnoraUser, group: Group): IO[UserServiceError, KnoraUser] = for {
_ <- ZIO
.fail(UserServiceError(s"User ${user.id.value} is not member of group ${group.groupIri.value}."))
.when(!user.isInGroup.contains(group.groupIri))
.unless(user.isInGroup.contains(group.groupIri))
user <- updateUser(user, UserChangeRequest(groups = Some(user.isInGroup.filterNot(_ == group.groupIri)))).orDie
} yield user

def removeUserFromKnoraGroup(user: User, group: KnoraGroup): IO[UserServiceError, KnoraUser] =
userRepo.findById(user.userIri).someOrFailException.orDie.flatMap(removeUserFromKnoraGroup(_, group))

def removeUserFromKnoraGroup(user: KnoraUser, group: KnoraGroup): IO[UserServiceError, KnoraUser] = for {
_ <- ZIO
.fail(UserServiceError(s"User ${user.id.value} is not member of group ${group.id.value}."))
mpro7 marked this conversation as resolved.
Show resolved Hide resolved
.unless(user.isInGroup.contains(group.id))
user <- updateUser(user, UserChangeRequest(groups = Some(user.isInGroup.filterNot(_ == group.id)))).orDie
} yield user

def addUserToProject(user: KnoraUser, project: Project): IO[UserServiceError, KnoraUser] = for {
_ <- ZIO
.fail(UserServiceError(s"User ${user.id.value} is already member of project ${project.projectIri.value}."))
Expand All @@ -163,7 +177,7 @@ case class KnoraUserService(
): IO[UserServiceError, KnoraUser] = for {
_ <- ZIO
.fail(UserServiceError(s"User ${user.id.value} is not member of project ${project.projectIri.value}."))
.when(!user.isInProject.contains(project.projectIri))
.unless(user.isInProject.contains(project.projectIri))
projectIri = project.projectIri
newIsInProject = user.isInProject.filterNot(_ == projectIri)
newIsInProjectAdminGroup = user.isInProjectAdminGroup.filterNot(_ == projectIri)
Expand Down Expand Up @@ -194,7 +208,7 @@ case class KnoraUserService(
val msg =
s"User ${user.id.value} is not a member of project ${project.projectIri.value}. A user needs to be a member of the project to be added as project admin."
UserServiceError(msg)
}.when(!user.isInProject.contains(project.projectIri))
}.unless(user.isInProject.contains(project.projectIri))
theChange = UserChangeRequest(projectsAdmin = Some(user.isInProjectAdminGroup :+ project.projectIri))
user <- updateUser(user, theChange).orDie
} yield user
Expand All @@ -213,7 +227,7 @@ case class KnoraUserService(
): IO[UserServiceError, KnoraUser] = for {
_ <- ZIO
.fail(UserServiceError(s"User ${user.id.value} is not admin member of project ${project.projectIri.value}."))
.when(!user.isInProjectAdminGroup.contains(project.projectIri))
.unless(user.isInProjectAdminGroup.contains(project.projectIri))
theChange = UserChangeRequest(projectsAdmin = Some(user.isInProjectAdminGroup.filterNot(_ == project.projectIri)))
user <- updateUser(user, theChange).orDie
} yield user
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -67,6 +67,10 @@ final case class KnoraUserRepoLive(
findAllByTriplePattern(_.has(isInProject, Rdf.iri(projectIri.value)))
.map(_ ++ KnoraUserRepo.builtIn.findAllBy(_.isInProject.contains(projectIri)))

override def findByGroupMembership(groupIri: GroupIri): Task[Chunk[KnoraUser]] =
findAllByTriplePattern(_.has(isInGroup, Rdf.iri(groupIri.value)))
.map(_ ++ KnoraUserRepo.builtIn.findAllBy(_.isInGroup.contains(groupIri)))

override def findByEmail(mail: Email): Task[Option[KnoraUser]] =
findOneByTriplePattern(_.has(email, Rdf.literalOf(mail.value)))
.map(_.orElse(KnoraUserRepo.builtIn.findOneBy(_.email == mail)))
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -19,10 +19,13 @@ import org.knora.webapi.responders.IriService
import org.knora.webapi.slice.admin.domain.model.User
import org.knora.webapi.slice.admin.domain.repo.KnoraProjectRepoInMemory
import org.knora.webapi.slice.admin.domain.service.KnoraGroupRepo
import org.knora.webapi.slice.admin.domain.service.KnoraGroupService.KnoraGroupService
import org.knora.webapi.slice.admin.domain.service.KnoraGroupService
import org.knora.webapi.slice.admin.domain.service.KnoraProjectRepo.builtIn.SystemProject
import org.knora.webapi.slice.admin.domain.service.KnoraProjectService
import org.knora.webapi.slice.admin.domain.service.KnoraUserService
import org.knora.webapi.slice.admin.domain.service.PasswordService
import org.knora.webapi.slice.admin.repo.service.KnoraGroupRepoInMemory
import org.knora.webapi.slice.admin.repo.service.KnoraUserRepoLive
import org.knora.webapi.slice.common.api.AuthorizationRestService
import org.knora.webapi.slice.resourceinfo.domain.IriConverter
import org.knora.webapi.store.cache.CacheService
Expand Down Expand Up @@ -131,16 +134,19 @@ object AuthorizationRestServiceSpec extends ZIOSpecDefault {
},
),
).provide(
AppConfig.layer,
AuthorizationRestService.layer,
CacheService.layer,
KnoraProjectService.layer,
KnoraProjectRepoInMemory.layer,
KnoraGroupService.layer,
KnoraGroupRepoInMemory.layer,
IriService.layer,
IriConverter.layer,
TriplestoreServiceLive.layer,
IriService.layer,
KnoraGroupRepoInMemory.layer,
KnoraGroupService.layer,
KnoraProjectRepoInMemory.layer,
KnoraProjectService.layer,
KnoraUserRepoLive.layer,
KnoraUserService.layer,
PasswordService.layer,
StringFormatter.live,
AppConfig.layer,
TriplestoreServiceLive.layer,
)
}
Loading
Loading