new import implementation with creating new configuration
This commit is contained in:
parent
da178edf50
commit
9bc8dfaf8b
13 changed files with 338 additions and 72 deletions
|
@ -9,32 +9,41 @@
|
||||||
package ch.ethz.seb.sebserver.gui.content;
|
package ch.ethz.seb.sebserver.gui.content;
|
||||||
|
|
||||||
import java.io.InputStream;
|
import java.io.InputStream;
|
||||||
import java.util.function.Consumer;
|
|
||||||
import java.util.function.Function;
|
import java.util.function.Function;
|
||||||
|
import java.util.function.Predicate;
|
||||||
import java.util.function.Supplier;
|
import java.util.function.Supplier;
|
||||||
|
|
||||||
|
import org.apache.commons.lang3.StringUtils;
|
||||||
import org.eclipse.swt.widgets.Composite;
|
import org.eclipse.swt.widgets.Composite;
|
||||||
import org.eclipse.swt.widgets.Control;
|
import org.eclipse.swt.widgets.Control;
|
||||||
|
|
||||||
import ch.ethz.seb.sebserver.gbl.api.API;
|
import ch.ethz.seb.sebserver.gbl.api.API;
|
||||||
|
import ch.ethz.seb.sebserver.gbl.model.Domain;
|
||||||
import ch.ethz.seb.sebserver.gbl.model.EntityKey;
|
import ch.ethz.seb.sebserver.gbl.model.EntityKey;
|
||||||
import ch.ethz.seb.sebserver.gbl.model.sebconfig.Configuration;
|
import ch.ethz.seb.sebserver.gbl.model.sebconfig.Configuration;
|
||||||
import ch.ethz.seb.sebserver.gbl.model.sebconfig.ConfigurationNode;
|
import ch.ethz.seb.sebserver.gbl.model.sebconfig.ConfigurationNode;
|
||||||
|
import ch.ethz.seb.sebserver.gbl.util.Result;
|
||||||
import ch.ethz.seb.sebserver.gui.form.Form;
|
import ch.ethz.seb.sebserver.gui.form.Form;
|
||||||
import ch.ethz.seb.sebserver.gui.form.FormBuilder;
|
import ch.ethz.seb.sebserver.gui.form.FormBuilder;
|
||||||
import ch.ethz.seb.sebserver.gui.form.FormHandle;
|
import ch.ethz.seb.sebserver.gui.form.FormHandle;
|
||||||
|
import ch.ethz.seb.sebserver.gui.service.ResourceService;
|
||||||
import ch.ethz.seb.sebserver.gui.service.i18n.LocTextKey;
|
import ch.ethz.seb.sebserver.gui.service.i18n.LocTextKey;
|
||||||
import ch.ethz.seb.sebserver.gui.service.page.ModalInputDialogComposer;
|
import ch.ethz.seb.sebserver.gui.service.page.ModalInputDialogComposer;
|
||||||
import ch.ethz.seb.sebserver.gui.service.page.PageContext;
|
import ch.ethz.seb.sebserver.gui.service.page.PageContext;
|
||||||
import ch.ethz.seb.sebserver.gui.service.page.PageService;
|
import ch.ethz.seb.sebserver.gui.service.page.PageService;
|
||||||
import ch.ethz.seb.sebserver.gui.service.page.impl.ModalInputDialog;
|
import ch.ethz.seb.sebserver.gui.service.page.impl.ModalInputDialog;
|
||||||
import ch.ethz.seb.sebserver.gui.service.page.impl.PageAction;
|
import ch.ethz.seb.sebserver.gui.service.page.impl.PageAction;
|
||||||
import ch.ethz.seb.sebserver.gui.service.remote.webservice.api.seb.examconfig.ImportExamConfig;
|
import ch.ethz.seb.sebserver.gui.service.remote.webservice.api.RestCall;
|
||||||
|
import ch.ethz.seb.sebserver.gui.service.remote.webservice.api.seb.examconfig.ImportExamConfigOnExistingConfig;
|
||||||
|
import ch.ethz.seb.sebserver.gui.service.remote.webservice.api.seb.examconfig.ImportNewExamConfig;
|
||||||
import ch.ethz.seb.sebserver.gui.widget.FileUploadSelection;
|
import ch.ethz.seb.sebserver.gui.widget.FileUploadSelection;
|
||||||
|
|
||||||
public final class SebExamConfigImport {
|
public final class SebExamConfigImport {
|
||||||
|
|
||||||
static Function<PageAction, PageAction> importConfigFunction(final PageService pageService) {
|
static Function<PageAction, PageAction> importFunction(
|
||||||
|
final PageService pageService,
|
||||||
|
final boolean newConfig) {
|
||||||
|
|
||||||
return action -> {
|
return action -> {
|
||||||
|
|
||||||
final ModalInputDialog<FormHandle<ConfigurationNode>> dialog =
|
final ModalInputDialog<FormHandle<ConfigurationNode>> dialog =
|
||||||
|
@ -45,13 +54,15 @@ public final class SebExamConfigImport {
|
||||||
|
|
||||||
final ImportFormContext importFormContext = new ImportFormContext(
|
final ImportFormContext importFormContext = new ImportFormContext(
|
||||||
pageService,
|
pageService,
|
||||||
action.pageContext());
|
action.pageContext(),
|
||||||
|
newConfig);
|
||||||
|
|
||||||
dialog.open(
|
dialog.open(
|
||||||
SebExamConfigPropForm.FORM_IMPORT_TEXT_KEY,
|
SebExamConfigPropForm.FORM_IMPORT_TEXT_KEY,
|
||||||
(Consumer<FormHandle<ConfigurationNode>>) formHandle -> doImport(
|
(Predicate<FormHandle<ConfigurationNode>>) formHandle -> doImport(
|
||||||
pageService,
|
pageService,
|
||||||
formHandle),
|
formHandle,
|
||||||
|
newConfig),
|
||||||
importFormContext::cancelUpload,
|
importFormContext::cancelUpload,
|
||||||
importFormContext);
|
importFormContext);
|
||||||
|
|
||||||
|
@ -59,39 +70,90 @@ public final class SebExamConfigImport {
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
private static final void doImport(
|
private static final boolean doImport(
|
||||||
final PageService pageService,
|
final PageService pageService,
|
||||||
final FormHandle<ConfigurationNode> formHandle) {
|
final FormHandle<ConfigurationNode> formHandle,
|
||||||
|
final boolean newConfig) {
|
||||||
|
|
||||||
final Form form = formHandle.getForm();
|
try {
|
||||||
final EntityKey entityKey = formHandle.getContext().getEntityKey();
|
final Form form = formHandle.getForm();
|
||||||
final Control fieldControl = form.getFieldControl(API.IMPORT_FILE_ATTR_NAME);
|
final EntityKey entityKey = formHandle.getContext().getEntityKey();
|
||||||
final PageContext context = formHandle.getContext();
|
final Control fieldControl = form.getFieldControl(API.IMPORT_FILE_ATTR_NAME);
|
||||||
if (fieldControl != null && fieldControl instanceof FileUploadSelection) {
|
final PageContext context = formHandle.getContext();
|
||||||
final FileUploadSelection fileUpload = (FileUploadSelection) fieldControl;
|
|
||||||
final InputStream inputStream = fileUpload.getInputStream();
|
|
||||||
if (inputStream != null) {
|
|
||||||
final Configuration configuration = pageService.getRestService()
|
|
||||||
.getBuilder(ImportExamConfig.class)
|
|
||||||
.withURIVariable(API.PARAM_MODEL_ID, entityKey.modelId)
|
|
||||||
.withHeader(
|
|
||||||
API.IMPORT_PASSWORD_ATTR_NAME,
|
|
||||||
form.getFieldValue(API.IMPORT_PASSWORD_ATTR_NAME))
|
|
||||||
.withBody(inputStream)
|
|
||||||
.call()
|
|
||||||
.get(e -> {
|
|
||||||
fileUpload.close();
|
|
||||||
return context.notifyError(e);
|
|
||||||
});
|
|
||||||
|
|
||||||
if (configuration != null) {
|
// Ad-hoc field validation
|
||||||
context.publishInfo(SebExamConfigPropForm.FORM_IMPORT_CONFIRM_TEXT_KEY);
|
formHandle.process(name -> true, field -> field.resetError());
|
||||||
}
|
final String fieldValue = form.getFieldValue(Domain.CONFIGURATION_NODE.ATTR_NAME);
|
||||||
} else {
|
if (StringUtils.isBlank(fieldValue)) {
|
||||||
formHandle.getContext().publishPageMessage(
|
form.setFieldError(
|
||||||
new LocTextKey("sebserver.error.unexpected"),
|
Domain.CONFIGURATION_NODE.ATTR_NAME,
|
||||||
new LocTextKey("Please selecte a valid SEB Exam Configuration File"));
|
pageService
|
||||||
|
.getI18nSupport()
|
||||||
|
.getText(new LocTextKey("sebserver.form.validation.fieldError.notNull")));
|
||||||
|
return false;
|
||||||
|
} else if (fieldValue.length() < 3 || fieldValue.length() > 255) {
|
||||||
|
form.setFieldError(
|
||||||
|
Domain.CONFIGURATION_NODE.ATTR_NAME,
|
||||||
|
pageService
|
||||||
|
.getI18nSupport()
|
||||||
|
.getText(new LocTextKey("sebserver.form.validation.fieldError.size",
|
||||||
|
null,
|
||||||
|
null,
|
||||||
|
null,
|
||||||
|
3,
|
||||||
|
255)));
|
||||||
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (fieldControl != null && fieldControl instanceof FileUploadSelection) {
|
||||||
|
final FileUploadSelection fileUpload = (FileUploadSelection) fieldControl;
|
||||||
|
final InputStream inputStream = fileUpload.getInputStream();
|
||||||
|
if (inputStream != null) {
|
||||||
|
final RestCall<Configuration>.RestCallBuilder restCall = (newConfig)
|
||||||
|
? pageService.getRestService()
|
||||||
|
.getBuilder(ImportNewExamConfig.class)
|
||||||
|
: pageService.getRestService()
|
||||||
|
.getBuilder(ImportExamConfigOnExistingConfig.class);
|
||||||
|
|
||||||
|
restCall
|
||||||
|
.withHeader(
|
||||||
|
API.IMPORT_PASSWORD_ATTR_NAME,
|
||||||
|
form.getFieldValue(API.IMPORT_PASSWORD_ATTR_NAME))
|
||||||
|
.withBody(inputStream);
|
||||||
|
|
||||||
|
if (newConfig) {
|
||||||
|
restCall
|
||||||
|
.withHeader(
|
||||||
|
Domain.CONFIGURATION_NODE.ATTR_NAME,
|
||||||
|
form.getFieldValue(Domain.CONFIGURATION_NODE.ATTR_NAME))
|
||||||
|
.withHeader(
|
||||||
|
Domain.CONFIGURATION_NODE.ATTR_DESCRIPTION,
|
||||||
|
form.getFieldValue(Domain.CONFIGURATION_NODE.ATTR_DESCRIPTION))
|
||||||
|
.withHeader(
|
||||||
|
Domain.CONFIGURATION_NODE.ATTR_TEMPLATE_ID,
|
||||||
|
form.getFieldValue(Domain.CONFIGURATION_NODE.ATTR_TEMPLATE_ID));
|
||||||
|
} else {
|
||||||
|
restCall.withURIVariable(API.PARAM_MODEL_ID, entityKey.modelId);
|
||||||
|
}
|
||||||
|
|
||||||
|
final Result<Configuration> configuration = restCall
|
||||||
|
.call();
|
||||||
|
|
||||||
|
if (!configuration.hasError()) {
|
||||||
|
context.publishInfo(SebExamConfigPropForm.FORM_IMPORT_CONFIRM_TEXT_KEY);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
formHandle.getContext().publishPageMessage(
|
||||||
|
new LocTextKey("sebserver.error.unexpected"),
|
||||||
|
new LocTextKey("Please selecte a valid SEB Exam Configuration File"));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return false;
|
||||||
|
} catch (final Exception e) {
|
||||||
|
formHandle.getContext().notifyError(e);
|
||||||
|
return true;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -99,17 +161,25 @@ public final class SebExamConfigImport {
|
||||||
|
|
||||||
private final PageService pageService;
|
private final PageService pageService;
|
||||||
private final PageContext pageContext;
|
private final PageContext pageContext;
|
||||||
|
private final boolean newConfig;
|
||||||
|
|
||||||
private Form form = null;
|
private Form form = null;
|
||||||
|
|
||||||
protected ImportFormContext(final PageService pageService, final PageContext pageContext) {
|
protected ImportFormContext(
|
||||||
|
final PageService pageService,
|
||||||
|
final PageContext pageContext,
|
||||||
|
final boolean newConfig) {
|
||||||
|
|
||||||
this.pageService = pageService;
|
this.pageService = pageService;
|
||||||
this.pageContext = pageContext;
|
this.pageContext = pageContext;
|
||||||
|
this.newConfig = newConfig;
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public Supplier<FormHandle<ConfigurationNode>> compose(final Composite parent) {
|
public Supplier<FormHandle<ConfigurationNode>> compose(final Composite parent) {
|
||||||
|
|
||||||
|
final ResourceService resourceService = this.pageService.getResourceService();
|
||||||
|
|
||||||
final FormHandle<ConfigurationNode> formHandle = this.pageService.formBuilder(
|
final FormHandle<ConfigurationNode> formHandle = this.pageService.formBuilder(
|
||||||
this.pageContext.copyOf(parent), 4)
|
this.pageContext.copyOf(parent), 4)
|
||||||
.readonly(false)
|
.readonly(false)
|
||||||
|
@ -118,6 +188,26 @@ public final class SebExamConfigImport {
|
||||||
SebExamConfigPropForm.FORM_IMPORT_SELECT_TEXT_KEY,
|
SebExamConfigPropForm.FORM_IMPORT_SELECT_TEXT_KEY,
|
||||||
null,
|
null,
|
||||||
API.SEB_FILE_EXTENSION))
|
API.SEB_FILE_EXTENSION))
|
||||||
|
|
||||||
|
.addFieldIf(
|
||||||
|
() -> this.newConfig,
|
||||||
|
() -> FormBuilder.text(
|
||||||
|
Domain.CONFIGURATION_NODE.ATTR_NAME,
|
||||||
|
SebExamConfigPropForm.FORM_NAME_TEXT_KEY))
|
||||||
|
.addFieldIf(
|
||||||
|
() -> this.newConfig,
|
||||||
|
() -> FormBuilder.text(
|
||||||
|
Domain.CONFIGURATION_NODE.ATTR_DESCRIPTION,
|
||||||
|
SebExamConfigPropForm.FORM_DESCRIPTION_TEXT_KEY)
|
||||||
|
.asArea())
|
||||||
|
.addFieldIf(
|
||||||
|
() -> this.newConfig,
|
||||||
|
() -> FormBuilder.singleSelection(
|
||||||
|
Domain.CONFIGURATION_NODE.ATTR_TEMPLATE_ID,
|
||||||
|
SebExamConfigPropForm.FORM_TEMPLATE_TEXT_KEY,
|
||||||
|
null,
|
||||||
|
resourceService::getExamConfigTemplateResources))
|
||||||
|
|
||||||
.addField(FormBuilder.text(
|
.addField(FormBuilder.text(
|
||||||
API.IMPORT_PASSWORD_ATTR_NAME,
|
API.IMPORT_PASSWORD_ATTR_NAME,
|
||||||
SebExamConfigPropForm.FORM_IMPORT_PASSWORD_TEXT_KEY,
|
SebExamConfigPropForm.FORM_IMPORT_PASSWORD_TEXT_KEY,
|
||||||
|
|
|
@ -207,6 +207,11 @@ public class SebExamConfigList implements TemplateComposer {
|
||||||
PageAction::applySingleSelection, EMPTY_SELECTION_TEXT_KEY)
|
PageAction::applySingleSelection, EMPTY_SELECTION_TEXT_KEY)
|
||||||
.publishIf(() -> examConfigGrant.im() && configTable.hasAnyContent())
|
.publishIf(() -> examConfigGrant.im() && configTable.hasAnyContent())
|
||||||
|
|
||||||
|
.newAction(ActionDefinition.SEB_EXAM_CONFIG_IMPORT_TO_NEW_CONFIG)
|
||||||
|
.withExec(SebExamConfigImport.importFunction(this.pageService, true))
|
||||||
|
.noEventPropagation()
|
||||||
|
.publishIf(() -> examConfigGrant.im())
|
||||||
|
|
||||||
// Exam Configuration template actions...
|
// Exam Configuration template actions...
|
||||||
.newAction(ActionDefinition.SEB_EXAM_CONFIG_TEMPLATE_NEW)
|
.newAction(ActionDefinition.SEB_EXAM_CONFIG_TEMPLATE_NEW)
|
||||||
.publishIf(examConfigGrant::iw)
|
.publishIf(examConfigGrant::iw)
|
||||||
|
|
|
@ -246,9 +246,9 @@ public class SebExamConfigPropForm implements TemplateComposer {
|
||||||
.noEventPropagation()
|
.noEventPropagation()
|
||||||
.publishIf(() -> modifyGrant && isReadonly)
|
.publishIf(() -> modifyGrant && isReadonly)
|
||||||
|
|
||||||
.newAction(ActionDefinition.SEB_EXAM_CONFIG_IMPORT_CONFIG)
|
.newAction(ActionDefinition.SEB_EXAM_CONFIG_IMPORT_TO_EXISTING_CONFIG)
|
||||||
.withEntityKey(entityKey)
|
.withEntityKey(entityKey)
|
||||||
.withExec(SebExamConfigImport.importConfigFunction(this.pageService))
|
.withExec(SebExamConfigImport.importFunction(this.pageService, false))
|
||||||
.noEventPropagation()
|
.noEventPropagation()
|
||||||
.publishIf(() -> modifyGrant && isReadonly && !isAttachedToExam)
|
.publishIf(() -> modifyGrant && isReadonly && !isAttachedToExam)
|
||||||
|
|
||||||
|
|
|
@ -368,6 +368,10 @@ public enum ActionDefinition {
|
||||||
ImageIcon.SHOW,
|
ImageIcon.SHOW,
|
||||||
PageStateDefinitionImpl.SEB_EXAM_CONFIG_PROP_VIEW,
|
PageStateDefinitionImpl.SEB_EXAM_CONFIG_PROP_VIEW,
|
||||||
ActionCategory.FORM),
|
ActionCategory.FORM),
|
||||||
|
SEB_EXAM_CONFIG_IMPORT_TO_NEW_CONFIG(
|
||||||
|
new LocTextKey("sebserver.examconfig.action.import-config"),
|
||||||
|
ImageIcon.IMPORT,
|
||||||
|
ActionCategory.VARIA),
|
||||||
|
|
||||||
SEB_EXAM_CONFIG_MODIFY_PROP_FROM_LIST(
|
SEB_EXAM_CONFIG_MODIFY_PROP_FROM_LIST(
|
||||||
new LocTextKey("sebserver.examconfig.action.list.modify.properties"),
|
new LocTextKey("sebserver.examconfig.action.list.modify.properties"),
|
||||||
|
@ -407,10 +411,11 @@ public enum ActionDefinition {
|
||||||
new LocTextKey("sebserver.examconfig.action.get-config-key"),
|
new LocTextKey("sebserver.examconfig.action.get-config-key"),
|
||||||
ImageIcon.SECURE,
|
ImageIcon.SECURE,
|
||||||
ActionCategory.FORM),
|
ActionCategory.FORM),
|
||||||
SEB_EXAM_CONFIG_IMPORT_CONFIG(
|
SEB_EXAM_CONFIG_IMPORT_TO_EXISTING_CONFIG(
|
||||||
new LocTextKey("sebserver.examconfig.action.import-config"),
|
new LocTextKey("sebserver.examconfig.action.import-config"),
|
||||||
ImageIcon.IMPORT,
|
ImageIcon.IMPORT,
|
||||||
ActionCategory.FORM),
|
ActionCategory.FORM),
|
||||||
|
|
||||||
SEB_EXAM_CONFIG_COPY_CONFIG(
|
SEB_EXAM_CONFIG_COPY_CONFIG(
|
||||||
new LocTextKey("sebserver.examconfig.action.copy"),
|
new LocTextKey("sebserver.examconfig.action.copy"),
|
||||||
ImageIcon.COPY,
|
ImageIcon.COPY,
|
||||||
|
|
|
@ -227,6 +227,15 @@ public final class Form implements FormBinding {
|
||||||
.isPresent();
|
.isPresent();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public void setFieldError(final String fieldName, final String errorMessage) {
|
||||||
|
final List<FormFieldAccessor> list = this.formFields.get(fieldName);
|
||||||
|
if (list != null) {
|
||||||
|
list
|
||||||
|
.stream()
|
||||||
|
.forEach(ffa -> ffa.setError(errorMessage));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
public void process(
|
public void process(
|
||||||
final Predicate<String> nameFilter,
|
final Predicate<String> nameFilter,
|
||||||
final Consumer<FormFieldAccessor> processor) {
|
final Consumer<FormFieldAccessor> processor) {
|
||||||
|
|
|
@ -24,9 +24,9 @@ import ch.ethz.seb.sebserver.gui.service.remote.webservice.api.RestCall;
|
||||||
@Lazy
|
@Lazy
|
||||||
@Component
|
@Component
|
||||||
@GuiProfile
|
@GuiProfile
|
||||||
public class ImportExamConfig extends RestCall<Configuration> {
|
public class ImportExamConfigOnExistingConfig extends RestCall<Configuration> {
|
||||||
|
|
||||||
public ImportExamConfig() {
|
public ImportExamConfigOnExistingConfig() {
|
||||||
super(new TypeKey<>(
|
super(new TypeKey<>(
|
||||||
CallType.UNDEFINED,
|
CallType.UNDEFINED,
|
||||||
EntityType.CONFIGURATION,
|
EntityType.CONFIGURATION,
|
|
@ -0,0 +1,40 @@
|
||||||
|
/*
|
||||||
|
* Copyright (c) 2019 ETH Zürich, Educational Development and Technology (LET)
|
||||||
|
*
|
||||||
|
* This Source Code Form is subject to the terms of the Mozilla Public
|
||||||
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
||||||
|
* file, You can obtain one at http://mozilla.org/MPL/2.0/.
|
||||||
|
*/
|
||||||
|
|
||||||
|
package ch.ethz.seb.sebserver.gui.service.remote.webservice.api.seb.examconfig;
|
||||||
|
|
||||||
|
import org.springframework.context.annotation.Lazy;
|
||||||
|
import org.springframework.http.HttpMethod;
|
||||||
|
import org.springframework.http.MediaType;
|
||||||
|
import org.springframework.stereotype.Component;
|
||||||
|
|
||||||
|
import com.fasterxml.jackson.core.type.TypeReference;
|
||||||
|
|
||||||
|
import ch.ethz.seb.sebserver.gbl.api.API;
|
||||||
|
import ch.ethz.seb.sebserver.gbl.api.EntityType;
|
||||||
|
import ch.ethz.seb.sebserver.gbl.model.sebconfig.Configuration;
|
||||||
|
import ch.ethz.seb.sebserver.gbl.profile.GuiProfile;
|
||||||
|
import ch.ethz.seb.sebserver.gui.service.remote.webservice.api.RestCall;
|
||||||
|
|
||||||
|
@Lazy
|
||||||
|
@Component
|
||||||
|
@GuiProfile
|
||||||
|
public class ImportNewExamConfig extends RestCall<Configuration> {
|
||||||
|
|
||||||
|
public ImportNewExamConfig() {
|
||||||
|
super(new TypeKey<>(
|
||||||
|
CallType.UNDEFINED,
|
||||||
|
EntityType.CONFIGURATION,
|
||||||
|
new TypeReference<Configuration>() {
|
||||||
|
}),
|
||||||
|
HttpMethod.POST,
|
||||||
|
MediaType.APPLICATION_OCTET_STREAM,
|
||||||
|
API.CONFIGURATION_NODE_ENDPOINT + API.CONFIGURATION_IMPORT_PATH_SEGMENT);
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
|
@ -13,6 +13,7 @@ import java.util.Set;
|
||||||
|
|
||||||
import ch.ethz.seb.sebserver.gbl.model.EntityKey;
|
import ch.ethz.seb.sebserver.gbl.model.EntityKey;
|
||||||
import ch.ethz.seb.sebserver.gbl.model.sebconfig.Configuration;
|
import ch.ethz.seb.sebserver.gbl.model.sebconfig.Configuration;
|
||||||
|
import ch.ethz.seb.sebserver.gbl.model.sebconfig.ConfigurationNode;
|
||||||
import ch.ethz.seb.sebserver.gbl.util.Result;
|
import ch.ethz.seb.sebserver.gbl.util.Result;
|
||||||
|
|
||||||
public interface ConfigurationDAO extends EntityDAO<Configuration, Configuration> {
|
public interface ConfigurationDAO extends EntityDAO<Configuration, Configuration> {
|
||||||
|
@ -48,6 +49,31 @@ public interface ConfigurationDAO extends EntityDAO<Configuration, Configuration
|
||||||
* @return the current and reseted follow-up version */
|
* @return the current and reseted follow-up version */
|
||||||
Result<Configuration> undo(Long configurationNodeId);
|
Result<Configuration> undo(Long configurationNodeId);
|
||||||
|
|
||||||
|
/** Restores the attribute values to the default values that have been set for the specified configuration
|
||||||
|
* on initialization. This are the base default values if the configuration has no template or the default
|
||||||
|
* values from the template if there is one assigned to the configuration.
|
||||||
|
*
|
||||||
|
* In fact. this just gets the initial configuration values and reset the current values with that one
|
||||||
|
*
|
||||||
|
* @param configurationNodeId the ConfigurationNode identifier
|
||||||
|
* @return the Configuration instance for which the attribute values have been reset */
|
||||||
|
Result<Configuration> restoreToDefaultValues(final Long configurationNodeId);
|
||||||
|
|
||||||
|
/** Restores the attribute values to the default values that have been set for the specified configuration
|
||||||
|
* on initialization. This are the base default values if the configuration has no template or the default
|
||||||
|
* values from the template if there is one assigned to the configuration.
|
||||||
|
*
|
||||||
|
* In fact. this just gets the initial configuration values and reset the current values with that one
|
||||||
|
*
|
||||||
|
* @param configuration the Configuration that defines the ConfigurationNode identifier
|
||||||
|
* @return the Configuration instance for which the attribute values have been reset */
|
||||||
|
default Result<Configuration> restoreToDefaultValues(final Configuration configuration) {
|
||||||
|
if (configuration == null) {
|
||||||
|
return Result.ofError(new NullPointerException("configuration"));
|
||||||
|
}
|
||||||
|
return restoreToDefaultValues(configuration.configurationNodeId);
|
||||||
|
}
|
||||||
|
|
||||||
/** Restores the current follow-up Configuration to the values of a given Configuration
|
/** Restores the current follow-up Configuration to the values of a given Configuration
|
||||||
* in the history of the specified ConfigurationNode.
|
* in the history of the specified ConfigurationNode.
|
||||||
*
|
*
|
||||||
|
@ -62,6 +88,17 @@ public interface ConfigurationDAO extends EntityDAO<Configuration, Configuration
|
||||||
* @return the current follow-up configuration */
|
* @return the current follow-up configuration */
|
||||||
Result<Configuration> getFollowupConfiguration(Long configNodeId);
|
Result<Configuration> getFollowupConfiguration(Long configNodeId);
|
||||||
|
|
||||||
|
/** Use this to get the follow-up configuration for a specified configuration node.
|
||||||
|
*
|
||||||
|
* @param configNode ConfigurationNode to get the current follow-up configuration from
|
||||||
|
* @return the current follow-up configuration */
|
||||||
|
default Result<Configuration> getFollowupConfiguration(final ConfigurationNode configurationNode) {
|
||||||
|
if (configurationNode == null) {
|
||||||
|
return Result.ofError(new NullPointerException("configurationNode"));
|
||||||
|
}
|
||||||
|
return getFollowupConfiguration(configurationNode.id);
|
||||||
|
}
|
||||||
|
|
||||||
/** Use this to get the last version of a configuration that is not the follow-up.
|
/** Use this to get the last version of a configuration that is not the follow-up.
|
||||||
*
|
*
|
||||||
* @param configNodeId ConfigurationNode identifier to get the last version of configuration from
|
* @param configNodeId ConfigurationNode identifier to get the last version of configuration from
|
||||||
|
|
|
@ -266,6 +266,25 @@ class ConfigurationDAOBatchService {
|
||||||
.flatMap(rec -> restoreToVersion(configurationNodeId, rec.getId()));
|
.flatMap(rec -> restoreToVersion(configurationNodeId, rec.getId()));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Result<Configuration> restoreToDefaultValues(final Long configurationNodeId) {
|
||||||
|
return Result.tryCatch(() -> {
|
||||||
|
// get initial version that contains the default values either from base or from template
|
||||||
|
return this.batchConfigurationRecordMapper.selectIdsByExample()
|
||||||
|
.where(
|
||||||
|
ConfigurationRecordDynamicSqlSupport.configurationNodeId,
|
||||||
|
isEqualTo(configurationNodeId))
|
||||||
|
.and(
|
||||||
|
ConfigurationRecordDynamicSqlSupport.version,
|
||||||
|
isEqualTo(INITIAL_VERSION_NAME))
|
||||||
|
.build()
|
||||||
|
.execute()
|
||||||
|
.stream()
|
||||||
|
.collect(Utils.toSingleton());
|
||||||
|
|
||||||
|
})
|
||||||
|
.flatMap(configId -> restoreToVersion(configurationNodeId, configId));
|
||||||
|
}
|
||||||
|
|
||||||
Result<Configuration> restoreToVersion(final Long configurationNodeId, final Long configId) {
|
Result<Configuration> restoreToVersion(final Long configurationNodeId, final Long configId) {
|
||||||
return Result.tryCatch(() -> {
|
return Result.tryCatch(() -> {
|
||||||
|
|
||||||
|
|
|
@ -193,6 +193,14 @@ public class ConfigurationDAOImpl implements ConfigurationDAO {
|
||||||
.onError(TransactionHandler::rollback);
|
.onError(TransactionHandler::rollback);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
@Transactional
|
||||||
|
public Result<Configuration> restoreToDefaultValues(final Long configurationNodeId) {
|
||||||
|
return this.configurationDAOBatchService
|
||||||
|
.restoreToDefaultValues(configurationNodeId)
|
||||||
|
.onError(TransactionHandler::rollback);
|
||||||
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
@Transactional
|
@Transactional
|
||||||
public Result<Configuration> restoreToVersion(final Long configurationNodeId, final Long configId) {
|
public Result<Configuration> restoreToVersion(final Long configurationNodeId, final Long configId) {
|
||||||
|
|
|
@ -118,10 +118,10 @@ public interface SebExamConfigService {
|
||||||
*
|
*
|
||||||
* Then parses the XML and adds each attribute to the new Configuration.
|
* Then parses the XML and adds each attribute to the new Configuration.
|
||||||
*
|
*
|
||||||
* @param configNodeId The identifier of the configuration node on which the import should take place
|
* @param config The Configuration to import the attribute values to
|
||||||
* @param input The InputStream to get the SEB config file as byte-stream
|
* @param input The InputStream to get the SEB config file as byte-stream
|
||||||
* @param password A password is only needed if the file is in an encrypted format
|
* @param password A password is only needed if the file is in an encrypted format
|
||||||
* @return The newly created Configuration instance */
|
* @return The newly created Configuration instance */
|
||||||
Result<Configuration> importFromSEBFile(Long configNodeId, InputStream input, CharSequence password);
|
Result<Configuration> importFromSEBFile(Configuration config, InputStream input, CharSequence password);
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -330,16 +330,12 @@ public class SebExamConfigServiceImpl implements SebExamConfigService {
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public Result<Configuration> importFromSEBFile(
|
public Result<Configuration> importFromSEBFile(
|
||||||
final Long configNodeId,
|
final Configuration config,
|
||||||
final InputStream input,
|
final InputStream input,
|
||||||
final CharSequence password) {
|
final CharSequence password) {
|
||||||
|
|
||||||
return Result.tryCatch(() -> {
|
return Result.tryCatch(() -> {
|
||||||
|
|
||||||
final Configuration newConfig = this.configurationDAO
|
|
||||||
.saveToHistory(configNodeId)
|
|
||||||
.getOrThrow();
|
|
||||||
|
|
||||||
Future<Exception> streamDecrypted = null;
|
Future<Exception> streamDecrypted = null;
|
||||||
InputStream cryptIn = null;
|
InputStream cryptIn = null;
|
||||||
PipedInputStream plainIn = null;
|
PipedInputStream plainIn = null;
|
||||||
|
@ -363,16 +359,16 @@ public class SebExamConfigServiceImpl implements SebExamConfigService {
|
||||||
// parse XML and import
|
// parse XML and import
|
||||||
this.examConfigIO.importPlainXML(
|
this.examConfigIO.importPlainXML(
|
||||||
unzippedIn,
|
unzippedIn,
|
||||||
newConfig.institutionId,
|
config.institutionId,
|
||||||
newConfig.id);
|
config.id);
|
||||||
|
|
||||||
return newConfig;
|
return config;
|
||||||
|
|
||||||
} catch (final Exception e) {
|
} catch (final Exception e) {
|
||||||
log.error("Unexpected error while trying to import SEB Exam Configuration: ", e);
|
log.error("Unexpected error while trying to import SEB Exam Configuration: ", e);
|
||||||
log.debug("Make an undo on the ConfigurationNode to rollback the changes");
|
log.debug("Make an undo on the ConfigurationNode to rollback the changes");
|
||||||
this.configurationDAO
|
this.configurationDAO
|
||||||
.undo(configNodeId)
|
.undo(config.configurationNodeId)
|
||||||
.getOrThrow();
|
.getOrThrow();
|
||||||
|
|
||||||
if (streamDecrypted != null) {
|
if (streamDecrypted != null) {
|
||||||
|
|
|
@ -20,6 +20,7 @@ import javax.servlet.http.HttpServletResponse;
|
||||||
import javax.validation.Valid;
|
import javax.validation.Valid;
|
||||||
|
|
||||||
import org.apache.commons.io.IOUtils;
|
import org.apache.commons.io.IOUtils;
|
||||||
|
import org.apache.commons.lang3.StringUtils;
|
||||||
import org.mybatis.dynamic.sql.SqlTable;
|
import org.mybatis.dynamic.sql.SqlTable;
|
||||||
import org.slf4j.Logger;
|
import org.slf4j.Logger;
|
||||||
import org.slf4j.LoggerFactory;
|
import org.slf4j.LoggerFactory;
|
||||||
|
@ -39,12 +40,14 @@ import ch.ethz.seb.sebserver.gbl.api.API;
|
||||||
import ch.ethz.seb.sebserver.gbl.api.APIMessage;
|
import ch.ethz.seb.sebserver.gbl.api.APIMessage;
|
||||||
import ch.ethz.seb.sebserver.gbl.api.POSTMapper;
|
import ch.ethz.seb.sebserver.gbl.api.POSTMapper;
|
||||||
import ch.ethz.seb.sebserver.gbl.api.authorization.PrivilegeType;
|
import ch.ethz.seb.sebserver.gbl.api.authorization.PrivilegeType;
|
||||||
|
import ch.ethz.seb.sebserver.gbl.model.Domain;
|
||||||
import ch.ethz.seb.sebserver.gbl.model.Domain.EXAM;
|
import ch.ethz.seb.sebserver.gbl.model.Domain.EXAM;
|
||||||
import ch.ethz.seb.sebserver.gbl.model.Page;
|
import ch.ethz.seb.sebserver.gbl.model.Page;
|
||||||
import ch.ethz.seb.sebserver.gbl.model.sebconfig.ConfigCopyInfo;
|
import ch.ethz.seb.sebserver.gbl.model.sebconfig.ConfigCopyInfo;
|
||||||
import ch.ethz.seb.sebserver.gbl.model.sebconfig.ConfigKey;
|
import ch.ethz.seb.sebserver.gbl.model.sebconfig.ConfigKey;
|
||||||
import ch.ethz.seb.sebserver.gbl.model.sebconfig.Configuration;
|
import ch.ethz.seb.sebserver.gbl.model.sebconfig.Configuration;
|
||||||
import ch.ethz.seb.sebserver.gbl.model.sebconfig.ConfigurationNode;
|
import ch.ethz.seb.sebserver.gbl.model.sebconfig.ConfigurationNode;
|
||||||
|
import ch.ethz.seb.sebserver.gbl.model.sebconfig.ConfigurationNode.ConfigurationStatus;
|
||||||
import ch.ethz.seb.sebserver.gbl.model.sebconfig.ConfigurationNode.ConfigurationType;
|
import ch.ethz.seb.sebserver.gbl.model.sebconfig.ConfigurationNode.ConfigurationType;
|
||||||
import ch.ethz.seb.sebserver.gbl.model.sebconfig.TemplateAttribute;
|
import ch.ethz.seb.sebserver.gbl.model.sebconfig.TemplateAttribute;
|
||||||
import ch.ethz.seb.sebserver.gbl.model.user.UserLogActivityType;
|
import ch.ethz.seb.sebserver.gbl.model.user.UserLogActivityType;
|
||||||
|
@ -222,11 +225,50 @@ public class ConfigurationNodeController extends EntityController<ConfigurationN
|
||||||
}
|
}
|
||||||
|
|
||||||
@RequestMapping(
|
@RequestMapping(
|
||||||
path = API.MODEL_ID_VAR_PATH_SEGMENT + API.CONFIGURATION_IMPORT_PATH_SEGMENT,
|
path = API.CONFIGURATION_IMPORT_PATH_SEGMENT,
|
||||||
method = RequestMethod.POST,
|
method = RequestMethod.POST,
|
||||||
consumes = MediaType.APPLICATION_OCTET_STREAM_VALUE,
|
consumes = MediaType.APPLICATION_OCTET_STREAM_VALUE,
|
||||||
produces = MediaType.APPLICATION_JSON_UTF8_VALUE)
|
produces = MediaType.APPLICATION_JSON_UTF8_VALUE)
|
||||||
public Object importExamConfig(
|
public Object importExamConfig(
|
||||||
|
@RequestHeader(name = Domain.CONFIGURATION_NODE.ATTR_NAME, required = false) final String name,
|
||||||
|
@RequestHeader(name = Domain.CONFIGURATION_NODE.ATTR_DESCRIPTION,
|
||||||
|
required = false) final String description,
|
||||||
|
@RequestHeader(name = Domain.CONFIGURATION_NODE.ATTR_TEMPLATE_ID, required = false) final String templateId,
|
||||||
|
@RequestHeader(name = API.IMPORT_PASSWORD_ATTR_NAME, required = false) final String password,
|
||||||
|
@RequestParam(
|
||||||
|
name = API.PARAM_INSTITUTION_ID,
|
||||||
|
required = true,
|
||||||
|
defaultValue = UserService.USERS_INSTITUTION_AS_DEFAULT) final Long institutionId,
|
||||||
|
final HttpServletRequest request) throws IOException {
|
||||||
|
|
||||||
|
this.checkModifyPrivilege(institutionId);
|
||||||
|
|
||||||
|
final SEBServerUser currentUser = this.authorization.getUserService().getCurrentUser();
|
||||||
|
|
||||||
|
final ConfigurationNode configurationNode = new ConfigurationNode(
|
||||||
|
null,
|
||||||
|
institutionId,
|
||||||
|
StringUtils.isNotBlank(templateId) ? Long.parseLong(templateId) : null,
|
||||||
|
name,
|
||||||
|
description,
|
||||||
|
ConfigurationType.EXAM_CONFIG,
|
||||||
|
currentUser.uuid(),
|
||||||
|
ConfigurationStatus.CONSTRUCTION);
|
||||||
|
|
||||||
|
final Configuration followup = this.beanValidationService.validateBean(configurationNode)
|
||||||
|
.flatMap(this.entityDAO::createNew)
|
||||||
|
.flatMap(this.configurationDAO::getFollowupConfiguration)
|
||||||
|
.getOrThrow();
|
||||||
|
|
||||||
|
return doImport(password, request, followup);
|
||||||
|
}
|
||||||
|
|
||||||
|
@RequestMapping(
|
||||||
|
path = API.MODEL_ID_VAR_PATH_SEGMENT + API.CONFIGURATION_IMPORT_PATH_SEGMENT,
|
||||||
|
method = RequestMethod.POST,
|
||||||
|
consumes = MediaType.APPLICATION_OCTET_STREAM_VALUE,
|
||||||
|
produces = MediaType.APPLICATION_JSON_UTF8_VALUE)
|
||||||
|
public Object importExamConfigOnExistingConfig(
|
||||||
@PathVariable final Long modelId,
|
@PathVariable final Long modelId,
|
||||||
@RequestHeader(name = API.IMPORT_PASSWORD_ATTR_NAME, required = false) final String password,
|
@RequestHeader(name = API.IMPORT_PASSWORD_ATTR_NAME, required = false) final String password,
|
||||||
@RequestParam(
|
@RequestParam(
|
||||||
|
@ -235,27 +277,15 @@ public class ConfigurationNodeController extends EntityController<ConfigurationN
|
||||||
defaultValue = UserService.USERS_INSTITUTION_AS_DEFAULT) final Long institutionId,
|
defaultValue = UserService.USERS_INSTITUTION_AS_DEFAULT) final Long institutionId,
|
||||||
final HttpServletRequest request) throws IOException {
|
final HttpServletRequest request) throws IOException {
|
||||||
|
|
||||||
final InputStream inputStream = new BufferedInputStream(request.getInputStream());
|
this.entityDAO.byPK(modelId)
|
||||||
try {
|
.flatMap(this.authorization::checkModify);
|
||||||
|
|
||||||
return this.sebExamConfigService.importFromSEBFile(
|
final Configuration newConfig = this.configurationDAO
|
||||||
modelId,
|
.saveToHistory(modelId)
|
||||||
inputStream,
|
.flatMap(this.configurationDAO::restoreToDefaultValues)
|
||||||
password)
|
.getOrThrow();
|
||||||
.getOrThrow();
|
|
||||||
|
|
||||||
} catch (final Exception e) {
|
return doImport(password, request, newConfig);
|
||||||
// NOTE: It seems that this has to be manually closed on error case
|
|
||||||
// We expected that this is closed by the API but if this manual close is been left
|
|
||||||
// some left-overs will affect strange behavior.
|
|
||||||
// TODO: find a better solution for this
|
|
||||||
IOUtils.closeQuietly(inputStream);
|
|
||||||
//throw e;
|
|
||||||
return new ResponseEntity<>(
|
|
||||||
Arrays.asList(APIMessage.ErrorMessage.UNEXPECTED.of(e.getMessage())),
|
|
||||||
Utils.createJsonContentHeader(),
|
|
||||||
HttpStatus.BAD_REQUEST);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@RequestMapping(
|
@RequestMapping(
|
||||||
|
@ -444,4 +474,31 @@ public class ConfigurationNodeController extends EntityController<ConfigurationN
|
||||||
return node;
|
return node;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private Object doImport(
|
||||||
|
final String password,
|
||||||
|
final HttpServletRequest request,
|
||||||
|
final Configuration configuration) throws IOException {
|
||||||
|
final InputStream inputStream = new BufferedInputStream(request.getInputStream());
|
||||||
|
try {
|
||||||
|
|
||||||
|
return this.sebExamConfigService.importFromSEBFile(
|
||||||
|
configuration,
|
||||||
|
inputStream,
|
||||||
|
password)
|
||||||
|
.getOrThrow();
|
||||||
|
|
||||||
|
} catch (final Exception e) {
|
||||||
|
// NOTE: It seems that this has to be manually closed on error case
|
||||||
|
// We expected that this is closed by the API but if this manual close is been left
|
||||||
|
// some left-overs will affect strange behavior.
|
||||||
|
// TODO: find a better solution for this
|
||||||
|
IOUtils.closeQuietly(inputStream);
|
||||||
|
|
||||||
|
return new ResponseEntity<>(
|
||||||
|
Arrays.asList(APIMessage.ErrorMessage.UNEXPECTED.of(e.getMessage())),
|
||||||
|
Utils.createJsonContentHeader(),
|
||||||
|
HttpStatus.BAD_REQUEST);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in a new issue