com.netgrif.application.engine.workflow.service.TaskSearchService Maven / Gradle / Ivy
package com.netgrif.application.engine.workflow.service;
import com.netgrif.application.engine.auth.domain.LoggedUser;
import com.netgrif.application.engine.petrinet.domain.PetriNetSearch;
import com.netgrif.application.engine.petrinet.service.interfaces.IPetriNetService;
import com.netgrif.application.engine.petrinet.web.responsebodies.PetriNetReference;
import com.netgrif.application.engine.utils.FullPageRequest;
import com.netgrif.application.engine.workflow.domain.QTask;
import com.netgrif.application.engine.workflow.domain.Task;
import com.netgrif.application.engine.workflow.web.requestbodies.TaskSearchRequest;
import com.netgrif.application.engine.workflow.web.requestbodies.taskSearch.TaskSearchCaseRequest;
import com.querydsl.core.BooleanBuilder;
import com.querydsl.core.types.Predicate;
import org.bson.types.ObjectId;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Service;
import java.util.*;
import java.util.stream.Collectors;
@Service
public class TaskSearchService extends MongoSearchService {
@Autowired
private IPetriNetService petriNetService;
public Predicate buildQuery(List requests, LoggedUser user, Locale locale, Boolean isIntersection) {
LoggedUser loggedOrImpersonated = user.getSelfOrImpersonated();
List singleQueries = requests.stream().map(r -> this.buildSingleQuery(r, loggedOrImpersonated, locale)).collect(Collectors.toList());
if (isIntersection && !singleQueries.stream().allMatch(Objects::nonNull)) {
// one of the queries evaluates to empty set => the entire result is an empty set
return null;
} else if (!isIntersection) {
singleQueries = singleQueries.stream().filter(Objects::nonNull).collect(Collectors.toList());
if (singleQueries.size() == 0) {
// all queries result in an empty set => the entire result is an empty set
return null;
}
}
BooleanBuilder builder = constructPredicateTree(singleQueries, isIntersection ? BooleanBuilder::and : BooleanBuilder::or);
BooleanBuilder constraints = new BooleanBuilder(buildRolesQueryConstraint(loggedOrImpersonated));
constraints.or(buildUserRefQueryConstraint(loggedOrImpersonated));
builder.and(constraints);
BooleanBuilder permissionConstraints = new BooleanBuilder(buildViewRoleQueryConstraint(loggedOrImpersonated));
permissionConstraints.andNot(buildNegativeViewRoleQueryConstraint(loggedOrImpersonated));
permissionConstraints.or(buildViewUserQueryConstraint(loggedOrImpersonated));
permissionConstraints.andNot(buildNegativeViewUsersQueryConstraint(loggedOrImpersonated));
builder.and(permissionConstraints);
return builder;
}
protected Predicate buildRolesQueryConstraint(LoggedUser user) {
List roleConstraints = user.getProcessRoles().stream().map(this::roleQuery).collect(Collectors.toList());
return constructPredicateTree(roleConstraints, BooleanBuilder::or);
}
protected Predicate buildUserRefQueryConstraint(LoggedUser user) {
Predicate userRefConstraints = userRefQuery(user.getId());
return constructPredicateTree(Collections.singletonList(userRefConstraints), BooleanBuilder::or);
}
protected Predicate buildViewRoleQueryConstraint(LoggedUser user) {
List roleConstraints = user.getProcessRoles().stream().map(this::viewRoleQuery).collect(Collectors.toList());
return constructPredicateTree(roleConstraints, BooleanBuilder::or);
}
public Predicate viewRoleQuery(String role) {
return QTask.task.viewUserRefs.isEmpty().and(QTask.task.viewRoles.isEmpty()).or(QTask.task.viewRoles.contains(role));
}
protected Predicate buildViewUserQueryConstraint(LoggedUser user) {
Predicate userConstraints = viewUsersQuery(user.getId());
return constructPredicateTree(Collections.singletonList(userConstraints), BooleanBuilder::or);
}
public Predicate viewUsersQuery(String userId) {
return QTask.task.negativeViewRoles.isEmpty().and(QTask.task.viewUserRefs.isEmpty()).and(QTask.task.viewRoles.isEmpty()).or(QTask.task.viewUsers.contains(userId));
}
protected Predicate buildNegativeViewRoleQueryConstraint(LoggedUser user) {
List roleConstraints = user.getProcessRoles().stream().map(this::negativeViewRoleQuery).collect(Collectors.toList());
return constructPredicateTree(roleConstraints, BooleanBuilder::or);
}
public Predicate negativeViewRoleQuery(String role) {
return QTask.task.negativeViewRoles.contains(role);
}
protected Predicate buildNegativeViewUsersQueryConstraint(LoggedUser user) {
Predicate userConstraints = negativeViewUsersQuery(user.getId());
return constructPredicateTree(Collections.singletonList(userConstraints), BooleanBuilder::or);
}
public Predicate negativeViewUsersQuery(String userId) {
return QTask.task.negativeViewUsers.contains(userId);
}
private Predicate buildSingleQuery(TaskSearchRequest request, LoggedUser user, Locale locale) {
BooleanBuilder builder = new BooleanBuilder();
buildStringIdQuery(request, builder);
buildRoleQuery(request, builder);
buildCaseQuery(request, builder);
buildTitleQuery(request, builder);
buildUserQuery(request, builder);
buildProcessQuery(request, builder);
buildFullTextQuery(request, builder);
buildTransitionQuery(request, builder);
buildTagsQuery(request, builder);
boolean resultAlwaysEmpty = buildGroupQuery(request, user, locale, builder);
if (resultAlwaysEmpty)
return null;
else
return builder;
}
private void buildStringIdQuery(TaskSearchRequest request, BooleanBuilder query) {
if (request.stringId == null || request.stringId.isEmpty()) {
return;
}
query.and(
constructPredicateTree(
request.stringId.stream().map(this::stringIdQuery).collect(Collectors.toList()),
BooleanBuilder::or)
);
}
private void buildRoleQuery(TaskSearchRequest request, BooleanBuilder query) {
if (request.role == null || request.role.isEmpty()) {
return;
}
query.and(
constructPredicateTree(
request.role.stream().map(this::roleQuery).collect(Collectors.toList()),
BooleanBuilder::or)
);
}
public Predicate roleQuery(String role) {
return QTask.task.roles.containsKey(role);
}
public Predicate stringIdQuery(String id) {
return QTask.task._id.eq(new ObjectId(id));
}
public Predicate userRefQuery(String userId) {
return QTask.task.users.containsKey(userId);
}
private void buildCaseQuery(TaskSearchRequest request, BooleanBuilder query) {
if (request.useCase == null || request.useCase.isEmpty()) {
return;
}
query.and(
constructPredicateTree(
request.useCase.stream().map(this::caseRequestQuery).filter(Objects::nonNull).collect(Collectors.toList()),
BooleanBuilder::or)
);
}
/**
* @return Predicate for ID if only ID is present. Predicate for title if only title is present.
* If both are present an ID predicate is returned. If neither are present null is returned.
*/
private Predicate caseRequestQuery(TaskSearchCaseRequest caseRequest) {
if (caseRequest.id != null) {
return caseIdQuery(caseRequest.id);
} else if (caseRequest.title != null) {
return caseTitleQuery(caseRequest.title);
}
return null;
}
public Predicate caseIdQuery(String caseId) {
return QTask.task.caseId.eq(caseId);
}
public Predicate caseTitleQuery(String caseTitle) {
return QTask.task.caseTitle.containsIgnoreCase(caseTitle);
}
private void buildTitleQuery(TaskSearchRequest request, BooleanBuilder query) {
if (request.title == null || request.title.isEmpty()) {
return;
}
query.and(
constructPredicateTree(
request.title.stream().map(this::titleQuery).collect(Collectors.toList()),
BooleanBuilder::or)
);
}
public Predicate titleQuery(String query) {
return QTask.task.title.defaultValue.containsIgnoreCase(query);
}
private void buildUserQuery(TaskSearchRequest request, BooleanBuilder query) {
if (request.user == null || request.user.isEmpty()) {
return;
}
query.and(
constructPredicateTree(
request.user.stream().map(this::userQuery).collect(Collectors.toList()),
BooleanBuilder::or)
);
}
public Predicate userQuery(String userId) {
return QTask.task.userId.eq(userId);
}
private void buildProcessQuery(TaskSearchRequest request, BooleanBuilder query) {
if (request.process == null || request.process.isEmpty()) {
return;
}
query.and(
constructPredicateTree(
request.process.stream().map(p -> processQuery(p.identifier)).collect(Collectors.toList()),
BooleanBuilder::or)
);
}
public Predicate processQuery(String processId) {
return QTask.task.processId.eq(processId);
}
private void buildFullTextQuery(TaskSearchRequest request, BooleanBuilder query) {
if (request.fullText == null || request.fullText.isEmpty()) {
return;
}
query.and(fullTextQuery(request.fullText));
}
public Predicate fullTextQuery(String searchedText) {
BooleanBuilder builder = new BooleanBuilder();
builder.or(QTask.task.title.defaultValue.containsIgnoreCase(searchedText));
builder.or(QTask.task.caseTitle.containsIgnoreCase(searchedText));
return builder;
}
private void buildTransitionQuery(TaskSearchRequest request, BooleanBuilder query) {
if (request.transitionId == null || request.transitionId.isEmpty()) {
return;
}
query.and(
constructPredicateTree(
request.transitionId.stream().map(this::transitionQuery).collect(Collectors.toList()),
BooleanBuilder::or)
);
}
public Predicate transitionQuery(String transitionId) {
return QTask.task.transitionId.eq(transitionId);
}
public boolean buildGroupQuery(TaskSearchRequest request, LoggedUser user, Locale locale, BooleanBuilder query) {
if (request.group == null || request.group.isEmpty())
return false;
PetriNetSearch processQuery = new PetriNetSearch();
processQuery.setGroup(request.group);
List groupProcesses = this.petriNetService.search(processQuery, user, new FullPageRequest(), locale).getContent();
if (groupProcesses.size() == 0)
return true;
query.and(
constructPredicateTree(
groupProcesses.stream().map(PetriNetReference::getStringId).map(QTask.task.processId::eq).collect(Collectors.toList()),
BooleanBuilder::or
)
);
return false;
}
private void buildTagsQuery(TaskSearchRequest request, BooleanBuilder query) {
if (request.tags == null || request.tags.isEmpty()) {
return;
}
query.and(
constructPredicateTree(
request.tags.entrySet().stream().map(entry -> this.tagQuery(entry.getKey(), entry.getValue())).collect(Collectors.toList()),
BooleanBuilder::and)
);
}
public Predicate tagQuery(String key, String value) {
return QTask.task.tags.get(key).eq(value);
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy