From 82d54c4bc0fe4abbecf25d002c4d77987b1b5f2a Mon Sep 17 00:00:00 2001
From: Adrian Fish
Date: Tue, 3 Dec 2024 18:20:21 +0000
Subject: [PATCH] SAK-50724 rubric Implement archive/merge
https://sakaiproject.atlassian.net/browse/SAK-50724
---
.../archive/impl/SiteArchiver.java | 2 +-
.../impl2/src/webapp/WEB-INF/components.xml | 1 +
.../config/bundle/default.sakai.properties | 2 +-
.../rubrics/api/RubricsConstants.java | 55 ++---
.../rubrics/api/RubricsService.java | 3 +-
.../rubrics/api/model/Rubric.java | 8 +-
rubrics/impl/pom.xml | 5 +
.../rubrics/impl/RubricsServiceImpl.java | 193 ++++++++++++++-
.../impl/test/RubricsServiceTests.java | 219 +++++++++++++++++-
.../src/test/resources/archive/rubrics.xml | 6 +
.../controllers/RubricsRestController.java | 8 +-
.../packages/sakai-editor/src/SakaiEditor.js | 8 +-
12 files changed, 455 insertions(+), 55 deletions(-)
create mode 100644 rubrics/impl/src/test/resources/archive/rubrics.xml
diff --git a/common/archive-impl/impl2/src/java/org/sakaiproject/archive/impl/SiteArchiver.java b/common/archive-impl/impl2/src/java/org/sakaiproject/archive/impl/SiteArchiver.java
index 7fd154ccd889..372be1cf6df8 100644
--- a/common/archive-impl/impl2/src/java/org/sakaiproject/archive/impl/SiteArchiver.java
+++ b/common/archive-impl/impl2/src/java/org/sakaiproject/archive/impl/SiteArchiver.java
@@ -312,7 +312,7 @@ protected String archiveSite(Site site, Document doc, Stack stack, String fromSy
if ( pattern != null ) {
NodeList nl = siteNode.getElementsByTagName("property");
- List toRemove = new ArrayList();
+ List toRemove = new ArrayList<>();
for(int i = 0; i < nl.getLength(); i++) {
Element proptag = (Element)nl.item(i);
diff --git a/common/archive-impl/impl2/src/webapp/WEB-INF/components.xml b/common/archive-impl/impl2/src/webapp/WEB-INF/components.xml
index b3d1df40d644..a697d2bce781 100644
--- a/common/archive-impl/impl2/src/webapp/WEB-INF/components.xml
+++ b/common/archive-impl/impl2/src/webapp/WEB-INF/components.xml
@@ -32,6 +32,7 @@
DiscussionForumService
WebService
LessonBuilderEntityProducer
+ rubrics
diff --git a/config/configuration/bundles/src/bundle/org/sakaiproject/config/bundle/default.sakai.properties b/config/configuration/bundles/src/bundle/org/sakaiproject/config/bundle/default.sakai.properties
index 0826c4c7b0b0..236f6b3b2fd5 100644
--- a/config/configuration/bundles/src/bundle/org/sakaiproject/config/bundle/default.sakai.properties
+++ b/config/configuration/bundles/src/bundle/org/sakaiproject/config/bundle/default.sakai.properties
@@ -533,7 +533,7 @@
# archive.merge.filter.services=false
# List of data service types that can merge in data from an archive
-# DEFAULT: AnnouncementService,AssignmentService,ContentHostingService,CalendarService,ChatEntityProducer,DiscussionService,MailArchiveService,SyllabusService,RWikiObjectService,DiscussionForumService,WebService,LessonBuilderEntityProducer
+# DEFAULT: AnnouncementService,AssignmentService,ContentHostingService,CalendarService,ChatEntityProducer,DiscussionService,MailArchiveService,SyllabusService,RWikiObjectService,DiscussionForumService,WebService,LessonBuilderEntityProducer,RubricsService
# archive.merge.filtered.services={list of service names}
# Controls if user role filtering is enabled. If enabled, any user roles not in the list cannot archive or merge data
diff --git a/rubrics/api/src/main/java/org/sakaiproject/rubrics/api/RubricsConstants.java b/rubrics/api/src/main/java/org/sakaiproject/rubrics/api/RubricsConstants.java
index 3d80f5a34e40..c2afba871743 100644
--- a/rubrics/api/src/main/java/org/sakaiproject/rubrics/api/RubricsConstants.java
+++ b/rubrics/api/src/main/java/org/sakaiproject/rubrics/api/RubricsConstants.java
@@ -24,27 +24,27 @@
public interface RubricsConstants {
- public static final String RBCS_TOOL = "sakai.rubrics";
- public static final String RBCS_TOOL_ASSIGNMENT_GRADES = "sakai.assignment.grades";
- public static final String RBCS_TOOL_FORUMS = "sakai.forums";
- public static final String RBCS_TOOL_GRADEBOOKNG = "sakai.gradebookng";
- public static final String RBCS_TOOL_SAMIGO = "sakai.samigo";
- public static final String RBCS_TOOL_LESSONBUILDERTOOL = "sakai.lessonbuildertool";
+ public static final String RBCS_TOOL = "sakai.rubrics";
+ public static final String RBCS_TOOL_ASSIGNMENT_GRADES = "sakai.assignment.grades";
+ public static final String RBCS_TOOL_FORUMS = "sakai.forums";
+ public static final String RBCS_TOOL_GRADEBOOKNG = "sakai.gradebookng";
+ public static final String RBCS_TOOL_SAMIGO = "sakai.samigo";
+ public static final String RBCS_TOOL_LESSONBUILDERTOOL = "sakai.lessonbuildertool";
- public static final String RBCS_PREFIX = "rbcs-";
- public static final String RBCS_CONFIG_PREFIX = "config-";
- public static final String RBCS_CONFIG = RBCS_PREFIX + RBCS_CONFIG_PREFIX;
- public static final String RBCS_MULTIPLE_OPTIONS_CONFIG = RBCS_PREFIX + "multiple-options-config-";
- public static final String RBCS_ASSOCIATION_STATE_DETAILS = RBCS_PREFIX + "state-details";
- public static final String RBCS_ASSOCIATE_SUFFIX = "associate";
- public static final String RBCS_ASSOCIATE = RBCS_PREFIX + RBCS_ASSOCIATE_SUFFIX;// values: 0 or empty no association, 1 regular association, 2 dynamic rubric in Samigo
- public static final String RBCS_LIST_SUFFIX = "rubricslist";
- public static final String RBCS_LIST = RBCS_PREFIX + RBCS_LIST_SUFFIX;
- public static final String RBCS_SOFT_DELETED = RBCS_PREFIX + "soft-deleted";
- public static final String RBCS_ASSESSOR_ID = "assessorId";
- public static final String RBCS_HAS_ASSOCIATED_RUBRIC = "hasAssociatedRubric";
- public static final String RBCS_STUDENT_SELF_REPORT = "studentSelfReport";
- public static final String RBCS_STUDENT_SELF_REPORT_MODE = "studentSelfReportMode";
+ public static final String RBCS_PREFIX = "rbcs-";
+ public static final String RBCS_CONFIG_PREFIX = "config-";
+ public static final String RBCS_CONFIG = RBCS_PREFIX + RBCS_CONFIG_PREFIX;
+ public static final String RBCS_MULTIPLE_OPTIONS_CONFIG = RBCS_PREFIX + "multiple-options-config-";
+ public static final String RBCS_ASSOCIATION_STATE_DETAILS = RBCS_PREFIX + "state-details";
+ public static final String RBCS_ASSOCIATE_SUFFIX = "associate";
+ public static final String RBCS_ASSOCIATE = RBCS_PREFIX + RBCS_ASSOCIATE_SUFFIX;// values: 0 or empty no association, 1 regular association, 2 dynamic rubric in Samigo
+ public static final String RBCS_LIST_SUFFIX = "rubricslist";
+ public static final String RBCS_LIST = RBCS_PREFIX + RBCS_LIST_SUFFIX;
+ public static final String RBCS_SOFT_DELETED = RBCS_PREFIX + "soft-deleted";
+ public static final String RBCS_ASSESSOR_ID = "assessorId";
+ public static final String RBCS_HAS_ASSOCIATED_RUBRIC = "hasAssociatedRubric";
+ public static final String RBCS_STUDENT_SELF_REPORT = "studentSelfReport";
+ public static final String RBCS_STUDENT_SELF_REPORT_MODE = "studentSelfReportMode";
public static final String RBCS_PERMISSIONS_EVALUATOR = "rubrics.evaluator";
public static final String RBCS_PERMISSIONS_EDITOR = "rubrics.editor";
@@ -53,12 +53,13 @@ public interface RubricsConstants {
public static final String RBCS_EXPORT_PDF = "rubrics.export.pdf";
- //samigo custom props
- public static final String RBCS_PUBLISHED_ASSESSMENT_ENTITY_PREFIX = "pub.";
+ //samigo custom props
+ public static final String RBCS_PUBLISHED_ASSESSMENT_ENTITY_PREFIX = "pub.";
- //forums custom props
- public static final String RBCS_FORUM_ENTITY_PREFIX = "for.";
- public static final String RBCS_TOPIC_ENTITY_PREFIX = "top.";
- public static final String RBCS_MSG_ENTITY_PREFIX = "msg.";
-
+ //forums custom props
+ public static final String RBCS_FORUM_ENTITY_PREFIX = "for.";
+ public static final String RBCS_TOPIC_ENTITY_PREFIX = "top.";
+ public static final String RBCS_MSG_ENTITY_PREFIX = "msg.";
+
+ public static final String LINE_SEPARATOR = System.getProperty("line.separator");
}
diff --git a/rubrics/api/src/main/java/org/sakaiproject/rubrics/api/RubricsService.java b/rubrics/api/src/main/java/org/sakaiproject/rubrics/api/RubricsService.java
index 547033ffe5b7..16f634b65536 100644
--- a/rubrics/api/src/main/java/org/sakaiproject/rubrics/api/RubricsService.java
+++ b/rubrics/api/src/main/java/org/sakaiproject/rubrics/api/RubricsService.java
@@ -29,6 +29,7 @@
import java.util.Optional;
import org.sakaiproject.entity.api.Entity;
+import org.sakaiproject.entity.api.EntityProducer;
import org.sakaiproject.rubrics.api.beans.AssociationTransferBean;
import org.sakaiproject.rubrics.api.beans.CriterionTransferBean;
import org.sakaiproject.rubrics.api.beans.EvaluationTransferBean;
@@ -37,7 +38,7 @@
import org.sakaiproject.rubrics.api.model.Rubric;
import org.sakaiproject.rubrics.api.model.ToolItemRubricAssociation;
-public interface RubricsService {
+public interface RubricsService extends EntityProducer {
public static final String REFERENCE_ROOT = Entity.SEPARATOR + "rubrics";
diff --git a/rubrics/api/src/main/java/org/sakaiproject/rubrics/api/model/Rubric.java b/rubrics/api/src/main/java/org/sakaiproject/rubrics/api/model/Rubric.java
index 7ba8d49f9a10..7adb886621e6 100644
--- a/rubrics/api/src/main/java/org/sakaiproject/rubrics/api/model/Rubric.java
+++ b/rubrics/api/src/main/java/org/sakaiproject/rubrics/api/model/Rubric.java
@@ -65,6 +65,7 @@ public class Rubric implements PersistableEntity, Serializable, Cloneable
@GeneratedValue(strategy = GenerationType.AUTO, generator ="rbc_seq" )
private Long id;
+ @Column(nullable = false)
private String title;
private Boolean weighted = Boolean.FALSE;
@@ -81,14 +82,15 @@ public class Rubric implements PersistableEntity, Serializable, Cloneable
@OneToMany(fetch = FetchType.LAZY, cascade = CascadeType.ALL, orphanRemoval = true, mappedBy = "rubric")
private List associations = new ArrayList<>();
- private Instant created;
+ @Column(nullable = false)
+ private Instant created = Instant.now();
private Instant modified;
- @Column(length = 99)
+ @Column(length = 99, nullable = false)
private String ownerId;
- @Column(length = 99)
+ @Column(length = 99, nullable = false)
private String creatorId;
private Boolean shared = Boolean.FALSE;
diff --git a/rubrics/impl/pom.xml b/rubrics/impl/pom.xml
index 8a920dbf7778..1cc33e80ddac 100644
--- a/rubrics/impl/pom.xml
+++ b/rubrics/impl/pom.xml
@@ -119,6 +119,11 @@
org.sakaiproject.edu-services.sections
sections-api
+
+ org.sakaiproject.common
+ archive-api
+ test
+
diff --git a/rubrics/impl/src/main/java/org/sakaiproject/rubrics/impl/RubricsServiceImpl.java b/rubrics/impl/src/main/java/org/sakaiproject/rubrics/impl/RubricsServiceImpl.java
index 3c81f03cbb90..c47487272029 100644
--- a/rubrics/impl/src/main/java/org/sakaiproject/rubrics/impl/RubricsServiceImpl.java
+++ b/rubrics/impl/src/main/java/org/sakaiproject/rubrics/impl/RubricsServiceImpl.java
@@ -25,6 +25,7 @@
import static org.sakaiproject.rubrics.api.RubricsConstants.RBCS_CONFIG;
import static org.sakaiproject.rubrics.api.RubricsConstants.RBCS_MULTIPLE_OPTIONS_CONFIG;
import static org.sakaiproject.rubrics.api.RubricsConstants.RBCS_PREFIX;
+import static org.sakaiproject.rubrics.api.RubricsConstants.LINE_SEPARATOR;
import java.awt.Color;
import java.io.ByteArrayOutputStream;
@@ -43,10 +44,11 @@
import java.util.Map;
import java.util.Objects;
import java.util.Optional;
+import java.util.Set;
+import java.util.Stack;
import java.util.function.Predicate;
import java.util.stream.Collectors;
-import org.apache.commons.lang3.BooleanUtils;
import org.apache.commons.lang3.StringUtils;
import org.apache.commons.lang3.math.NumberUtils;
import org.jsoup.Jsoup;
@@ -58,7 +60,6 @@
import org.sakaiproject.component.api.ServerConfigurationService;
import org.sakaiproject.entity.api.Entity;
import org.sakaiproject.entity.api.EntityManager;
-import org.sakaiproject.entity.api.EntityProducer;
import org.sakaiproject.entity.api.EntityTransferrer;
import org.sakaiproject.entity.api.Reference;
import org.sakaiproject.event.api.EventTrackingService;
@@ -88,11 +89,8 @@
import org.sakaiproject.rubrics.api.repository.RatingRepository;
import org.sakaiproject.rubrics.api.repository.ReturnedEvaluationRepository;
import org.sakaiproject.rubrics.api.repository.RubricRepository;
-import org.sakaiproject.site.api.Group;
-import org.sakaiproject.site.api.Site;
import org.sakaiproject.site.api.SiteService;
import org.sakaiproject.time.api.UserTimeService;
-import org.sakaiproject.tool.assessment.data.ifc.assessment.PublishedAssessmentIfc;
import org.sakaiproject.tool.assessment.facade.PublishedAssessmentFacadeQueriesAPI;
import org.sakaiproject.tool.api.SessionManager;
import org.sakaiproject.tool.api.ToolManager;
@@ -107,7 +105,6 @@
import com.lowagie.text.Chunk;
import com.lowagie.text.Document;
-import com.lowagie.text.Element;
import com.lowagie.text.Font;
import com.lowagie.text.FontFactory;
import com.lowagie.text.PageSize;
@@ -119,10 +116,14 @@
import lombok.Setter;
import lombok.extern.slf4j.Slf4j;
+import org.w3c.dom.CDATASection;
+import org.w3c.dom.Element;
+import org.w3c.dom.NodeList;
+
@Slf4j
@Setter
@Transactional
-public class RubricsServiceImpl implements RubricsService, EntityProducer, EntityTransferrer {
+public class RubricsServiceImpl implements RubricsService, EntityTransferrer {
private static final Font BOLD_FONT = FontFactory.getFont(FontFactory.HELVETICA, 10, Font.BOLD);
private static final Font NORMAL_FONT = FontFactory.getFont(FontFactory.HELVETICA, 7, Font.NORMAL);
@@ -1280,7 +1281,7 @@ public byte[] createPdf(String siteId, Long rubricId, String toolId, String item
PdfPCell header = new PdfPCell();
Paragraph paragraph = new Paragraph(resourceLoader.getFormattedMessage("export_rubric_title", rubric.getTitle() + "\n"), BOLD_FONT);
- paragraph.setAlignment(Element.ALIGN_LEFT);
+ paragraph.setAlignment(com.lowagie.text.Element.ALIGN_LEFT);
try {
String siteTitle = siteService.getSite(rubric.getOwnerId()).getTitle();
paragraph.add(resourceLoader.getFormattedMessage("export_rubric_site", siteTitle));
@@ -1522,6 +1523,181 @@ public Map transferCopyEntities(String fromContext, String toCon
return transferCopyEntities(fromContext, toContext, ids, null);
}
+ @Override
+ public String getLabel() {
+ return "rubrics";
+ }
+
+ @Override
+ public boolean willArchiveMerge() {
+ return true;
+ }
+
+ @Override
+ public String archive(String siteId, org.w3c.dom.Document doc, Stack stack, String archivePath, List attachments) {
+
+ StringBuilder results = new StringBuilder();
+ results.append("begin archiving ").append(getLabel()).append(" for site ").append(siteId).append(LINE_SEPARATOR);
+
+ Element rubrics = doc.createElement(getLabel());
+ stack.peek().appendChild(rubrics);
+ stack.push(rubrics);
+
+ rubricRepository.findByOwnerId(siteId).stream().sorted((r1, r2) -> r1.getTitle().compareTo(r2.getTitle())).forEach(rubric -> {
+
+ Element rubricEl = doc.createElement("rubric");
+ rubrics.appendChild(rubricEl);
+
+ rubricEl.setAttribute("title", rubric.getTitle());
+ rubricEl.setAttribute("created", Long.toString(rubric.getCreated().getEpochSecond()));
+ rubricEl.setAttribute("creator", rubric.getCreatorId());
+ rubricEl.setAttribute("weighted", Boolean.toString(rubric.getWeighted()));
+ rubricEl.setAttribute("adhoc", Boolean.toString(rubric.getAdhoc()));
+ rubricEl.setAttribute("max-points", Double.toString(rubric.getMaxPoints()));
+
+ Element criteriaEl = doc.createElement("criteria");
+ rubricEl.appendChild(criteriaEl);
+
+ rubric.getCriteria().forEach(criterion -> {
+
+ Element criterionEl = doc.createElement("criterion");
+ criteriaEl.appendChild(criterionEl);
+
+ criterionEl.setAttribute("title", criterion.getTitle());
+ Float weight = criterion.getWeight();
+ if (weight != null) {
+ criterionEl.setAttribute("weight", Float.toString(weight));
+ }
+
+ String description = criterion.getDescription();
+ if (StringUtils.isNotBlank(description)) {
+ Element descriptionEl = doc.createElement("description");
+ criterionEl.appendChild(descriptionEl);
+ descriptionEl.appendChild(doc.createCDATASection(description));
+ }
+
+ Element ratingsEl = doc.createElement("ratings");
+ criterionEl.appendChild(ratingsEl);
+
+ criterion.getRatings().forEach(rating -> {
+
+ Element ratingEl = doc.createElement("rating");
+ ratingsEl.appendChild(ratingEl);
+ ratingEl.setAttribute("title", rating.getTitle());
+ ratingEl.setAttribute("points", Double.toString(rating.getPoints()));
+ String ratingDescription = rating.getDescription();
+ if (StringUtils.isNotBlank(ratingDescription)) {
+ Element ratingDescriptionEl = doc.createElement("description");
+ ratingEl.appendChild(ratingDescriptionEl);
+ ratingDescriptionEl.appendChild(doc.createCDATASection(ratingDescription));
+ }
+ });
+ });
+ });
+
+ stack.pop();
+
+ results.append("completed archiving ").append(getLabel()).append(" for site ").append(siteId).append(LINE_SEPARATOR);
+ return results.toString();
+ }
+
+ @Override
+ public String merge(String toSiteId, Element root, String archivePath, String fromSiteId, Map attachmentNames, Map userIdTrans, Set userListAllowImport) {
+
+ StringBuilder results = new StringBuilder();
+ results.append("begin merging ").append(getLabel()).append(" for site ").append(toSiteId).append(LINE_SEPARATOR);
+
+ String currentUserId = sessionManager.getCurrentSessionUserId();
+
+ if (!root.getTagName().equals(getLabel())) {
+ log.warn("Tried to merge a non <{}> xml document", getLabel());
+ return "Invalid xml document";
+ }
+
+ NodeList rubricNodes = root.getElementsByTagName("rubric");
+
+ for (int i = 0; i < rubricNodes.getLength(); i++) {
+ Element rubricEl = (Element) rubricNodes.item(i);
+
+ String creatorId = currentUserId;
+
+ String originalCreatorId = rubricEl.getAttribute("creator");
+ if (StringUtils.isNotBlank(originalCreatorId)) {
+ try {
+ userDirectoryService.getUser(originalCreatorId);
+ creatorId = originalCreatorId;
+ } catch (UserNotDefinedException unde) {
+ }
+ }
+
+ RubricTransferBean rubricBean = new RubricTransferBean();
+ rubricBean.setOwnerId(toSiteId);
+ // TODO: Do we honour the original creator, or the current user?
+ rubricBean.setCreatorId(creatorId);
+ // TODO: Do we honour the original created date, or use now?
+ rubricBean.setCreated(Instant.ofEpochSecond(Long.parseLong(rubricEl.getAttribute("created"))));
+ rubricBean.setTitle(rubricEl.getAttribute("title"));
+ rubricBean.setMaxPoints(Double.parseDouble(rubricEl.getAttribute("max-points")));
+ rubricBean.setWeighted(Boolean.parseBoolean(rubricEl.getAttribute("weighted")));
+ rubricBean.setAdhoc(Boolean.parseBoolean(rubricEl.getAttribute("adhoc")));
+ rubricBean.setDraft(false);
+
+ NodeList criteriaNodes = rubricEl.getElementsByTagName("criteria");
+ if (criteriaNodes.getLength() != 1) {
+ log.warn("No criteria element in rubrics archive XML");
+ continue;
+ }
+
+ NodeList criterionNodes = ((Element) criteriaNodes.item(0)).getElementsByTagName("criterion");
+
+ List criteria = new ArrayList<>();
+ for (int j = 0; j < criterionNodes.getLength(); j++) {
+ Element criterionEl = (Element) criterionNodes.item(j);
+ CriterionTransferBean criterionBean = new CriterionTransferBean();
+ criterionBean.setTitle(criterionEl.getAttribute("title"));
+
+ NodeList descriptionNodes = criterionEl.getElementsByTagName("description");
+ if (descriptionNodes.getLength() == 1) {
+ CDATASection descriptionSection = (CDATASection) descriptionNodes.item(0).getFirstChild();
+ if (descriptionSection != null) {
+ criterionBean.setDescription(descriptionSection.getNodeValue());
+ }
+ }
+
+ NodeList ratingsNodes = criterionEl.getElementsByTagName("ratings");
+ if (ratingsNodes.getLength() == 1) {
+ NodeList ratingNodes = ((Element) ratingsNodes.item(0)).getElementsByTagName("rating");
+ List ratings = new ArrayList<>();
+ for (int k = 0; k < ratingNodes.getLength(); k++) {
+ Element ratingEl = (Element) ratingNodes.item(k);
+ RatingTransferBean ratingBean = new RatingTransferBean();
+ ratingBean.setTitle(ratingEl.getAttribute("title"));
+ ratingBean.setPoints(Double.parseDouble(ratingEl.getAttribute("points")));
+
+ NodeList ratingDescriptionNodes = ratingEl.getElementsByTagName("description");
+ if (ratingDescriptionNodes.getLength() == 1) {
+ CDATASection ratingDescriptionSection = (CDATASection) ratingDescriptionNodes.item(0).getFirstChild();
+ if (ratingDescriptionSection != null) {
+ ratingBean.setDescription(ratingDescriptionSection.getNodeValue());
+ }
+ }
+ ratings.add(ratingBean);
+ }
+ criterionBean.setRatings(ratings);
+ }
+
+ criteria.add(criterionBean);
+ }
+
+ rubricBean.setCriteria(criteria);
+
+ saveRubric(rubricBean);
+ }
+
+ results.append("completed merging ").append(getLabel()).append(" for site ").append(toSiteId).append(LINE_SEPARATOR);
+ return results.toString();
+ }
+
@Override
public String[] myToolIds() {
return new String[] { RubricsConstants.RBCS_TOOL };
@@ -1636,7 +1812,6 @@ private Optional getCriterionPoints(Criterion cri, Evaluation evaluation
}
private boolean isEditor(String siteId) {
-
return securityService.unlock(RubricsConstants.RBCS_PERMISSIONS_EDITOR, siteService.siteReference(siteId));
}
diff --git a/rubrics/impl/src/test/org/sakaiproject/rubrics/impl/test/RubricsServiceTests.java b/rubrics/impl/src/test/org/sakaiproject/rubrics/impl/test/RubricsServiceTests.java
index 9e419037c75b..6d00bfd9ca87 100644
--- a/rubrics/impl/src/test/org/sakaiproject/rubrics/impl/test/RubricsServiceTests.java
+++ b/rubrics/impl/src/test/org/sakaiproject/rubrics/impl/test/RubricsServiceTests.java
@@ -15,6 +15,8 @@
*/
package org.sakaiproject.rubrics.impl.test;
+import org.apache.commons.lang3.StringUtils;
+
import static org.junit.Assert.assertEquals;
import static org.junit.Assert.assertFalse;
import static org.junit.Assert.assertNotNull;
@@ -30,11 +32,14 @@
import java.util.Locale;
import java.util.Map;
import java.util.Optional;
+import java.util.Stack;
import org.hibernate.SessionFactory;
import org.junit.Before;
import org.junit.Test;
import org.junit.runner.RunWith;
+
+import org.sakaiproject.archive.api.ArchiveService;
import org.sakaiproject.authz.api.SecurityService;
import org.sakaiproject.rubrics.api.RubricsConstants;
import org.sakaiproject.rubrics.api.RubricsService;
@@ -47,12 +52,14 @@
import org.sakaiproject.rubrics.api.model.EvaluatedItemOwnerType;
import org.sakaiproject.rubrics.api.model.Evaluation;
import org.sakaiproject.rubrics.api.model.EvaluationStatus;
+import org.sakaiproject.rubrics.api.model.Rating;
import org.sakaiproject.rubrics.api.model.ReturnedEvaluation;
import org.sakaiproject.rubrics.api.model.Rubric;
import org.sakaiproject.rubrics.api.model.ToolItemRubricAssociation;
import org.sakaiproject.rubrics.api.repository.AssociationRepository;
import org.sakaiproject.rubrics.api.repository.CriterionRepository;
import org.sakaiproject.rubrics.api.repository.EvaluationRepository;
+import org.sakaiproject.rubrics.api.repository.RubricRepository;
import org.sakaiproject.rubrics.api.repository.ReturnedEvaluationRepository;
import org.sakaiproject.rubrics.impl.RubricsServiceImpl;
import org.sakaiproject.site.api.Site;
@@ -64,12 +71,24 @@
import org.sakaiproject.user.api.UserDirectoryService;
import org.sakaiproject.user.api.UserNotDefinedException;
import org.sakaiproject.util.ResourceLoader;
+import org.sakaiproject.util.Xml;
+
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.test.context.ContextConfiguration;
import org.springframework.test.context.junit4.AbstractTransactionalJUnit4SpringContextTests;
import org.springframework.test.context.junit4.SpringJUnit4ClassRunner;
import org.springframework.test.util.AopTestUtils;
+import javax.xml.parsers.DocumentBuilder;
+import javax.xml.parsers.DocumentBuilderFactory;
+import javax.xml.parsers.ParserConfigurationException;
+
+import org.w3c.dom.CDATASection;
+import org.w3c.dom.Document;
+import org.w3c.dom.Element;
+import org.w3c.dom.Node;
+import org.w3c.dom.NodeList;
+
import lombok.extern.slf4j.Slf4j;
@Slf4j
@@ -81,6 +100,7 @@ public class RubricsServiceTests extends AbstractTransactionalJUnit4SpringContex
@Autowired private CriterionRepository criterionRepository;
@Autowired private EvaluationRepository evaluationRepository;
@Autowired private ReturnedEvaluationRepository returnedEvaluationRepository;
+ @Autowired private RubricRepository rubricRepository;
@Autowired private RubricsService rubricsService;
@Autowired private SecurityService securityService;
@Autowired private SessionManager sessionManager;
@@ -471,8 +491,6 @@ public void deleteRating() {
int initialRatingsSize = criterionBean.getRatings().size();
RatingTransferBean ratingBean = criterionBean.getRatings().get(2);
- System.out.println(ratingBean.getPoints());
-
switchToUser1();
assertThrows(SecurityException.class, () -> rubricsService.deleteRating(ratingBean.getId(), criterionBean.getId(), siteId, rubricBean.getId()));
@@ -767,6 +785,7 @@ public void saveEvaluationAsTeachingAssistant() {
@Test
public void contextualFilenameNoEvaluation() {
+
switchToInstructor();
RubricTransferBean rubric = rubricsService.createDefaultRubric(siteId);
String toolId = "sakai.assignment";
@@ -782,6 +801,7 @@ public void contextualFilenameNoEvaluation() {
@Test
public void contextualFilename() throws UserNotDefinedException {
+
switchToInstructor();
RubricTransferBean rubric = rubricsService.createDefaultRubric(siteId);
String toolId = "sakai.assignment";
@@ -801,6 +821,184 @@ public void contextualFilename() throws UserNotDefinedException {
assertEquals(rubric.getTitle() + '_' + user1SortName, filename);
}
+ @Test
+ public void archive() {
+
+ switchToInstructor();
+
+ RubricTransferBean rubric1 = rubricsService.createDefaultRubric(siteId);
+ String title1 = "Cheese Sandwich";
+ rubric1.setTitle(title1);
+ String rubric1Criteria1Rating1Description = "This is great";
+ rubric1.getCriteria().get(0).getRatings().get(0).setDescription(rubric1Criteria1Rating1Description);
+ rubricsService.saveRubric(rubric1);
+
+ String rubric2CriteriaDescription = "Rate those sandwiches";
+ RubricTransferBean rubric2 = rubricsService.createDefaultRubric(siteId);
+ String title2 = "Egg Sandwich";
+ rubric2.setTitle(title2);
+ rubric2.getCriteria().get(0).setDescription(rubric2CriteriaDescription);
+ rubricsService.saveRubric(rubric2);
+
+ RubricTransferBean rubric3 = rubricsService.createDefaultRubric(siteId);
+ String title3 = "Ham Sandwich";
+ rubric3.setTitle(title3);
+ rubricsService.saveRubric(rubric3);
+
+ RubricTransferBean[] rubrics = new RubricTransferBean[] { rubric1, rubric2, rubric3 };
+
+ Document doc = Xml.createDocument();
+ Stack stack = new Stack<>();
+
+ Element root = doc.createElement("archive");
+ doc.appendChild(root);
+ root.setAttribute("source", siteId);
+ root.setAttribute("xmlns:sakai", ArchiveService.SAKAI_ARCHIVE_NS);
+ root.setAttribute("xmlns:CHEF", ArchiveService.SAKAI_ARCHIVE_NS.concat("CHEF"));
+ root.setAttribute("xmlns:DAV", ArchiveService.SAKAI_ARCHIVE_NS.concat("DAV"));
+ stack.push(root);
+
+ String results = rubricsService.archive(siteId, doc, stack, "", null);
+
+ assertEquals(1, stack.size());
+
+ NodeList rubricsNode = root.getElementsByTagName(rubricsService.getLabel());
+ assertEquals(1, rubricsNode.getLength());
+
+ NodeList rubricNodes = ((Element) rubricsNode.item(0)).getElementsByTagName("rubric");
+ assertEquals(3, rubricNodes.getLength());
+
+ for (int i = 0; i < rubricNodes.getLength(); i++) {
+
+ RubricTransferBean rubricBean = rubrics[i];
+ Element rubricEl = (Element) rubricNodes.item(i);
+
+ assertEquals(rubricBean.getTitle(), rubricEl.getAttribute("title"));
+ assertEquals(Double.toString(rubricBean.getMaxPoints()), rubricEl.getAttribute("max-points"));
+ assertEquals(Long.toString(rubricBean.getCreated().getEpochSecond()), rubricEl.getAttribute("created"));
+ assertEquals(Boolean.toString(rubricBean.getWeighted()), rubricEl.getAttribute("weighted"));
+ assertEquals(Boolean.toString(rubricBean.getAdhoc()), rubricEl.getAttribute("adhoc"));
+
+ NodeList criteriaNodes = rubricEl.getElementsByTagName("criteria");
+ assertEquals(1, criteriaNodes.getLength());
+
+ NodeList criterionNodes = ((Element) criteriaNodes.item(0)).getElementsByTagName("criterion");
+ assertEquals(rubricBean.getCriteria().size(), criterionNodes.getLength());
+
+ for (int j = 0; j < criterionNodes.getLength(); j++) {
+
+ CriterionTransferBean criterionBean = rubrics[i].getCriteria().get(j);
+ Element criterionEl = (Element) criterionNodes.item(j);
+
+ assertEquals(criterionBean.getTitle(), criterionEl.getAttribute("title"));
+ assertEquals(Float.toString(criterionBean.getWeight()), criterionEl.getAttribute("weight"));
+
+ if (i == 1 && j == 0) {
+ // We've only set a description on the first criteria of the second rubric, and
+ // descriptions are optional
+ NodeList descriptionNodes = criterionEl.getElementsByTagName("description");
+ assertEquals(1, descriptionNodes.getLength());
+ CDATASection descriptionNode = (CDATASection) descriptionNodes.item(0).getFirstChild();
+ assertNotNull(descriptionNode);
+ assertEquals(criterionBean.getDescription(), descriptionNode.getNodeValue());
+ }
+
+ NodeList ratingsNodes = criterionEl.getElementsByTagName("ratings");
+ assertEquals(1, ratingsNodes.getLength());
+
+ NodeList ratingNodes = ((Element) ratingsNodes.item(0)).getElementsByTagName("rating");
+ assertEquals(criterionBean.getRatings().size(), ratingNodes.getLength());
+
+ for (int k = 0; k < ratingNodes.getLength(); k++) {
+
+ RatingTransferBean ratingBean = criterionBean.getRatings().get(k);
+ Element ratingEl = (Element) ratingNodes.item(k);
+
+ assertEquals(ratingBean.getTitle(), ratingEl.getAttribute("title"));
+ assertEquals(Double.toString(ratingBean.getPoints()), ratingEl.getAttribute("points"));
+
+ String ratingDescription = ratingBean.getDescription();
+ if (StringUtils.isNotBlank(ratingDescription)) {
+ NodeList ratingDescriptionNodes = ratingEl.getElementsByTagName("description");
+ assertEquals(1, ratingDescriptionNodes.getLength());
+ }
+ }
+ }
+ }
+ }
+
+ @Test
+ public void merge() {
+
+ Document doc = Xml.readDocumentFromStream(this.getClass().getResourceAsStream("/archive/rubrics.xml"));
+
+ Element root = doc.getDocumentElement();
+
+ String fromSite = root.getAttribute("source");
+ String toSite = "my-new-site";
+
+ switchToInstructor(toSite);
+
+ Element rubricsElement = doc.createElement("not-rubrics");
+
+ rubricsService.merge(toSite, rubricsElement, "", fromSite, null, null, null);
+
+ assertEquals("Invalid xml document", rubricsService.merge(siteId, rubricsElement, "", fromSite, null, null, null));
+
+ rubricsElement = (Element) root.getElementsByTagName(rubricsService.getLabel()).item(0);
+
+ rubricsService.merge(toSite, rubricsElement, "", fromSite, null, null, null);
+
+ NodeList rubricNodes = rubricsElement.getElementsByTagName("rubric");
+
+ List rubrics = rubricRepository.findByOwnerId(toSite);
+
+ for (int i = 0; i < rubricNodes.getLength(); i++) {
+ Element rubricEl = (Element) rubricNodes.item(i);
+ String rubricTitle = rubricEl.getAttribute("title");
+ Optional optRubric = rubrics.stream().filter(r -> r.getTitle().equals(rubricTitle)).findAny();
+ assertTrue(optRubric.isPresent());
+ Rubric rubric = optRubric.get();
+ //assertEquals(Double.valueOf(rubricEl.getAttribute("max-points")), rubric.getMaxPoints());
+ assertEquals(Boolean.valueOf(rubricEl.getAttribute("adhoc")), rubric.getAdhoc());
+ assertEquals(Boolean.valueOf(rubricEl.getAttribute("weighted")), rubric.getWeighted());
+
+ NodeList criterionNodes = ((Element) rubricEl.getElementsByTagName("criteria").item(0))
+ .getElementsByTagName("criterion");
+ List criteria = rubric.getCriteria();
+ assertEquals(criterionNodes.getLength(), criteria.size());
+
+ for (int j = 0; j < criterionNodes.getLength(); j++) {
+ Element criterionEl = (Element) criterionNodes.item(j);
+ String criterionTitle = criterionEl.getAttribute("title");
+ Optional optCriterion = criteria.stream().filter(c -> c.getTitle().equals(criterionEl.getAttribute("title"))).findAny();
+ assertTrue(optCriterion.isPresent());
+ Criterion criterion = optCriterion.get();
+ NodeList descriptionNodes = criterionEl.getElementsByTagName("description");
+ if (descriptionNodes.getLength() == 1) {
+ assertEquals(descriptionNodes.item(0).getFirstChild().getNodeValue(), criterion.getDescription());
+ }
+
+ NodeList ratingNodes = ((Element) criterionEl.getElementsByTagName("ratings").item(0))
+ .getElementsByTagName("rating");
+ List ratings = criterion.getRatings();
+ for (int k = 0; k < ratingNodes.getLength(); k++) {
+ Element ratingEl = (Element) ratingNodes.item(j);
+ String ratingTitle = ratingEl.getAttribute("title");
+ Optional optRating = ratings.stream().filter(r -> r.getTitle().equals(ratingEl.getAttribute("title"))).findAny();
+ assertTrue(optRating.isPresent());
+ Rating rating = optRating.get();
+ assertEquals(Double.valueOf(ratingEl.getAttribute("points")), rating.getPoints());
+
+ NodeList ratingDescriptionNodes = ratingEl.getElementsByTagName("description");
+ if (ratingDescriptionNodes.getLength() == 1) {
+ assertEquals(ratingDescriptionNodes.item(0).getFirstChild().getNodeValue(), rating.getDescription());
+ }
+ }
+ }
+ }
+ }
+
private EvaluationTransferBean buildEvaluation(Long associationId, RubricTransferBean rubricBean, String toolItemId) {
EvaluationTransferBean etb = new EvaluationTransferBean();
@@ -824,6 +1022,7 @@ private EvaluationTransferBean buildEvaluation(Long associationId, RubricTransfe
}
private void setupStudentPermissions() {
+
when(securityService.unlock(RubricsConstants.RBCS_PERMISSIONS_EDITOR, siteRef)).thenReturn(false);
when(securityService.unlock(RubricsConstants.RBCS_PERMISSIONS_EVALUATOR, siteRef)).thenReturn(false);
when(securityService.unlock(RubricsConstants.RBCS_PERMISSIONS_EVALUEE, siteRef)).thenReturn(true);
@@ -866,10 +1065,20 @@ private void switchToUser3() {
}
private void switchToInstructor() {
+ switchToInstructor(null);
+ }
- when(securityService.unlock(RubricsConstants.RBCS_PERMISSIONS_EDITOR, siteRef)).thenReturn(true);
- when(securityService.unlock(RubricsConstants.RBCS_PERMISSIONS_EVALUATOR, siteRef)).thenReturn(true);
- when(securityService.unlock(RubricsConstants.RBCS_PERMISSIONS_EVALUEE, siteRef)).thenReturn(false);
+ private void switchToInstructor(String siteId) {
+
+ String ref = siteId != null ? "/site/" + siteId : siteRef;
+
+ if (siteId != null) {
+ when(siteService.siteReference(siteId)).thenReturn(ref);
+ }
+
+ when(securityService.unlock(RubricsConstants.RBCS_PERMISSIONS_EDITOR, ref)).thenReturn(true);
+ when(securityService.unlock(RubricsConstants.RBCS_PERMISSIONS_EVALUATOR, ref)).thenReturn(true);
+ when(securityService.unlock(RubricsConstants.RBCS_PERMISSIONS_EVALUEE, ref)).thenReturn(false);
when(sessionManager.getCurrentSessionUserId()).thenReturn(instructor);
when(userDirectoryService.getCurrentUser()).thenReturn(instructorUser);
diff --git a/rubrics/impl/src/test/resources/archive/rubrics.xml b/rubrics/impl/src/test/resources/archive/rubrics.xml
new file mode 100644
index 000000000000..27af7247a308
--- /dev/null
+++ b/rubrics/impl/src/test/resources/archive/rubrics.xml
@@ -0,0 +1,6 @@
+
+spin it around
+]]>some foods
+]]>rackets
+]]>pitch
+]]>
\ No newline at end of file
diff --git a/webapi/src/main/java/org/sakaiproject/webapi/controllers/RubricsRestController.java b/webapi/src/main/java/org/sakaiproject/webapi/controllers/RubricsRestController.java
index 06dfcf9dc47a..12739ec42cbf 100644
--- a/webapi/src/main/java/org/sakaiproject/webapi/controllers/RubricsRestController.java
+++ b/webapi/src/main/java/org/sakaiproject/webapi/controllers/RubricsRestController.java
@@ -85,7 +85,13 @@ List> getRubricsForSite(@PathVariable String sit
checkSakaiSession();
- return rubricsService.getRubricsForSite(siteId).stream().map(b -> entityModelForRubricBean(b)).collect(Collectors.toList());
+ try {
+ return rubricsService.getRubricsForSite(siteId).stream().map(b -> entityModelForRubricBean(b)).collect(Collectors.toList());
+ } catch (Exception e) {
+ e.printStackTrace();
+ }
+
+ return null;
}
@GetMapping(value = "/rubrics/shared", produces = MediaType.APPLICATION_JSON_VALUE)
diff --git a/webcomponents/tool/src/main/frontend/packages/sakai-editor/src/SakaiEditor.js b/webcomponents/tool/src/main/frontend/packages/sakai-editor/src/SakaiEditor.js
index 76d97bc6de44..04f162f91320 100644
--- a/webcomponents/tool/src/main/frontend/packages/sakai-editor/src/SakaiEditor.js
+++ b/webcomponents/tool/src/main/frontend/packages/sakai-editor/src/SakaiEditor.js
@@ -131,14 +131,8 @@ export class SakaiEditor extends SakaiElement {
render() {
- if (this.textarea) {
- return html `
-
- `;
- }
-
return html `
-
+
`;
}
}