From fb26109f6bfb08ee8d4235b06f7b229574cb0fbe Mon Sep 17 00:00:00 2001
From: Adrian Fish
Date: Tue, 10 Dec 2024 17:56:52 +0000
Subject: [PATCH] SAK-50748 conversations Implement archive/merge
https://sakaiproject.atlassian.net/browse/SAK-50748
---
.../impl2/src/webapp/WEB-INF/components.xml | 1 +
.../api/ConversationsService.java | 3 +-
conversations/impl/pom.xml | 5 +
.../impl/ConversationsServiceImpl.java | 113 +++++++++++-
.../impl/ConversationsServiceTests.java | 168 +++++++++++++++++-
.../test/resources/archive/conversations.xml | 5 +
.../test/resources/archive/conversations2.xml | 5 +
7 files changed, 293 insertions(+), 7 deletions(-)
create mode 100644 conversations/impl/src/test/resources/archive/conversations.xml
create mode 100644 conversations/impl/src/test/resources/archive/conversations2.xml
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 6e61ec57d3d8..a3c895456d6f 100644
--- a/common/archive-impl/impl2/src/webapp/WEB-INF/components.xml
+++ b/common/archive-impl/impl2/src/webapp/WEB-INF/components.xml
@@ -23,6 +23,7 @@
AssignmentService
AssessmentEntityProducer
ContentHostingService
+ conversations
CalendarService
ChatEntityProducer
DiscussionService
diff --git a/conversations/api/src/main/java/org/sakaiproject/conversations/api/ConversationsService.java b/conversations/api/src/main/java/org/sakaiproject/conversations/api/ConversationsService.java
index 5cd548e6db4a..1194a04dd2a7 100644
--- a/conversations/api/src/main/java/org/sakaiproject/conversations/api/ConversationsService.java
+++ b/conversations/api/src/main/java/org/sakaiproject/conversations/api/ConversationsService.java
@@ -34,8 +34,9 @@
import org.sakaiproject.conversations.api.model.Tag;
import org.sakaiproject.conversations.api.model.ConversationsTopic;
import org.sakaiproject.entity.api.Entity;
+import org.sakaiproject.entity.api.EntityProducer;
-public interface ConversationsService {
+public interface ConversationsService extends EntityProducer {
public static final String TOOL_ID = "sakai.conversations";
public static final String REFERENCE_ROOT = Entity.SEPARATOR + "conversations";
diff --git a/conversations/impl/pom.xml b/conversations/impl/pom.xml
index ab16cb493880..d0b6af59f8f1 100644
--- a/conversations/impl/pom.xml
+++ b/conversations/impl/pom.xml
@@ -129,6 +129,11 @@
org.apache.commons
commons-lang3
+
+ org.sakaiproject.common
+ archive-api
+ test
+
org.opensearch
opensearch
diff --git a/conversations/impl/src/main/java/org/sakaiproject/conversations/impl/ConversationsServiceImpl.java b/conversations/impl/src/main/java/org/sakaiproject/conversations/impl/ConversationsServiceImpl.java
index a62191fc6c03..df8d141949b6 100644
--- a/conversations/impl/src/main/java/org/sakaiproject/conversations/impl/ConversationsServiceImpl.java
+++ b/conversations/impl/src/main/java/org/sakaiproject/conversations/impl/ConversationsServiceImpl.java
@@ -32,8 +32,13 @@
import java.util.Observer;
import java.util.Optional;
import java.util.Set;
+import java.util.Stack;
import java.util.stream.Collectors;
+import org.w3c.dom.Document;
+import org.w3c.dom.Element;
+import org.w3c.dom.NodeList;
+
import org.sakaiproject.api.app.scheduler.ScheduledInvocationManager;
import org.sakaiproject.authz.api.AuthzGroup;
import org.sakaiproject.authz.api.AuthzGroupService;
@@ -85,7 +90,6 @@
import org.sakaiproject.conversations.api.repository.TopicStatusRepository;
import org.sakaiproject.entity.api.Entity;
import org.sakaiproject.entity.api.EntityManager;
-import org.sakaiproject.entity.api.EntityProducer;
import org.sakaiproject.entity.api.Reference;
import org.sakaiproject.event.api.Event;
import org.sakaiproject.event.api.EventTrackingService;
@@ -135,7 +139,7 @@
@Slf4j
@Setter
@Transactional
-public class ConversationsServiceImpl implements ConversationsService, EntityProducer, EntityTransferrer, Observer {
+public class ConversationsServiceImpl implements ConversationsService, EntityTransferrer, Observer {
private AuthzGroupService authzGroupService;
@@ -423,7 +427,7 @@ public Optional getCommentPortalUrl(String commentId) {
@Transactional
public TopicTransferBean saveTopic(final TopicTransferBean topicBean, boolean sendMessage) throws ConversationsPermissionsException {
- String currentUserId = getCheckedCurrentUserId();
+ String currentUserId = StringUtils.isNotBlank(topicBean.creator) ? topicBean.creator : getCheckedCurrentUserId();
String siteRef = siteService.siteReference(topicBean.siteId);
@@ -2540,6 +2544,7 @@ public Map transferCopyEntities(String fromContext, String toCon
return traversalMap;
}
+ @Override
public Map transferCopyEntities(String fromContext, String toContext, List ids, List transferOptions, boolean cleanup) {
if (cleanup) {
@@ -2556,6 +2561,108 @@ public Map transferCopyEntities(String fromContext, String toCon
return transferCopyEntities(fromContext, toContext, ids, transferOptions);
}
+ @Override
+ public boolean willArchiveMerge() {
+ return true;
+ }
+
+ @Override
+ public String getLabel() {
+ return "conversations";
+ }
+
+ @Override
+ public String archive(String siteId, Document doc, Stack stack, String archivePath, List attachments) {
+
+ StringBuilder results = new StringBuilder();
+ results.append("begin archiving ").append(getLabel()).append(" for site ").append(siteId).append(System.lineSeparator());
+
+ Element element = doc.createElement(getLabel());
+ stack.peek().appendChild(element);
+ stack.push(element);
+
+ Element topicsEl = doc.createElement("topics");
+ element.appendChild(topicsEl);
+
+ topicRepository.findBySiteId(siteId).stream().sorted((t1, t2) -> t1.getTitle().compareTo(t2.getTitle())).forEach(topic -> {
+
+ Element topicEl = doc.createElement("topic");
+ topicsEl.appendChild(topicEl);
+ topicEl.setAttribute("title", topic.getTitle());
+ topicEl.setAttribute("type", topic.getType().name());
+ topicEl.setAttribute("post-before-viewing", Boolean.toString(topic.getMustPostBeforeViewing()));
+ topicEl.setAttribute("allow-anonymous-posts", Boolean.toString(topic.getAllowAnonymousPosts()));
+ topicEl.setAttribute("pinned", Boolean.toString(topic.getPinned()));
+ topicEl.setAttribute("draft", Boolean.toString(topic.getDraft()));
+ topicEl.setAttribute("visibility", topic.getVisibility().name());
+ topicEl.setAttribute("creator", topic.getMetadata().getCreator());
+ topicEl.setAttribute("created", Long.toString(topic.getMetadata().getCreated().getEpochSecond()));
+
+ Element messageEl = doc.createElement("message");
+ messageEl.appendChild(doc.createCDATASection(topic.getMessage()));
+ topicEl.appendChild(messageEl);
+ });
+
+ results.append("completed archiving ").append(getLabel()).append(" for site ").append(siteId).append(System.lineSeparator());
+ 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(System.lineSeparator());
+
+ if (!root.getTagName().equals(getLabel())) {
+ log.warn("Tried to merge a non <{}> xml document", getLabel());
+ return "Invalid xml document";
+ }
+
+ Set currentTitles = topicRepository.findBySiteId(toSiteId)
+ .stream().map(ConversationsTopic::getTitle).collect(Collectors.toSet());
+
+ NodeList topicNodes = root.getElementsByTagName("topic");
+
+ Instant now = Instant.now();
+
+ for (int i = 0; i < topicNodes.getLength(); i++) {
+
+ Element topicEl = (Element) topicNodes.item(i);
+ String title = topicEl.getAttribute("title");
+
+ if (currentTitles.contains(title)) {
+ log.debug("Topic \"{}\" already exists in site {}. Skipping merge ...", title, toSiteId);
+ continue;
+ }
+
+ TopicTransferBean topicBean = new TopicTransferBean();
+ topicBean.siteId = toSiteId;
+ topicBean.title = title;
+ topicBean.type = topicEl.getAttribute("type");
+ topicBean.created = now;
+ topicBean.mustPostBeforeViewing = Boolean.parseBoolean(topicEl.getAttribute("post-before-viewing"));
+ topicBean.anonymous = Boolean.parseBoolean(topicEl.getAttribute("anonymous"));
+ topicBean.allowAnonymousPosts = Boolean.parseBoolean(topicEl.getAttribute("allow-anonymous-posts"));
+ topicBean.draft = Boolean.parseBoolean(topicEl.getAttribute("draft"));
+ topicBean.pinned = Boolean.parseBoolean(topicEl.getAttribute("pinned"));
+ topicBean.visibility = topicEl.getAttribute("visibility");
+
+ NodeList messageNodes = topicEl.getElementsByTagName("message");
+ if (messageNodes.getLength() == 1) {
+ topicBean.message = ((Element) messageNodes.item(0)).getFirstChild().getNodeValue();
+ }
+
+ try {
+ saveTopic(topicBean, false);
+ } catch (Exception e) {
+ log.warn("Failed to merge topic \"{}\": {}", topicBean.title, e.toString());
+ }
+ }
+
+ return "";
+ }
+
+ @Override
public boolean parseEntityReference(String referenceString, Reference ref) {
if (referenceString.startsWith(REFERENCE_ROOT)) {
diff --git a/conversations/impl/src/test/org/sakaiproject/conversations/impl/ConversationsServiceTests.java b/conversations/impl/src/test/org/sakaiproject/conversations/impl/ConversationsServiceTests.java
index be3be75adef7..71f4d61e93f9 100644
--- a/conversations/impl/src/test/org/sakaiproject/conversations/impl/ConversationsServiceTests.java
+++ b/conversations/impl/src/test/org/sakaiproject/conversations/impl/ConversationsServiceTests.java
@@ -15,7 +15,7 @@
*/
package org.sakaiproject.conversations.impl;
-import org.junit.Assume;
+import org.sakaiproject.archive.api.ArchiveService;
import org.sakaiproject.authz.api.AuthzGroup;
import org.sakaiproject.authz.api.AuthzGroupService;
import org.sakaiproject.authz.api.SecurityService;
@@ -56,6 +56,7 @@
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;
@@ -76,16 +77,20 @@
import java.util.Map;
import java.util.Optional;
import java.util.Set;
+import java.util.Stack;
import java.util.stream.Collectors;
import java.time.Instant;
import java.time.temporal.ChronoUnit;
-import static org.mockito.Mockito.*;
-
+import org.w3c.dom.Document;
+import org.w3c.dom.Element;
+import org.w3c.dom.NodeList;
import lombok.extern.slf4j.Slf4j;
+import static org.mockito.Mockito.*;
import static org.junit.Assert.*;
+import org.junit.Assume;
import org.junit.Before;
import org.junit.Test;
import org.junit.runner.RunWith;
@@ -1968,6 +1973,163 @@ public void grading() {
assertNull(savedBean.gradingItemId);
}
+ @Test
+ public void archive() {
+
+ switchToInstructor(null);
+
+ String title1 = "Topic 1";
+ TopicTransferBean topic1 = new TopicTransferBean();
+ topic1.aboutReference = site1Ref;
+ topic1.title = title1;
+ topic1.message = "Something about topic1";
+ topic1.siteId = site1Id;
+ topic1 = saveTopic(topic1);
+
+ String title2 = "Topic 2";
+ TopicTransferBean topic2 = new TopicTransferBean();
+ topic2.aboutReference = site1Ref;
+ topic2.title = title2;
+ topic2.siteId = site1Id;
+ topic2 = saveTopic(topic2);
+
+ String title3 = "Topic 3";
+ TopicTransferBean topic3 = new TopicTransferBean();
+ topic3.aboutReference = site1Ref;
+ topic3.title = title3;
+ topic3.siteId = site1Id;
+ topic3 = saveTopic(topic3);
+
+ String title4 = "Topic 4";
+ TopicTransferBean topic4 = new TopicTransferBean();
+ topic4.aboutReference = site1Ref;
+ topic4.title = title4;
+ topic4.siteId = site1Id;
+ topic4 = saveTopic(topic4);
+
+ TopicTransferBean[] topicBeans = new TopicTransferBean[] { topic1, topic2, topic3, topic4 };
+
+ Document doc = Xml.createDocument();
+ Stack stack = new Stack<>();
+
+ Element root = doc.createElement("archive");
+ doc.appendChild(root);
+ root.setAttribute("source", site1Id);
+ 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);
+
+ assertEquals(1, stack.size());
+
+ String results = conversationsService.archive(site1Id, doc, stack, "", null);
+
+ assertEquals(2, stack.size());
+
+ NodeList conversationsNode = root.getElementsByTagName(conversationsService.getLabel());
+ assertEquals(1, conversationsNode.getLength());
+
+ NodeList topicsNode = ((Element) conversationsNode.item(0)).getElementsByTagName("topics");
+ assertEquals(1, topicsNode.getLength());
+
+ NodeList topicNodes = ((Element) topicsNode.item(0)).getElementsByTagName("topic");
+ assertEquals(topicBeans.length, topicNodes.getLength());
+
+ for (int i = 0; i < topicNodes.getLength(); i++) {
+ Element topicEl = (Element) topicNodes.item(i);
+ assertEquals(topicBeans[i].title, topicEl.getAttribute("title"));
+ assertEquals(topicBeans[i].type, topicEl.getAttribute("type"));
+ assertEquals(topicBeans[i].anonymous, Boolean.parseBoolean(topicEl.getAttribute("anonymous")));
+ assertEquals(topicBeans[i].allowAnonymousPosts, Boolean.parseBoolean(topicEl.getAttribute("allow-anonymous-posts")));
+ assertEquals(topicBeans[i].pinned, Boolean.parseBoolean(topicEl.getAttribute("pinned")));
+ assertEquals(topicBeans[i].draft, Boolean.parseBoolean(topicEl.getAttribute("draft")));
+ assertEquals(topicBeans[i].visibility, topicEl.getAttribute("visibility"));
+ assertEquals(topicBeans[i].creator, topicEl.getAttribute("creator"));
+ assertEquals(topicBeans[i].created.getEpochSecond(), Long.parseLong(topicEl.getAttribute("created")));
+
+ NodeList messageNodes = topicEl.getElementsByTagName("message");
+ assertEquals(1, messageNodes.getLength());
+
+ assertEquals(topicBeans[i].message, ((Element) messageNodes.item(0)).getFirstChild().getNodeValue());
+ }
+ }
+
+ @Test
+ public void merge() {
+
+ Document doc = Xml.readDocumentFromStream(this.getClass().getResourceAsStream("/archive/conversations.xml"));
+
+ Element root = doc.getDocumentElement();
+
+ String fromSite = root.getAttribute("source");
+ String toSite = "my-new-site";
+
+ String toSiteRef = "/site/" + toSite;
+ switchToInstructor(toSiteRef);
+
+ when(siteService.siteReference(toSite)).thenReturn(toSiteRef);
+
+ Element conversationsElement = doc.createElement("not-conversations");
+
+ conversationsService.merge(toSite, conversationsElement, "", fromSite, null, null, null);
+
+ assertEquals("Invalid xml document", conversationsService.merge(toSite, conversationsElement, "", fromSite, null, null, null));
+
+ conversationsElement = (Element) root.getElementsByTagName(conversationsService.getLabel()).item(0);
+
+ conversationsService.merge(toSite, conversationsElement, "", fromSite, null, null, null);
+
+ NodeList topicNodes = ((Element) conversationsElement.getElementsByTagName("topics").item(0)).getElementsByTagName("topic");
+
+ List topics = topicRepository.findBySiteId(toSite);
+
+ assertEquals(topics.size(), topicNodes.getLength());
+
+ for (int i = 0; i < topicNodes.getLength(); i++) {
+
+ Element topicEl = (Element) topicNodes.item(i);
+
+ String title = topicEl.getAttribute("title");
+ Optional optTopic = topics.stream().filter(t -> t.getTitle().equals(title)).findAny();
+ assertTrue(optTopic.isPresent());
+
+ ConversationsTopic topic = optTopic.get();
+
+ assertEquals(topic.getType().name(), topicEl.getAttribute("type"));
+ assertEquals(topic.getPinned(), Boolean.parseBoolean(topicEl.getAttribute("pinned")));
+ assertEquals(topic.getAnonymous(), Boolean.parseBoolean(topicEl.getAttribute("anonymous")));
+ assertEquals(topic.getDraft(), Boolean.parseBoolean(topicEl.getAttribute("draft")));
+ assertEquals(topic.getMustPostBeforeViewing(), Boolean.parseBoolean(topicEl.getAttribute("post-before-viewing")));
+
+ NodeList messageNodes = topicEl.getElementsByTagName("message");
+ assertEquals(1, messageNodes.getLength());
+
+ assertEquals(topic.getMessage(), messageNodes.item(0).getFirstChild().getNodeValue());
+ }
+
+ Set oldTitles = topics.stream().map(ConversationsTopic::getTitle).collect(Collectors.toSet());
+
+ // Now let's try and merge this set of rubrics. It has one with a different title, but the
+ // rest the same, so we should end up with only one rubric being added.
+ Document doc2 = Xml.readDocumentFromStream(this.getClass().getResourceAsStream("/archive/conversations2.xml"));
+
+ Element root2 = doc2.getDocumentElement();
+
+ conversationsElement = (Element) root2.getElementsByTagName(conversationsService.getLabel()).item(0);
+
+ conversationsService.merge(toSite, conversationsElement, "", fromSite, null, null, null);
+
+ String extraTitle = "Smurfs";
+
+ assertEquals(topics.size() + 1, topicRepository.findBySiteId(toSite).size());
+
+ Set newTitles = topicRepository.findBySiteId(toSite)
+ .stream().map(ConversationsTopic::getTitle).collect(Collectors.toSet());
+
+ assertFalse(oldTitles.contains(extraTitle));
+ assertTrue(newTitles.contains(extraTitle));
+ }
+
private TopicTransferBean saveTopic(TopicTransferBean topicBean) {
try {
diff --git a/conversations/impl/src/test/resources/archive/conversations.xml b/conversations/impl/src/test/resources/archive/conversations.xml
new file mode 100644
index 000000000000..88ef43f84aac
--- /dev/null
+++ b/conversations/impl/src/test/resources/archive/conversations.xml
@@ -0,0 +1,5 @@
+Let's discuss aliens, right here.
+]]>It's philosophy, innit?
+]]>Does anybody know where the toilets actually are?
+]]>sporting stuff
+]]>
\ No newline at end of file
diff --git a/conversations/impl/src/test/resources/archive/conversations2.xml b/conversations/impl/src/test/resources/archive/conversations2.xml
new file mode 100644
index 000000000000..548cf7137aca
--- /dev/null
+++ b/conversations/impl/src/test/resources/archive/conversations2.xml
@@ -0,0 +1,5 @@
+
+Let's discuss aliens, right here.
+]]>It's philosophy, innit?
+]]>Does anybody know where the toilets actually are?
+]]>