com.netgrif.application.engine.petrinet.service.PetriNetService Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of application-engine Show documentation
Show all versions of application-engine Show documentation
System provides workflow management functions including user, role and data management.
package com.netgrif.application.engine.petrinet.service;
import com.fasterxml.jackson.core.type.TypeReference;
import com.fasterxml.jackson.databind.type.TypeFactory;
import com.netgrif.application.engine.auth.domain.LoggedUser;
import com.netgrif.application.engine.auth.service.interfaces.IUserService;
import com.netgrif.application.engine.history.domain.petrinetevents.DeletePetriNetEventLog;
import com.netgrif.application.engine.history.domain.petrinetevents.ImportPetriNetEventLog;
import com.netgrif.application.engine.history.service.IHistoryService;
import com.netgrif.application.engine.importer.service.Importer;
import com.netgrif.application.engine.importer.service.throwable.MissingIconKeyException;
import com.netgrif.application.engine.orgstructure.groups.interfaces.INextGroupService;
import com.netgrif.application.engine.petrinet.domain.PetriNet;
import com.netgrif.application.engine.petrinet.domain.Transition;
import com.netgrif.application.engine.petrinet.domain.VersionType;
import com.netgrif.application.engine.petrinet.domain.dataset.logic.action.Action;
import com.netgrif.application.engine.petrinet.domain.dataset.logic.action.FieldActionsRunner;
import com.netgrif.application.engine.petrinet.domain.events.EventPhase;
import com.netgrif.application.engine.petrinet.domain.events.ProcessEventType;
import com.netgrif.application.engine.petrinet.domain.repositories.PetriNetRepository;
import com.netgrif.application.engine.petrinet.domain.roles.ProcessRole;
import com.netgrif.application.engine.petrinet.domain.throwable.MissingPetriNetMetaDataException;
import com.netgrif.application.engine.petrinet.domain.version.Version;
import com.netgrif.application.engine.petrinet.service.interfaces.IPetriNetService;
import com.netgrif.application.engine.petrinet.service.interfaces.IProcessRoleService;
import com.netgrif.application.engine.petrinet.web.responsebodies.DataFieldReference;
import com.netgrif.application.engine.petrinet.web.responsebodies.PetriNetReference;
import com.netgrif.application.engine.petrinet.web.responsebodies.TransitionReference;
import com.netgrif.application.engine.rules.domain.facts.NetImportedFact;
import com.netgrif.application.engine.rules.service.interfaces.IRuleEngine;
import com.netgrif.application.engine.workflow.domain.FileStorageConfiguration;
import com.netgrif.application.engine.workflow.domain.eventoutcomes.petrinetoutcomes.ImportPetriNetEventOutcome;
import com.netgrif.application.engine.workflow.service.interfaces.IEventService;
import com.netgrif.application.engine.workflow.service.interfaces.IFieldActionsCacheService;
import com.netgrif.application.engine.workflow.service.interfaces.IWorkflowService;
import org.apache.tomcat.util.http.fileupload.IOUtils;
import org.bson.Document;
import org.bson.types.ObjectId;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.context.ApplicationEventPublisher;
import org.springframework.core.io.FileSystemResource;
import org.springframework.data.domain.Page;
import org.springframework.data.domain.PageRequest;
import org.springframework.data.domain.Pageable;
import org.springframework.data.domain.Sort;
import org.springframework.data.mongodb.core.MongoTemplate;
import org.springframework.data.mongodb.core.aggregation.Aggregation;
import org.springframework.data.mongodb.core.aggregation.AggregationResults;
import org.springframework.data.mongodb.core.aggregation.GroupOperation;
import org.springframework.data.mongodb.core.query.Criteria;
import org.springframework.data.mongodb.core.query.Query;
import org.springframework.data.repository.support.PageableExecutionUtils;
import org.springframework.stereotype.Service;
import org.springframework.transaction.annotation.Transactional;
import javax.inject.Provider;
import java.io.*;
import java.nio.file.Path;
import java.util.*;
import java.util.stream.Collectors;
import static com.netgrif.application.engine.petrinet.service.interfaces.IPetriNetService.transformToReference;
@Service
public class PetriNetService implements IPetriNetService {
private static final Logger log = LoggerFactory.getLogger(PetriNetService.class);
@Autowired
private IProcessRoleService processRoleService;
@Autowired
private PetriNetRepository repository;
@Autowired
private MongoTemplate mongoTemplate;
@Autowired
private ApplicationEventPublisher publisher;
@Autowired
private FileStorageConfiguration fileStorageConfiguration;
@Autowired
private IRuleEngine ruleEngine;
@Autowired
private IWorkflowService workflowService;
@Autowired
private INextGroupService groupService;
@Autowired
private Provider importerProvider;
@Autowired
private FieldActionsRunner actionsRunner;
@Autowired
private IFieldActionsCacheService functionCacheService;
@Autowired
private IUserService userService;
@Autowired
private IEventService eventService;
@Autowired
private IHistoryService historyService;
private Map cache = new HashMap<>();
protected Importer getImporter() {
return importerProvider.get();
}
@Override
public void evictCache() {
cache = new HashMap<>();
}
/**
* Get read only Petri net.
*/
@Override
public PetriNet get(ObjectId petriNetId) {
PetriNet net = cache.get(petriNetId);
if (net == null) {
Optional optional = repository.findById(petriNetId.toString());
if (!optional.isPresent()) {
throw new IllegalArgumentException("Petri net with id [" + petriNetId + "] not found");
}
net = optional.get();
cache.put(petriNetId, net);
}
return net;
}
@Override
public List get(Collection petriNetIds) {
return petriNetIds.stream().map(this::get).collect(Collectors.toList());
}
@Override
public List get(List petriNetIds) {
return this.get(petriNetIds.stream().map(ObjectId::new).collect(Collectors.toList()));
}
@Override
public PetriNet clone(ObjectId petriNetId) {
return get(petriNetId).clone();
}
@Override
@Deprecated
public ImportPetriNetEventOutcome importPetriNet(InputStream xmlFile, String releaseType, LoggedUser author) throws IOException, MissingPetriNetMetaDataException, MissingIconKeyException{
return importPetriNet(xmlFile, VersionType.valueOf(releaseType.trim().toUpperCase()), author);
}
@Override
public ImportPetriNetEventOutcome importPetriNet(InputStream xmlFile, VersionType releaseType, LoggedUser author) throws IOException, MissingPetriNetMetaDataException, MissingIconKeyException {
Optional imported = getImporter().importPetriNet(copy(xmlFile));
ImportPetriNetEventOutcome outcome = new ImportPetriNetEventOutcome();
if (!imported.isPresent()) {
return outcome;
}
PetriNet net = imported.get();
PetriNet existingNet = getNewestVersionByIdentifier(net.getIdentifier());
if (existingNet != null) {
net.setVersion(existingNet.getVersion());
net.incrementVersion(releaseType);
}
processRoleService.saveAll(net.getRoles().values());
net.setAuthor(author.transformToAuthor());
functionCacheService.cachePetriNetFunctions(net);
Path savedPath = getImporter().saveNetFile(net, xmlFile);
log.info("Petri net " + net.getTitle() + " (" + net.getInitials() + " v" + net.getVersion() + ") imported successfully");
outcome.setOutcomes(eventService.runActions(net.getPreUploadActions(), null, Optional.empty()));
evaluateRules(net, EventPhase.PRE);
save(net);
historyService.save(new ImportPetriNetEventLog(null, EventPhase.PRE, net.getObjectId()));
outcome.setOutcomes(eventService.runActions(net.getPostUploadActions(), null, Optional.empty()));
evaluateRules(net, EventPhase.POST);
save(net);
cache.put(net.getObjectId(), net);
historyService.save(new ImportPetriNetEventLog(null, EventPhase.POST, net.getObjectId()));
addMessageToOutcome(net, ProcessEventType.UPLOAD, outcome);
outcome.setNet(imported.get());
return outcome;
}
private ImportPetriNetEventOutcome addMessageToOutcome(PetriNet net, ProcessEventType type, ImportPetriNetEventOutcome outcome) {
if(net.getProcessEvents().containsKey(type)){
outcome.setMessage(net.getProcessEvents().get(type).getMessage());
}
return outcome;
}
protected void evaluateRules(PetriNet net, EventPhase phase) {
ruleEngine.evaluateRules(net, new NetImportedFact(net.getStringId(), phase));
}
private InputStream copy(InputStream xmlFile) throws IOException {
ByteArrayOutputStream baos = new ByteArrayOutputStream();
IOUtils.copy(xmlFile, baos);
byte[] bytes = baos.toByteArray();
return new ByteArrayInputStream(bytes);
}
@Override
public Optional save(PetriNet petriNet) {
petriNet.initializeArcs();
return Optional.of(repository.save(petriNet));
}
@Override
public PetriNet getPetriNet(String id) {
Optional net = repository.findById(id);
if (!net.isPresent())
throw new IllegalArgumentException("No Petri net with id: " + id + " was found.");
net.get().initializeArcs();
return net.get();
}
@Override
public PetriNet getPetriNet(String identifier, Version version) {
PetriNet net = repository.findByIdentifierAndVersion(identifier, version);
if (net == null)
return null;
net.initializeArcs();
return net;
}
@Override
public List getByIdentifier(String identifier) {
List nets = repository.findAllByIdentifier(identifier);
nets.forEach(PetriNet::initializeArcs);
return nets;
}
@Override
public PetriNet getNewestVersionByIdentifier(String identifier) {
List nets = repository.findByIdentifier(identifier, PageRequest.of(0, 1, Sort.Direction.DESC, "version.major", "version.minor", "version.patch")).getContent();
if (nets.isEmpty())
return null;
return nets.get(0);
}
/**
* Determines which of the provided Strings are identifiers of {@link PetriNet}s uploaded in the system.
*
* @param identifiers a list of Strings that represent potential PetriNet identifiers
* @return a list containing a subset of the input strings that correspond to identifiers of PetriNets that are present in the system
*/
@Override
public List getExistingPetriNetIdentifiersFromIdentifiersList(List identifiers) {
Aggregation agg = Aggregation.newAggregation(
Aggregation.match(Criteria.where("identifier").in(identifiers)),
Aggregation.group("identifier"),
Aggregation.project("identifier").and("identifier").previousOperation()
);
AggregationResults> groupResults = mongoTemplate.aggregate(
agg,
PetriNet.class,
TypeFactory.defaultInstance().constructType(new TypeReference
© 2015 - 2025 Weber Informatics LLC | Privacy Policy