Some refactoring

pull/10/head
Jens Pelzetter 2021-03-18 19:24:35 +01:00
parent f9eac9c1c2
commit a6119e09a2
6 changed files with 900 additions and 471 deletions

View File

@ -5,17 +5,9 @@
*/ */
package org.librecms.ui.contentsections.documents; package org.librecms.ui.contentsections.documents;
import org.libreccm.api.Identifier;
import org.libreccm.api.IdentifierParser;
import org.libreccm.l10n.GlobalizationHelper; import org.libreccm.l10n.GlobalizationHelper;
import org.libreccm.security.AuthorizationRequired; import org.libreccm.security.AuthorizationRequired;
import org.libreccm.security.PermissionChecker; import org.libreccm.security.PermissionChecker;
import org.libreccm.workflow.AssignableTask;
import org.libreccm.workflow.AssignableTaskManager;
import org.libreccm.workflow.TaskManager;
import org.libreccm.workflow.Workflow;
import org.libreccm.workflow.WorkflowManager;
import org.libreccm.workflow.WorkflowRepository;
import org.librecms.contentsection.ContentItem; import org.librecms.contentsection.ContentItem;
import org.librecms.contentsection.ContentItemManager; import org.librecms.contentsection.ContentItemManager;
import org.librecms.contentsection.ContentItemRepository; import org.librecms.contentsection.ContentItemRepository;
@ -25,17 +17,11 @@ import org.librecms.contentsection.FolderRepository;
import org.librecms.contentsection.FolderType; import org.librecms.contentsection.FolderType;
import org.librecms.contentsection.privileges.ItemPrivileges; import org.librecms.contentsection.privileges.ItemPrivileges;
import org.librecms.lifecycle.LifecycleDefinition; import org.librecms.lifecycle.LifecycleDefinition;
import org.librecms.lifecycle.LifecycleManager;
import org.librecms.lifecycle.Phase; import org.librecms.lifecycle.Phase;
import org.librecms.lifecycle.PhaseRepository;
import org.librecms.ui.contentsections.ContentSectionsUi; import org.librecms.ui.contentsections.ContentSectionsUi;
import java.time.LocalDateTime;
import java.time.ZoneId; import java.time.ZoneId;
import java.time.ZoneOffset;
import java.time.format.DateTimeFormatter; import java.time.format.DateTimeFormatter;
import java.time.temporal.TemporalAccessor;
import java.util.Date;
import java.util.Optional; import java.util.Optional;
import java.util.stream.Collectors; import java.util.stream.Collectors;
@ -70,6 +56,9 @@ public class DocumentController {
@Inject @Inject
private ContentSectionsUi sectionsUi; private ContentSectionsUi sectionsUi;
@Inject
private DocumentUi documentUi;
@Inject @Inject
private FolderRepository folderRepo; private FolderRepository folderRepo;
@ -85,39 +74,18 @@ public class DocumentController {
@Inject @Inject
private GlobalizationHelper globalizationHelper; private GlobalizationHelper globalizationHelper;
@Inject
private IdentifierParser identifierParser;
@Inject
private LifecycleManager lifecycleManager;
@Inject @Inject
private Models models; private Models models;
@Inject @Inject
private PermissionChecker permissionChecker; private PermissionChecker permissionChecker;
@Inject
private PhaseRepository phaseRepository;
@Inject @Inject
private PublishStepModel publishStepModel; private PublishStepModel publishStepModel;
@Inject @Inject
private SelectedDocumentModel selectedDocumentModel; private SelectedDocumentModel selectedDocumentModel;
@Inject
private WorkflowManager workflowManager;
@Inject
private AssignableTaskManager assignableTaskManager;
@Inject
private TaskManager taskManager;
@Inject
private WorkflowRepository workflowRepository;
@GET @GET
@Path("/") @Path("/")
@AuthorizationRequired @AuthorizationRequired
@ -287,7 +255,7 @@ public class DocumentController {
final Optional<ContentItem> itemResult = itemRepo final Optional<ContentItem> itemResult = itemRepo
.findByPath(section, documentPath); .findByPath(section, documentPath);
if (!itemResult.isPresent()) { if (!itemResult.isPresent()) {
return showDocumentNotFound(section, documentPath); return documentUi.showDocumentNotFound(section, documentPath);
} }
final ContentItem item = itemResult.get(); final ContentItem item = itemResult.get();
if (!permissionChecker.isPermitted(ItemPrivileges.EDIT, item)) { if (!permissionChecker.isPermitted(ItemPrivileges.EDIT, item)) {
@ -324,7 +292,7 @@ public class DocumentController {
final Optional<ContentItem> itemResult = itemRepo final Optional<ContentItem> itemResult = itemRepo
.findByPath(section, documentPath); .findByPath(section, documentPath);
if (!itemResult.isPresent()) { if (!itemResult.isPresent()) {
return showDocumentNotFound(section, documentPath); return documentUi.showDocumentNotFound(section, documentPath);
} }
final ContentItem item = itemResult.get(); final ContentItem item = itemResult.get();
if (!permissionChecker.isPermitted(ItemPrivileges.PUBLISH, item)) { if (!permissionChecker.isPermitted(ItemPrivileges.PUBLISH, item)) {
@ -355,84 +323,6 @@ public class DocumentController {
return "org/librecms/ui/contentsection/documents/publish.xhtml"; return "org/librecms/ui/contentsection/documents/publish.xhtml";
} }
@POST
@Path("/{documentPath:(.+)?}/@lifecycle/phases/{phaseId}")
@AuthorizationRequired
@Transactional(Transactional.TxType.REQUIRED)
public String updatePhaseDates(
@PathParam("sectionIdentifider") final String sectionIdentifier,
@PathParam("documentPath") final String documentPath,
@PathParam("phaseId") final long phaseId,
@FormParam("startDate") final String startDateParam,
@FormParam("endDate") final String endDateParam
) {
final Optional<ContentSection> sectionResult = sectionsUi
.findContentSection(sectionIdentifier);
if (!sectionResult.isPresent()) {
return sectionsUi.showContentSectionNotFound(sectionIdentifier);
}
final ContentSection section = sectionResult.get();
final Optional<ContentItem> itemResult = itemRepo
.findByPath(section, documentPath);
if (!itemResult.isPresent()) {
return showDocumentNotFound(section, documentPath);
}
final ContentItem item = itemResult.get();
if (!permissionChecker.isPermitted(ItemPrivileges.PUBLISH, item)) {
return sectionsUi.showAccessDenied(
"sectionIdentifier", sectionIdentifier,
"documentPath", documentPath
);
}
if (item.getLifecycle() != null) {
final Optional<Phase> phaseResult = item
.getLifecycle()
.getPhases()
.stream()
.filter(phase -> phase.getPhaseId() == phaseId)
.findAny();
if (!phaseResult.isPresent()) {
models.put("section", section.getLabel());
models.put("phaseId", phaseId);
return "org/librecms/ui/contentsection/phase-not-found.xhtml";
}
final Phase phase = phaseResult.get();
final DateTimeFormatter dateTimeFormatter
= DateTimeFormatter.ISO_DATE_TIME
.withZone(ZoneId.systemDefault());
final LocalDateTime startLocalDateTime = LocalDateTime
.parse(startDateParam, dateTimeFormatter);
phase.setStartDateTime(
Date.from(
startLocalDateTime.toInstant(
ZoneOffset.from(startLocalDateTime)
)
)
);
final LocalDateTime endLocalDateTime = LocalDateTime
.parse(endDateParam, dateTimeFormatter);
phase.setEndDateTime(
Date.from(
endLocalDateTime.toInstant(
ZoneOffset.from(endLocalDateTime)
)
)
);
phaseRepository.save(phase);
}
return String.format(
"redirect:/%s/documents/%s/@publish",
sectionIdentifier,
documentPath
);
}
@POST @POST
@Path("/{documentPath:(.+)?}/@publish") @Path("/{documentPath:(.+)?}/@publish")
@AuthorizationRequired @AuthorizationRequired
@ -453,7 +343,7 @@ public class DocumentController {
final Optional<ContentItem> itemResult = itemRepo final Optional<ContentItem> itemResult = itemRepo
.findByPath(section, documentPath); .findByPath(section, documentPath);
if (!itemResult.isPresent()) { if (!itemResult.isPresent()) {
return showDocumentNotFound(section, documentPath); return documentUi.showDocumentNotFound(section, documentPath);
} }
final ContentItem item = itemResult.get(); final ContentItem item = itemResult.get();
if (!permissionChecker.isPermitted(ItemPrivileges.PUBLISH, item)) { if (!permissionChecker.isPermitted(ItemPrivileges.PUBLISH, item)) {
@ -506,7 +396,7 @@ public class DocumentController {
final Optional<ContentItem> itemResult = itemRepo final Optional<ContentItem> itemResult = itemRepo
.findByPath(section, documentPath); .findByPath(section, documentPath);
if (!itemResult.isPresent()) { if (!itemResult.isPresent()) {
return showDocumentNotFound(section, documentPath); return documentUi.showDocumentNotFound(section, documentPath);
} }
final ContentItem item = itemResult.get(); final ContentItem item = itemResult.get();
if (!permissionChecker.isPermitted(ItemPrivileges.PUBLISH, item)) { if (!permissionChecker.isPermitted(ItemPrivileges.PUBLISH, item)) {
@ -543,7 +433,7 @@ public class DocumentController {
final Optional<ContentItem> itemResult = itemRepo final Optional<ContentItem> itemResult = itemRepo
.findByPath(section, documentPath); .findByPath(section, documentPath);
if (!itemResult.isPresent()) { if (!itemResult.isPresent()) {
return showDocumentNotFound(section, documentPath); return documentUi.showDocumentNotFound(section, documentPath);
} }
final ContentItem item = itemResult.get(); final ContentItem item = itemResult.get();
if (!permissionChecker.isPermitted(ItemPrivileges.PUBLISH, item)) { if (!permissionChecker.isPermitted(ItemPrivileges.PUBLISH, item)) {
@ -562,250 +452,248 @@ public class DocumentController {
); );
} }
@POST // @POST
@Path("/{documentPath:(.+)?}/@workflow/tasks/${taskIdentifier}/@lock") // @Path("/{documentPath:(.+)?}/@workflow/tasks/${taskIdentifier}/@lock")
@AuthorizationRequired // @AuthorizationRequired
@Transactional(Transactional.TxType.REQUIRED) // @Transactional(Transactional.TxType.REQUIRED)
public String lockTask( // public String lockTask(
@PathParam("sectionIdentifider") final String sectionIdentifier, // @PathParam("sectionIdentifider") final String sectionIdentifier,
@PathParam("documentPath") final String documentPath, // @PathParam("documentPath") final String documentPath,
@PathParam("taskIdentifier") final String taskIdentifier, // @PathParam("taskIdentifier") final String taskIdentifier,
@FormParam("returnUrl") final String returnUrl // @FormParam("returnUrl") final String returnUrl
) { // ) {
final Optional<ContentSection> sectionResult = sectionsUi // final Optional<ContentSection> sectionResult = sectionsUi
.findContentSection(sectionIdentifier); // .findContentSection(sectionIdentifier);
if (!sectionResult.isPresent()) { // if (!sectionResult.isPresent()) {
return sectionsUi.showContentSectionNotFound(sectionIdentifier); // return sectionsUi.showContentSectionNotFound(sectionIdentifier);
} // }
final ContentSection section = sectionResult.get(); // final ContentSection section = sectionResult.get();
//
final Optional<ContentItem> itemResult = itemRepo // final Optional<ContentItem> itemResult = itemRepo
.findByPath(section, documentPath); // .findByPath(section, documentPath);
if (!itemResult.isPresent()) { // if (!itemResult.isPresent()) {
return showDocumentNotFound(section, documentPath); // return documentUi.showDocumentNotFound(section, documentPath);
} // }
final ContentItem item = itemResult.get(); // final ContentItem item = itemResult.get();
selectedDocumentModel.setContentItem(item); // selectedDocumentModel.setContentItem(item);
//
final Optional<AssignableTask> taskResult = findTask( // final Optional<AssignableTask> taskResult = findTask(
item, taskIdentifier // item, taskIdentifier
); // );
if (!taskResult.isPresent()) { // if (!taskResult.isPresent()) {
return showTaskNotFound(section, documentPath, taskIdentifier); // return showTaskNotFound(section, documentPath, taskIdentifier);
} // }
//
final AssignableTask task = taskResult.get(); // final AssignableTask task = taskResult.get();
assignableTaskManager.lockTask(task); // assignableTaskManager.lockTask(task);
//
return String.format("redirect:%s", returnUrl); // return String.format("redirect:%s", returnUrl);
} // }
// @POST
@POST // @Path("/{documentPath:(.+)?}/@workflow/tasks/${taskIdentifier}/@unlock")
@Path("/{documentPath:(.+)?}/@workflow/tasks/${taskIdentifier}/@unlock") // @AuthorizationRequired
@AuthorizationRequired // @Transactional(Transactional.TxType.REQUIRED)
@Transactional(Transactional.TxType.REQUIRED) // public String unlockTask(
public String unlockTask( // @PathParam("sectionIdentifider") final String sectionIdentifier,
@PathParam("sectionIdentifider") final String sectionIdentifier, // @PathParam("documentPath") final String documentPath,
@PathParam("documentPath") final String documentPath, // @PathParam("taskIdentifier") final String taskIdentifier,
@PathParam("taskIdentifier") final String taskIdentifier, // @FormParam("returnUrl") final String returnUrl
@FormParam("returnUrl") final String returnUrl // ) {
) { // final Optional<ContentSection> sectionResult = sectionsUi
final Optional<ContentSection> sectionResult = sectionsUi // .findContentSection(sectionIdentifier);
.findContentSection(sectionIdentifier); // if (!sectionResult.isPresent()) {
if (!sectionResult.isPresent()) { // return sectionsUi.showContentSectionNotFound(sectionIdentifier);
return sectionsUi.showContentSectionNotFound(sectionIdentifier); // }
} // final ContentSection section = sectionResult.get();
final ContentSection section = sectionResult.get(); //
// final Optional<ContentItem> itemResult = itemRepo
final Optional<ContentItem> itemResult = itemRepo // .findByPath(section, documentPath);
.findByPath(section, documentPath); // if (!itemResult.isPresent()) {
if (!itemResult.isPresent()) { // return documentUi.showDocumentNotFound(section, documentPath);
return showDocumentNotFound(section, documentPath); // }
} // final ContentItem item = itemResult.get();
final ContentItem item = itemResult.get(); // selectedDocumentModel.setContentItem(item);
selectedDocumentModel.setContentItem(item); //
// final Optional<AssignableTask> taskResult = findTask(
final Optional<AssignableTask> taskResult = findTask( // item, taskIdentifier
item, taskIdentifier // );
); // if (!taskResult.isPresent()) {
if (!taskResult.isPresent()) { // return showTaskNotFound(section, documentPath, taskIdentifier);
return showTaskNotFound(section, documentPath, taskIdentifier); // }
} //
// final AssignableTask task = taskResult.get();
final AssignableTask task = taskResult.get(); // assignableTaskManager.unlockTask(task);
assignableTaskManager.unlockTask(task); //
// return String.format("redirect:%s", returnUrl);
return String.format("redirect:%s", returnUrl); // }
} //
// @POST
@POST // @Path("/{documentPath:(.+)?}/@workflow/tasks/${taskIdentifier}/@finish")
@Path("/{documentPath:(.+)?}/@workflow/tasks/${taskIdentifier}/@finish") // @AuthorizationRequired
@AuthorizationRequired // @Transactional(Transactional.TxType.REQUIRED)
@Transactional(Transactional.TxType.REQUIRED) // public String finishTask(
public String finishTask( // @PathParam("sectionIdentifider") final String sectionIdentifier,
@PathParam("sectionIdentifider") final String sectionIdentifier, // @PathParam("documentPath") final String documentPath,
@PathParam("documentPath") final String documentPath, // @PathParam("taskIdentifier") final String taskIdentifier,
@PathParam("taskIdentifier") final String taskIdentifier, // @FormParam("comment") @DefaultValue("") final String comment,
@FormParam("comment") @DefaultValue("") final String comment, // @FormParam("returnUrl") final String returnUrl
@FormParam("returnUrl") final String returnUrl // ) {
) { // final Optional<ContentSection> sectionResult = sectionsUi
final Optional<ContentSection> sectionResult = sectionsUi // .findContentSection(sectionIdentifier);
.findContentSection(sectionIdentifier); // if (!sectionResult.isPresent()) {
if (!sectionResult.isPresent()) { // return sectionsUi.showContentSectionNotFound(sectionIdentifier);
return sectionsUi.showContentSectionNotFound(sectionIdentifier); // }
} // final ContentSection section = sectionResult.get();
final ContentSection section = sectionResult.get(); //
// final Optional<ContentItem> itemResult = itemRepo
final Optional<ContentItem> itemResult = itemRepo // .findByPath(section, documentPath);
.findByPath(section, documentPath); // if (!itemResult.isPresent()) {
if (!itemResult.isPresent()) { // return documentUi.showDocumentNotFound(section, documentPath);
return showDocumentNotFound(section, documentPath); // }
} // final ContentItem item = itemResult.get();
final ContentItem item = itemResult.get(); // selectedDocumentModel.setContentItem(item);
selectedDocumentModel.setContentItem(item); //
// final Optional<AssignableTask> taskResult = findTask(
final Optional<AssignableTask> taskResult = findTask( // item, taskIdentifier
item, taskIdentifier // );
); // if (!taskResult.isPresent()) {
if (!taskResult.isPresent()) { // return showTaskNotFound(section, documentPath, taskIdentifier);
return showTaskNotFound(section, documentPath, taskIdentifier); // }
} //
// final AssignableTask task = taskResult.get();
final AssignableTask task = taskResult.get(); // if (comment.isEmpty()) {
if (comment.isEmpty()) { // assignableTaskManager.finish(task);
assignableTaskManager.finish(task); // } else {
} else { // assignableTaskManager.finish(task, comment);
assignableTaskManager.finish(task, comment); // }
} //
// return String.format("redirect:%s", returnUrl);
return String.format("redirect:%s", returnUrl); // }
} //
// @POST
@POST // @Path(
@Path( // "/{documentPath:(.+)?}/@workflow/@applyAlternative/{workflowIdentifier}")
"/{documentPath:(.+)?}/@workflow/@applyAlternative/{workflowIdentifier}") // @AuthorizationRequired
@AuthorizationRequired // @Transactional(Transactional.TxType.REQUIRED)
@Transactional(Transactional.TxType.REQUIRED) // public String applyAlternateWorkflow(
public String applyAlternateWorkflow( // @PathParam("sectionIdentifider") final String sectionIdentifier,
@PathParam("sectionIdentifider") final String sectionIdentifier, // @PathParam("documentPath") final String documentPath,
@PathParam("documentPath") final String documentPath, // @FormParam("newWorkflowUuid") final String newWorkflowUuid,
@FormParam("newWorkflowUuid") final String newWorkflowUuid, // @FormParam("returnUrl") final String returnUrl
@FormParam("returnUrl") final String returnUrl // ) {
) { // final Optional<ContentSection> sectionResult = sectionsUi
final Optional<ContentSection> sectionResult = sectionsUi // .findContentSection(sectionIdentifier);
.findContentSection(sectionIdentifier); // if (!sectionResult.isPresent()) {
if (!sectionResult.isPresent()) { // return sectionsUi.showContentSectionNotFound(sectionIdentifier);
return sectionsUi.showContentSectionNotFound(sectionIdentifier); // }
} // final ContentSection section = sectionResult.get();
final ContentSection section = sectionResult.get(); //
// final Optional<ContentItem> itemResult = itemRepo
final Optional<ContentItem> itemResult = itemRepo // .findByPath(section, documentPath);
.findByPath(section, documentPath); // if (!itemResult.isPresent()) {
if (!itemResult.isPresent()) { // return documentUi.showDocumentNotFound(section, documentPath);
return showDocumentNotFound(section, documentPath); // }
} // final ContentItem item = itemResult.get();
final ContentItem item = itemResult.get(); // selectedDocumentModel.setContentItem(item);
selectedDocumentModel.setContentItem(item); //
// if (!permissionChecker.isPermitted(
if (!permissionChecker.isPermitted( // ItemPrivileges.APPLY_ALTERNATE_WORKFLOW, item)) {
ItemPrivileges.APPLY_ALTERNATE_WORKFLOW, item)) { // return sectionsUi.showAccessDenied(
return sectionsUi.showAccessDenied( // "sectionIdentifier", sectionIdentifier,
"sectionIdentifier", sectionIdentifier, // "documentPath", documentPath
"documentPath", documentPath // );
); // }
} //
// final Optional<Workflow> workflowResult = section
final Optional<Workflow> workflowResult = section // .getWorkflowTemplates()
.getWorkflowTemplates() // .stream()
.stream() // .filter(template -> template.getUuid().equals(newWorkflowUuid))
.filter(template -> template.getUuid().equals(newWorkflowUuid)) // .findAny();
.findAny(); // if (!workflowResult.isPresent()) {
if (!workflowResult.isPresent()) { // models.put("section", section.getLabel());
models.put("section", section.getLabel()); // models.put("workflowUuid", newWorkflowUuid);
models.put("workflowUuid", newWorkflowUuid); // return "org/librecms/ui/contentsection/workflow-not-found.xhtml";
return "org/librecms/ui/contentsection/workflow-not-found.xhtml"; // }
} //
// workflowManager.createWorkflow(workflowResult.get(), item);
workflowManager.createWorkflow(workflowResult.get(), item); // return String.format("redirect:%s", returnUrl);
return String.format("redirect:%s", returnUrl); // }
} //
// @POST
@POST // @Path("/{documentPath:(.+)?}/@workflow/@restart")
@Path("/{documentPath:(.+)?}/@workflow/@restart") // @AuthorizationRequired
@AuthorizationRequired // @Transactional(Transactional.TxType.REQUIRED)
@Transactional(Transactional.TxType.REQUIRED) // public String restartWorkflow(
public String restartWorkflow( // @PathParam("sectionIdentifider") final String sectionIdentifier,
@PathParam("sectionIdentifider") final String sectionIdentifier, // @PathParam("documentPath") final String documentPath,
@PathParam("documentPath") final String documentPath, // @FormParam("returnUrl") final String returnUrl
@FormParam("returnUrl") final String returnUrl // ) {
) { // final Optional<ContentSection> sectionResult = sectionsUi
final Optional<ContentSection> sectionResult = sectionsUi // .findContentSection(sectionIdentifier);
.findContentSection(sectionIdentifier); // if (!sectionResult.isPresent()) {
if (!sectionResult.isPresent()) { // return sectionsUi.showContentSectionNotFound(sectionIdentifier);
return sectionsUi.showContentSectionNotFound(sectionIdentifier); // }
} // final ContentSection section = sectionResult.get();
final ContentSection section = sectionResult.get(); //
// final Optional<ContentItem> itemResult = itemRepo
final Optional<ContentItem> itemResult = itemRepo // .findByPath(section, documentPath);
.findByPath(section, documentPath); // if (!itemResult.isPresent()) {
if (!itemResult.isPresent()) { // return documentUi.showDocumentNotFound(section, documentPath);
return showDocumentNotFound(section, documentPath); // }
} // final ContentItem item = itemResult.get();
final ContentItem item = itemResult.get(); // selectedDocumentModel.setContentItem(item);
selectedDocumentModel.setContentItem(item); //
// if (!permissionChecker.isPermitted(
if (!permissionChecker.isPermitted( // ItemPrivileges.APPLY_ALTERNATE_WORKFLOW, item)) {
ItemPrivileges.APPLY_ALTERNATE_WORKFLOW, item)) { // return sectionsUi.showAccessDenied(
return sectionsUi.showAccessDenied( // "sectionIdentifier", sectionIdentifier,
"sectionIdentifier", sectionIdentifier, // "documentPath", documentPath
"documentPath", documentPath // );
); // }
} //
// if (item.getWorkflow() != null) {
if (item.getWorkflow() != null) { // workflowManager.start(item.getWorkflow());
workflowManager.start(item.getWorkflow()); // }
} //
// return String.format("redirect:%s", returnUrl);
return String.format("redirect:%s", returnUrl); // }
} //
// private Optional<AssignableTask> findTask(
private Optional<AssignableTask> findTask( // final ContentItem item,
final ContentItem item, // final String taskIdentifier
final String taskIdentifier // ) {
) { // final Workflow workflow = item.getWorkflow();
final Workflow workflow = item.getWorkflow(); // if (workflow == null) {
if (workflow == null) { // return Optional.empty();
return Optional.empty(); // }
} //
// final Identifier identifier = identifierParser.parseIdentifier(
final Identifier identifier = identifierParser.parseIdentifier( // taskIdentifier
taskIdentifier // );
); // switch (identifier.getType()) {
switch (identifier.getType()) { // case ID:
case ID: // return workflow
return workflow // .getTasks()
.getTasks() // .stream()
.stream() // .filter(task -> task instanceof AssignableTask)
.filter(task -> task instanceof AssignableTask) // .map(task -> (AssignableTask) task)
.map(task -> (AssignableTask) task) // .filter(
.filter( // task -> task.getTaskId() == Long
task -> task.getTaskId() == Long // .parseLong(identifier.getIdentifier())
.parseLong(identifier.getIdentifier()) // ).findAny();
).findAny(); // default:
default: // return workflow
return workflow // .getTasks()
.getTasks() // .stream()
.stream() // .filter(task -> task instanceof AssignableTask)
.filter(task -> task instanceof AssignableTask) // .map(task -> (AssignableTask) task)
.map(task -> (AssignableTask) task) // .filter(
.filter( // task -> task.getUuid().equals(
task -> task.getUuid().equals( // identifier.getIdentifier()
identifier.getIdentifier() // )
) // ).findAny();
).findAny(); // }
} // }
}
private LifecycleListEntry buildLifecycleListEntry( private LifecycleListEntry buildLifecycleListEntry(
final LifecycleDefinition definition final LifecycleDefinition definition
) { ) {
@ -851,25 +739,6 @@ public class DocumentController {
return entry; return entry;
} }
private String showDocumentNotFound(
final ContentSection section, final String documentPath
) {
models.put("section", section.getLabel());
models.put("documentPath", documentPath);
return "org/librecms/ui/contentsection/document-not-found.xhtml";
}
private String showTaskNotFound(
final ContentSection section,
final String documentPath,
final String taskIdentifier
) {
models.put("section", section.getLabel());
models.put("documentPath", documentPath);
models.put("taskIdentifier", taskIdentifier);
return "org/librecms/ui/contentsection/task-not-found.xhtml";
}
private static class CreateDocumentOfTypeLiteral private static class CreateDocumentOfTypeLiteral
extends AnnotationLiteral<CreatesDocumentOfType> extends AnnotationLiteral<CreatesDocumentOfType>
implements CreatesDocumentOfType { implements CreatesDocumentOfType {

View File

@ -0,0 +1,140 @@
/*
* To change this license header, choose License Headers in Project Properties.
* To change this template file, choose Tools | Templates
* and open the template in the editor.
*/
package org.librecms.ui.contentsections.documents;
import org.libreccm.security.AuthorizationRequired;
import org.libreccm.security.PermissionChecker;
import org.librecms.contentsection.ContentItem;
import org.librecms.contentsection.ContentItemRepository;
import org.librecms.contentsection.ContentSection;
import org.librecms.contentsection.privileges.ItemPrivileges;
import org.librecms.lifecycle.Phase;
import org.librecms.lifecycle.PhaseRepository;
import org.librecms.ui.contentsections.ContentSectionsUi;
import java.time.LocalDateTime;
import java.time.ZoneId;
import java.time.ZoneOffset;
import java.time.format.DateTimeFormatter;
import java.util.Date;
import java.util.Optional;
import javax.enterprise.context.RequestScoped;
import javax.inject.Inject;
import javax.inject.Named;
import javax.mvc.Controller;
import javax.mvc.Models;
import javax.transaction.Transactional;
import javax.ws.rs.FormParam;
import javax.ws.rs.POST;
import javax.ws.rs.Path;
import javax.ws.rs.PathParam;
/**
*
* @author <a href="mailto:jens.pelzetter@googlemail.com">Jens Pelzetter</a>
*/
@RequestScoped
@Path("/{sectionIdentifier}/documents/{documentPath:(.+)?}/@lifecycle/")
@Controller
public class DocumentLifecyclesController {
@Inject
private ContentItemRepository itemRepo;
@Inject
private ContentSectionsUi sectionsUi;
@Inject
private DocumentUi documentUi;
@Inject
private Models models;
@Inject
private PermissionChecker permissionChecker;
@Inject
private PhaseRepository phaseRepository;
@POST
@Path("/phases/{phaseId}")
@AuthorizationRequired
@Transactional(Transactional.TxType.REQUIRED)
public String updatePhaseDates(
@PathParam("sectionIdentifider") final String sectionIdentifier,
@PathParam("documentPath") final String documentPath,
@PathParam("phaseId") final long phaseId,
@FormParam("startDate") final String startDateParam,
@FormParam("endDate") final String endDateParam
) {
final Optional<ContentSection> sectionResult = sectionsUi
.findContentSection(sectionIdentifier);
if (!sectionResult.isPresent()) {
return sectionsUi.showContentSectionNotFound(sectionIdentifier);
}
final ContentSection section = sectionResult.get();
final Optional<ContentItem> itemResult = itemRepo
.findByPath(section, documentPath);
if (!itemResult.isPresent()) {
return documentUi.showDocumentNotFound(section, documentPath);
}
final ContentItem item = itemResult.get();
if (!permissionChecker.isPermitted(ItemPrivileges.PUBLISH, item)) {
return sectionsUi.showAccessDenied(
"sectionIdentifier", sectionIdentifier,
"documentPath", documentPath
);
}
if (item.getLifecycle() != null) {
final Optional<Phase> phaseResult = item
.getLifecycle()
.getPhases()
.stream()
.filter(phase -> phase.getPhaseId() == phaseId)
.findAny();
if (!phaseResult.isPresent()) {
models.put("section", section.getLabel());
models.put("phaseId", phaseId);
return "org/librecms/ui/contentsection/phase-not-found.xhtml";
}
final Phase phase = phaseResult.get();
final DateTimeFormatter dateTimeFormatter
= DateTimeFormatter.ISO_DATE_TIME
.withZone(ZoneId.systemDefault());
final LocalDateTime startLocalDateTime = LocalDateTime
.parse(startDateParam, dateTimeFormatter);
phase.setStartDateTime(
Date.from(
startLocalDateTime.toInstant(
ZoneOffset.from(startLocalDateTime)
)
)
);
final LocalDateTime endLocalDateTime = LocalDateTime
.parse(endDateParam, dateTimeFormatter);
phase.setEndDateTime(
Date.from(
endLocalDateTime.toInstant(
ZoneOffset.from(endLocalDateTime)
)
)
);
phaseRepository.save(phase);
}
return String.format(
"redirect:/%s/documents/%s/@publish",
sectionIdentifier,
documentPath
);
}
}

View File

@ -0,0 +1,34 @@
/*
* To change this license header, choose License Headers in Project Properties.
* To change this template file, choose Tools | Templates
* and open the template in the editor.
*/
package org.librecms.ui.contentsections.documents;
import org.librecms.contentsection.ContentSection;
import javax.enterprise.context.RequestScoped;
import javax.inject.Inject;
import javax.mvc.Models;
/**
*
* @author <a href="mailto:jens.pelzetter@googlemail.com">Jens Pelzetter</a>
*/
@RequestScoped
class DocumentUi {
@Inject
private Models models;
public String showDocumentNotFound(
final ContentSection section, final String documentPath
) {
models.put("section", section.getLabel());
models.put("documentPath", documentPath);
return "org/librecms/ui/contentsection/document-not-found.xhtml";
}
}

View File

@ -0,0 +1,325 @@
/*
* To change this license header, choose License Headers in Project Properties.
* To change this template file, choose Tools | Templates
* and open the template in the editor.
*/
package org.librecms.ui.contentsections.documents;
import org.libreccm.api.Identifier;
import org.libreccm.api.IdentifierParser;
import org.libreccm.security.AuthorizationRequired;
import org.libreccm.security.PermissionChecker;
import org.libreccm.workflow.AssignableTask;
import org.libreccm.workflow.AssignableTaskManager;
import org.libreccm.workflow.Workflow;
import org.libreccm.workflow.WorkflowManager;
import org.librecms.contentsection.ContentItem;
import org.librecms.contentsection.ContentItemRepository;
import org.librecms.contentsection.ContentSection;
import org.librecms.contentsection.privileges.ItemPrivileges;
import org.librecms.ui.contentsections.ContentSectionsUi;
import java.util.Optional;
import javax.enterprise.context.RequestScoped;
import javax.inject.Inject;
import javax.mvc.Controller;
import javax.mvc.Models;
import javax.transaction.Transactional;
import javax.ws.rs.DefaultValue;
import javax.ws.rs.FormParam;
import javax.ws.rs.POST;
import javax.ws.rs.Path;
import javax.ws.rs.PathParam;
/**
*
* @author <a href="mailto:jens.pelzetter@googlemail.com">Jens Pelzetter</a>
*/
@RequestScoped
@Path("/{sectionIdentifier}/documents/{documentPath:(.+)?}/@workflow/")
@Controller
public class DocumentWorkflowController {
@Inject
private AssignableTaskManager assignableTaskManager;
@Inject
private ContentItemRepository itemRepo;
@Inject
private ContentSectionsUi sectionsUi;
@Inject
private DocumentUi documentUi;
@Inject
private IdentifierParser identifierParser;
@Inject
private PermissionChecker permissionChecker;
@Inject
private SelectedDocumentModel selectedDocumentModel;
@Inject
private Models models;
@Inject
private WorkflowManager workflowManager;
@POST
@Path("/tasks/${taskIdentifier}/@lock")
@AuthorizationRequired
@Transactional(Transactional.TxType.REQUIRED)
public String lockTask(
@PathParam("sectionIdentifider") final String sectionIdentifier,
@PathParam("documentPath") final String documentPath,
@PathParam("taskIdentifier") final String taskIdentifier,
@FormParam("returnUrl") final String returnUrl
) {
final Optional<ContentSection> sectionResult = sectionsUi
.findContentSection(sectionIdentifier);
if (!sectionResult.isPresent()) {
return sectionsUi.showContentSectionNotFound(sectionIdentifier);
}
final ContentSection section = sectionResult.get();
final Optional<ContentItem> itemResult = itemRepo
.findByPath(section, documentPath);
if (!itemResult.isPresent()) {
return documentUi.showDocumentNotFound(section, documentPath);
}
final ContentItem item = itemResult.get();
selectedDocumentModel.setContentItem(item);
final Optional<AssignableTask> taskResult = findTask(
item, taskIdentifier
);
if (!taskResult.isPresent()) {
return showTaskNotFound(section, documentPath, taskIdentifier);
}
final AssignableTask task = taskResult.get();
assignableTaskManager.lockTask(task);
return String.format("redirect:%s", returnUrl);
}
@POST
@Path("/tasks/${taskIdentifier}/@unlock")
@AuthorizationRequired
@Transactional(Transactional.TxType.REQUIRED)
public String unlockTask(
@PathParam("sectionIdentifider") final String sectionIdentifier,
@PathParam("documentPath") final String documentPath,
@PathParam("taskIdentifier") final String taskIdentifier,
@FormParam("returnUrl") final String returnUrl
) {
final Optional<ContentSection> sectionResult = sectionsUi
.findContentSection(sectionIdentifier);
if (!sectionResult.isPresent()) {
return sectionsUi.showContentSectionNotFound(sectionIdentifier);
}
final ContentSection section = sectionResult.get();
final Optional<ContentItem> itemResult = itemRepo
.findByPath(section, documentPath);
if (!itemResult.isPresent()) {
return documentUi.showDocumentNotFound(section, documentPath);
}
final ContentItem item = itemResult.get();
selectedDocumentModel.setContentItem(item);
final Optional<AssignableTask> taskResult = findTask(
item, taskIdentifier
);
if (!taskResult.isPresent()) {
return showTaskNotFound(section, documentPath, taskIdentifier);
}
final AssignableTask task = taskResult.get();
assignableTaskManager.unlockTask(task);
return String.format("redirect:%s", returnUrl);
}
@POST
@Path("/tasks/${taskIdentifier}/@finish")
@AuthorizationRequired
@Transactional(Transactional.TxType.REQUIRED)
public String finishTask(
@PathParam("sectionIdentifider") final String sectionIdentifier,
@PathParam("documentPath") final String documentPath,
@PathParam("taskIdentifier") final String taskIdentifier,
@FormParam("comment") @DefaultValue("") final String comment,
@FormParam("returnUrl") final String returnUrl
) {
final Optional<ContentSection> sectionResult = sectionsUi
.findContentSection(sectionIdentifier);
if (!sectionResult.isPresent()) {
return sectionsUi.showContentSectionNotFound(sectionIdentifier);
}
final ContentSection section = sectionResult.get();
final Optional<ContentItem> itemResult = itemRepo
.findByPath(section, documentPath);
if (!itemResult.isPresent()) {
return documentUi.showDocumentNotFound(section, documentPath);
}
final ContentItem item = itemResult.get();
selectedDocumentModel.setContentItem(item);
final Optional<AssignableTask> taskResult = findTask(
item, taskIdentifier
);
if (!taskResult.isPresent()) {
return showTaskNotFound(section, documentPath, taskIdentifier);
}
final AssignableTask task = taskResult.get();
if (comment.isEmpty()) {
assignableTaskManager.finish(task);
} else {
assignableTaskManager.finish(task, comment);
}
return String.format("redirect:%s", returnUrl);
}
@POST
@Path("@workflow/@applyAlternative/{workflowIdentifier}")
@AuthorizationRequired
@Transactional(Transactional.TxType.REQUIRED)
public String applyAlternateWorkflow(
@PathParam("sectionIdentifider") final String sectionIdentifier,
@PathParam("documentPath") final String documentPath,
@FormParam("newWorkflowUuid") final String newWorkflowUuid,
@FormParam("returnUrl") final String returnUrl
) {
final Optional<ContentSection> sectionResult = sectionsUi
.findContentSection(sectionIdentifier);
if (!sectionResult.isPresent()) {
return sectionsUi.showContentSectionNotFound(sectionIdentifier);
}
final ContentSection section = sectionResult.get();
final Optional<ContentItem> itemResult = itemRepo
.findByPath(section, documentPath);
if (!itemResult.isPresent()) {
return documentUi.showDocumentNotFound(section, documentPath);
}
final ContentItem item = itemResult.get();
selectedDocumentModel.setContentItem(item);
if (!permissionChecker.isPermitted(
ItemPrivileges.APPLY_ALTERNATE_WORKFLOW, item)) {
return sectionsUi.showAccessDenied(
"sectionIdentifier", sectionIdentifier,
"documentPath", documentPath
);
}
final Optional<Workflow> workflowResult = section
.getWorkflowTemplates()
.stream()
.filter(template -> template.getUuid().equals(newWorkflowUuid))
.findAny();
if (!workflowResult.isPresent()) {
models.put("section", section.getLabel());
models.put("workflowUuid", newWorkflowUuid);
return "org/librecms/ui/contentsection/workflow-not-found.xhtml";
}
workflowManager.createWorkflow(workflowResult.get(), item);
return String.format("redirect:%s", returnUrl);
}
@POST
@Path("/@restart")
@AuthorizationRequired
@Transactional(Transactional.TxType.REQUIRED)
public String restartWorkflow(
@PathParam("sectionIdentifider") final String sectionIdentifier,
@PathParam("documentPath") final String documentPath,
@FormParam("returnUrl") final String returnUrl
) {
final Optional<ContentSection> sectionResult = sectionsUi
.findContentSection(sectionIdentifier);
if (!sectionResult.isPresent()) {
return sectionsUi.showContentSectionNotFound(sectionIdentifier);
}
final ContentSection section = sectionResult.get();
final Optional<ContentItem> itemResult = itemRepo
.findByPath(section, documentPath);
if (!itemResult.isPresent()) {
return documentUi.showDocumentNotFound(section, documentPath);
}
final ContentItem item = itemResult.get();
selectedDocumentModel.setContentItem(item);
if (!permissionChecker.isPermitted(
ItemPrivileges.APPLY_ALTERNATE_WORKFLOW, item)) {
return sectionsUi.showAccessDenied(
"sectionIdentifier", sectionIdentifier,
"documentPath", documentPath
);
}
if (item.getWorkflow() != null) {
workflowManager.start(item.getWorkflow());
}
return String.format("redirect:%s", returnUrl);
}
private Optional<AssignableTask> findTask(
final ContentItem item,
final String taskIdentifier
) {
final Workflow workflow = item.getWorkflow();
if (workflow == null) {
return Optional.empty();
}
final Identifier identifier = identifierParser.parseIdentifier(
taskIdentifier
);
switch (identifier.getType()) {
case ID:
return workflow
.getTasks()
.stream()
.filter(task -> task instanceof AssignableTask)
.map(task -> (AssignableTask) task)
.filter(
task -> task.getTaskId() == Long
.parseLong(identifier.getIdentifier())
).findAny();
default:
return workflow
.getTasks()
.stream()
.filter(task -> task instanceof AssignableTask)
.map(task -> (AssignableTask) task)
.filter(
task -> task.getUuid().equals(
identifier.getIdentifier()
)
).findAny();
}
}
private String showTaskNotFound(
final ContentSection section,
final String documentPath,
final String taskIdentifier
) {
models.put("section", section.getLabel());
models.put("documentPath", documentPath);
models.put("taskIdentifier", taskIdentifier);
return "org/librecms/ui/contentsection/task-not-found.xhtml";
}
}

View File

@ -0,0 +1,61 @@
/*
* To change this license header, choose License Headers in Project Properties.
* To change this template file, choose Tools | Templates
* and open the template in the editor.
*/
package org.librecms.ui.contentsections.documents;
import org.junit.After;
import org.junit.AfterClass;
import org.junit.Before;
import org.junit.BeforeClass;
import org.junit.Test;
import static org.junit.Assert.*;
/**
*
* @author <a href="mailto:jens.pelzetter@googlemail.com">Jens Pelzetter</a>
*/
public class LifecyclesControllerTest {
public LifecyclesControllerTest() {
}
@BeforeClass
public static void setUpClass() {
}
@AfterClass
public static void tearDownClass() {
}
@Before
public void setUp() {
}
@After
public void tearDown() {
}
/**
* Test of updatePhaseDates method, of class DocumentLifecyclesController.
*/
@Test
public void testUpdatePhaseDates() {
System.out.println("updatePhaseDates");
String sectionIdentifier = "";
String documentPath = "";
long phaseId = 0L;
String startDateParam = "";
String endDateParam = "";
DocumentLifecyclesController instance = new DocumentLifecyclesController();
String expResult = "";
String result
= instance.updatePhaseDates(sectionIdentifier, documentPath, phaseId,
startDateParam, endDateParam);
assertEquals(expResult, result);
// TODO review the generated test code and remove the default call to fail.
fail("The test case is a prototype.");
}
}

View File

@ -16,7 +16,7 @@
"integrity": "sha512-mOrlCEdwX3seT3n0AXNt4KNPAZZxcsABUHwBgFXOt+nvFUXkxCAO6UBJHPrDxWEa2KDMil86355fjo8jbZ+K0Q==", "integrity": "sha512-mOrlCEdwX3seT3n0AXNt4KNPAZZxcsABUHwBgFXOt+nvFUXkxCAO6UBJHPrDxWEa2KDMil86355fjo8jbZ+K0Q==",
"dev": true, "dev": true,
"requires": { "requires": {
"@types/react": "16.4.11" "@types/react": "*"
} }
}, },
"@types/react": { "@types/react": {
@ -25,8 +25,8 @@
"integrity": "sha512-1DQnmwO8u8N3ucvRX2ZLDEjQ2VctkAvL/rpbm2ev4uaZA0z4ysU+I0tk+K8ZLblC6p7MCgFyF+cQlSNIPUHzeQ==", "integrity": "sha512-1DQnmwO8u8N3ucvRX2ZLDEjQ2VctkAvL/rpbm2ev4uaZA0z4ysU+I0tk+K8ZLblC6p7MCgFyF+cQlSNIPUHzeQ==",
"dev": true, "dev": true,
"requires": { "requires": {
"@types/prop-types": "15.5.5", "@types/prop-types": "*",
"csstype": "2.5.6" "csstype": "^2.2.0"
} }
}, },
"@types/react-dom": { "@types/react-dom": {
@ -35,8 +35,8 @@
"integrity": "sha512-vaq4vMaJOaNgFff1t3LnHYr6vRa09vRspMkmLdXtFZmO1fwDI2snP+dpOkwrtlU8UC8qsqemCu4RmVM2OLq/fA==", "integrity": "sha512-vaq4vMaJOaNgFff1t3LnHYr6vRa09vRspMkmLdXtFZmO1fwDI2snP+dpOkwrtlU8UC8qsqemCu4RmVM2OLq/fA==",
"dev": true, "dev": true,
"requires": { "requires": {
"@types/node": "10.7.1", "@types/node": "*",
"@types/react": "16.4.11" "@types/react": "*"
} }
}, },
"@types/react-modal": { "@types/react-modal": {
@ -45,7 +45,7 @@
"integrity": "sha512-EhRC3xjsehC0e8OKz/NmEyjc/ggHxVj4rNu8p/AfSohbn5NHY0V58fj0OZgQPZzXY42v+rLxiO7PL1uOeBfimg==", "integrity": "sha512-EhRC3xjsehC0e8OKz/NmEyjc/ggHxVj4rNu8p/AfSohbn5NHY0V58fj0OZgQPZzXY42v+rLxiO7PL1uOeBfimg==",
"dev": true, "dev": true,
"requires": { "requires": {
"@types/react": "16.4.11" "@types/react": "*"
} }
}, },
"ansi-regex": { "ansi-regex": {
@ -66,7 +66,7 @@
"integrity": "sha512-o5Roy6tNG4SL/FOkCAN6RzjiakZS25RLYFrcMttJqbdd8BWrnA+fGz57iN5Pb06pvBGvl5gQ0B48dJlslXvoTg==", "integrity": "sha512-o5Roy6tNG4SL/FOkCAN6RzjiakZS25RLYFrcMttJqbdd8BWrnA+fGz57iN5Pb06pvBGvl5gQ0B48dJlslXvoTg==",
"dev": true, "dev": true,
"requires": { "requires": {
"sprintf-js": "1.0.3" "sprintf-js": "~1.0.2"
} }
}, },
"asap": { "asap": {
@ -80,9 +80,9 @@
"integrity": "sha1-Y/1D99weO7fONZR9uP42mj9Yx0s=", "integrity": "sha1-Y/1D99weO7fONZR9uP42mj9Yx0s=",
"dev": true, "dev": true,
"requires": { "requires": {
"chalk": "1.1.3", "chalk": "^1.1.3",
"esutils": "2.0.2", "esutils": "^2.0.2",
"js-tokens": "3.0.2" "js-tokens": "^3.0.2"
}, },
"dependencies": { "dependencies": {
"chalk": { "chalk": {
@ -91,11 +91,11 @@
"integrity": "sha1-qBFcVeSnAv5NFQq9OHKCKn4J/Jg=", "integrity": "sha1-qBFcVeSnAv5NFQq9OHKCKn4J/Jg=",
"dev": true, "dev": true,
"requires": { "requires": {
"ansi-styles": "2.2.1", "ansi-styles": "^2.2.1",
"escape-string-regexp": "1.0.5", "escape-string-regexp": "^1.0.2",
"has-ansi": "2.0.0", "has-ansi": "^2.0.0",
"strip-ansi": "3.0.1", "strip-ansi": "^3.0.0",
"supports-color": "2.0.0" "supports-color": "^2.0.0"
} }
} }
} }
@ -112,7 +112,7 @@
"integrity": "sha512-iCuPHDFgrHX7H2vEI/5xpz07zSHB00TpugqhmYtVmMO6518mCuRMoOYFldEBl0g187ufozdaHgWKcYFb61qGiA==", "integrity": "sha512-iCuPHDFgrHX7H2vEI/5xpz07zSHB00TpugqhmYtVmMO6518mCuRMoOYFldEBl0g187ufozdaHgWKcYFb61qGiA==",
"dev": true, "dev": true,
"requires": { "requires": {
"balanced-match": "1.0.0", "balanced-match": "^1.0.0",
"concat-map": "0.0.1" "concat-map": "0.0.1"
} }
}, },
@ -128,9 +128,9 @@
"integrity": "sha512-ObN6h1v2fTJSmUXoS3nMQ92LbDK9be4TV+6G+omQlGJFdcUX5heKi1LZ1YnRMIgwTLEj3E24bT6tYni50rlCfQ==", "integrity": "sha512-ObN6h1v2fTJSmUXoS3nMQ92LbDK9be4TV+6G+omQlGJFdcUX5heKi1LZ1YnRMIgwTLEj3E24bT6tYni50rlCfQ==",
"dev": true, "dev": true,
"requires": { "requires": {
"ansi-styles": "3.2.1", "ansi-styles": "^3.2.1",
"escape-string-regexp": "1.0.5", "escape-string-regexp": "^1.0.5",
"supports-color": "5.4.0" "supports-color": "^5.3.0"
}, },
"dependencies": { "dependencies": {
"ansi-styles": { "ansi-styles": {
@ -139,7 +139,7 @@
"integrity": "sha512-VT0ZI6kZRdTh8YyJw3SMbYm/u+NqfsAxEpWO0Pf9sq8/e94WxxOpPKx9FR1FlyCtOVDNOQ+8ntlqFxiRc+r5qA==", "integrity": "sha512-VT0ZI6kZRdTh8YyJw3SMbYm/u+NqfsAxEpWO0Pf9sq8/e94WxxOpPKx9FR1FlyCtOVDNOQ+8ntlqFxiRc+r5qA==",
"dev": true, "dev": true,
"requires": { "requires": {
"color-convert": "1.9.2" "color-convert": "^1.9.0"
} }
}, },
"supports-color": { "supports-color": {
@ -148,7 +148,7 @@
"integrity": "sha512-zjaXglF5nnWpsq470jSv6P9DwPvgLkuapYmfDm3JWOm0vkNTVF2tI4UrN2r6jH1qM/uc/WtxYY1hYoA2dOKj5w==", "integrity": "sha512-zjaXglF5nnWpsq470jSv6P9DwPvgLkuapYmfDm3JWOm0vkNTVF2tI4UrN2r6jH1qM/uc/WtxYY1hYoA2dOKj5w==",
"dev": true, "dev": true,
"requires": { "requires": {
"has-flag": "3.0.0" "has-flag": "^3.0.0"
} }
} }
} }
@ -202,7 +202,7 @@
"resolved": "https://registry.npmjs.org/encoding/-/encoding-0.1.12.tgz", "resolved": "https://registry.npmjs.org/encoding/-/encoding-0.1.12.tgz",
"integrity": "sha1-U4tm8+5izRq1HsMjgp0flIDHS+s=", "integrity": "sha1-U4tm8+5izRq1HsMjgp0flIDHS+s=",
"requires": { "requires": {
"iconv-lite": "0.4.23" "iconv-lite": "~0.4.13"
} }
}, },
"escape-string-regexp": { "escape-string-regexp": {
@ -233,13 +233,13 @@
"resolved": "https://registry.npmjs.org/fbjs/-/fbjs-0.8.16.tgz", "resolved": "https://registry.npmjs.org/fbjs/-/fbjs-0.8.16.tgz",
"integrity": "sha1-XmdDL1UNxBtXK/VYR7ispk5TN9s=", "integrity": "sha1-XmdDL1UNxBtXK/VYR7ispk5TN9s=",
"requires": { "requires": {
"core-js": "1.2.7", "core-js": "^1.0.0",
"isomorphic-fetch": "2.2.1", "isomorphic-fetch": "^2.1.1",
"loose-envify": "1.3.1", "loose-envify": "^1.0.0",
"object-assign": "4.1.1", "object-assign": "^4.1.0",
"promise": "7.3.1", "promise": "^7.1.1",
"setimmediate": "1.0.5", "setimmediate": "^1.0.5",
"ua-parser-js": "0.7.18" "ua-parser-js": "^0.7.9"
} }
}, },
"fs.realpath": { "fs.realpath": {
@ -254,12 +254,12 @@
"integrity": "sha512-MJTUg1kjuLeQCJ+ccE4Vpa6kKVXkPYJ2mOCQyUuKLcLQsdrMCpBPUi8qVE6+YuaJkozeA9NusTAw3hLr8Xe5EQ==", "integrity": "sha512-MJTUg1kjuLeQCJ+ccE4Vpa6kKVXkPYJ2mOCQyUuKLcLQsdrMCpBPUi8qVE6+YuaJkozeA9NusTAw3hLr8Xe5EQ==",
"dev": true, "dev": true,
"requires": { "requires": {
"fs.realpath": "1.0.0", "fs.realpath": "^1.0.0",
"inflight": "1.0.6", "inflight": "^1.0.4",
"inherits": "2.0.3", "inherits": "2",
"minimatch": "3.0.4", "minimatch": "^3.0.4",
"once": "1.4.0", "once": "^1.3.0",
"path-is-absolute": "1.0.1" "path-is-absolute": "^1.0.0"
} }
}, },
"has-ansi": { "has-ansi": {
@ -268,7 +268,7 @@
"integrity": "sha1-NPUEnOHs3ysGSa8+8k5F7TVBbZE=", "integrity": "sha1-NPUEnOHs3ysGSa8+8k5F7TVBbZE=",
"dev": true, "dev": true,
"requires": { "requires": {
"ansi-regex": "2.1.1" "ansi-regex": "^2.0.0"
} }
}, },
"has-flag": { "has-flag": {
@ -287,7 +287,7 @@
"resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.4.23.tgz", "resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.4.23.tgz",
"integrity": "sha512-neyTUVFtahjf0mB3dZT77u+8O0QB89jFdnBkd5P1JgYPbPaia3gXXOVL2fq8VyU2gMMD7SaN7QukTB/pmXYvDA==", "integrity": "sha512-neyTUVFtahjf0mB3dZT77u+8O0QB89jFdnBkd5P1JgYPbPaia3gXXOVL2fq8VyU2gMMD7SaN7QukTB/pmXYvDA==",
"requires": { "requires": {
"safer-buffer": "2.1.2" "safer-buffer": ">= 2.1.2 < 3"
} }
}, },
"inflight": { "inflight": {
@ -296,8 +296,8 @@
"integrity": "sha1-Sb1jMdfQLQwJvJEKEHW6gWW1bfk=", "integrity": "sha1-Sb1jMdfQLQwJvJEKEHW6gWW1bfk=",
"dev": true, "dev": true,
"requires": { "requires": {
"once": "1.4.0", "once": "^1.3.0",
"wrappy": "1.0.2" "wrappy": "1"
} }
}, },
"inherits": { "inherits": {
@ -311,7 +311,7 @@
"resolved": "https://registry.npmjs.org/invariant/-/invariant-2.2.4.tgz", "resolved": "https://registry.npmjs.org/invariant/-/invariant-2.2.4.tgz",
"integrity": "sha512-phJfQVBuaJM5raOpJjSfkiD6BpbCE4Ns//LaXl6wGYtUBY83nWS6Rf9tXm2e8VaK60JEjYldbPif/A2B1C2gNA==", "integrity": "sha512-phJfQVBuaJM5raOpJjSfkiD6BpbCE4Ns//LaXl6wGYtUBY83nWS6Rf9tXm2e8VaK60JEjYldbPif/A2B1C2gNA==",
"requires": { "requires": {
"loose-envify": "1.3.1" "loose-envify": "^1.0.0"
} }
}, },
"is-stream": { "is-stream": {
@ -324,8 +324,8 @@
"resolved": "https://registry.npmjs.org/isomorphic-fetch/-/isomorphic-fetch-2.2.1.tgz", "resolved": "https://registry.npmjs.org/isomorphic-fetch/-/isomorphic-fetch-2.2.1.tgz",
"integrity": "sha1-YRrhrPFPXoH3KVB0coGf6XM1WKk=", "integrity": "sha1-YRrhrPFPXoH3KVB0coGf6XM1WKk=",
"requires": { "requires": {
"node-fetch": "1.7.3", "node-fetch": "^1.0.1",
"whatwg-fetch": "2.0.4" "whatwg-fetch": ">=0.10.0"
} }
}, },
"js-tokens": { "js-tokens": {
@ -339,8 +339,8 @@
"integrity": "sha512-PIt2cnwmPfL4hKNwqeiuz4bKfnzHTBv6HyVgjahA6mPLwPDzjDWrplJBMjHUFxku/N3FlmrbyPclad+I+4mJ3A==", "integrity": "sha512-PIt2cnwmPfL4hKNwqeiuz4bKfnzHTBv6HyVgjahA6mPLwPDzjDWrplJBMjHUFxku/N3FlmrbyPclad+I+4mJ3A==",
"dev": true, "dev": true,
"requires": { "requires": {
"argparse": "1.0.10", "argparse": "^1.0.7",
"esprima": "4.0.1" "esprima": "^4.0.0"
} }
}, },
"lodash": { "lodash": {
@ -358,7 +358,7 @@
"resolved": "https://registry.npmjs.org/loose-envify/-/loose-envify-1.3.1.tgz", "resolved": "https://registry.npmjs.org/loose-envify/-/loose-envify-1.3.1.tgz",
"integrity": "sha1-0aitM/qc4OcT1l/dCsi3SNR4yEg=", "integrity": "sha1-0aitM/qc4OcT1l/dCsi3SNR4yEg=",
"requires": { "requires": {
"js-tokens": "3.0.2" "js-tokens": "^3.0.0"
} }
}, },
"minimatch": { "minimatch": {
@ -367,7 +367,7 @@
"integrity": "sha512-yJHVQEhyqPLUTgt9B83PXu6W3rx4MvvHvSUvToogpwoGDOUQ+yDrR0HRot+yOCdCO7u4hX3pWft6kWBBcqh0UA==", "integrity": "sha512-yJHVQEhyqPLUTgt9B83PXu6W3rx4MvvHvSUvToogpwoGDOUQ+yDrR0HRot+yOCdCO7u4hX3pWft6kWBBcqh0UA==",
"dev": true, "dev": true,
"requires": { "requires": {
"brace-expansion": "1.1.11" "brace-expansion": "^1.1.7"
} }
}, },
"node-fetch": { "node-fetch": {
@ -375,8 +375,8 @@
"resolved": "https://registry.npmjs.org/node-fetch/-/node-fetch-1.7.3.tgz", "resolved": "https://registry.npmjs.org/node-fetch/-/node-fetch-1.7.3.tgz",
"integrity": "sha512-NhZ4CsKx7cYm2vSrBAr2PvFOe6sWDf0UYLRqA6svUYg7+/TSfVAu49jYC4BvQ4Sms9SZgdqGBgroqfDhJdTyKQ==", "integrity": "sha512-NhZ4CsKx7cYm2vSrBAr2PvFOe6sWDf0UYLRqA6svUYg7+/TSfVAu49jYC4BvQ4Sms9SZgdqGBgroqfDhJdTyKQ==",
"requires": { "requires": {
"encoding": "0.1.12", "encoding": "^0.1.11",
"is-stream": "1.1.0" "is-stream": "^1.0.1"
} }
}, },
"object-assign": { "object-assign": {
@ -390,7 +390,7 @@
"integrity": "sha1-WDsap3WWHUsROsF9nFC6753Xa9E=", "integrity": "sha1-WDsap3WWHUsROsF9nFC6753Xa9E=",
"dev": true, "dev": true,
"requires": { "requires": {
"wrappy": "1.0.2" "wrappy": "1"
} }
}, },
"path-is-absolute": { "path-is-absolute": {
@ -410,7 +410,7 @@
"resolved": "https://registry.npmjs.org/promise/-/promise-7.3.1.tgz", "resolved": "https://registry.npmjs.org/promise/-/promise-7.3.1.tgz",
"integrity": "sha512-nolQXZ/4L+bP/UGlkfaIujX9BKxGwmQ9OT4mOt5yvy8iK1h3wqTEJCijzGANTCCl9nWjY41juyAn2K3Q1hLLTg==", "integrity": "sha512-nolQXZ/4L+bP/UGlkfaIujX9BKxGwmQ9OT4mOt5yvy8iK1h3wqTEJCijzGANTCCl9nWjY41juyAn2K3Q1hLLTg==",
"requires": { "requires": {
"asap": "2.0.6" "asap": "~2.0.3"
} }
}, },
"prop-types": { "prop-types": {
@ -418,9 +418,9 @@
"resolved": "https://registry.npmjs.org/prop-types/-/prop-types-15.6.1.tgz", "resolved": "https://registry.npmjs.org/prop-types/-/prop-types-15.6.1.tgz",
"integrity": "sha512-4ec7bY1Y66LymSUOH/zARVYObB23AT2h8cf6e/O6ZALB/N0sqZFEx7rq6EYPX2MkOdKORuooI/H5k9TlR4q7kQ==", "integrity": "sha512-4ec7bY1Y66LymSUOH/zARVYObB23AT2h8cf6e/O6ZALB/N0sqZFEx7rq6EYPX2MkOdKORuooI/H5k9TlR4q7kQ==",
"requires": { "requires": {
"fbjs": "0.8.16", "fbjs": "^0.8.16",
"loose-envify": "1.3.1", "loose-envify": "^1.3.1",
"object-assign": "4.1.1" "object-assign": "^4.1.1"
} }
}, },
"react": { "react": {
@ -428,10 +428,10 @@
"resolved": "https://registry.npmjs.org/react/-/react-16.4.2.tgz", "resolved": "https://registry.npmjs.org/react/-/react-16.4.2.tgz",
"integrity": "sha512-dMv7YrbxO4y2aqnvA7f/ik9ibeLSHQJTI6TrYAenPSaQ6OXfb+Oti+oJiy8WBxgRzlKatYqtCjphTgDSCEiWFg==", "integrity": "sha512-dMv7YrbxO4y2aqnvA7f/ik9ibeLSHQJTI6TrYAenPSaQ6OXfb+Oti+oJiy8WBxgRzlKatYqtCjphTgDSCEiWFg==",
"requires": { "requires": {
"fbjs": "0.8.16", "fbjs": "^0.8.16",
"loose-envify": "1.3.1", "loose-envify": "^1.1.0",
"object-assign": "4.1.1", "object-assign": "^4.1.1",
"prop-types": "15.6.1" "prop-types": "^15.6.0"
} }
}, },
"react-dom": { "react-dom": {
@ -439,10 +439,10 @@
"resolved": "https://registry.npmjs.org/react-dom/-/react-dom-16.4.2.tgz", "resolved": "https://registry.npmjs.org/react-dom/-/react-dom-16.4.2.tgz",
"integrity": "sha512-Usl73nQqzvmJN+89r97zmeUpQDKDlh58eX6Hbs/ERdDHzeBzWy+ENk7fsGQ+5KxArV1iOFPT46/VneklK9zoWw==", "integrity": "sha512-Usl73nQqzvmJN+89r97zmeUpQDKDlh58eX6Hbs/ERdDHzeBzWy+ENk7fsGQ+5KxArV1iOFPT46/VneklK9zoWw==",
"requires": { "requires": {
"fbjs": "0.8.16", "fbjs": "^0.8.16",
"loose-envify": "1.3.1", "loose-envify": "^1.1.0",
"object-assign": "4.1.1", "object-assign": "^4.1.1",
"prop-types": "15.6.1" "prop-types": "^15.6.0"
} }
}, },
"react-lifecycles-compat": { "react-lifecycles-compat": {
@ -455,10 +455,10 @@
"resolved": "https://registry.npmjs.org/react-modal/-/react-modal-3.5.1.tgz", "resolved": "https://registry.npmjs.org/react-modal/-/react-modal-3.5.1.tgz",
"integrity": "sha512-GxL7ycOgKC+p641cR+V1bw5dC1faL2N86/AJlzbMVmvt1totoylgkJmn9zvLuHeuarGbB7CLfHMGpeRowaj2jQ==", "integrity": "sha512-GxL7ycOgKC+p641cR+V1bw5dC1faL2N86/AJlzbMVmvt1totoylgkJmn9zvLuHeuarGbB7CLfHMGpeRowaj2jQ==",
"requires": { "requires": {
"exenv": "1.2.2", "exenv": "^1.2.0",
"prop-types": "15.6.1", "prop-types": "^15.5.10",
"react-lifecycles-compat": "3.0.4", "react-lifecycles-compat": "^3.0.0",
"warning": "3.0.0" "warning": "^3.0.0"
} }
}, },
"react-redux": { "react-redux": {
@ -466,12 +466,12 @@
"resolved": "https://registry.npmjs.org/react-redux/-/react-redux-5.0.7.tgz", "resolved": "https://registry.npmjs.org/react-redux/-/react-redux-5.0.7.tgz",
"integrity": "sha512-5VI8EV5hdgNgyjfmWzBbdrqUkrVRKlyTKk1sGH3jzM2M2Mhj/seQgPXaz6gVAj2lz/nz688AdTqMO18Lr24Zhg==", "integrity": "sha512-5VI8EV5hdgNgyjfmWzBbdrqUkrVRKlyTKk1sGH3jzM2M2Mhj/seQgPXaz6gVAj2lz/nz688AdTqMO18Lr24Zhg==",
"requires": { "requires": {
"hoist-non-react-statics": "2.5.5", "hoist-non-react-statics": "^2.5.0",
"invariant": "2.2.4", "invariant": "^2.0.0",
"lodash": "4.17.10", "lodash": "^4.17.5",
"lodash-es": "4.17.10", "lodash-es": "^4.17.5",
"loose-envify": "1.3.1", "loose-envify": "^1.1.0",
"prop-types": "15.6.1" "prop-types": "^15.6.0"
} }
}, },
"redux": { "redux": {
@ -479,8 +479,8 @@
"resolved": "https://registry.npmjs.org/redux/-/redux-4.0.0.tgz", "resolved": "https://registry.npmjs.org/redux/-/redux-4.0.0.tgz",
"integrity": "sha512-NnnHF0h0WVE/hXyrB6OlX67LYRuaf/rJcbWvnHHEPCF/Xa/AZpwhs/20WyqzQae5x4SD2F9nPObgBh2rxAgLiA==", "integrity": "sha512-NnnHF0h0WVE/hXyrB6OlX67LYRuaf/rJcbWvnHHEPCF/Xa/AZpwhs/20WyqzQae5x4SD2F9nPObgBh2rxAgLiA==",
"requires": { "requires": {
"loose-envify": "1.3.1", "loose-envify": "^1.1.0",
"symbol-observable": "1.2.0" "symbol-observable": "^1.2.0"
} }
}, },
"resolve": { "resolve": {
@ -489,7 +489,7 @@
"integrity": "sha512-AicPrAC7Qu1JxPCZ9ZgCZlY35QgFnNqc+0LtbRNxnVw4TXvjQ72wnuL9JQcEBgXkI9JM8MsT9kaQoHcpCRJOYA==", "integrity": "sha512-AicPrAC7Qu1JxPCZ9ZgCZlY35QgFnNqc+0LtbRNxnVw4TXvjQ72wnuL9JQcEBgXkI9JM8MsT9kaQoHcpCRJOYA==",
"dev": true, "dev": true,
"requires": { "requires": {
"path-parse": "1.0.6" "path-parse": "^1.0.5"
} }
}, },
"safer-buffer": { "safer-buffer": {
@ -520,7 +520,7 @@
"integrity": "sha1-ajhfuIU9lS1f8F0Oiq+UJ43GPc8=", "integrity": "sha1-ajhfuIU9lS1f8F0Oiq+UJ43GPc8=",
"dev": true, "dev": true,
"requires": { "requires": {
"ansi-regex": "2.1.1" "ansi-regex": "^2.0.0"
} }
}, },
"supports-color": { "supports-color": {
@ -546,18 +546,18 @@
"integrity": "sha1-mPMMAurjzecAYgHkwzywi0hYHu0=", "integrity": "sha1-mPMMAurjzecAYgHkwzywi0hYHu0=",
"dev": true, "dev": true,
"requires": { "requires": {
"babel-code-frame": "6.26.0", "babel-code-frame": "^6.22.0",
"builtin-modules": "1.1.1", "builtin-modules": "^1.1.1",
"chalk": "2.4.1", "chalk": "^2.3.0",
"commander": "2.17.1", "commander": "^2.12.1",
"diff": "3.5.0", "diff": "^3.2.0",
"glob": "7.1.2", "glob": "^7.1.1",
"js-yaml": "3.12.0", "js-yaml": "^3.7.0",
"minimatch": "3.0.4", "minimatch": "^3.0.4",
"resolve": "1.8.1", "resolve": "^1.3.2",
"semver": "5.5.0", "semver": "^5.3.0",
"tslib": "1.9.3", "tslib": "^1.8.0",
"tsutils": "2.29.0" "tsutils": "^2.27.2"
} }
}, },
"tsutils": { "tsutils": {
@ -566,7 +566,7 @@
"integrity": "sha512-g5JVHCIJwzfISaXpXE1qvNalca5Jwob6FjI4AoPlqMusJ6ftFE7IkkFoMhVLRgK+4Kx3gkzb8UZK5t5yTTvEmA==", "integrity": "sha512-g5JVHCIJwzfISaXpXE1qvNalca5Jwob6FjI4AoPlqMusJ6ftFE7IkkFoMhVLRgK+4Kx3gkzb8UZK5t5yTTvEmA==",
"dev": true, "dev": true,
"requires": { "requires": {
"tslib": "1.9.3" "tslib": "^1.8.1"
} }
}, },
"typescript": { "typescript": {
@ -585,7 +585,7 @@
"resolved": "https://registry.npmjs.org/warning/-/warning-3.0.0.tgz", "resolved": "https://registry.npmjs.org/warning/-/warning-3.0.0.tgz",
"integrity": "sha1-MuU3fLVy3kqwR1O9+IIcAe1gW3w=", "integrity": "sha1-MuU3fLVy3kqwR1O9+IIcAe1gW3w=",
"requires": { "requires": {
"loose-envify": "1.3.1" "loose-envify": "^1.0.0"
} }
}, },
"whatwg-fetch": { "whatwg-fetch": {