From aca259ea1d699c88ad7fabcfb9ce0fc7d2b77c08 Mon Sep 17 00:00:00 2001 From: joffinjoy <35325730+joffinjoy@users.noreply.github.com> Date: Sat, 26 Mar 2022 10:22:22 +0530 Subject: [PATCH 1/9] Issue #SB-29145 feat: Added Question & QuestionSet Copy APIs & Test-Cases --- .../org/sunbird/actors/QuestionActor.scala | 8 +- .../org/sunbird/actors/QuestionSetActor.scala | 8 +- .../org/sunbird/managers/CopyManager.scala | 293 ++++++++++++++++++ .../sunbird/utils/AssessmentContants.scala | 39 +++ .../src/test/resources/application.conf | 10 + .../sunbird/actors/QuestionActorTest.scala | 41 ++- .../sunbird/actors/QuestionSetActorTest.scala | 58 +++- .../scala/org/sunbird/actors/copyTrait.scala | 264 ++++++++++++++++ .../controllers/v4/QuestionController.scala | 13 + .../v4/QuestionSetController.scala | 11 + .../assessment-service/app/utils/ApiId.scala | 3 +- .../app/utils/QuestionOperations.scala | 2 +- .../app/utils/QuestionSetOperations.scala | 6 +- .../assessment-service/conf/application.conf | 12 +- assessment-api/assessment-service/conf/routes | 8 +- schemas/question/1.0/schema.json | 6 + schemas/questionset/1.0/schema.json | 6 + 17 files changed, 774 insertions(+), 14 deletions(-) create mode 100644 assessment-api/assessment-actors/src/main/scala/org/sunbird/managers/CopyManager.scala create mode 100644 assessment-api/assessment-actors/src/main/scala/org/sunbird/utils/AssessmentContants.scala create mode 100644 assessment-api/assessment-actors/src/test/scala/org/sunbird/actors/copyTrait.scala diff --git a/assessment-api/assessment-actors/src/main/scala/org/sunbird/actors/QuestionActor.scala b/assessment-api/assessment-actors/src/main/scala/org/sunbird/actors/QuestionActor.scala index 8f429f13a..718dc83d6 100644 --- a/assessment-api/assessment-actors/src/main/scala/org/sunbird/actors/QuestionActor.scala +++ b/assessment-api/assessment-actors/src/main/scala/org/sunbird/actors/QuestionActor.scala @@ -8,7 +8,7 @@ import org.sunbird.common.{DateUtils, Platform} import org.sunbird.graph.OntologyEngineContext import org.sunbird.graph.nodes.DataNode import org.sunbird.graph.utils.NodeUtil -import org.sunbird.managers.AssessmentManager +import org.sunbird.managers.{AssessmentManager, CopyManager} import org.sunbird.utils.RequestUtil import java.util @@ -36,6 +36,7 @@ class QuestionActor @Inject()(implicit oec: OntologyEngineContext) extends BaseA case "systemUpdateQuestion" => systemUpdate(request) case "listQuestions" => listQuestions(request) case "rejectQuestion" => reject(request) + case "copyQuestion" => copy(request) case _ => ERROR(request.getOperation) } @@ -127,4 +128,9 @@ class QuestionActor @Inject()(implicit oec: OntologyEngineContext) extends BaseA AssessmentManager.updateNode(updateRequest) }) } + + def copy(request: Request): Future[Response] ={ + RequestUtil.restrictProperties(request) + CopyManager.copy(request) + } } diff --git a/assessment-api/assessment-actors/src/main/scala/org/sunbird/actors/QuestionSetActor.scala b/assessment-api/assessment-actors/src/main/scala/org/sunbird/actors/QuestionSetActor.scala index f5b5742d9..ea022c3fa 100644 --- a/assessment-api/assessment-actors/src/main/scala/org/sunbird/actors/QuestionSetActor.scala +++ b/assessment-api/assessment-actors/src/main/scala/org/sunbird/actors/QuestionSetActor.scala @@ -1,7 +1,6 @@ package org.sunbird.actors import java.util - import javax.inject.Inject import org.apache.commons.collections4.CollectionUtils import org.sunbird.`object`.importer.{ImportConfig, ImportManager} @@ -13,7 +12,7 @@ import org.sunbird.graph.OntologyEngineContext import org.sunbird.graph.nodes.DataNode import org.sunbird.graph.dac.model.Node import org.sunbird.managers.HierarchyManager.hierarchyPrefix -import org.sunbird.managers.{AssessmentManager, HierarchyManager, UpdateHierarchyManager} +import org.sunbird.managers.{AssessmentManager, CopyManager, HierarchyManager, UpdateHierarchyManager} import org.sunbird.utils.RequestUtil import scala.collection.JavaConverters._ @@ -40,6 +39,7 @@ class QuestionSetActor @Inject()(implicit oec: OntologyEngineContext) extends Ba case "rejectQuestionSet" => reject(request) case "importQuestionSet" => importQuestionSet(request) case "systemUpdateQuestionSet" => systemUpdate(request) + case "copyQuestionSet" => copy(request) case _ => ERROR(request.getOperation) } @@ -156,4 +156,8 @@ class QuestionSetActor @Inject()(implicit oec: OntologyEngineContext) extends Ba }).map(node => ResponseHandler.OK.put("identifier", identifier).put("status", "success")) } + def copy(request: Request): Future[Response] ={ + RequestUtil.restrictProperties(request) + CopyManager.copy(request) + } } diff --git a/assessment-api/assessment-actors/src/main/scala/org/sunbird/managers/CopyManager.scala b/assessment-api/assessment-actors/src/main/scala/org/sunbird/managers/CopyManager.scala new file mode 100644 index 000000000..5c5a5fa73 --- /dev/null +++ b/assessment-api/assessment-actors/src/main/scala/org/sunbird/managers/CopyManager.scala @@ -0,0 +1,293 @@ +package org.sunbird.managers + +import org.apache.commons.collections.CollectionUtils +import org.apache.commons.collections4.MapUtils +import org.apache.commons.lang.StringUtils +import org.sunbird.common.Platform +import org.sunbird.common.dto.{Request, Response, ResponseHandler} +import org.sunbird.common.exception.{ClientException, ServerException} +import org.sunbird.graph.OntologyEngineContext +import org.sunbird.graph.common.Identifier +import org.sunbird.graph.dac.model.Node +import org.sunbird.graph.nodes.DataNode +import org.sunbird.graph.schema.DefinitionNode +import org.sunbird.graph.utils.{NodeUtil, ScalaJsonUtils} +import org.sunbird.telemetry.logger.TelemetryManager +import org.sunbird.utils.AssessmentConstants + +import java.util +import java.util.concurrent.{CompletionException, TimeUnit} +import java.util.{Optional, UUID} +import scala.collection.JavaConverters._ +import scala.concurrent.duration.{Duration, FiniteDuration} +import scala.concurrent.{Await, ExecutionContext, Future} + +object CopyManager { + + private val originMetadataKeys: util.List[String] = Platform.getStringList("assessment.copy.origin_data", new util.ArrayList[String]()) + private val internalHierarchyProps = List("identifier", "parent", "index", "depth") + private val metadataNotTobeCopied = Platform.config.getStringList("assessment.copy.props_to_remove") + + def copy(request: Request)(implicit ec: ExecutionContext, oec: OntologyEngineContext): Future[Response] = { + validateRequest(request) + DataNode.read(request).map(node => { + validateExistingNode(request, node) + val copiedNodeFuture: Future[Node] = node.getMetadata.get(AssessmentConstants.MIME_TYPE) match { + case AssessmentConstants.QUESTIONSET_MIME_TYPE => + node.setInRelations(null) + node.setOutRelations(null) + validateShallowCopyReq(node, request) + copyQuestionSet(node, request) + case AssessmentConstants.QUESTION_MIME_TYPE => + node.setInRelations(null) + validateCopyQuestionReq(request, node) //Check if the question has got "Default" visibility. + copyNode(node, request) + } + copiedNodeFuture.map(copiedNode => { + val response = ResponseHandler.OK() + response.put("node_id", new util.HashMap[String, AnyRef]() { + { + put(node.getIdentifier, copiedNode.getIdentifier) + } + }) + response.put(AssessmentConstants.VERSION_KEY, copiedNode.getMetadata.get(AssessmentConstants.VERSION_KEY)) + response + }) + }).flatMap(f => f) recoverWith { case e: CompletionException => throw e.getCause } + } + + def validateCopyQuestionReq(request: Request, node: Node) = { + val visibility = node.getMetadata.getOrDefault(AssessmentConstants.VISIBILITY, AssessmentConstants.VISIBILITY_PARENT).asInstanceOf[String] + if (StringUtils.equalsIgnoreCase(visibility, AssessmentConstants.VISIBILITY_PARENT)) { + throw new ClientException(AssessmentConstants.ERR_INVALID_REQUEST, "Question With Visibility Parent Cannot Be Copied Individually!") + } + } + + def validateExistingNode(request: Request, node: Node) = { + val requestObjectType = request.getObjectType + val nodeObjectType = node.getObjectType + if (!StringUtils.equalsIgnoreCase(requestObjectType, nodeObjectType)) { + throw new ClientException(AssessmentConstants.ERR_INVALID_OBJECT_TYPE, "Invalid Object Type: " + requestObjectType) + } + } + + def copyQuestionSet(originNode: Node, request: Request)(implicit ex: ExecutionContext, oec: OntologyEngineContext): Future[Node] = { + val copyType = request.getRequest.get(AssessmentConstants.COPY_TYPE).asInstanceOf[String] + copyNode(originNode, request).map(node => { + val req = new Request(request) + req.put(AssessmentConstants.ROOT_ID, request.get(AssessmentConstants.IDENTIFIER)) + req.put(AssessmentConstants.MODE, request.get(AssessmentConstants.MODE)) + HierarchyManager.getHierarchy(req).map(response => { + val originHierarchy = response.getResult.getOrDefault(AssessmentConstants.QUESTIONSET, new util.HashMap[String, AnyRef]()).asInstanceOf[java.util.Map[String, AnyRef]] + copyType match { + case AssessmentConstants.COPY_TYPE_SHALLOW => updateShallowHierarchy(request, node, originNode, originHierarchy) + case _ => updateHierarchy(request, node, originNode, originHierarchy, copyType) + } + }).flatMap(f => f) + }).flatMap(f => f) recoverWith { case e: CompletionException => throw e.getCause } + } + + + def copyNode(node: Node, request: Request)(implicit ec: ExecutionContext, oec: OntologyEngineContext): Future[Node] = { + val copyCreateReq: Future[Request] = getCopyRequest(node, request) + copyCreateReq.map(req => { + DataNode.create(req).map(copiedNode => { + Future(copiedNode) + }).flatMap(f => f) + }).flatMap(f => f) + } + + def updateHierarchy(request: Request, node: Node, originNode: Node, originHierarchy: util.Map[String, AnyRef], copyType: String)(implicit ec: ExecutionContext, oec: OntologyEngineContext): Future[Node] = { + prepareHierarchyRequest(originHierarchy, originNode, node, copyType, request).map(req => { + val hierarchyRequest = new Request(request) + hierarchyRequest.putAll(req) + hierarchyRequest.getContext.put(AssessmentConstants.SCHEMA_NAME, AssessmentConstants.QUESTIONSET_SCHEMA_NAME) + hierarchyRequest.getContext.put(AssessmentConstants.VERSION, AssessmentConstants.SCHEMA_VERSION) + UpdateHierarchyManager.updateHierarchy(hierarchyRequest).map(response => { + if (!ResponseHandler.checkError(response)) node else { + TelemetryManager.info(s"Update Hierarchy Failed For Copy Question Set Having Identifier: ${node.getIdentifier} | Response is : " + response) + throw new ServerException("ERR_QUESTIONSET_COPY", "Something Went Wrong, Please Try Again") + } + }) + }).flatMap(f => f) + } + + def prepareHierarchyRequest(originHierarchy: util.Map[String, AnyRef], originNode: Node, node: Node, copyType: String, request: Request)(implicit ec: ExecutionContext, oec: OntologyEngineContext): Future[util.Map[String, AnyRef]] = { + val children: util.List[util.Map[String, AnyRef]] = originHierarchy.get("children").asInstanceOf[util.List[util.Map[String, AnyRef]]] + if (null != children && !children.isEmpty) { + val nodesModified = new util.HashMap[String, AnyRef]() + val hierarchy = new util.HashMap[String, AnyRef]() + val idMap = new util.HashMap[String, String]() + hierarchy.put(node.getIdentifier, new util.HashMap[String, AnyRef]() { + { + put(AssessmentConstants.CHILDREN, new util.ArrayList[String]()) + put(AssessmentConstants.ROOT, true.asInstanceOf[AnyRef]) + put(AssessmentConstants.PRIMARY_CATEGORY, node.getMetadata.get(AssessmentConstants.PRIMARY_CATEGORY)) + } + }) + populateHierarchyRequest(children, nodesModified, hierarchy, node.getIdentifier, copyType, request, idMap) + getExternalData(idMap.keySet().asScala.toList, request).map(exData => { + idMap.asScala.toMap.foreach(entry => { + nodesModified.get(entry._2).asInstanceOf[java.util.Map[String, AnyRef]].get("metadata").asInstanceOf[util.Map[String, AnyRef]].putAll(exData.getOrDefault(entry._1, new util.HashMap[String, AnyRef]()).asInstanceOf[util.Map[String, AnyRef]]) + }) + new util.HashMap[String, AnyRef]() { + { + put(AssessmentConstants.NODES_MODIFIED, nodesModified) + put(AssessmentConstants.HIERARCHY, hierarchy) + } + } + }) + + } else Future(new util.HashMap[String, AnyRef]()) + } + + def getExternalData(identifiers: List[String], request: Request)(implicit ec: ExecutionContext, oec: OntologyEngineContext): Future[util.Map[String, AnyRef]] = { + val req = new Request(request) + req.getContext().putAll(Map("objectType" -> "Question", "schemaName" -> "question").asJava) + req.put("identifiers", identifiers) + val result = new util.HashMap[String, AnyRef]() + val externalProps = DefinitionNode.getExternalProps(req.getContext.getOrDefault("graph_id", "domain").asInstanceOf[String], req.getContext.getOrDefault("version", "1.0").asInstanceOf[String], req.getContext.getOrDefault("schemaName", "question").asInstanceOf[String]) + val externalPropsResponse = oec.graphService.readExternalProps(req, externalProps) + externalPropsResponse.map(response => { + identifiers.map(id => { + val externalData = Optional.ofNullable(response.get(id).asInstanceOf[util.Map[String, AnyRef]]).orElse(new util.HashMap[String, AnyRef]()) + result.put(id, externalData) + }) + result + }) + } + + def populateHierarchyRequest(children: util.List[util.Map[String, AnyRef]], nodesModified: util.HashMap[String, AnyRef], hierarchy: util.HashMap[String, AnyRef], parentId: String, copyType: String, request: Request, idMap: java.util.Map[String, String])(implicit ec: ExecutionContext, oec: OntologyEngineContext): Unit = { + if (null != children && !children.isEmpty) { + children.asScala.toList.foreach(child => { + val id = if ("Parent".equalsIgnoreCase(child.get(AssessmentConstants.VISIBILITY).asInstanceOf[String])) { + val identifier = UUID.randomUUID().toString + nodesModified.put(identifier, new util.HashMap[String, AnyRef]() { + { + put(AssessmentConstants.METADATA, cleanUpCopiedData(new util.HashMap[String, AnyRef]() { + { + putAll(child) + put(AssessmentConstants.CHILDREN, new util.ArrayList()) + internalHierarchyProps.map(key => remove(key)) + } + }, copyType)) + put(AssessmentConstants.ROOT, false.asInstanceOf[AnyRef]) + put(AssessmentConstants.OBJECT_TYPE, child.getOrDefault(AssessmentConstants.OBJECT_TYPE, "")) + put("isNew", true.asInstanceOf[AnyRef]) + put("setDefaultValue", false.asInstanceOf[AnyRef]) + } + }) + if (StringUtils.equalsIgnoreCase(AssessmentConstants.QUESTION_MIME_TYPE, child.getOrDefault("mimeType", "").asInstanceOf[String])) + idMap.put(child.getOrDefault("identifier", "").asInstanceOf[String], identifier) + identifier + } else + child.get(AssessmentConstants.IDENTIFIER).asInstanceOf[String] + if (StringUtils.equalsIgnoreCase(child.getOrDefault(AssessmentConstants.MIME_TYPE, "").asInstanceOf[String], AssessmentConstants.QUESTIONSET_MIME_TYPE)) + hierarchy.put(id, new util.HashMap[String, AnyRef]() { + { + put(AssessmentConstants.CHILDREN, new util.ArrayList[String]()) + put(AssessmentConstants.ROOT, false.asInstanceOf[AnyRef]) + put(AssessmentConstants.PRIMARY_CATEGORY, child.get(AssessmentConstants.PRIMARY_CATEGORY)) + } + }) + hierarchy.get(parentId).asInstanceOf[util.Map[String, AnyRef]].get(AssessmentConstants.CHILDREN).asInstanceOf[util.List[String]].add(id) + populateHierarchyRequest(child.get(AssessmentConstants.CHILDREN).asInstanceOf[util.List[util.Map[String, AnyRef]]], nodesModified, hierarchy, id, copyType, request, idMap) + }) + } + } + + def updateShallowHierarchy(request: Request, node: Node, originNode: Node, originHierarchy: util.Map[String, AnyRef])(implicit ec: ExecutionContext, oec: OntologyEngineContext): Future[Node] = { + val childrenHierarchy = originHierarchy.get("children").asInstanceOf[util.List[util.Map[String, AnyRef]]] + val req = new Request(request) + req.getContext.put(AssessmentConstants.SCHEMA_NAME, AssessmentConstants.QUESTIONSET_SCHEMA_NAME) + req.getContext.put(AssessmentConstants.VERSION, AssessmentConstants.SCHEMA_VERSION) + req.getContext.put(AssessmentConstants.IDENTIFIER, node.getIdentifier) + req.put(AssessmentConstants.HIERARCHY, ScalaJsonUtils.serialize(new java.util.HashMap[String, AnyRef]() { + { + put(AssessmentConstants.IDENTIFIER, node.getIdentifier) + put(AssessmentConstants.CHILDREN, childrenHierarchy) + } + })) + DataNode.update(req).map(node => node) + } + + def getCopyRequest(node: Node, request: Request)(implicit ec: ExecutionContext, oec: OntologyEngineContext): Future[Request] = { + val metadata: util.Map[String, AnyRef] = NodeUtil.serialize(node, new util.ArrayList(), node.getObjectType.toLowerCase.replace("image", ""), AssessmentConstants.SCHEMA_VERSION) + val requestMap = request.getRequest + requestMap.remove(AssessmentConstants.MODE) + requestMap.remove(AssessmentConstants.COPY_SCHEME).asInstanceOf[String] + val copyType = requestMap.remove(AssessmentConstants.COPY_TYPE).asInstanceOf[String] + val originData: java.util.Map[String, AnyRef] = getOriginData(metadata, copyType) + cleanUpCopiedData(metadata, copyType) + metadata.putAll(requestMap) + metadata.put(AssessmentConstants.STATUS, "Draft") + metadata.put(AssessmentConstants.ORIGIN, node.getIdentifier) + metadata.put(AssessmentConstants.IDENTIFIER, Identifier.getIdentifier(request.getContext.get("graph_id").asInstanceOf[String], Identifier.getUniqueIdFromTimestamp)) + if (MapUtils.isNotEmpty(originData)) + metadata.put(AssessmentConstants.ORIGIN_DATA, originData) + request.getContext().put(AssessmentConstants.SCHEMA_NAME, node.getObjectType.toLowerCase.replace("image", "")) + val req = new Request(request) + req.setRequest(metadata) + + + val graphId = request.getContext.getOrDefault("graph_id", "").asInstanceOf[String] + val version = request.getContext.getOrDefault("version", "").asInstanceOf[String] + val externalProps = if (StringUtils.equalsIgnoreCase(AssessmentConstants.QUESTIONSET_MIME_TYPE, node.getMetadata.getOrDefault("mimeType", "").asInstanceOf[String])) { + DefinitionNode.getExternalProps(graphId, version, AssessmentConstants.QUESTIONSET_SCHEMA_NAME).diff(List("hierarchy")) + } else { + DefinitionNode.getExternalProps(graphId, version, AssessmentConstants.QUESTION_SCHEMA_NAME) + } + val readReq = new Request() + readReq.setContext(request.getContext) + readReq.put("identifier", node.getIdentifier) + readReq.put("fields", externalProps.asJava) + val maxWaitTime: FiniteDuration = Duration(5, TimeUnit.SECONDS) + Await.result( + DataNode.read(readReq).map(node => { + val metadata: util.Map[String, AnyRef] = NodeUtil.serialize(node, externalProps.asJava, node.getObjectType.toLowerCase.replace("image", ""), request.getContext.get("version").asInstanceOf[String]) + externalProps.foreach(prop => { + val propValue = metadata.get(prop) + if (metadata.containsKey(prop) && propValue != null) { + req.put(prop, propValue) + } + }) + }), maxWaitTime) + Future(req) + } + + def getOriginData(metadata: util.Map[String, AnyRef], copyType: String): java.util.Map[String, AnyRef] = { + new java.util.HashMap[String, AnyRef]() { + { + putAll(originMetadataKeys.asScala.filter(key => metadata.containsKey(key)).map(key => key -> metadata.get(key)).toMap.asJava) + put(AssessmentConstants.COPY_TYPE, copyType) + } + } + } + + def validateRequest(request: Request)(implicit ec: ExecutionContext, oec: OntologyEngineContext): Unit = { + val keysNotPresent = AssessmentConstants.REQUIRED_KEYS.filter(key => emptyCheckFilter(request.getRequest.getOrDefault(key, ""))) + if (keysNotPresent.nonEmpty) + throw new ClientException(AssessmentConstants.ERR_INVALID_REQUEST, "Please provide valid value for " + keysNotPresent) + } + + def validateShallowCopyReq(node: Node, request: Request) = { + val copyType: String = request.getRequest.get("copyType").asInstanceOf[String] + if (StringUtils.equalsIgnoreCase("shallow", copyType) && !StringUtils.equalsIgnoreCase("Live", node.getMetadata.get("status").asInstanceOf[String])) + throw new ClientException(AssessmentConstants.ERR_INVALID_REQUEST, "QuestionSet with status " + node.getMetadata.get(AssessmentConstants.STATUS).asInstanceOf[String].toLowerCase + " cannot be partially (shallow) copied.") + //TODO: check if need to throw client exception for combination of copyType=shallow and mode=edit + } + + def emptyCheckFilter(key: AnyRef): Boolean = key match { + case k: String => k.asInstanceOf[String].isEmpty + case k: util.Map[String, AnyRef] => MapUtils.isEmpty(k.asInstanceOf[util.Map[String, AnyRef]]) + case k: util.List[String] => CollectionUtils.isEmpty(k.asInstanceOf[util.List[String]]) + case _ => true + } + + def cleanUpCopiedData(metadata: util.Map[String, AnyRef], copyType: String): util.Map[String, AnyRef] = { + if (StringUtils.equalsIgnoreCase(AssessmentConstants.COPY_TYPE_SHALLOW, copyType)) { + metadata.keySet().removeAll(metadataNotTobeCopied.asScala.toList.filter(str => !str.contains("dial")).asJava) + } else metadata.keySet().removeAll(metadataNotTobeCopied) + metadata + } +} diff --git a/assessment-api/assessment-actors/src/main/scala/org/sunbird/utils/AssessmentContants.scala b/assessment-api/assessment-actors/src/main/scala/org/sunbird/utils/AssessmentContants.scala new file mode 100644 index 000000000..c422c3256 --- /dev/null +++ b/assessment-api/assessment-actors/src/main/scala/org/sunbird/utils/AssessmentContants.scala @@ -0,0 +1,39 @@ +package org.sunbird.utils + +object AssessmentConstants { + val REQUIRED_KEYS: List[String] = List("createdBy", "createdFor") + val ERR_INVALID_REQUEST: String = "ERR_INVALID_REQUEST" + val ERR_INVALID_OBJECT_TYPE: String = "ERR_INVALID_OBJECT_TYPE" + val COPY_SCHEME: String = "copyScheme" + val MIME_TYPE: String = "mimeType" + val QUESTIONSET_MIME_TYPE: String = "application/vnd.sunbird.questionset" + val STATUS: String = "status" + val COPY_TYPE: String = "copyType" + val SCHEMA_NAME: String = "schemaName" + val VERSION: String = "version" + val ROOT_ID: String = "rootId" + val MODE: String = "mode" + val COPY_TYPE_SHALLOW: String = "shallow" + val QUESTIONSET_SCHEMA_NAME: String = "questionset" + val SCHEMA_VERSION: String = "1.0" + val IDENTIFIER: String = "identifier" + val QUESTION: String = "question" + val HIERARCHY: String = "hierarchy" + val CHILDREN: String = "children" + val ORIGIN: String = "origin" + val ORIGIN_DATA: String = "originData" + val CONTENT_TYPE: String = "contentType" + val ROOT: String = "root" + val NODES_MODIFIED: String = "nodesModified" + val VISIBILITY: String = "visibility" + val METADATA: String = "metadata" + val VERSION_KEY: String = "versionKey" + val PRIMARY_CATEGORY : String = "primaryCategory" + val QUESTIONSET : String = "questionSet" + val OBJECT_TYPE : String = "objectType" + val COPY_TYPE_DEEP: String = "deep" + val QUESTION_MIME_TYPE: String = "application/vnd.sunbird.question" + val QUESTION_SCHEMA_NAME: String = "question" + val VISIBILITY_PARENT: String = "Parent" + val VISIBILITY_DEFAULT: String = "Default" +} diff --git a/assessment-api/assessment-actors/src/test/resources/application.conf b/assessment-api/assessment-actors/src/test/resources/application.conf index 24d3a74f2..b9ad82663 100644 --- a/assessment-api/assessment-actors/src/test/resources/application.conf +++ b/assessment-api/assessment-actors/src/test/resources/application.conf @@ -412,5 +412,15 @@ import { } } +assessment.copy.origin_data=["name", "author", "license", "organisation"] +assessment.copy.props_to_remove=["downloadUrl", "artifactUrl", "variants", + "createdOn", "collections", "children", "lastUpdatedOn", "SYS_INTERNAL_LAST_UPDATED_ON", + "versionKey", "s3Key", "status", "pkgVersion", "toc_url", "mimeTypesCount", + "contentTypesCount", "leafNodesCount", "childNodes", "prevState", "lastPublishedOn", + "flagReasons", "compatibilityLevel", "size", "publishChecklist", "publishComment", + "LastPublishedBy", "rejectReasons", "rejectComment", "gradeLevel", "subject", + "medium", "board", "topic", "purpose", "subtopic", "contentCredits", + "owner", "collaborators", "creators", "contributors", "badgeAssertions", "dialcodes", + "concepts", "keywords", "reservedDialcodes", "dialcodeRequired", "leafNodes", "sYS_INTERNAL_LAST_UPDATED_ON", "prevStatus", "lastPublishedBy", "streamingUrl"] diff --git a/assessment-api/assessment-actors/src/test/scala/org/sunbird/actors/QuestionActorTest.scala b/assessment-api/assessment-actors/src/test/scala/org/sunbird/actors/QuestionActorTest.scala index 3adc82110..11cc17f53 100644 --- a/assessment-api/assessment-actors/src/test/scala/org/sunbird/actors/QuestionActorTest.scala +++ b/assessment-api/assessment-actors/src/test/scala/org/sunbird/actors/QuestionActorTest.scala @@ -16,7 +16,7 @@ import scala.collection.JavaConverters._ import scala.concurrent.Future import scala.concurrent.ExecutionContext.Implicits.global -class QuestionActorTest extends BaseSpec with MockFactory { +class QuestionActorTest extends BaseSpec with MockFactory with copyTrait { "questionActor" should "return failed response for 'unknown' operation" in { implicit val oec: OntologyEngineContext = new OntologyEngineContext @@ -314,6 +314,45 @@ class QuestionActorTest extends BaseSpec with MockFactory { assert("successful".equals(response.getParams.getStatus)) } + it should "return success response for 'copyQuestion'" in { + implicit val oec: OntologyEngineContext = mock[OntologyEngineContext] + val graphDB = mock[GraphService] + (oec.graphService _).expects().returns(graphDB).anyNumberOfTimes() + val nodes: util.List[Node] = getCategoryNode() + (graphDB.getNodeByUniqueIds(_: String, _: SearchCriteria)).expects(*, *).returns(Future(nodes)).anyNumberOfTimes() + (graphDB.getNodeByUniqueId(_: String, _: String, _: Boolean, _: Request)).expects("domain", "do_1234", false, *).returns(Future(getExistingQuestionNode())).anyNumberOfTimes() + (graphDB.readExternalProps(_: Request, _: List[String])).expects(*, List("objectMetadata")).returns(Future(getSuccessfulResponse())).anyNumberOfTimes() + (graphDB.readExternalProps(_: Request, _: List[String])).expects(*, *).returns(Future(getReadPropsResponseForQuestion())).anyNumberOfTimes() + (graphDB.addNode(_: String, _: Node)).expects(*, *).returns(Future(getNewQuestionNode())) + (graphDB.saveExternalProps(_: Request)).expects(*).returns(Future(getSuccessfulResponse())).anyNumberOfTimes + val request = getQuestionCopyRequest() + request.putAll(mapAsJavaMap(Map("identifier" -> "do_1234", "mode" -> "", "copyType"-> "deep"))) + request.setOperation("copyQuestion") + val response = callActor(request, Props(new QuestionActor())) + assert("successful".equals(response.getParams.getStatus)) + } + + it should "return error response for 'copyQuestion' when createdFor & createdBy is missing" in { + implicit val oec: OntologyEngineContext = mock[OntologyEngineContext] + val request = getInvalidQuestionSetCopyRequest() + request.putAll(mapAsJavaMap(Map("identifier" -> "do_1234", "mode" -> "", "copyType"-> "deep"))) + request.setOperation("copyQuestion") + val response = callActor(request, Props(new QuestionActor())) + assert("failed".equals(response.getParams.getStatus)) + } + + it should "return error response for 'copyQuestion' when visibility is Parent" in { + implicit val oec: OntologyEngineContext = mock[OntologyEngineContext] + val graphDB = mock[GraphService] + (oec.graphService _).expects().returns(graphDB).anyNumberOfTimes() + val request = getQuestionCopyRequest() + (graphDB.getNodeByUniqueId(_: String, _: String, _: Boolean, _: Request)).expects("domain", "do_1234", false, *).returns(Future(getQuestionNode())).anyNumberOfTimes() + request.putAll(mapAsJavaMap(Map("identifier" -> "do_1234", "mode" -> "", "copyType"-> "deep"))) + request.setOperation("copyQuestion") + val response = callActor(request, Props(new QuestionActor())) + assert("failed".equals(response.getParams.getStatus)) + } + private def getQuestionRequest(): Request = { val request = new Request() request.setContext(new java.util.HashMap[String, AnyRef]() { diff --git a/assessment-api/assessment-actors/src/test/scala/org/sunbird/actors/QuestionSetActorTest.scala b/assessment-api/assessment-actors/src/test/scala/org/sunbird/actors/QuestionSetActorTest.scala index 74f7d2dad..25cf6c6d1 100644 --- a/assessment-api/assessment-actors/src/test/scala/org/sunbird/actors/QuestionSetActorTest.scala +++ b/assessment-api/assessment-actors/src/test/scala/org/sunbird/actors/QuestionSetActorTest.scala @@ -19,7 +19,7 @@ import scala.collection.JavaConverters._ import scala.concurrent.ExecutionContext.Implicits.global import scala.concurrent.Future -class QuestionSetActorTest extends BaseSpec with MockFactory { +class QuestionSetActorTest extends BaseSpec with MockFactory with copyTrait { "questionSetActor" should "return failed response for 'unknown' operation" in { implicit val oec: OntologyEngineContext = new OntologyEngineContext @@ -538,6 +538,62 @@ class QuestionSetActorTest extends BaseSpec with MockFactory { assert("successful".equals(response.getParams.getStatus)) } + it should "return success response for 'copyQuestionSet' (Deep)" in { + implicit val oec: OntologyEngineContext = mock[OntologyEngineContext] + val graphDB = mock[GraphService] + (oec.graphService _).expects().returns(graphDB).anyNumberOfTimes() + val nodes: util.List[Node] = getCategoryNode() + (graphDB.getNodeByUniqueIds(_: String, _: SearchCriteria)).expects(*, *).returns(Future(nodes)).anyNumberOfTimes() + (graphDB.getNodeByUniqueId(_: String, _: String, _: Boolean, _: Request)).expects("domain", "do_1234", false, *).returns(Future(getExistingRootNode())).anyNumberOfTimes() + (graphDB.getNodeByUniqueId(_: String, _: String, _: Boolean, _: Request)).expects("domain", "do_9876", false, *).returns(Future(getNewRootNode())).anyNumberOfTimes() + (graphDB.getNodeByUniqueId(_: String, _: String, _: Boolean, _: Request)).expects("domain", "do_9876.img", false, *).returns(Future(getNewRootNode())).anyNumberOfTimes() + (graphDB.getNodeByUniqueId(_: String, _: String, _: Boolean, _: Request)).expects("domain", *, false, *).returns(Future(getQuestionNode())).anyNumberOfTimes() + (graphDB.readExternalProps(_: Request, _: List[String])).expects(*, List("objectMetadata")).returns(Future(getSuccessfulResponse())).anyNumberOfTimes() + (graphDB.readExternalProps(_: Request, _: List[String])).expects(*, *).returns(Future(getExternalPropsResponseWithData())).anyNumberOfTimes() + (graphDB.updateExternalProps(_: Request)).expects(*).returns(Future(getSuccessfulResponse())).anyNumberOfTimes + (graphDB.saveExternalProps(_: Request)).expects(*).returns(Future(getSuccessfulResponse())).anyNumberOfTimes + (graphDB.upsertNode(_: String, _: Node, _: Request)).expects(*, *, *).returns(Future(getUpsertNode())).anyNumberOfTimes() + inSequence { + (graphDB.addNode(_: String, _: Node)).expects(*, *).returns(Future(getNewRootNode())) + (graphDB.addNode(_: String, _: Node)).expects(*, *).returns(Future(getQuestionNode())) + } + val request = getQuestionSetCopyRequest() + request.putAll(mapAsJavaMap(Map("identifier" -> "do_1234", "mode" -> "", "copyType"-> "deep"))) + request.setOperation("copyQuestionSet") + val response = callActor(request, Props(new QuestionSetActor())) + assert("successful".equals(response.getParams.getStatus)) + } + + it should "return success response for 'copyQuestionSet' (Shallow)" in { + implicit val oec: OntologyEngineContext = mock[OntologyEngineContext] + val graphDB = mock[GraphService] + (oec.graphService _).expects().returns(graphDB).anyNumberOfTimes() + val nodes: util.List[Node] = getCategoryNode() + (graphDB.getNodeByUniqueIds(_: String, _: SearchCriteria)).expects(*, *).returns(Future(nodes)).anyNumberOfTimes() + (graphDB.getNodeByUniqueId(_: String, _: String, _: Boolean, _: Request)).expects("domain", "do_1234", false, *).returns(Future(getExistingRootNode())).anyNumberOfTimes() + (graphDB.getNodeByUniqueId(_: String, _: String, _: Boolean, _: Request)).expects("domain", "do_5678", false, *).returns(Future(getNewRootNode())).anyNumberOfTimes() + (graphDB.readExternalProps(_: Request, _: List[String])).expects(*, List("objectMetadata")).returns(Future(getSuccessfulResponse())).anyNumberOfTimes() + (graphDB.readExternalProps(_: Request, _: List[String])).expects(*, *).returns(Future(getExternalPropsResponseWithData())).anyNumberOfTimes() + (graphDB.addNode(_: String, _: Node)).expects(*, *).returns(Future(getQuestionNode())) + (graphDB.saveExternalProps(_: Request)).expects(*).returns(Future(getSuccessfulResponse())).anyNumberOfTimes + (graphDB.upsertNode(_: String, _: Node, _: Request)).expects(*, *, *).returns(Future(getUpsertNode())).anyNumberOfTimes() + (graphDB.updateExternalProps(_: Request)).expects(*).returns(Future(getSuccessfulResponse())).anyNumberOfTimes + val request = getQuestionSetCopyRequest() + request.putAll(mapAsJavaMap(Map("identifier" -> "do_1234", "mode" -> "", "copyType"-> "shallow"))) + request.setOperation("copyQuestionSet") + val response = callActor(request, Props(new QuestionSetActor())) + assert("successful".equals(response.getParams.getStatus)) + } + + it should "return error response for 'copyQuestionSet' when createdFor & createdBy is missing" in { + implicit val oec: OntologyEngineContext = mock[OntologyEngineContext] + val request = getInvalidQuestionCopyRequest() + request.putAll(mapAsJavaMap(Map("identifier" -> "do_1234", "mode" -> "", "copyType"-> "deep"))) + request.setOperation("copyQuestionSet") + val response = callActor(request, Props(new QuestionSetActor())) + assert("failed".equals(response.getParams.getStatus)) + } + private def getQuestionSetRequest(): Request = { val request = new Request() request.setContext(new java.util.HashMap[String, AnyRef]() { diff --git a/assessment-api/assessment-actors/src/test/scala/org/sunbird/actors/copyTrait.scala b/assessment-api/assessment-actors/src/test/scala/org/sunbird/actors/copyTrait.scala new file mode 100644 index 000000000..5f5478c4d --- /dev/null +++ b/assessment-api/assessment-actors/src/test/scala/org/sunbird/actors/copyTrait.scala @@ -0,0 +1,264 @@ +package org.sunbird.actors + +import org.mortbay.util.StringUtil +import org.sunbird.common.dto.{Request, Response, ResponseParams} +import org.sunbird.graph.dac.model.Node +import org.sunbird.utils.AssessmentConstants +import org.sunbird.common.exception.{ResponseCode} + +import java.util + +trait copyTrait { + + private def getQuestionSetRequest(): Request = { + val request = new Request() + request.setContext(new java.util.HashMap[String, AnyRef]() { + { + put("graph_id", "domain") + put("version", "1.0") + put("objectType", "QuestionSet") + put("schemaName", "questionset") + } + }) + request.setObjectType("QuestionSet") + request + } + + def getQuestionSetCopyRequest(): Request = { + val request = getQuestionSetRequest() + request.putAll(new util.HashMap[String, AnyRef]() { + { + put("createdBy", "Shikshalokam") + put("createdFor", new util.ArrayList[String]() { + { + add("Shikshalokam") + } + }) + put("name", "NewRootNode") + } + }) + request + } + + def getInvalidQuestionSetCopyRequest(): Request = { + val request = getQuestionSetRequest() + request.putAll(new util.HashMap[String, AnyRef]() { + { + put("name", "NewRootNode") + } + }) + request + } + + def getInvalidQuestionCopyRequest(): Request = { + val request = getQuestionRequest() + request.putAll(new util.HashMap[String, AnyRef]() { + { + put("name", "NewQuestion") + } + }) + request + } + + private def getQuestionRequest(): Request = { + val request = new Request() + request.setContext(new java.util.HashMap[String, AnyRef]() { + { + put("graph_id", "domain") + put("version", "1.0") + put("objectType", "Question") + put("schemaName", "question") + } + }) + request.setObjectType("Question") + request + } + + def getQuestionCopyRequest(): Request = { + val request = getQuestionRequest() + request.putAll(new util.HashMap[String, AnyRef]() { + { + put("createdBy", "Shikshalokam") + put("createdFor", new util.ArrayList[String]() { + { + add("Shikshalokam") + } + }) + put("name", "NewQuestion") + } + }) + request + } + + private def getNode(objectType: String, identifier: String, primaryCategory: String, visibility: String, name: String, id: Long, + status: String): Node = { + val node = new Node("domain", "DATA_NODE", objectType) + node.setGraphId("domain") + node.setIdentifier(identifier) + node.setId(id) + node.setNodeType("DATA_NODE") + node.setObjectType(objectType) + node.setMetadata(new util.HashMap[String, AnyRef]() { + { + put("code", "xyz") + put("mimeType", { + if (StringUtil.endsWithIgnoreCase(objectType, AssessmentConstants.QUESTIONSET_SCHEMA_NAME)) { + AssessmentConstants.QUESTIONSET_MIME_TYPE + } else { + AssessmentConstants.QUESTION_MIME_TYPE + } + }) + put("createdOn", "2022-03-16T14:35:11.040+0530") + put("objectType", objectType) + put("primaryCategory", primaryCategory) + put("contentDisposition", "inline") + put("contentEncoding", "gzip") + put("lastUpdatedOn", "2022-03-16T14:38:51.287+0530") + put("showSolutions", "No") + put("allowAnonymousAccess", "Yes") + put("identifier", identifier) + put("lastStatusChangedOn", "2022-03-16T14:35:11.040+0530") + put("visibility", visibility) + put("showTimer", "No") + put("version", 1.asInstanceOf[Number]) + put("showFeedback", "No") + put("versionKey", "1234") + put("license", "CC BY 4.0") + put("compatibilityLevel", 5.asInstanceOf[Number]) + put("name", name) + put("status", status) + } + }) + node + } + + def getExistingRootNode(): Node = { + val node = getNode("QuestionSet", "do_1234", "Observation", AssessmentConstants.VISIBILITY_DEFAULT, "ExistingRootNode", 1234, + "Live") + node.getMetadata.put("childNodes", Array("do_5678")) + node + } + + def getNewRootNode(): Node = { + val node = getNode("QuestionSet", "do_9876", "Observation", AssessmentConstants.VISIBILITY_DEFAULT, "NewRootNode", 0, "Draft") + node.getMetadata.put("origin", "do_1234") + node.getMetadata.put("originData", "{\"name\":\"ExistingRootNode\",\"copyType\":\"deep\"}") + node.getMetadata.put("createdFor", Array("ShikshaLokam")) + node.getMetadata.put("createdBy", "ShikshaLokam") + node.setExternalData(new util.HashMap[String, AnyRef]() { + { + put("instructions", "This is the instruction.") + put("outcomeDeclaration", "This is outcomeDeclaration.") + } + }) + node + } + + def getExistingQuestionNode(): Node = { + val node = getNode("Question", "do_1234", "Slider", AssessmentConstants.VISIBILITY_DEFAULT, "ExistingQuestionNode", 1234, + "Live") + node + } + + def getQuestionNode(): Node = { + val node = getNode("Question", "do_5678", "Slider", AssessmentConstants.VISIBILITY_PARENT, "Question1", 0, "Draft") + node.setExternalData(new util.HashMap[String, AnyRef]() { + { + put("answer", "This is Answer.") + put("body", "This is Body.") + } + }) + node + } + + def getNewQuestionNode(): Node = { + val node = getNode("Question", "do_5678", "Slider", AssessmentConstants.VISIBILITY_DEFAULT, "NewQuestion", 0, "Draft") + node.setExternalData(new util.HashMap[String, AnyRef]() { + { + put("answer", "This is Answer.") + put("body", "This is Body.") + } + }) + node.getMetadata.put("origin", "do_1234") + node.getMetadata.put("originData", "{\\\"name\\\":\\\"Q2\\\",\\\"copyType\\\":\\\"deep\\\",\\\"license\\\":\\\"CC BY 4.0\\\"}") + node.getMetadata.put("createdFor", Array("ShikshaLokam")) + node.getMetadata.put("createdBy", "ShikshaLokam") + node + } + + def getSuccessfulResponse(): Response = { + val response = new Response + response.setVer("3.0") + val responseParams = new ResponseParams + responseParams.setStatus("successful") + response.setParams(responseParams) + response.setResponseCode(ResponseCode.OK) + response + } + + def getExternalPropsRequest(): Request = { + val request = getQuestionSetRequest() + request.putAll(new util.HashMap[String, AnyRef]() { + { + put("instructions", "This is the instruction.") + put("outcomeDeclaration", "This is outcomeDeclaration.") + } + }) + request + } + + def getExternalPropsResponseWithData(): Response = { + val response = getSuccessfulResponse() + response.put("instructions", "This is the instruction for this QuestionSet") + response.put("outcomeDeclaration", "This is the outcomeDeclaration for this QuestionSet") + response.put("hierarchy", "{\"code\":\"ExistingRootNode\",\"allowSkip\":\"Yes\",\"containsUserData\":\"No\"," + + "\"channel\":\"{{all}}\",\"language\":[\"English\"],\"showHints\":\"No\",\"mimeType\":\"application/vnd" + "" + ".sunbird" + "" + + ".questionset\",\"createdOn\":\"2022-03-16T14:35:11.040+0530\",\"objectType\":\"QuestionSet\"," + + "\"primaryCategory\":\"Observation\",\"contentDisposition\":\"inline\",\"contentEncoding\":\"gzip\"," + + "\"lastUpdatedOn\":\"2022-03-16T14:38:51.287+0530\",\"generateDIALCodes\":\"No\",\"showSolutions\":\"No\"," + + "\"allowAnonymousAccess\":\"Yes\",\"identifier\":\"do_1234\"," + "\"lastStatusChangedOn\":\"2022-03-16T14:35:11.040+0530\"," + + "\"requiresSubmit\":\"No\",\"visibility\":\"Default\"," + "" + "" + "\"IL_SYS_NODE_TYPE\":\"DATA_NODE\",\"showTimer\":\"No\"," + + "\"childNodes\":[\"do_113495678820704256110\"]," + "\"setType\":\"materialised\",\"version\":1," + "\"showFeedback\":\"No\"," + + "\"versionKey\":\"1647421731287\"," + "\"license\":\"CC BY 4.0\",\"depth\":0," + "\"compatibilityLevel\":5," + + "\"IL_FUNC_OBJECT_TYPE\":\"QuestionSet\"," + "\"allowBranching\":\"No\"," + "\"navigationMode\":\"non-linear\"," + + "\"name\":\"CopyQuestionSet\",\"shuffle\":true," + "\"IL_UNIQUE_ID\":\"do_11349567701798912019\",\"status\":\"Live\"," + + "\"children\":[{\"parent\":\"do_11349567701798912019\",\"code\":\"Q1\",\"channel\":\"{{channel_id}}\"," + + "\"language\":[\"English\"],\"mimeType\":\"application/vnd.sunbird.question\"," + + "\"createdOn\":\"2022-03-16T14:38:51.043+0530\",\"objectType\":\"Question\",\"primaryCategory\":\"Slider\"," + + "\"contentDisposition\":\"inline\",\"lastUpdatedOn\":\"2022-03-16T14:38:51.042+0530\"," + "\"contentEncoding\":\"gzip\"," + + "\"showSolutions\":\"No\",\"allowAnonymousAccess\":\"Yes\"," + "\"identifier\":\"do_113495678820704256110\"," + + "\"lastStatusChangedOn\":\"2022-03-16T14:38:51.043+0530\"," + "\"visibility\":\"Parent\",\"showTimer\":\"No\",\"index\":1," + + "\"languageCode\":[\"en\"],\"version\":1," + "\"versionKey\":\"1647421731066\",\"showFeedback\":\"No\",\"license\":\"CC BY " + + "4.0\",\"depth\":1," + "\"compatibilityLevel\":4,\"name\":\"Q1\",\"status\":\"Live\"}]}") + response.put("body", "This is Body") + response.put("answer", "This is Answer") + response + } + + def getReadPropsResponseForQuestion(): Response = { + val response = getSuccessfulResponse() + response.put("answer", "This is Answer 2") + response.put("body", "This is Body 2") + response + } + + def getUpsertNode(): Node = { + val node = getNewRootNode() + node.setExternalData(new util.HashMap[String, AnyRef]() { + { + put("hierarchy", "{\\\"identifier\\\":\\\"do_9876\\\"," + "\\\"children\\\":[{\\\"parent\\\":\\\"do_9876\\\"," + + "\\\"code\\\":\\\"b65f36d1-a243-4043-9df7-da14a2dd83b9\\\",\\\"channel\\\":\\\"{{channel_id}}\\\"," + + "\\\"language\\\":[\\\"English\\\"],\\\"mimeType\\\":\\\"application/vnd.sunbird.question\\\"," + + "\\\"createdOn\\\":\\\"2022-03-23T15:45:28.620+0530\\\",\\\"objectType\\\":\\\"Question\\\"," + + "\\\"primaryCategory\\\":\\\"Slider\\\",\\\"contentDisposition\\\":\\\"inline\\\"," + + "\\\"lastUpdatedOn\\\":\\\"2022-03-23T15:45:28.616+0530\\\",\\\"contentEncoding\\\":\\\"gzip\\\"," + + "\\\"showSolutions\\\":\\\"No\\\",\\\"allowAnonymousAccess\\\":\\\"Yes\\\"," + + "\\\"identifier\\\":\\\"do_11350066609045504013\\\",\\\"lastStatusChangedOn\\\":\\\"2022-03-23T15:45:28.621+0530\\\"," + + "\\\"visibility\\\":\\\"Parent\\\",\\\"showTimer\\\":\\\"No\\\",\\\"index\\\":1,\\\"languageCode\\\":[\\\"en\\\"]," + + "\\\"version\\\":1,\\\"versionKey\\\":\\\"1648030746815\\\",\\\"showFeedback\\\":\\\"No\\\",\\\"license\\\":\\\"CC BY " + + "4.0\\\",\\\"depth\\\":1,\\\"compatibilityLevel\\\":4,\\\"name\\\":\\\"Q1\\\",\\\"status\\\":\\\"Draft\\\"}]}") + } + }) + node + } +} diff --git a/assessment-api/assessment-service/app/controllers/v4/QuestionController.scala b/assessment-api/assessment-service/app/controllers/v4/QuestionController.scala index 0af9f3676..d929bb272 100644 --- a/assessment-api/assessment-service/app/controllers/v4/QuestionController.scala +++ b/assessment-api/assessment-service/app/controllers/v4/QuestionController.scala @@ -2,6 +2,8 @@ package controllers.v4 import akka.actor.{ActorRef, ActorSystem} import controllers.BaseController +import org.sunbird.utils.AssessmentConstants + import javax.inject.{Inject, Named} import play.api.mvc.ControllerComponents import utils.{ActorNames, ApiId, QuestionOperations} @@ -130,4 +132,15 @@ class QuestionController @Inject()(@Named(ActorNames.QUESTION_ACTOR) questionAct questionRequest.getContext.put("identifier", identifier) getResult(ApiId.REJECT_QUESTION, questionActor, questionRequest) } + + def copy(identifier: String, mode: Option[String]) = Action.async { implicit request => + val headers = commonHeaders() + val body = requestBody() + val question = body.getOrDefault("question", new java.util.HashMap()).asInstanceOf[java.util.Map[String, Object]]; + question.putAll(headers) + question.putAll(Map("identifier" -> identifier, "mode" -> mode.getOrElse(""), "copyType" -> AssessmentConstants.COPY_TYPE_DEEP).asJava) + val questionRequest = getRequest(question, headers, QuestionOperations.copyQuestion.toString) + setRequestContext(questionRequest, version, objectType, schemaName) + getResult(ApiId.COPY_QUESTION, questionActor, questionRequest) + } } diff --git a/assessment-api/assessment-service/app/controllers/v4/QuestionSetController.scala b/assessment-api/assessment-service/app/controllers/v4/QuestionSetController.scala index a82f320ae..f2fa9cdce 100644 --- a/assessment-api/assessment-service/app/controllers/v4/QuestionSetController.scala +++ b/assessment-api/assessment-service/app/controllers/v4/QuestionSetController.scala @@ -158,4 +158,15 @@ class QuestionSetController @Inject()(@Named(ActorNames.QUESTION_SET_ACTOR) ques questionSetRequest.getContext.put("identifier", identifier); getResult(ApiId.SYSTEM_UPDATE_QUESTION_SET, questionSetActor, questionSetRequest) } + + def copy(identifier: String, mode: Option[String], copyType: String) = Action.async { implicit request => + val headers = commonHeaders() + val body = requestBody() + val questionSet = body.getOrDefault("questionset", new java.util.HashMap()).asInstanceOf[java.util.Map[String, Object]]; + questionSet.putAll(headers) + questionSet.putAll(Map("identifier" -> identifier, "mode" -> mode.getOrElse(""), "copyType" -> copyType).asJava) + val questionSetRequest = getRequest(questionSet, headers, QuestionSetOperations.copyQuestionSet.toString) + setRequestContext(questionSetRequest, version, objectType, schemaName) + getResult(ApiId.COPY_QUESTION_SET, questionSetActor, questionSetRequest) + } } diff --git a/assessment-api/assessment-service/app/utils/ApiId.scala b/assessment-api/assessment-service/app/utils/ApiId.scala index d45e57597..062340338 100644 --- a/assessment-api/assessment-service/app/utils/ApiId.scala +++ b/assessment-api/assessment-service/app/utils/ApiId.scala @@ -24,6 +24,7 @@ object ApiId { val SYSTEM_UPDATE_QUESTION = "api.question.system.update" val LIST_QUESTIONS = "api.questions.list" val REJECT_QUESTION = "api.question.reject" + val COPY_QUESTION = "api.question.copy" //QuestionSet APIs val CREATE_QUESTION_SET = "api.questionset.create" @@ -40,5 +41,5 @@ object ApiId { val REJECT_QUESTION_SET = "api.questionset.reject" val IMPORT_QUESTION_SET = "api.questionset.import" val SYSTEM_UPDATE_QUESTION_SET = "api.questionset.system.update" - + val COPY_QUESTION_SET = "api.questionset.copy" } diff --git a/assessment-api/assessment-service/app/utils/QuestionOperations.scala b/assessment-api/assessment-service/app/utils/QuestionOperations.scala index 57e9e3815..8d71d88e1 100644 --- a/assessment-api/assessment-service/app/utils/QuestionOperations.scala +++ b/assessment-api/assessment-service/app/utils/QuestionOperations.scala @@ -1,5 +1,5 @@ package utils object QuestionOperations extends Enumeration { - val createQuestion, readQuestion, readPrivateQuestion, updateQuestion, reviewQuestion, publishQuestion, retireQuestion, importQuestion, systemUpdateQuestion, listQuestions, rejectQuestion = Value + val createQuestion, readQuestion, readPrivateQuestion, updateQuestion, reviewQuestion, publishQuestion, retireQuestion, importQuestion, systemUpdateQuestion, listQuestions, rejectQuestion, copyQuestion = Value } diff --git a/assessment-api/assessment-service/app/utils/QuestionSetOperations.scala b/assessment-api/assessment-service/app/utils/QuestionSetOperations.scala index afcd22e2b..43f0266b1 100644 --- a/assessment-api/assessment-service/app/utils/QuestionSetOperations.scala +++ b/assessment-api/assessment-service/app/utils/QuestionSetOperations.scala @@ -1,7 +1,7 @@ package utils object QuestionSetOperations extends Enumeration { - val createQuestionSet, readQuestionSet, readPrivateQuestionSet, updateQuestionSet, reviewQuestionSet, publishQuestionSet, - retireQuestionSet, addQuestion, removeQuestion, updateHierarchyQuestion, readHierarchyQuestion, - rejectQuestionSet, importQuestionSet, systemUpdateQuestionSet = Value + val createQuestionSet, readQuestionSet, readPrivateQuestionSet, updateQuestionSet, reviewQuestionSet, publishQuestionSet, + retireQuestionSet, addQuestion, removeQuestion, updateHierarchyQuestion, readHierarchyQuestion, + rejectQuestionSet, importQuestionSet, systemUpdateQuestionSet, copyQuestionSet = Value } diff --git a/assessment-api/assessment-service/conf/application.conf b/assessment-api/assessment-service/conf/application.conf index 82fe873b5..5e82dfa27 100644 --- a/assessment-api/assessment-service/conf/application.conf +++ b/assessment-api/assessment-service/conf/application.conf @@ -420,4 +420,14 @@ import { } } -root_node_visibility=["Default","Private"] \ No newline at end of file +root_node_visibility=["Default","Private"] +assessment.copy.origin_data=["name", "author", "license", "organisation"] +assessment.copy.props_to_remove=["downloadUrl", "artifactUrl", "variants", + "createdOn", "collections", "children", "lastUpdatedOn", "SYS_INTERNAL_LAST_UPDATED_ON", + "versionKey", "s3Key", "status", "pkgVersion", "toc_url", "mimeTypesCount", + "contentTypesCount", "leafNodesCount", "childNodes", "prevState", "lastPublishedOn", + "flagReasons", "compatibilityLevel", "size", "publishChecklist", "publishComment", + "LastPublishedBy", "rejectReasons", "rejectComment", "gradeLevel", "subject", + "medium", "board", "topic", "purpose", "subtopic", "contentCredits", + "owner", "collaborators", "creators", "contributors", "badgeAssertions", "dialcodes", + "concepts", "keywords", "reservedDialcodes", "dialcodeRequired", "leafNodes", "sYS_INTERNAL_LAST_UPDATED_ON", "prevStatus", "lastPublishedBy", "streamingUrl"] \ No newline at end of file diff --git a/assessment-api/assessment-service/conf/routes b/assessment-api/assessment-service/conf/routes index e33c9c1da..b76bb70a9 100644 --- a/assessment-api/assessment-service/conf/routes +++ b/assessment-api/assessment-service/conf/routes @@ -19,10 +19,11 @@ PATCH /question/v4/update/:identifier controllers.v4.QuestionControl POST /question/v4/review/:identifier controllers.v4.QuestionController.review(identifier:String) POST /question/v4/publish/:identifier controllers.v4.QuestionController.publish(identifier:String) DELETE /question/v4/retire/:identifier controllers.v4.QuestionController.retire(identifier:String) -POST /question/v4/import controllers.v4.QuestionController.importQuestion() +POST /question/v4/import controllers.v4.QuestionController.importQuestion() PATCH /question/v4/system/update/:identifier controllers.v4.QuestionController.systemUpdate(identifier:String) POST /question/v4/list controllers.v4.QuestionController.list(fields:Option[String]) POST /question/v4/reject/:identifier controllers.v4.QuestionController.reject(identifier:String) +POST /question/v4/copy/:identifier controllers.v4.QuestionController.copy(identifier:String, mode:Option[String]) # QuestionSet API's POST /questionset/v4/create controllers.v4.QuestionSetController.create @@ -37,5 +38,6 @@ DELETE /questionset/v4/remove controllers.v4.QuestionSetC PATCH /questionset/v4/hierarchy/update controllers.v4.QuestionSetController.updateHierarchy GET /questionset/v4/hierarchy/:identifier controllers.v4.QuestionSetController.getHierarchy(identifier:String, mode:Option[String]) POST /questionset/v4/reject/:identifier controllers.v4.QuestionSetController.reject(identifier:String) -POST /questionset/v4/import controllers.v4.QuestionSetController.importQuestionSet() -PATCH /questionset/v4/system/update/:identifier controllers.v4.QuestionSetController.systemUpdate(identifier:String) \ No newline at end of file +POST /questionset/v4/import controllers.v4.QuestionSetController.importQuestionSet() +PATCH /questionset/v4/system/update/:identifier controllers.v4.QuestionSetController.systemUpdate(identifier:String) +POST /questionset/v4/copy/:identifier controllers.v4.QuestionSetController.copy(identifier:String, mode:Option[String], type:String?="deep") \ No newline at end of file diff --git a/schemas/question/1.0/schema.json b/schemas/question/1.0/schema.json index 670d548e5..6054a39e3 100644 --- a/schemas/question/1.0/schema.json +++ b/schemas/question/1.0/schema.json @@ -574,6 +574,12 @@ "items": { "type": "object" } + }, + "origin": { + "type": "string" + }, + "originData": { + "type": "object" } }, "additionalProperties": false diff --git a/schemas/questionset/1.0/schema.json b/schemas/questionset/1.0/schema.json index 02516dc47..fcc691a25 100644 --- a/schemas/questionset/1.0/schema.json +++ b/schemas/questionset/1.0/schema.json @@ -650,6 +650,12 @@ "outcomeDeclaration": { "type": "object", "description": "External Property" + }, + "origin": { + "type": "string" + }, + "originData": { + "type": "object" } }, "additionalProperties": false From 76401a1166162f32bd44de941f300e46b1ef7a27 Mon Sep 17 00:00:00 2001 From: joffinjoy <35325730+joffinjoy@users.noreply.github.com> Date: Fri, 1 Apr 2022 20:17:50 +0530 Subject: [PATCH 2/9] Issue #SB-29145 feat: Added BranchingLogic Copy --- .../org/sunbird/managers/CopyManager.scala | 148 +++++++++++++++++- .../sunbird/utils/AssessmentContants.scala | 9 ++ 2 files changed, 150 insertions(+), 7 deletions(-) diff --git a/assessment-api/assessment-actors/src/main/scala/org/sunbird/managers/CopyManager.scala b/assessment-api/assessment-actors/src/main/scala/org/sunbird/managers/CopyManager.scala index 5c5a5fa73..609acdc71 100644 --- a/assessment-api/assessment-actors/src/main/scala/org/sunbird/managers/CopyManager.scala +++ b/assessment-api/assessment-actors/src/main/scala/org/sunbird/managers/CopyManager.scala @@ -1,9 +1,11 @@ package org.sunbird.managers +import com.google.gson.{Gson, GsonBuilder} +import com.google.gson.reflect.TypeToken import org.apache.commons.collections.CollectionUtils import org.apache.commons.collections4.MapUtils import org.apache.commons.lang.StringUtils -import org.sunbird.common.Platform +import org.sunbird.common.{JsonUtils, Platform} import org.sunbird.common.dto.{Request, Response, ResponseHandler} import org.sunbird.common.exception.{ClientException, ServerException} import org.sunbird.graph.OntologyEngineContext @@ -13,11 +15,13 @@ import org.sunbird.graph.nodes.DataNode import org.sunbird.graph.schema.DefinitionNode import org.sunbird.graph.utils.{NodeUtil, ScalaJsonUtils} import org.sunbird.telemetry.logger.TelemetryManager -import org.sunbird.utils.AssessmentConstants +import org.sunbird.utils.{AssessmentConstants, HierarchyConstants} import java.util import java.util.concurrent.{CompletionException, TimeUnit} +import java.util.stream.Collectors import java.util.{Optional, UUID} +import scala.collection.JavaConversions.{asScalaBuffer, mapAsScalaMap} import scala.collection.JavaConverters._ import scala.concurrent.duration.{Duration, FiniteDuration} import scala.concurrent.{Await, ExecutionContext, Future} @@ -97,15 +101,144 @@ object CopyManager { }).flatMap(f => f) } - def updateHierarchy(request: Request, node: Node, originNode: Node, originHierarchy: util.Map[String, AnyRef], copyType: String)(implicit ec: ExecutionContext, oec: OntologyEngineContext): Future[Node] = { + + def generateNodeBLRecord(nodesModified: util.HashMap[String, AnyRef]): util.HashMap[String, AnyRef] = { + val idSet = nodesModified.keySet().asScala.toList + val nodeBLRecord = new util.HashMap[String, AnyRef]() + idSet.map(id => { + val nodeMetaData = nodesModified.getOrDefault(id, new util.HashMap()).asInstanceOf[util.Map[String, AnyRef]].getOrDefault(AssessmentConstants.METADATA, new util.HashMap()).asInstanceOf[util.Map[String, AnyRef]] + val containsBL = nodeMetaData.containsKey(AssessmentConstants.BRANCHING_LOGIC) + nodeBLRecord.put(id, new util.HashMap[String, AnyRef]() { + { + if (containsBL) put(AssessmentConstants.BRANCHING_LOGIC, nodeMetaData.get(AssessmentConstants.BRANCHING_LOGIC)) + put(AssessmentConstants.CONTAINS_BL, containsBL.asInstanceOf[AnyRef]) + put(AssessmentConstants.COPY_OF, nodeMetaData.get(AssessmentConstants.COPY_OF).asInstanceOf[String]) + } + }) + if (containsBL) nodeMetaData.remove(AssessmentConstants.BRANCHING_LOGIC) + nodeMetaData.remove(AssessmentConstants.COPY_OF) + }) + nodeBLRecord + } + + def branchingLogicArrayHandler(nodeBL: util.HashMap[String, AnyRef], name: String, oldToNewIdMap: util.Map[String, String]) = { + val array = nodeBL.getOrDefault(name, new util.ArrayList[String]).asInstanceOf[util.ArrayList[String]] + val newArray = new util.ArrayList[String]() + array.map(id => { + if (oldToNewIdMap.containsKey(id)) { + newArray.add(oldToNewIdMap.get(id)) + } else newArray.add(id) + }) + nodeBL.remove(name) + nodeBL.put(name, newArray) + } + + def preConditionHandler(nodeBL: util.HashMap[String, AnyRef], oldToNewIdMap: util.Map[String, String]): Unit = { + val preCondition = nodeBL.get(AssessmentConstants.PRE_CONDITION).asInstanceOf[util.HashMap[String, AnyRef]] + preCondition.keySet().asScala.toList.map(key => { + val conjunctionArray = preCondition.get(key).asInstanceOf[util.ArrayList[String]] + val condition = conjunctionArray.get(0).asInstanceOf[util.HashMap[String, AnyRef]] + condition.keySet().asScala.toList.map(logicOp => { + val conditionArray = condition.get(logicOp).asInstanceOf[util.ArrayList[String]] + val sourceQuestionRecord = conditionArray.get(0).asInstanceOf[util.HashMap[String, AnyRef]] + val preConditionVar = sourceQuestionRecord.get(AssessmentConstants.PRE_CONDITION_VAR).asInstanceOf[String] + val stringArray = preConditionVar.split("\\.") + if (oldToNewIdMap.containsKey(stringArray(0))) { + val newString = oldToNewIdMap.get(stringArray(0)) + "." + stringArray.drop(1).mkString(".") + sourceQuestionRecord.remove(AssessmentConstants.PRE_CONDITION_VAR) + sourceQuestionRecord.put(AssessmentConstants.PRE_CONDITION_VAR, newString) + } + }) + }) + } + + def branchingLogicModifier(branchingLogic: util.HashMap[String, AnyRef], oldToNewIdMap: util.Map[String, String]): Unit = { + branchingLogic.keySet().asScala.toList.map(identifier => { + val nodeBL = branchingLogic.get(identifier).asInstanceOf[util.HashMap[String, AnyRef]] + nodeBL.keySet().asScala.toList.map(key => { + if (StringUtils.equalsIgnoreCase(key, AssessmentConstants.TARGET)) branchingLogicArrayHandler(nodeBL, AssessmentConstants.TARGET, oldToNewIdMap) + else if (StringUtils.equalsIgnoreCase(key, AssessmentConstants.PRE_CONDITION)) preConditionHandler(nodeBL, oldToNewIdMap) + else if (StringUtils.equalsIgnoreCase(key, AssessmentConstants.SOURCE)) branchingLogicArrayHandler(nodeBL, AssessmentConstants.SOURCE, oldToNewIdMap) + }) + if (oldToNewIdMap.containsKey(identifier)) { + branchingLogic.put(oldToNewIdMap.get(identifier), nodeBL) + branchingLogic.remove(identifier) + } + }) + } + + def generateOldToNewIdMap(nodeBLRecord: util.HashMap[String, AnyRef], identifiers: util.Map[String, String]): util.Map[String, String] = { + val oldToNewIdMap = new util.HashMap[String, String]() + nodeBLRecord.keySet().asScala.toList.map(id => { + val nodeInfo = nodeBLRecord.get(id).asInstanceOf[util.HashMap[String, AnyRef]] + val newId = identifiers.get(id) + val oldId = nodeInfo.get(AssessmentConstants.COPY_OF).asInstanceOf[String] + oldToNewIdMap.put(oldId, newId) + }) + oldToNewIdMap + } + + def hierarchyRequestModifier(request: Request, nodeBLRecord: util.HashMap[String, AnyRef], identifiers: util.Map[String, String]): Unit = { + val nodesModified: java.util.HashMap[String, AnyRef] = request.getRequest.get(HierarchyConstants.NODES_MODIFIED).asInstanceOf[java.util.HashMap[String, AnyRef]] + val hierarchy: java.util.HashMap[String, AnyRef] = request.getRequest.get(HierarchyConstants.HIERARCHY).asInstanceOf[java.util.HashMap[String, AnyRef]] + val oldToNewIdMap = generateOldToNewIdMap(nodeBLRecord, identifiers) + nodeBLRecord.keySet().asScala.toList.map(id => { + val nodeInfo = nodeBLRecord.get(id).asInstanceOf[util.HashMap[String, AnyRef]] + val node = nodesModified.get(id).asInstanceOf[util.HashMap[String, AnyRef]] + val nodeMetaData = node.get(AssessmentConstants.METADATA).asInstanceOf[util.HashMap[String, AnyRef]] + val newId = identifiers.get(id) + if (nodeInfo.get(AssessmentConstants.CONTAINS_BL).asInstanceOf[Boolean]) { + val branchingLogic = nodeInfo.get(AssessmentConstants.BRANCHING_LOGIC).asInstanceOf[util.HashMap[String, AnyRef]] + branchingLogicModifier(branchingLogic, oldToNewIdMap) + nodeMetaData.put(AssessmentConstants.BRANCHING_LOGIC, branchingLogic) + } + node.remove(AssessmentConstants.IS_NEW) + node.put(AssessmentConstants.IS_NEW, false.asInstanceOf[AnyRef]) + nodesModified.remove(id) + nodesModified.put(newId, node) + }) + hierarchy.keySet().asScala.toList.map(id => { + val nodeHierarchy = hierarchy.get(id).asInstanceOf[util.HashMap[String, AnyRef]] + val children = nodeHierarchy.get(AssessmentConstants.CHILDREN).asInstanceOf[util.ArrayList[String]] + val newChildrenList = new util.ArrayList[String] + children.map(identifier => { + if (identifiers.containsKey(identifier)) newChildrenList.add(identifiers.get(identifier)) else newChildrenList.add(identifier) + }) + nodeHierarchy.remove(AssessmentConstants.CHILDREN) + nodeHierarchy.put(AssessmentConstants.CHILDREN, newChildrenList) + if (identifiers.containsKey(id)) { + hierarchy.remove(id) + hierarchy.put(identifiers.get(id), nodeHierarchy) + } + }) + } + + def updateHierarchy(request: Request, node: Node, originNode: Node, originHierarchy: util.Map[String, AnyRef], copyType: String) + (implicit ec: ExecutionContext, oec: OntologyEngineContext): Future[Node] = { prepareHierarchyRequest(originHierarchy, originNode, node, copyType, request).map(req => { val hierarchyRequest = new Request(request) hierarchyRequest.putAll(req) - hierarchyRequest.getContext.put(AssessmentConstants.SCHEMA_NAME, AssessmentConstants.QUESTIONSET_SCHEMA_NAME) - hierarchyRequest.getContext.put(AssessmentConstants.VERSION, AssessmentConstants.SCHEMA_VERSION) + val nodesModified: java.util.HashMap[String, AnyRef] = hierarchyRequest.getRequest.get(HierarchyConstants.NODES_MODIFIED) + .asInstanceOf[java.util.HashMap[String, AnyRef]] + val nodeBLRecord = generateNodeBLRecord(nodesModified) + val newUpdateRequest = JsonUtils.deserialize(ScalaJsonUtils.serialize(hierarchyRequest), classOf[Request]) UpdateHierarchyManager.updateHierarchy(hierarchyRequest).map(response => { - if (!ResponseHandler.checkError(response)) node else { - TelemetryManager.info(s"Update Hierarchy Failed For Copy Question Set Having Identifier: ${node.getIdentifier} | Response is : " + response) + if (!ResponseHandler.checkError(response)) { + val identifiers = response.getResult.get(AssessmentConstants.IDENTIFIERS).asInstanceOf[util.Map[String, String]] + hierarchyRequestModifier(newUpdateRequest, nodeBLRecord, identifiers) + UpdateHierarchyManager.updateHierarchy(newUpdateRequest).map(response_ => { + if (!ResponseHandler.checkError(response_)) { + node + } else { + TelemetryManager.info(s"Update Hierarchy Failed For Copy Question Set Having Identifier: ${node.getIdentifier} | Response " + + s"is " + s": " + response) + throw new ServerException("ERR_QUESTIONSET_COPY", "Something Went Wrong, Please Try Again") + } + }) + node + } else { + TelemetryManager.info(s"Update Hierarchy Failed For Copy Question Set Having Identifier: ${node.getIdentifier} | Response is " + + s": " + response) throw new ServerException("ERR_QUESTIONSET_COPY", "Something Went Wrong, Please Try Again") } }) @@ -167,6 +300,7 @@ object CopyManager { put(AssessmentConstants.METADATA, cleanUpCopiedData(new util.HashMap[String, AnyRef]() { { putAll(child) + put("copyOf", child.getOrDefault(AssessmentConstants.IDENTIFIER,"")) put(AssessmentConstants.CHILDREN, new util.ArrayList()) internalHierarchyProps.map(key => remove(key)) } diff --git a/assessment-api/assessment-actors/src/main/scala/org/sunbird/utils/AssessmentContants.scala b/assessment-api/assessment-actors/src/main/scala/org/sunbird/utils/AssessmentContants.scala index c422c3256..16e47ccde 100644 --- a/assessment-api/assessment-actors/src/main/scala/org/sunbird/utils/AssessmentContants.scala +++ b/assessment-api/assessment-actors/src/main/scala/org/sunbird/utils/AssessmentContants.scala @@ -36,4 +36,13 @@ object AssessmentConstants { val QUESTION_SCHEMA_NAME: String = "question" val VISIBILITY_PARENT: String = "Parent" val VISIBILITY_DEFAULT: String = "Default" + val BRANCHING_LOGIC: String = "branchingLogic" + val COPY_OF: String = "copyOf" + val CONTAINS_BL: String = "containsBL" + val IDENTIFIERS: String = "identifiers" + val IS_NEW: String = "isNew" + val TARGET: String = "target" + val PRE_CONDITION: String = "preCondition" + val SOURCE: String = "source" + val PRE_CONDITION_VAR : String = "var" } From f05328de25acba194e6210dda606e333ba12f6c7 Mon Sep 17 00:00:00 2001 From: joffinjoy <35325730+joffinjoy@users.noreply.github.com> Date: Thu, 7 Apr 2022 10:19:21 +0530 Subject: [PATCH 3/9] Issue #SB-29145 test: Added BranchingLogic Copy Test-Case --- .../sunbird/actors/QuestionSetActorTest.scala | 33 +++ .../scala/org/sunbird/actors/copyTrait.scala | 229 ++++++++++++++++++ 2 files changed, 262 insertions(+) diff --git a/assessment-api/assessment-actors/src/test/scala/org/sunbird/actors/QuestionSetActorTest.scala b/assessment-api/assessment-actors/src/test/scala/org/sunbird/actors/QuestionSetActorTest.scala index 25cf6c6d1..9d6019aea 100644 --- a/assessment-api/assessment-actors/src/test/scala/org/sunbird/actors/QuestionSetActorTest.scala +++ b/assessment-api/assessment-actors/src/test/scala/org/sunbird/actors/QuestionSetActorTest.scala @@ -594,6 +594,39 @@ class QuestionSetActorTest extends BaseSpec with MockFactory with copyTrait { assert("failed".equals(response.getParams.getStatus)) } + it should "return success response for 'copyQuestionSet' (Branching Logic Copy)" in { + implicit val oec: OntologyEngineContext = mock[OntologyEngineContext] + val graphDB = mock[GraphService] + (oec.graphService _).expects().returns(graphDB).anyNumberOfTimes() + val nodes: util.List[Node] = getCategoryNode() + (graphDB.getNodeByUniqueIds(_: String, _: SearchCriteria)).expects(*, *).returns(Future(nodes)).anyNumberOfTimes() + (graphDB.getNodeByUniqueId(_: String, _: String, _: Boolean, _: Request)).expects(*, "do_1234", *, *).returns(Future(getRootNodeWithBL("do_1234", "do_2222", true, true))).anyNumberOfTimes() + (graphDB.getNodeByUniqueId(_: String, _: String, _: Boolean, _: Request)).expects(*, "do_9876", *, *).returns(Future(getRootNodeWithBL("do_9876", "do_3333", false, false))).anyNumberOfTimes() + (graphDB.getNodeByUniqueId(_: String, _: String, _: Boolean, _: Request)).expects(*, "do_9876.img", *, *).returns(Future(getRootNodeWithBL("do_9876", "do_3333", false, false))).anyNumberOfTimes() + (graphDB.getNodeByUniqueId(_: String, _: String, _: Boolean, _: Request)).expects(*, "do_5555", *, *).returns(Future(getQuestionNodeBL("do_5555"))).anyNumberOfTimes() + (graphDB.getNodeByUniqueId(_: String, _: String, _: Boolean, _: Request)).expects(*, "do_7777", *, *).returns(Future(getQuestionNodeBL("do_7777"))).anyNumberOfTimes() + (graphDB.readExternalProps(_: Request, _: List[String])).expects(*, List("objectMetadata")).returns(Future(getSuccessfulResponse)).anyNumberOfTimes() + (graphDB.readExternalProps(_: Request, _: List[String])).expects(*, List("instructions", "outcomeDeclaration")).returns(Future(getSuccessfulResponse)).anyNumberOfTimes() + (graphDB.readExternalProps(_: Request, _: List[String])).expects(*, List("solutions", "body", "editorState", "interactions", "hints", "responseDeclaration", "media", "answer", "instructions")).returns(Future(getResourceNotFoundResponse)).anyNumberOfTimes() + (graphDB.addNode(_: String, _: Node)).expects(*, *).returns(Future(getRootNodeWithBL("do_9876", "do_3333", false, false))).anyNumberOfTimes + (graphDB.saveExternalProps(_: Request)).expects(*).returns(Future(getSuccessfulResponse)).anyNumberOfTimes + (graphDB.updateExternalProps(_: Request)).expects(*).returns(Future(getSuccessfulResponse)).anyNumberOfTimes + (graphDB.getNodeByUniqueId(_: String, _: String, _: Boolean, _: Request)).expects(*, *, *, *).returns(Future(getRootNodeWithBL("do_9876", "do_3333", true, true))).anyNumberOfTimes() + inSequence { + (graphDB.upsertNode(_: String, _: Node, _: Request)).expects(*, *, *).returns(Future(getUpsertNodeBLWithoutBL)) + (graphDB.upsertNode(_: String, _: Node, _: Request)).expects(*, *, *).returns(Future(getUpsertNodeBLWithBL)).anyNumberOfTimes + } + inSequence { + (graphDB.readExternalProps(_: Request, _: List[String])).expects(*, List("hierarchy")).returns(Future(getRootExternalPropsResponseBL)) + (graphDB.readExternalProps(_: Request, _: List[String])).expects(*, List("hierarchy")).returns(Future(getNewRootExternalPropsResponseBL)).anyNumberOfTimes + } + val request = getQuestionSetCopyRequest() + request.putAll(mapAsJavaMap(Map("identifier" -> "do_1234", "mode" -> "", "copyType" -> "deep"))) + request.setOperation("copyQuestionSet") + val response = callActor(request, Props(new QuestionSetActor())) + assert("successful".equals(response.getParams.getStatus)) + } + private def getQuestionSetRequest(): Request = { val request = new Request() request.setContext(new java.util.HashMap[String, AnyRef]() { diff --git a/assessment-api/assessment-actors/src/test/scala/org/sunbird/actors/copyTrait.scala b/assessment-api/assessment-actors/src/test/scala/org/sunbird/actors/copyTrait.scala index 5f5478c4d..3bf221a77 100644 --- a/assessment-api/assessment-actors/src/test/scala/org/sunbird/actors/copyTrait.scala +++ b/assessment-api/assessment-actors/src/test/scala/org/sunbird/actors/copyTrait.scala @@ -261,4 +261,233 @@ trait copyTrait { }) node } + + def getRootNodeWithBL(rootId: String, sectionId: String, addBranchingLogic: Boolean, withChildren: Boolean): Node = { + val node = getNode("QuestionSet", rootId, "Observation", AssessmentConstants.VISIBILITY_DEFAULT, "ExistingRootNode", 1234, "Live") + if (withChildren) { + val section = getNode("QuestionSet", sectionId, "Observation", AssessmentConstants.VISIBILITY_DEFAULT, "Section_1", 1234, "Live") + val children = new util.ArrayList[util.Map[String, AnyRef]]() + children.add(getNode("Question", "do_5555", "Slider", AssessmentConstants.VISIBILITY_DEFAULT, "Question1", 1234, "Live") + .getMetadata) + children.add(getNode("Question", "do_7777", "Slider", AssessmentConstants.VISIBILITY_DEFAULT, "Question2", 1234, "Live") + .getMetadata) + if (addBranchingLogic) { + section.getMetadata.put("branchingLogic", new util.HashMap[String, AnyRef]() { + { + put("do_5555", new util.HashMap[String, AnyRef]() { + put("target", new util.ArrayList[String]() { + { + add("do_7777") + } + }) + put("preCondition", new util.HashMap[String, AnyRef]()) + put("source", new util.ArrayList[String]()) + }) + put("do_7777", new util.HashMap[String, AnyRef]() { + put("target", new util.ArrayList[String]()) + put("preCondition", new util.HashMap[String, AnyRef]() { + { + put("and", new util.ArrayList[util.HashMap[String, AnyRef]]() { + add(new util.HashMap[String, AnyRef]() { + put("eq", new util.ArrayList[AnyRef]() { + { + add(new util.HashMap[String, String]() { + put("var", "do_5555" + ".response1.value") + put("type", "responseDeclaration") + }) + add("0") + } + }) + }) + }) + } + }) + put("source", new util.ArrayList[String]() { + { + add("do_5555") + } + }) + }) + } + }) + } + node.getMetadata.put("childNodes", new util.ArrayList[String]() { + { + add(sectionId) + add("do_5555") + add("do_7777") + } + }) + section.getMetadata.put("children", children) + node.getMetadata.put("children", new util.ArrayList[util.Map[String, AnyRef]]() { + { + add(section.getMetadata) + } + }) + } + node + } + + def getQuestionNodeBL(identifier: String): Node = { + val node = getNode("Question", identifier, "Slider", AssessmentConstants.VISIBILITY_DEFAULT, identifier, 1234, "Live") + node + } + + def getUpsertNodeBLWithoutBL(): Node = { + val node = getRootNodeWithBL("do_9876", "do_3333", false, false) + node.setExternalData(new util.HashMap[String, AnyRef]() { + { + put("hierarchy", "{\"identifier\":\"do_9876\",\"children\":[{\"parent\":\"do_9876\"," + + "\"code\":\"9f0332ad-c3e3-4803-b673-50174aff24e3\",\"allowSkip\":\"Yes\",\"containsUserData\":\"No\"," + + "\"channel\":\"{{channel_id}}\",\"language\":[\"English\"],\"mimeType\":\"application/vnd.sunbird.questionset\"," + + "\"showHints\":\"No\",\"createdOn\":\"2022-04-06T12:51:53.592+0530\",\"objectType\":\"QuestionSet\"," + + "\"primaryCategory\":\"Observation\",\"children\":[{\"parent\":\"do_3333\",\"code\":\"Q1\"," + + "\"channel\":\"{{channel_id}}\"," + "\"language\":[\"English\"],\"mimeType\":\"application/vnd.sunbird.question\"," + + "\"createdOn\":\"2022-04-06T10:13:32.859+0530\"," + "\"objectType\":\"Question\",\"primaryCategory\":\"Slider\"," + + "\"contentDisposition\":\"inline\"," + "\"lastUpdatedOn\":\"2022-04-06T10:13:32.911+0530\",\"contentEncoding\":\"gzip\"," + + "" + "\"showSolutions\":\"No\"," + "\"allowAnonymousAccess\":\"Yes\",\"identifier\":\"do_5555\"," + + "\"lastStatusChangedOn\":\"2022-04-06T10:13:32.859+0530\"," + "\"visibility\":\"Default\",\"showTimer\":\"No\"," + + "\"index\":1," + "\"languageCode\":[\"en\"],\"version\":1," + "\"versionKey\":\"1649220212911\",\"showFeedback\":\"No\"," + + "\"license\":\"CC BY " + "4.0\",\"depth\":2,\"compatibilityLevel\":4," + "\"name\":\"Q1\",\"status\":\"Live\"}," + + "{\"parent\":\"do_3333\",\"code\":\"Q2\",\"channel\":\"{{channel_id}}\"," + "\"language\":[\"English\"]," + + "\"mimeType\":\"application/vnd.sunbird.question\",\"createdOn\":\"2022-04-06T10:13:32.896+0530\"," + + "\"objectType\":\"Question\",\"primaryCategory\":\"Slider\",\"contentDisposition\":\"inline\"," + + "\"lastUpdatedOn\":\"2022-04-06T10:13:32.954+0530\",\"contentEncoding\":\"gzip\",\"showSolutions\":\"No\"," + + "\"allowAnonymousAccess\":\"Yes\",\"identifier\":\"do_7777\"," + + "\"lastStatusChangedOn\":\"2022-04-06T10:13:32.896+0530\"," + "\"visibility\":\"Default\",\"showTimer\":\"No\"," + + "\"index\":2," + "\"languageCode\":[\"en\"],\"version\":1," + "\"versionKey\":\"1649220212954\",\"showFeedback\":\"No\"," + + "\"license\":\"CC BY " + "4.0\",\"depth\":2,\"compatibilityLevel\":4," + "\"name\":\"Q2\",\"status\":\"Live\"}]," + + "\"contentDisposition\":\"inline\"," + "\"lastUpdatedOn\":\"2022-04-06T12:51:53.591+0530\"," + + "\"contentEncoding\":\"gzip\",\"generateDIALCodes\":\"No\"," + "\"showSolutions\":\"No\"," + + "\"allowAnonymousAccess\":\"Yes\"," + "\"identifier\":\"do_3333\"," + + "\"lastStatusChangedOn\":\"2022-04-06T12:51:53.592+0530\",\"requiresSubmit\":\"No\"," + "\"visibility\":\"Parent\"," + + "\"showTimer\":\"No\",\"index\":1,\"setType\":\"materialised\",\"languageCode\":[\"en\"],\"version\":1," + "" + "" + + "\"versionKey\":\"1649229713592\",\"showFeedback\":\"No\",\"license\":\"CC BY 4.0\",\"depth\":1,\"name\":\"S1\"," + + "\"navigationMode\":\"non-linear\",\"allowBranching\":\"Yes\",\"shuffle\":true,\"status\":\"Draft\"}]}") + } + }) + node + } + + def getUpsertNodeBLWithBL(): Node = { + val node = getRootNodeWithBL("do_9876", "do_3333", false, false) + node.setExternalData(new util.HashMap[String, AnyRef]() { + { + put("hierarchy", "{\"identifier\":\"do_9876\",\"children\":[{\"parent\":\"do_9876\"," + "\"code\":\"S1\"," + + "\"allowSkip\":\"Yes\",\"containsUserData\":\"No\",\"channel\":\"{{channel_id}}\"," + + "\"branchingLogic\":{\"do_7777\":{\"preCondition\":{\"and\":[{\"eq\":[{\"type\":\"responseDeclaration" + "\"," + + "\"var\":\"do_5555.response1.value\"},\"0\"]}]},\"target\":[]," + "\"source\":[\"do_5555\"]}," + + "\"do_5555\":{\"preCondition\":{}," + "\"target\":[\"do_7777\"],\"source\":[]}},\"description\":\"Section 1\"," + + "\"language\":[\"English\"]," + "\"mimeType\":\"application/vnd" + ".sunbird.questionset\",\"showHints\":\"No\"," + + "\"createdOn\":\"2022-04-04T16:30:59.566+0530\"," + "\"objectType\":\"QuestionSet\"," + + "\"primaryCategory\":\"Observation\"," + "\"children\":[{\"parent\":\"do_3333\",\"code\":\"Q1\"," + + "\"channel\":\"{{channel_id}}\"," + "\"description\":\"Q1\",\"language\":[\"English\"],\"mimeType\":\"application/vnd" + + ".sunbird" + ".question\"," + "\"createdOn\":\"2022-04-04T16:30:59.539+0530\",\"objectType\":\"Question\"," + + "\"primaryCategory\":\"Slider\"," + "\"contentDisposition\":\"inline\"," + + "\"lastUpdatedOn\":\"2022-04-04T16:32:46.200+0530\",\"contentEncoding\":\"gzip\"," + "\"showSolutions\":\"No\"," + + "\"allowAnonymousAccess\":\"Yes\",\"identifier\":\"do_5555\"," + + "\"lastStatusChangedOn\":\"2022-03-29T15:37:42.837+0530\",\"visibility\":\"Parent\",\"showTimer\":\"No\",\"index\":1," + + "\"languageCode\":[\"en\"],\"version\":1,\"versionKey\":\"1649070166325\",\"showFeedback\":\"No\",\"license\":\"CC BY " + + "4.0\"," + "\"depth\":2,\"compatibilityLevel\":4,\"name\":\"Q1\",\"status\":\"Draft\"}," + "{\"parent\":\"do_3333\"," + + "\"code\":\"Q2\"," + "\"channel\":\"{{channel_id}}\",\"description\":\"Q2\"," + "\"language\":[\"English\"]," + + "\"mimeType\":\"application/vnd.sunbird" + "" + ".question\"," + "\"createdOn\":\"2022-03-29T15:37:42.852+0530\"," + + "\"objectType\":\"Question\",\"primaryCategory\":\"Slider\"," + "\"contentDisposition\":\"inline\"," + + "\"lastUpdatedOn\":\"2022-03-29T15:37:42.896+0530\",\"contentEncoding\":\"gzip\"," + "\"showSolutions\":\"No\"," + + "\"allowAnonymousAccess\":\"Yes\",\"identifier\":\"do_7777\"," + + "\"lastStatusChangedOn\":\"2022-03-29T15:37:42.852+0530\",\"visibility\":\"Default\",\"showTimer\":\"No\",\"index\":2," + + "\"languageCode\":[\"en\"],\"version\":1,\"versionKey\":\"1648548462896\",\"showFeedback\":\"No\",\"license\":\"CC BY " + + "" + "4.0\"," + "\"depth\":2,\"compatibilityLevel\":4,\"name\":\"Q2\",\"status\":\"Live\"}]," + + "\"contentDisposition\":\"inline\"," + "\"lastUpdatedOn\":\"2022-04-04T16:32:46.273+0530\",\"contentEncoding\":\"gzip\"," + + "\"generateDIALCodes\":\"No\"," + "\"showSolutions\":\"No\",\"allowAnonymousAccess\":\"Yes\",\"identifier\":\"do_3333\"," + + "" + "\"lastStatusChangedOn\":\"2022-03-29T15:37:42.872+0530\",\"requiresSubmit\":\"No\",\"visibility\":\"Parent\"," + + "\"showTimer\":\"No\",\"index\":1,\"setType\":\"materialised\",\"languageCode\":[\"en\"],\"version\":1," + + "\"versionKey\":\"1649070059566\",\"showFeedback\":\"No\",\"license\":\"CC BY 4.0\",\"depth\":1,\"name\":\"S1\"," + + "\"navigationMode\":\"non-linear\",\"allowBranching\":\"Yes\",\"shuffle\":true,\"status\":\"Draft\"}]}") + } + }) + node + } + + def getRootExternalPropsResponseBL(): Response = { + val response = getSuccessfulResponse() + response.put("hierarchy", "{\"code\":\"CopyQuestionSetv21\",\"allowSkip\":\"Yes\",\"containsUserData\":\"No\"," + + "\"channel\":\"{{channel_id}}\",\"language\":[\"English\"],\"showHints\":\"No\",\"mimeType\":\"application/vnd.sunbird" + "" + + "" + ".questionset\",\"createdOn\":\"2022-04-06T10:13:15.975+0530\",\"objectType\":\"QuestionSet\"," + + "\"primaryCategory\":\"Observation\",\"contentDisposition\":\"inline\",\"contentEncoding\":\"gzip\"," + + "\"lastUpdatedOn\":\"2022-04-06T10:16:05.263+0530\",\"generateDIALCodes\":\"No\",\"showSolutions\":\"No\"," + + "\"allowAnonymousAccess\":\"Yes\",\"identifier\":\"do_1234\"," + "\"lastStatusChangedOn\":\"2022-04-06T10:13:15.975+0530\"," + + "\"requiresSubmit\":\"No\",\"visibility\":\"Default\"," + "\"IL_SYS_NODE_TYPE\":\"DATA_NODE\",\"showTimer\":\"No\"," + + "\"childNodes\":[\"do_5555\"," + "\"do_2222\",\"do_7777\"],\"setType\":\"materialised\",\"version\":1,\"showFeedback\":\"No\"," + + "\"versionKey\":\"1649220365263\",\"license\":\"CC BY 4.0\",\"depth\":0,\"compatibilityLevel\":5," + + "\"IL_FUNC_OBJECT_TYPE\":\"QuestionSet\",\"allowBranching\":\"No\",\"navigationMode\":\"non-linear\"," + + "\"name\":\"CopyQuestionSetv21\",\"shuffle\":true,\"IL_UNIQUE_ID\":\"do_1234\",\"status\":\"Live\"," + + "\"children\":[{\"parent\":\"do_1234\",\"code\":\"S1\",\"allowSkip\":\"Yes\",\"containsUserData\":\"No\"," + + "\"channel\":\"{{channel_id}}\",\"branchingLogic\":{\"do_5555\":{\"target\":[\"do_7777\"]," + "\"preCondition\":{}," + + "\"source\":[]}," + "\"do_7777\":{\"target\":[]," + "\"preCondition\":{\"and\":[{\"eq\":[{\"var\":\"do_5555.response1.value\"," + + "\"type\":\"responseDeclaration\"}," + "\"0\"]}]},\"source\":[\"do_5555\"]}},\"language\":[\"English\"]," + + "\"mimeType\":\"application/vnd" + ".sunbird" + ".questionset\",\"showHints\":\"No\"," + + "\"createdOn\":\"2022-04-06T10:13:32.949+0530\"," + "\"objectType\":\"QuestionSet\"," + "\"primaryCategory\":\"Observation\"," + + "\"children\":[{\"parent\":\"do_2222\",\"code\":\"Q1\"," + "\"channel\":\"{{channel_id}}\"," + "\"language\":[\"English\"]," + + "\"mimeType\":\"application/vnd.sunbird.question\"," + "\"createdOn\":\"2022-04-06T10:13:32.859+0530\"," + + "\"objectType\":\"Question\",\"primaryCategory\":\"Slider\"," + "\"contentDisposition\":\"inline\"," + + "\"lastUpdatedOn\":\"2022-04-06T10:13:32.911+0530\",\"contentEncoding\":\"gzip\"," + "\"showSolutions\":\"No\"," + + "\"allowAnonymousAccess\":\"Yes\",\"identifier\":\"do_5555\"," + "\"lastStatusChangedOn\":\"2022-04-06T10:13:32.859+0530\"," + + "\"visibility\":\"Default\",\"showTimer\":\"No\",\"index\":1," + "\"languageCode\":[\"en\"],\"version\":1," + + "\"versionKey\":\"1649220212911\",\"showFeedback\":\"No\",\"license\":\"CC BY 4.0\"," + "\"depth\":2,\"compatibilityLevel\":4," + + "\"name\":\"Q1\",\"status\":\"Live\"},{\"parent\":\"do_2222\"," + "\"code\":\"Q2\"," + "\"channel\":\"{{channel_id}}\"," + + "\"language\":[\"English\"],\"mimeType\":\"application/vnd.sunbird.question\"," + + "\"createdOn\":\"2022-04-06T10:13:32.896+0530\"," + "\"objectType\":\"Question\",\"primaryCategory\":\"Slider\"," + + "\"contentDisposition\":\"inline\"," + "\"lastUpdatedOn\":\"2022-04-06T10:13:32.954+0530\",\"contentEncoding\":\"gzip\"," + + "\"showSolutions\":\"No\"," + "\"allowAnonymousAccess\":\"Yes\",\"identifier\":\"do_7777\"," + + "\"lastStatusChangedOn\":\"2022-04-06T10:13:32.896+0530\"," + "\"visibility\":\"Default\",\"showTimer\":\"No\",\"index\":2," + + "\"languageCode\":[\"en\"],\"version\":1," + "\"versionKey\":\"1649220212954\",\"showFeedback\":\"No\",\"license\":\"CC BY " + + "4.0\"," + "\"depth\":2,\"compatibilityLevel\":4," + "\"name\":\"Q2\",\"status\":\"Live\"}],\"contentDisposition\":\"inline\"," + + "\"lastUpdatedOn\":\"2022-04-06T10:16:05.061+0530\"," + "\"contentEncoding\":\"gzip\",\"generateDIALCodes\":\"No\"," + + "\"showSolutions\":\"No\",\"allowAnonymousAccess\":\"Yes\"," + "\"identifier\":\"do_2222\"," + + "\"lastStatusChangedOn\":\"2022-04-06T10:13:32.949+0530\",\"requiresSubmit\":\"No\"," + "\"visibility\":\"Parent\"," + + "\"showTimer\":\"No\",\"index\":1,\"setType\":\"materialised\",\"languageCode\":[\"en\"],\"version\":1," + "" + + "\"versionKey\":\"1649220212949\",\"showFeedback\":\"No\",\"license\":\"CC BY 4.0\",\"depth\":1,\"compatibilityLevel\":5," + + "\"name\":\"S1\",\"navigationMode\":\"non-linear\",\"allowBranching\":\"Yes\",\"shuffle\":true,\"status\":\"Live\"}]}") + response + } + + def getNewRootExternalPropsResponseBL(): Response = { + val response = getSuccessfulResponse() + response.put("hierarchy", "{\"identifier\":\"do_9876\",\"children\":[{\"parent\":\"do_9876\"," + + "\"code\":\"1911de43-48aa-4533-b93e-2e342e9f6ec7\",\"allowSkip\":\"Yes\",\"containsUserData\":\"No\"," + + "\"channel\":\"{{channel_id}}\",\"language\":[\"English\"],\"mimeType\":\"application/vnd.sunbird.questionset\"," + + "\"showHints\":\"No\",\"createdOn\":\"2022-04-06T14:10:31.187+0530\",\"objectType\":\"QuestionSet\"," + + "\"primaryCategory\":\"Observation\",\"children\":[{\"parent\":\"do_3333\",\"code\":\"Q1\"," + "\"channel\":\"{{channel_id}}\"," + + "\"language\":[\"English\"],\"mimeType\":\"application/vnd.sunbird.question\"," + + "\"createdOn\":\"2022-04-06T10:13:32.859+0530\"," + "\"objectType\":\"Question\",\"primaryCategory\":\"Slider\"," + + "\"contentDisposition\":\"inline\"," + "\"lastUpdatedOn\":\"2022-04-06T10:13:32.911+0530\",\"contentEncoding\":\"gzip\"," + + "\"showSolutions\":\"No\"," + "\"allowAnonymousAccess\":\"Yes\",\"identifier\":\"do_5555\"," + + "\"lastStatusChangedOn\":\"2022-04-06T10:13:32.859+0530\"," + "\"visibility\":\"Default\",\"showTimer\":\"No\",\"index\":1," + + "\"languageCode\":[\"en\"],\"version\":1," + "\"versionKey\":\"1649220212911\",\"showFeedback\":\"No\",\"license\":\"CC BY " + + "4.0\"," + "\"depth\":2,\"compatibilityLevel\":4," + "\"name\":\"Q1\",\"status\":\"Live\"},{\"parent\":\"do_3333\"," + + "\"code\":\"Q2\",\"channel\":\"{{channel_id}}\"," + "\"language\":[\"English\"],\"mimeType\":\"application/vnd.sunbird" + + ".question\"," + "\"createdOn\":\"2022-04-06T10:13:32.896+0530\"," + "\"objectType\":\"Question\"," + + "\"primaryCategory\":\"Slider\"," + "\"contentDisposition\":\"inline\"," + "\"lastUpdatedOn\":\"2022-04-06T10:13:32.954+0530\"," + + "\"contentEncoding\":\"gzip\"," + "\"showSolutions\":\"No\"," + "\"allowAnonymousAccess\":\"Yes\",\"identifier\":\"do_7777\"," + + "\"lastStatusChangedOn\":\"2022-04-06T10:13:32.896+0530\"," + "\"visibility\":\"Default\",\"showTimer\":\"No\",\"index\":2," + + "\"languageCode\":[\"en\"],\"version\":1," + "\"versionKey\":\"1649220212954\",\"showFeedback\":\"No\",\"license\":\"CC BY " + + "4.0\"," + "\"depth\":2,\"compatibilityLevel\":4," + "\"name\":\"Q2\",\"status\":\"Live\"}],\"contentDisposition\":\"inline\"," + + "\"lastUpdatedOn\":\"2022-04-06T14:10:31.185+0530\"," + "\"contentEncoding\":\"gzip\",\"generateDIALCodes\":\"No\"," + + "\"showSolutions\":\"No\",\"allowAnonymousAccess\":\"Yes\"," + "\"identifier\":\"do_3333\"," + + "\"lastStatusChangedOn\":\"2022-04-06T14:10:31.187+0530\",\"requiresSubmit\":\"No\"," + "\"visibility\":\"Parent\"," + + "\"showTimer\":\"No\",\"index\":1,\"setType\":\"materialised\",\"languageCode\":[\"en\"],\"version\":1," + "" + "" + + "\"versionKey\":\"1649234431187\",\"showFeedback\":\"No\",\"license\":\"CC BY 4.0\",\"depth\":1,\"name\":\"S1\"," + + "\"navigationMode\":\"non-linear\",\"allowBranching\":\"Yes\",\"shuffle\":true,\"status\":\"Draft\"}]}") + response + } + + def getResourceNotFoundResponse(): Response = { + val response = new Response + response.setVer("3.0") + val responseParams = new ResponseParams + responseParams.setStatus("failed") + response.setParams(responseParams) + response.setResponseCode(ResponseCode.RESOURCE_NOT_FOUND) + response + } } From 3b5a4558dba6eb51ae9bb9a8b2bc6eed37abf138 Mon Sep 17 00:00:00 2001 From: joffinjoy <35325730+joffinjoy@users.noreply.github.com> Date: Wed, 13 Apr 2022 09:18:11 +0530 Subject: [PATCH 4/9] Issue #SB-29145 refactor: Optimized UpdateHierarchy Method & Modified BranchingLogic Copy Test-Case --- .../org/sunbird/managers/CopyManager.scala | 47 ++- .../sunbird/actors/QuestionSetActorTest.scala | 44 +-- .../scala/org/sunbird/actors/copyTrait.scala | 303 ++++++------------ 3 files changed, 130 insertions(+), 264 deletions(-) diff --git a/assessment-api/assessment-actors/src/main/scala/org/sunbird/managers/CopyManager.scala b/assessment-api/assessment-actors/src/main/scala/org/sunbird/managers/CopyManager.scala index 609acdc71..06656c105 100644 --- a/assessment-api/assessment-actors/src/main/scala/org/sunbird/managers/CopyManager.scala +++ b/assessment-api/assessment-actors/src/main/scala/org/sunbird/managers/CopyManager.scala @@ -178,7 +178,7 @@ object CopyManager { oldToNewIdMap } - def hierarchyRequestModifier(request: Request, nodeBLRecord: util.HashMap[String, AnyRef], identifiers: util.Map[String, String]): Unit = { + def hierarchyRequestModifier(request: Request, nodeBLRecord: util.HashMap[String, AnyRef], identifiers: util.Map[String, String]) = { val nodesModified: java.util.HashMap[String, AnyRef] = request.getRequest.get(HierarchyConstants.NODES_MODIFIED).asInstanceOf[java.util.HashMap[String, AnyRef]] val hierarchy: java.util.HashMap[String, AnyRef] = request.getRequest.get(HierarchyConstants.HIERARCHY).asInstanceOf[java.util.HashMap[String, AnyRef]] val oldToNewIdMap = generateOldToNewIdMap(nodeBLRecord, identifiers) @@ -211,37 +211,36 @@ object CopyManager { hierarchy.put(identifiers.get(id), nodeHierarchy) } }) + request } - def updateHierarchy(request: Request, node: Node, originNode: Node, originHierarchy: util.Map[String, AnyRef], copyType: String) - (implicit ec: ExecutionContext, oec: OntologyEngineContext): Future[Node] = { + def updateHierarchy(request: Request, node: Node, originNode: Node, originHierarchy: util.Map[String, AnyRef], copyType: String)(implicit ec: ExecutionContext, oec: OntologyEngineContext): Future[Node] = { prepareHierarchyRequest(originHierarchy, originNode, node, copyType, request).map(req => { val hierarchyRequest = new Request(request) hierarchyRequest.putAll(req) - val nodesModified: java.util.HashMap[String, AnyRef] = hierarchyRequest.getRequest.get(HierarchyConstants.NODES_MODIFIED) - .asInstanceOf[java.util.HashMap[String, AnyRef]] + val nodesModified: java.util.HashMap[String, AnyRef] = hierarchyRequest.getRequest.get(HierarchyConstants.NODES_MODIFIED).asInstanceOf[java.util.HashMap[String, AnyRef]] val nodeBLRecord = generateNodeBLRecord(nodesModified) - val newUpdateRequest = JsonUtils.deserialize(ScalaJsonUtils.serialize(hierarchyRequest), classOf[Request]) - UpdateHierarchyManager.updateHierarchy(hierarchyRequest).map(response => { - if (!ResponseHandler.checkError(response)) { - val identifiers = response.getResult.get(AssessmentConstants.IDENTIFIERS).asInstanceOf[util.Map[String, String]] - hierarchyRequestModifier(newUpdateRequest, nodeBLRecord, identifiers) - UpdateHierarchyManager.updateHierarchy(newUpdateRequest).map(response_ => { - if (!ResponseHandler.checkError(response_)) { - node - } else { - TelemetryManager.info(s"Update Hierarchy Failed For Copy Question Set Having Identifier: ${node.getIdentifier} | Response " + - s"is " + s": " + response) + val originalRequest = JsonUtils.deserialize(ScalaJsonUtils.serialize(hierarchyRequest), classOf[Request]) + val BLExists = nodeBLRecord.exists(BLRecord => BLRecord._2.asInstanceOf[util.HashMap[String, AnyRef]].get(AssessmentConstants.CONTAINS_BL) == true) + val (firstUpdateRequest, secondUpdateRequest) = if (BLExists) (hierarchyRequest, JsonUtils.deserialize(ScalaJsonUtils.serialize(hierarchyRequest), classOf[Request])) else (originalRequest, new Request()) + UpdateHierarchyManager.updateHierarchy(firstUpdateRequest).map(response => { + if (!ResponseHandler.checkError(response)) response + else { + TelemetryManager.info(s"Update Hierarchy Failed For Copy Question Set Having Identifier: ${node.getIdentifier} | Response is: " + response) + throw new ServerException("ERR_QUESTIONSET_COPY", "Something Went Wrong, Please Try Again") + } + }).map(response => { + if (BLExists) { + hierarchyRequestModifier(secondUpdateRequest, nodeBLRecord, response.getResult.get(AssessmentConstants.IDENTIFIERS).asInstanceOf[util.Map[String, String]]) + UpdateHierarchyManager.updateHierarchy(secondUpdateRequest).map(response_ => { + if (!ResponseHandler.checkError(response_)) node + else { + TelemetryManager.info(s"Update Hierarchy Failed For Copy Question Set Having Identifier: ${node.getIdentifier} | Response is: " + response) throw new ServerException("ERR_QUESTIONSET_COPY", "Something Went Wrong, Please Try Again") } }) - node - } else { - TelemetryManager.info(s"Update Hierarchy Failed For Copy Question Set Having Identifier: ${node.getIdentifier} | Response is " - + s": " + response) - throw new ServerException("ERR_QUESTIONSET_COPY", "Something Went Wrong, Please Try Again") - } - }) + } else Future(node) + }).flatMap(f => f) }).flatMap(f => f) } @@ -300,7 +299,7 @@ object CopyManager { put(AssessmentConstants.METADATA, cleanUpCopiedData(new util.HashMap[String, AnyRef]() { { putAll(child) - put("copyOf", child.getOrDefault(AssessmentConstants.IDENTIFIER,"")) + put(AssessmentConstants.COPY_OF, child.getOrDefault(AssessmentConstants.IDENTIFIER,"")) put(AssessmentConstants.CHILDREN, new util.ArrayList()) internalHierarchyProps.map(key => remove(key)) } diff --git a/assessment-api/assessment-actors/src/test/scala/org/sunbird/actors/QuestionSetActorTest.scala b/assessment-api/assessment-actors/src/test/scala/org/sunbird/actors/QuestionSetActorTest.scala index 9d6019aea..721171d96 100644 --- a/assessment-api/assessment-actors/src/test/scala/org/sunbird/actors/QuestionSetActorTest.scala +++ b/assessment-api/assessment-actors/src/test/scala/org/sunbird/actors/QuestionSetActorTest.scala @@ -11,7 +11,8 @@ import org.sunbird.graph.nodes.DataNode.getRelationMap import org.sunbird.graph.utils.ScalaJsonUtils import org.sunbird.graph.{GraphService, OntologyEngineContext} import org.sunbird.kafka.client.KafkaClient -import org.sunbird.utils.JavaJsonUtils +import org.sunbird.managers.CopyManager +import org.sunbird.utils.{AssessmentConstants, JavaJsonUtils} import java.util import scala.collection.JavaConversions._ @@ -594,37 +595,16 @@ class QuestionSetActorTest extends BaseSpec with MockFactory with copyTrait { assert("failed".equals(response.getParams.getStatus)) } - it should "return success response for 'copyQuestionSet' (Branching Logic Copy)" in { - implicit val oec: OntologyEngineContext = mock[OntologyEngineContext] - val graphDB = mock[GraphService] - (oec.graphService _).expects().returns(graphDB).anyNumberOfTimes() - val nodes: util.List[Node] = getCategoryNode() - (graphDB.getNodeByUniqueIds(_: String, _: SearchCriteria)).expects(*, *).returns(Future(nodes)).anyNumberOfTimes() - (graphDB.getNodeByUniqueId(_: String, _: String, _: Boolean, _: Request)).expects(*, "do_1234", *, *).returns(Future(getRootNodeWithBL("do_1234", "do_2222", true, true))).anyNumberOfTimes() - (graphDB.getNodeByUniqueId(_: String, _: String, _: Boolean, _: Request)).expects(*, "do_9876", *, *).returns(Future(getRootNodeWithBL("do_9876", "do_3333", false, false))).anyNumberOfTimes() - (graphDB.getNodeByUniqueId(_: String, _: String, _: Boolean, _: Request)).expects(*, "do_9876.img", *, *).returns(Future(getRootNodeWithBL("do_9876", "do_3333", false, false))).anyNumberOfTimes() - (graphDB.getNodeByUniqueId(_: String, _: String, _: Boolean, _: Request)).expects(*, "do_5555", *, *).returns(Future(getQuestionNodeBL("do_5555"))).anyNumberOfTimes() - (graphDB.getNodeByUniqueId(_: String, _: String, _: Boolean, _: Request)).expects(*, "do_7777", *, *).returns(Future(getQuestionNodeBL("do_7777"))).anyNumberOfTimes() - (graphDB.readExternalProps(_: Request, _: List[String])).expects(*, List("objectMetadata")).returns(Future(getSuccessfulResponse)).anyNumberOfTimes() - (graphDB.readExternalProps(_: Request, _: List[String])).expects(*, List("instructions", "outcomeDeclaration")).returns(Future(getSuccessfulResponse)).anyNumberOfTimes() - (graphDB.readExternalProps(_: Request, _: List[String])).expects(*, List("solutions", "body", "editorState", "interactions", "hints", "responseDeclaration", "media", "answer", "instructions")).returns(Future(getResourceNotFoundResponse)).anyNumberOfTimes() - (graphDB.addNode(_: String, _: Node)).expects(*, *).returns(Future(getRootNodeWithBL("do_9876", "do_3333", false, false))).anyNumberOfTimes - (graphDB.saveExternalProps(_: Request)).expects(*).returns(Future(getSuccessfulResponse)).anyNumberOfTimes - (graphDB.updateExternalProps(_: Request)).expects(*).returns(Future(getSuccessfulResponse)).anyNumberOfTimes - (graphDB.getNodeByUniqueId(_: String, _: String, _: Boolean, _: Request)).expects(*, *, *, *).returns(Future(getRootNodeWithBL("do_9876", "do_3333", true, true))).anyNumberOfTimes() - inSequence { - (graphDB.upsertNode(_: String, _: Node, _: Request)).expects(*, *, *).returns(Future(getUpsertNodeBLWithoutBL)) - (graphDB.upsertNode(_: String, _: Node, _: Request)).expects(*, *, *).returns(Future(getUpsertNodeBLWithBL)).anyNumberOfTimes - } - inSequence { - (graphDB.readExternalProps(_: Request, _: List[String])).expects(*, List("hierarchy")).returns(Future(getRootExternalPropsResponseBL)) - (graphDB.readExternalProps(_: Request, _: List[String])).expects(*, List("hierarchy")).returns(Future(getNewRootExternalPropsResponseBL)).anyNumberOfTimes - } - val request = getQuestionSetCopyRequest() - request.putAll(mapAsJavaMap(Map("identifier" -> "do_1234", "mode" -> "", "copyType" -> "deep"))) - request.setOperation("copyQuestionSet") - val response = callActor(request, Props(new QuestionSetActor())) - assert("successful".equals(response.getParams.getStatus)) + it should "return expected result for 'generateNodeBLRecord'" in { + val result = CopyManager.generateNodeBLRecord(generateNodesModified("afa2bef1-b5db-45d9-b0d7-aeea757906c3", true)) + assert(result == generateNodeBLRecord) + } + + it should "return expected result for 'hierarchyRequestModifier'" in { + val result = CopyManager.hierarchyRequestModifier(generateUpdateRequest(false, "afa2bef1-b5db-45d9-b0d7-aeea757906c3"), generateNodeBLRecord(), generateIdentifiers()) + val expectedResult = generateUpdateRequest(true, "do_11351201604857856013") + assert(result.getRequest.get(AssessmentConstants.NODES_MODIFIED) == expectedResult.getRequest.get(AssessmentConstants.NODES_MODIFIED)) + assert(result.getRequest.get(AssessmentConstants.HIERARCHY) == expectedResult.getRequest.get(AssessmentConstants.HIERARCHY)) } private def getQuestionSetRequest(): Request = { diff --git a/assessment-api/assessment-actors/src/test/scala/org/sunbird/actors/copyTrait.scala b/assessment-api/assessment-actors/src/test/scala/org/sunbird/actors/copyTrait.scala index 3bf221a77..8ee744cd7 100644 --- a/assessment-api/assessment-actors/src/test/scala/org/sunbird/actors/copyTrait.scala +++ b/assessment-api/assessment-actors/src/test/scala/org/sunbird/actors/copyTrait.scala @@ -4,9 +4,12 @@ import org.mortbay.util.StringUtil import org.sunbird.common.dto.{Request, Response, ResponseParams} import org.sunbird.graph.dac.model.Node import org.sunbird.utils.AssessmentConstants -import org.sunbird.common.exception.{ResponseCode} +import org.sunbird.common.exception.ResponseCode import java.util +import scala.collection.JavaConversions.mapAsJavaMap +import scala.collection.JavaConverters.asJavaIterableConverter +import scala.collection.mutable trait copyTrait { @@ -262,232 +265,116 @@ trait copyTrait { node } - def getRootNodeWithBL(rootId: String, sectionId: String, addBranchingLogic: Boolean, withChildren: Boolean): Node = { - val node = getNode("QuestionSet", rootId, "Observation", AssessmentConstants.VISIBILITY_DEFAULT, "ExistingRootNode", 1234, "Live") - if (withChildren) { - val section = getNode("QuestionSet", sectionId, "Observation", AssessmentConstants.VISIBILITY_DEFAULT, "Section_1", 1234, "Live") - val children = new util.ArrayList[util.Map[String, AnyRef]]() - children.add(getNode("Question", "do_5555", "Slider", AssessmentConstants.VISIBILITY_DEFAULT, "Question1", 1234, "Live") - .getMetadata) - children.add(getNode("Question", "do_7777", "Slider", AssessmentConstants.VISIBILITY_DEFAULT, "Question2", 1234, "Live") - .getMetadata) - if (addBranchingLogic) { - section.getMetadata.put("branchingLogic", new util.HashMap[String, AnyRef]() { - { - put("do_5555", new util.HashMap[String, AnyRef]() { - put("target", new util.ArrayList[String]() { - { - add("do_7777") - } - }) - put("preCondition", new util.HashMap[String, AnyRef]()) - put("source", new util.ArrayList[String]()) - }) - put("do_7777", new util.HashMap[String, AnyRef]() { - put("target", new util.ArrayList[String]()) - put("preCondition", new util.HashMap[String, AnyRef]() { - { - put("and", new util.ArrayList[util.HashMap[String, AnyRef]]() { - add(new util.HashMap[String, AnyRef]() { - put("eq", new util.ArrayList[AnyRef]() { - { - add(new util.HashMap[String, String]() { - put("var", "do_5555" + ".response1.value") - put("type", "responseDeclaration") - }) - add("0") - } + private def generateStaticBranchingLogic(): util.HashMap[String, AnyRef] = { + new util.HashMap[String, AnyRef]() { + { + put("do_11351041198373273619", new util.HashMap[String, AnyRef]() { + put("target", new util.ArrayList[String]() { + { + add("do_113510411984044032111") + } + }) + put("preCondition", new util.HashMap[String, AnyRef]()) + put("source", new util.ArrayList[String]()) + }) + put("do_113510411984044032111", new util.HashMap[String, AnyRef]() { + put("target", new util.ArrayList[String]()) + put("preCondition", new util.HashMap[String, AnyRef]() { + { + put("and", new util.ArrayList[util.HashMap[String, AnyRef]]() { + add(new util.HashMap[String, AnyRef]() { + put("eq", new util.ArrayList[AnyRef]() { + { + add(new util.HashMap[String, String]() { + put("var", "do_11351041198373273619" + ".response1.value") + put("type", "responseDeclaration") }) - }) + add("0") + } }) - } - }) - put("source", new util.ArrayList[String]() { - { - add("do_5555") - } + }) }) - }) - } + } + }) + put("source", new util.ArrayList[String]() { + { + add("do_11351041198373273619") + } + }) }) } - node.getMetadata.put("childNodes", new util.ArrayList[String]() { - { - add(sectionId) - add("do_5555") - add("do_7777") - } - }) - section.getMetadata.put("children", children) - node.getMetadata.put("children", new util.ArrayList[util.Map[String, AnyRef]]() { - { - add(section.getMetadata) - } - }) } - node - } - - def getQuestionNodeBL(identifier: String): Node = { - val node = getNode("Question", identifier, "Slider", AssessmentConstants.VISIBILITY_DEFAULT, identifier, 1234, "Live") - node } - def getUpsertNodeBLWithoutBL(): Node = { - val node = getRootNodeWithBL("do_9876", "do_3333", false, false) - node.setExternalData(new util.HashMap[String, AnyRef]() { + def generateNodesModified(identifier: String, withBranchingLogic: Boolean): util.HashMap[String, AnyRef] = { + val nodesModified = new util.HashMap[String, AnyRef]() + nodesModified.put(identifier, new util.HashMap[String, AnyRef]() { { - put("hierarchy", "{\"identifier\":\"do_9876\",\"children\":[{\"parent\":\"do_9876\"," + - "\"code\":\"9f0332ad-c3e3-4803-b673-50174aff24e3\",\"allowSkip\":\"Yes\",\"containsUserData\":\"No\"," + - "\"channel\":\"{{channel_id}}\",\"language\":[\"English\"],\"mimeType\":\"application/vnd.sunbird.questionset\"," + - "\"showHints\":\"No\",\"createdOn\":\"2022-04-06T12:51:53.592+0530\",\"objectType\":\"QuestionSet\"," + - "\"primaryCategory\":\"Observation\",\"children\":[{\"parent\":\"do_3333\",\"code\":\"Q1\"," + - "\"channel\":\"{{channel_id}}\"," + "\"language\":[\"English\"],\"mimeType\":\"application/vnd.sunbird.question\"," + - "\"createdOn\":\"2022-04-06T10:13:32.859+0530\"," + "\"objectType\":\"Question\",\"primaryCategory\":\"Slider\"," + - "\"contentDisposition\":\"inline\"," + "\"lastUpdatedOn\":\"2022-04-06T10:13:32.911+0530\",\"contentEncoding\":\"gzip\"," + - "" + "\"showSolutions\":\"No\"," + "\"allowAnonymousAccess\":\"Yes\",\"identifier\":\"do_5555\"," + - "\"lastStatusChangedOn\":\"2022-04-06T10:13:32.859+0530\"," + "\"visibility\":\"Default\",\"showTimer\":\"No\"," + - "\"index\":1," + "\"languageCode\":[\"en\"],\"version\":1," + "\"versionKey\":\"1649220212911\",\"showFeedback\":\"No\"," + - "\"license\":\"CC BY " + "4.0\",\"depth\":2,\"compatibilityLevel\":4," + "\"name\":\"Q1\",\"status\":\"Live\"}," + - "{\"parent\":\"do_3333\",\"code\":\"Q2\",\"channel\":\"{{channel_id}}\"," + "\"language\":[\"English\"]," + - "\"mimeType\":\"application/vnd.sunbird.question\",\"createdOn\":\"2022-04-06T10:13:32.896+0530\"," + - "\"objectType\":\"Question\",\"primaryCategory\":\"Slider\",\"contentDisposition\":\"inline\"," + - "\"lastUpdatedOn\":\"2022-04-06T10:13:32.954+0530\",\"contentEncoding\":\"gzip\",\"showSolutions\":\"No\"," + - "\"allowAnonymousAccess\":\"Yes\",\"identifier\":\"do_7777\"," + - "\"lastStatusChangedOn\":\"2022-04-06T10:13:32.896+0530\"," + "\"visibility\":\"Default\",\"showTimer\":\"No\"," + - "\"index\":2," + "\"languageCode\":[\"en\"],\"version\":1," + "\"versionKey\":\"1649220212954\",\"showFeedback\":\"No\"," + - "\"license\":\"CC BY " + "4.0\",\"depth\":2,\"compatibilityLevel\":4," + "\"name\":\"Q2\",\"status\":\"Live\"}]," + - "\"contentDisposition\":\"inline\"," + "\"lastUpdatedOn\":\"2022-04-06T12:51:53.591+0530\"," + - "\"contentEncoding\":\"gzip\",\"generateDIALCodes\":\"No\"," + "\"showSolutions\":\"No\"," + - "\"allowAnonymousAccess\":\"Yes\"," + "\"identifier\":\"do_3333\"," + - "\"lastStatusChangedOn\":\"2022-04-06T12:51:53.592+0530\",\"requiresSubmit\":\"No\"," + "\"visibility\":\"Parent\"," + - "\"showTimer\":\"No\",\"index\":1,\"setType\":\"materialised\",\"languageCode\":[\"en\"],\"version\":1," + "" + "" + - "\"versionKey\":\"1649229713592\",\"showFeedback\":\"No\",\"license\":\"CC BY 4.0\",\"depth\":1,\"name\":\"S1\"," + - "\"navigationMode\":\"non-linear\",\"allowBranching\":\"Yes\",\"shuffle\":true,\"status\":\"Draft\"}]}") + put("setDefaultValue", false.asInstanceOf[AnyRef]) + put("metadata", new util.HashMap[String, AnyRef]() { + { + putAll((getNode("QuestionSet", "do_5678", "Observation", AssessmentConstants.VISIBILITY_PARENT, "Observation", 0, + "Draft").getMetadata)) + put("copyOf", "do_113510411984478208113") + if (withBranchingLogic) put("branchingLogic", generateStaticBranchingLogic) + } + }) + put("root", false.asInstanceOf[AnyRef]) + put("isNew", (!withBranchingLogic).asInstanceOf[AnyRef]) + put("objectType", "QuestionSet") } }) - node + nodesModified } - def getUpsertNodeBLWithBL(): Node = { - val node = getRootNodeWithBL("do_9876", "do_3333", false, false) - node.setExternalData(new util.HashMap[String, AnyRef]() { + def generateNodeBLRecord(): util.HashMap[String, AnyRef] = { + val nodeBLRecord = new util.HashMap[String, AnyRef]() + nodeBLRecord.put("afa2bef1-b5db-45d9-b0d7-aeea757906c3", new util.HashMap[String, AnyRef]() { { - put("hierarchy", "{\"identifier\":\"do_9876\",\"children\":[{\"parent\":\"do_9876\"," + "\"code\":\"S1\"," + - "\"allowSkip\":\"Yes\",\"containsUserData\":\"No\",\"channel\":\"{{channel_id}}\"," + - "\"branchingLogic\":{\"do_7777\":{\"preCondition\":{\"and\":[{\"eq\":[{\"type\":\"responseDeclaration" + "\"," + - "\"var\":\"do_5555.response1.value\"},\"0\"]}]},\"target\":[]," + "\"source\":[\"do_5555\"]}," + - "\"do_5555\":{\"preCondition\":{}," + "\"target\":[\"do_7777\"],\"source\":[]}},\"description\":\"Section 1\"," + - "\"language\":[\"English\"]," + "\"mimeType\":\"application/vnd" + ".sunbird.questionset\",\"showHints\":\"No\"," + - "\"createdOn\":\"2022-04-04T16:30:59.566+0530\"," + "\"objectType\":\"QuestionSet\"," + - "\"primaryCategory\":\"Observation\"," + "\"children\":[{\"parent\":\"do_3333\",\"code\":\"Q1\"," + - "\"channel\":\"{{channel_id}}\"," + "\"description\":\"Q1\",\"language\":[\"English\"],\"mimeType\":\"application/vnd" + - ".sunbird" + ".question\"," + "\"createdOn\":\"2022-04-04T16:30:59.539+0530\",\"objectType\":\"Question\"," + - "\"primaryCategory\":\"Slider\"," + "\"contentDisposition\":\"inline\"," + - "\"lastUpdatedOn\":\"2022-04-04T16:32:46.200+0530\",\"contentEncoding\":\"gzip\"," + "\"showSolutions\":\"No\"," + - "\"allowAnonymousAccess\":\"Yes\",\"identifier\":\"do_5555\"," + - "\"lastStatusChangedOn\":\"2022-03-29T15:37:42.837+0530\",\"visibility\":\"Parent\",\"showTimer\":\"No\",\"index\":1," + - "\"languageCode\":[\"en\"],\"version\":1,\"versionKey\":\"1649070166325\",\"showFeedback\":\"No\",\"license\":\"CC BY " - + "4.0\"," + "\"depth\":2,\"compatibilityLevel\":4,\"name\":\"Q1\",\"status\":\"Draft\"}," + "{\"parent\":\"do_3333\"," + - "\"code\":\"Q2\"," + "\"channel\":\"{{channel_id}}\",\"description\":\"Q2\"," + "\"language\":[\"English\"]," + - "\"mimeType\":\"application/vnd.sunbird" + "" + ".question\"," + "\"createdOn\":\"2022-03-29T15:37:42.852+0530\"," + - "\"objectType\":\"Question\",\"primaryCategory\":\"Slider\"," + "\"contentDisposition\":\"inline\"," + - "\"lastUpdatedOn\":\"2022-03-29T15:37:42.896+0530\",\"contentEncoding\":\"gzip\"," + "\"showSolutions\":\"No\"," + - "\"allowAnonymousAccess\":\"Yes\",\"identifier\":\"do_7777\"," + - "\"lastStatusChangedOn\":\"2022-03-29T15:37:42.852+0530\",\"visibility\":\"Default\",\"showTimer\":\"No\",\"index\":2," - + "\"languageCode\":[\"en\"],\"version\":1,\"versionKey\":\"1648548462896\",\"showFeedback\":\"No\",\"license\":\"CC BY " + - "" + "4.0\"," + "\"depth\":2,\"compatibilityLevel\":4,\"name\":\"Q2\",\"status\":\"Live\"}]," + - "\"contentDisposition\":\"inline\"," + "\"lastUpdatedOn\":\"2022-04-04T16:32:46.273+0530\",\"contentEncoding\":\"gzip\"," + - "\"generateDIALCodes\":\"No\"," + "\"showSolutions\":\"No\",\"allowAnonymousAccess\":\"Yes\",\"identifier\":\"do_3333\"," + - "" + "\"lastStatusChangedOn\":\"2022-03-29T15:37:42.872+0530\",\"requiresSubmit\":\"No\",\"visibility\":\"Parent\"," + - "\"showTimer\":\"No\",\"index\":1,\"setType\":\"materialised\",\"languageCode\":[\"en\"],\"version\":1," + - "\"versionKey\":\"1649070059566\",\"showFeedback\":\"No\",\"license\":\"CC BY 4.0\",\"depth\":1,\"name\":\"S1\"," + - "\"navigationMode\":\"non-linear\",\"allowBranching\":\"Yes\",\"shuffle\":true,\"status\":\"Draft\"}]}") + put("containsBL", true.asInstanceOf[AnyRef]) + put("branchingLogic", generateStaticBranchingLogic()) + put("copyOf", "do_113510411984478208113") } }) - node - } - - def getRootExternalPropsResponseBL(): Response = { - val response = getSuccessfulResponse() - response.put("hierarchy", "{\"code\":\"CopyQuestionSetv21\",\"allowSkip\":\"Yes\",\"containsUserData\":\"No\"," + - "\"channel\":\"{{channel_id}}\",\"language\":[\"English\"],\"showHints\":\"No\",\"mimeType\":\"application/vnd.sunbird" + "" + - "" + ".questionset\",\"createdOn\":\"2022-04-06T10:13:15.975+0530\",\"objectType\":\"QuestionSet\"," + - "\"primaryCategory\":\"Observation\",\"contentDisposition\":\"inline\",\"contentEncoding\":\"gzip\"," + - "\"lastUpdatedOn\":\"2022-04-06T10:16:05.263+0530\",\"generateDIALCodes\":\"No\",\"showSolutions\":\"No\"," + - "\"allowAnonymousAccess\":\"Yes\",\"identifier\":\"do_1234\"," + "\"lastStatusChangedOn\":\"2022-04-06T10:13:15.975+0530\"," + - "\"requiresSubmit\":\"No\",\"visibility\":\"Default\"," + "\"IL_SYS_NODE_TYPE\":\"DATA_NODE\",\"showTimer\":\"No\"," + - "\"childNodes\":[\"do_5555\"," + "\"do_2222\",\"do_7777\"],\"setType\":\"materialised\",\"version\":1,\"showFeedback\":\"No\"," - + "\"versionKey\":\"1649220365263\",\"license\":\"CC BY 4.0\",\"depth\":0,\"compatibilityLevel\":5," + - "\"IL_FUNC_OBJECT_TYPE\":\"QuestionSet\",\"allowBranching\":\"No\",\"navigationMode\":\"non-linear\"," + - "\"name\":\"CopyQuestionSetv21\",\"shuffle\":true,\"IL_UNIQUE_ID\":\"do_1234\",\"status\":\"Live\"," + - "\"children\":[{\"parent\":\"do_1234\",\"code\":\"S1\",\"allowSkip\":\"Yes\",\"containsUserData\":\"No\"," + - "\"channel\":\"{{channel_id}}\",\"branchingLogic\":{\"do_5555\":{\"target\":[\"do_7777\"]," + "\"preCondition\":{}," + - "\"source\":[]}," + "\"do_7777\":{\"target\":[]," + "\"preCondition\":{\"and\":[{\"eq\":[{\"var\":\"do_5555.response1.value\"," - + "\"type\":\"responseDeclaration\"}," + "\"0\"]}]},\"source\":[\"do_5555\"]}},\"language\":[\"English\"]," + - "\"mimeType\":\"application/vnd" + ".sunbird" + ".questionset\",\"showHints\":\"No\"," + - "\"createdOn\":\"2022-04-06T10:13:32.949+0530\"," + "\"objectType\":\"QuestionSet\"," + "\"primaryCategory\":\"Observation\"," + - "\"children\":[{\"parent\":\"do_2222\",\"code\":\"Q1\"," + "\"channel\":\"{{channel_id}}\"," + "\"language\":[\"English\"]," + - "\"mimeType\":\"application/vnd.sunbird.question\"," + "\"createdOn\":\"2022-04-06T10:13:32.859+0530\"," + - "\"objectType\":\"Question\",\"primaryCategory\":\"Slider\"," + "\"contentDisposition\":\"inline\"," + - "\"lastUpdatedOn\":\"2022-04-06T10:13:32.911+0530\",\"contentEncoding\":\"gzip\"," + "\"showSolutions\":\"No\"," + - "\"allowAnonymousAccess\":\"Yes\",\"identifier\":\"do_5555\"," + "\"lastStatusChangedOn\":\"2022-04-06T10:13:32.859+0530\"," + - "\"visibility\":\"Default\",\"showTimer\":\"No\",\"index\":1," + "\"languageCode\":[\"en\"],\"version\":1," + - "\"versionKey\":\"1649220212911\",\"showFeedback\":\"No\",\"license\":\"CC BY 4.0\"," + "\"depth\":2,\"compatibilityLevel\":4," - + "\"name\":\"Q1\",\"status\":\"Live\"},{\"parent\":\"do_2222\"," + "\"code\":\"Q2\"," + "\"channel\":\"{{channel_id}}\"," + - "\"language\":[\"English\"],\"mimeType\":\"application/vnd.sunbird.question\"," + - "\"createdOn\":\"2022-04-06T10:13:32.896+0530\"," + "\"objectType\":\"Question\",\"primaryCategory\":\"Slider\"," + - "\"contentDisposition\":\"inline\"," + "\"lastUpdatedOn\":\"2022-04-06T10:13:32.954+0530\",\"contentEncoding\":\"gzip\"," + - "\"showSolutions\":\"No\"," + "\"allowAnonymousAccess\":\"Yes\",\"identifier\":\"do_7777\"," + - "\"lastStatusChangedOn\":\"2022-04-06T10:13:32.896+0530\"," + "\"visibility\":\"Default\",\"showTimer\":\"No\",\"index\":2," + - "\"languageCode\":[\"en\"],\"version\":1," + "\"versionKey\":\"1649220212954\",\"showFeedback\":\"No\",\"license\":\"CC BY " + - "4.0\"," + "\"depth\":2,\"compatibilityLevel\":4," + "\"name\":\"Q2\",\"status\":\"Live\"}],\"contentDisposition\":\"inline\"," - + "\"lastUpdatedOn\":\"2022-04-06T10:16:05.061+0530\"," + "\"contentEncoding\":\"gzip\",\"generateDIALCodes\":\"No\"," + - "\"showSolutions\":\"No\",\"allowAnonymousAccess\":\"Yes\"," + "\"identifier\":\"do_2222\"," + - "\"lastStatusChangedOn\":\"2022-04-06T10:13:32.949+0530\",\"requiresSubmit\":\"No\"," + "\"visibility\":\"Parent\"," + - "\"showTimer\":\"No\",\"index\":1,\"setType\":\"materialised\",\"languageCode\":[\"en\"],\"version\":1," + "" + - "\"versionKey\":\"1649220212949\",\"showFeedback\":\"No\",\"license\":\"CC BY 4.0\",\"depth\":1,\"compatibilityLevel\":5," + - "\"name\":\"S1\",\"navigationMode\":\"non-linear\",\"allowBranching\":\"Yes\",\"shuffle\":true,\"status\":\"Live\"}]}") - response + nodeBLRecord } - def getNewRootExternalPropsResponseBL(): Response = { - val response = getSuccessfulResponse() - response.put("hierarchy", "{\"identifier\":\"do_9876\",\"children\":[{\"parent\":\"do_9876\"," + - "\"code\":\"1911de43-48aa-4533-b93e-2e342e9f6ec7\",\"allowSkip\":\"Yes\",\"containsUserData\":\"No\"," + - "\"channel\":\"{{channel_id}}\",\"language\":[\"English\"],\"mimeType\":\"application/vnd.sunbird.questionset\"," + - "\"showHints\":\"No\",\"createdOn\":\"2022-04-06T14:10:31.187+0530\",\"objectType\":\"QuestionSet\"," + - "\"primaryCategory\":\"Observation\",\"children\":[{\"parent\":\"do_3333\",\"code\":\"Q1\"," + "\"channel\":\"{{channel_id}}\"," - + "\"language\":[\"English\"],\"mimeType\":\"application/vnd.sunbird.question\"," + - "\"createdOn\":\"2022-04-06T10:13:32.859+0530\"," + "\"objectType\":\"Question\",\"primaryCategory\":\"Slider\"," + - "\"contentDisposition\":\"inline\"," + "\"lastUpdatedOn\":\"2022-04-06T10:13:32.911+0530\",\"contentEncoding\":\"gzip\"," + - "\"showSolutions\":\"No\"," + "\"allowAnonymousAccess\":\"Yes\",\"identifier\":\"do_5555\"," + - "\"lastStatusChangedOn\":\"2022-04-06T10:13:32.859+0530\"," + "\"visibility\":\"Default\",\"showTimer\":\"No\",\"index\":1," + - "\"languageCode\":[\"en\"],\"version\":1," + "\"versionKey\":\"1649220212911\",\"showFeedback\":\"No\",\"license\":\"CC BY " + - "4.0\"," + "\"depth\":2,\"compatibilityLevel\":4," + "\"name\":\"Q1\",\"status\":\"Live\"},{\"parent\":\"do_3333\"," + - "\"code\":\"Q2\",\"channel\":\"{{channel_id}}\"," + "\"language\":[\"English\"],\"mimeType\":\"application/vnd.sunbird" + - ".question\"," + "\"createdOn\":\"2022-04-06T10:13:32.896+0530\"," + "\"objectType\":\"Question\"," + - "\"primaryCategory\":\"Slider\"," + "\"contentDisposition\":\"inline\"," + "\"lastUpdatedOn\":\"2022-04-06T10:13:32.954+0530\"," + - "\"contentEncoding\":\"gzip\"," + "\"showSolutions\":\"No\"," + "\"allowAnonymousAccess\":\"Yes\",\"identifier\":\"do_7777\"," + - "\"lastStatusChangedOn\":\"2022-04-06T10:13:32.896+0530\"," + "\"visibility\":\"Default\",\"showTimer\":\"No\",\"index\":2," + - "\"languageCode\":[\"en\"],\"version\":1," + "\"versionKey\":\"1649220212954\",\"showFeedback\":\"No\",\"license\":\"CC BY " + - "4.0\"," + "\"depth\":2,\"compatibilityLevel\":4," + "\"name\":\"Q2\",\"status\":\"Live\"}],\"contentDisposition\":\"inline\"," - + "\"lastUpdatedOn\":\"2022-04-06T14:10:31.185+0530\"," + "\"contentEncoding\":\"gzip\",\"generateDIALCodes\":\"No\"," + - "\"showSolutions\":\"No\",\"allowAnonymousAccess\":\"Yes\"," + "\"identifier\":\"do_3333\"," + - "\"lastStatusChangedOn\":\"2022-04-06T14:10:31.187+0530\",\"requiresSubmit\":\"No\"," + "\"visibility\":\"Parent\"," + - "\"showTimer\":\"No\",\"index\":1,\"setType\":\"materialised\",\"languageCode\":[\"en\"],\"version\":1," + "" + "" + - "\"versionKey\":\"1649234431187\",\"showFeedback\":\"No\",\"license\":\"CC BY 4.0\",\"depth\":1,\"name\":\"S1\"," + - "\"navigationMode\":\"non-linear\",\"allowBranching\":\"Yes\",\"shuffle\":true,\"status\":\"Draft\"}]}") - response + def generateIdentifiers(): util.Map[String, String] = { + val idMap: mutable.Map[String, String] = mutable.Map() + idMap += ("afa2bef1-b5db-45d9-b0d7-aeea757906c3" -> "do_11351201604857856013") + mapAsJavaMap(idMap) } - def getResourceNotFoundResponse(): Response = { - val response = new Response - response.setVer("3.0") - val responseParams = new ResponseParams - responseParams.setStatus("failed") - response.setParams(responseParams) - response.setResponseCode(ResponseCode.RESOURCE_NOT_FOUND) - response + def generateUpdateRequest(withBranchingLogic: Boolean, identifier: String): Request = { + val request = getQuestionSetRequest() + request.put(AssessmentConstants.NODES_MODIFIED, generateNodesModified(identifier, withBranchingLogic)) + request.put(AssessmentConstants.HIERARCHY, new util.HashMap[String, AnyRef]() { + { + put("do_11351201402236108811", new util.HashMap[String, AnyRef]() { + { + put(AssessmentConstants.CHILDREN, new util.ArrayList[String]() { + { + add(identifier) + } + }) + put(AssessmentConstants.PRIMARY_CATEGORY, "Observation") + put(AssessmentConstants.ROOT, true.asInstanceOf[AnyRef]) + } + }) + put(identifier, new util.HashMap[String, AnyRef]() { + { + put(AssessmentConstants.CHILDREN, new util.ArrayList[String]() { + { + add("do_11351041198373273619") + add("do_113510411984044032111") + } + }) + put(AssessmentConstants.PRIMARY_CATEGORY, "Observation") + put(AssessmentConstants.ROOT, false.asInstanceOf[AnyRef]) + } + }) + } + }) + request } } From 96030e8910492105ea049ff015714faf8506315c Mon Sep 17 00:00:00 2001 From: joffinjoy <35325730+joffinjoy@users.noreply.github.com> Date: Fri, 22 Apr 2022 12:04:31 +0530 Subject: [PATCH 5/9] Issue #SB-29145 refactor: PR Review Based Changes --- .../org/sunbird/managers/CopyManager.scala | 52 ++++++++----------- 1 file changed, 21 insertions(+), 31 deletions(-) diff --git a/assessment-api/assessment-actors/src/main/scala/org/sunbird/managers/CopyManager.scala b/assessment-api/assessment-actors/src/main/scala/org/sunbird/managers/CopyManager.scala index 06656c105..1aa02a4f1 100644 --- a/assessment-api/assessment-actors/src/main/scala/org/sunbird/managers/CopyManager.scala +++ b/assessment-api/assessment-actors/src/main/scala/org/sunbird/managers/CopyManager.scala @@ -44,7 +44,6 @@ object CopyManager { copyQuestionSet(node, request) case AssessmentConstants.QUESTION_MIME_TYPE => node.setInRelations(null) - validateCopyQuestionReq(request, node) //Check if the question has got "Default" visibility. copyNode(node, request) } copiedNodeFuture.map(copiedNode => { @@ -60,19 +59,12 @@ object CopyManager { }).flatMap(f => f) recoverWith { case e: CompletionException => throw e.getCause } } - def validateCopyQuestionReq(request: Request, node: Node) = { - val visibility = node.getMetadata.getOrDefault(AssessmentConstants.VISIBILITY, AssessmentConstants.VISIBILITY_PARENT).asInstanceOf[String] - if (StringUtils.equalsIgnoreCase(visibility, AssessmentConstants.VISIBILITY_PARENT)) { - throw new ClientException(AssessmentConstants.ERR_INVALID_REQUEST, "Question With Visibility Parent Cannot Be Copied Individually!") - } - } - def validateExistingNode(request: Request, node: Node) = { val requestObjectType = request.getObjectType val nodeObjectType = node.getObjectType - if (!StringUtils.equalsIgnoreCase(requestObjectType, nodeObjectType)) { - throw new ClientException(AssessmentConstants.ERR_INVALID_OBJECT_TYPE, "Invalid Object Type: " + requestObjectType) - } + if (!StringUtils.equalsIgnoreCase(requestObjectType, nodeObjectType)) throw new ClientException(AssessmentConstants.ERR_INVALID_OBJECT_TYPE, s"Please Provide Valid ${requestObjectType} Identifier") + if (StringUtils.equalsIgnoreCase(node.getObjectType, AssessmentConstants.QUESTION) && StringUtils.equalsIgnoreCase(node.getMetadata.getOrDefault(AssessmentConstants.VISIBILITY, AssessmentConstants.VISIBILITY_PARENT).asInstanceOf[String], AssessmentConstants.VISIBILITY_PARENT)) + throw new ClientException(AssessmentConstants.ERR_INVALID_REQUEST, "Question With Visibility Parent Cannot Be Copied Individually!") } def copyQuestionSet(originNode: Node, request: Request)(implicit ex: ExecutionContext, oec: OntologyEngineContext): Future[Node] = { @@ -332,8 +324,8 @@ object CopyManager { def updateShallowHierarchy(request: Request, node: Node, originNode: Node, originHierarchy: util.Map[String, AnyRef])(implicit ec: ExecutionContext, oec: OntologyEngineContext): Future[Node] = { val childrenHierarchy = originHierarchy.get("children").asInstanceOf[util.List[util.Map[String, AnyRef]]] val req = new Request(request) - req.getContext.put(AssessmentConstants.SCHEMA_NAME, AssessmentConstants.QUESTIONSET_SCHEMA_NAME) - req.getContext.put(AssessmentConstants.VERSION, AssessmentConstants.SCHEMA_VERSION) + req.getContext.put(AssessmentConstants.SCHEMA_NAME, request.getContext.getOrDefault(AssessmentConstants.SCHEMA_NAME, AssessmentConstants.QUESTIONSET_SCHEMA_NAME)) + req.getContext.put(AssessmentConstants.VERSION, request.getContext.getOrDefault(AssessmentConstants.VERSION, AssessmentConstants.SCHEMA_VERSION)) req.getContext.put(AssessmentConstants.IDENTIFIER, node.getIdentifier) req.put(AssessmentConstants.HIERARCHY, ScalaJsonUtils.serialize(new java.util.HashMap[String, AnyRef]() { { @@ -345,7 +337,7 @@ object CopyManager { } def getCopyRequest(node: Node, request: Request)(implicit ec: ExecutionContext, oec: OntologyEngineContext): Future[Request] = { - val metadata: util.Map[String, AnyRef] = NodeUtil.serialize(node, new util.ArrayList(), node.getObjectType.toLowerCase.replace("image", ""), AssessmentConstants.SCHEMA_VERSION) + val metadata: util.Map[String, AnyRef] = NodeUtil.serialize(node, new util.ArrayList(), node.getObjectType.toLowerCase.replace("image", ""), request.getContext.getOrDefault(AssessmentConstants.VERSION, "").asInstanceOf[String]) val requestMap = request.getRequest requestMap.remove(AssessmentConstants.MODE) requestMap.remove(AssessmentConstants.COPY_SCHEME).asInstanceOf[String] @@ -361,8 +353,6 @@ object CopyManager { request.getContext().put(AssessmentConstants.SCHEMA_NAME, node.getObjectType.toLowerCase.replace("image", "")) val req = new Request(request) req.setRequest(metadata) - - val graphId = request.getContext.getOrDefault("graph_id", "").asInstanceOf[String] val version = request.getContext.getOrDefault("version", "").asInstanceOf[String] val externalProps = if (StringUtils.equalsIgnoreCase(AssessmentConstants.QUESTIONSET_MIME_TYPE, node.getMetadata.getOrDefault("mimeType", "").asInstanceOf[String])) { @@ -374,18 +364,17 @@ object CopyManager { readReq.setContext(request.getContext) readReq.put("identifier", node.getIdentifier) readReq.put("fields", externalProps.asJava) - val maxWaitTime: FiniteDuration = Duration(5, TimeUnit.SECONDS) - Await.result( - DataNode.read(readReq).map(node => { - val metadata: util.Map[String, AnyRef] = NodeUtil.serialize(node, externalProps.asJava, node.getObjectType.toLowerCase.replace("image", ""), request.getContext.get("version").asInstanceOf[String]) - externalProps.foreach(prop => { - val propValue = metadata.get(prop) - if (metadata.containsKey(prop) && propValue != null) { - req.put(prop, propValue) - } - }) - }), maxWaitTime) - Future(req) + DataNode.read(readReq).map(node => { + val metadata: util.Map[String, AnyRef] = NodeUtil.serialize(node, externalProps.asJava, node.getObjectType.toLowerCase.replace + ("image", ""), request.getContext.get("version").asInstanceOf[String]) + externalProps.foreach(prop => { + val propValue = metadata.get(prop) + if (metadata.containsKey(prop) && propValue != null) { + req.put(prop, propValue) + } + }) + Future(req) + }).flatMap(f=>f) } def getOriginData(metadata: util.Map[String, AnyRef], copyType: String): java.util.Map[String, AnyRef] = { @@ -406,8 +395,9 @@ object CopyManager { def validateShallowCopyReq(node: Node, request: Request) = { val copyType: String = request.getRequest.get("copyType").asInstanceOf[String] if (StringUtils.equalsIgnoreCase("shallow", copyType) && !StringUtils.equalsIgnoreCase("Live", node.getMetadata.get("status").asInstanceOf[String])) - throw new ClientException(AssessmentConstants.ERR_INVALID_REQUEST, "QuestionSet with status " + node.getMetadata.get(AssessmentConstants.STATUS).asInstanceOf[String].toLowerCase + " cannot be partially (shallow) copied.") - //TODO: check if need to throw client exception for combination of copyType=shallow and mode=edit + throw new ClientException(AssessmentConstants.ERR_INVALID_REQUEST, "QuestionSet With Status " + node.getMetadata.get(AssessmentConstants.STATUS).asInstanceOf[String].toLowerCase + " Cannot Be Partially (Shallow) Copied.") + if(StringUtils.equalsIgnoreCase("shallow", copyType) && StringUtils.equalsIgnoreCase(request.get(AssessmentConstants.MODE).asInstanceOf[String], "edit")) + request.getRequest.remove(AssessmentConstants.MODE) } def emptyCheckFilter(key: AnyRef): Boolean = key match { @@ -423,4 +413,4 @@ object CopyManager { } else metadata.keySet().removeAll(metadataNotTobeCopied) metadata } -} +} \ No newline at end of file From 4916195d601720fbf7ac981bf7827407138cdae5 Mon Sep 17 00:00:00 2001 From: joffinjoy <35325730+joffinjoy@users.noreply.github.com> Date: Fri, 22 Apr 2022 12:51:32 +0530 Subject: [PATCH 6/9] Issue #SB-29145 refactor: Origin Data Map Fix --- .../main/scala/org/sunbird/managers/CopyManager.scala | 9 +-------- 1 file changed, 1 insertion(+), 8 deletions(-) diff --git a/assessment-api/assessment-actors/src/main/scala/org/sunbird/managers/CopyManager.scala b/assessment-api/assessment-actors/src/main/scala/org/sunbird/managers/CopyManager.scala index 1aa02a4f1..269b53040 100644 --- a/assessment-api/assessment-actors/src/main/scala/org/sunbird/managers/CopyManager.scala +++ b/assessment-api/assessment-actors/src/main/scala/org/sunbird/managers/CopyManager.scala @@ -377,14 +377,7 @@ object CopyManager { }).flatMap(f=>f) } - def getOriginData(metadata: util.Map[String, AnyRef], copyType: String): java.util.Map[String, AnyRef] = { - new java.util.HashMap[String, AnyRef]() { - { - putAll(originMetadataKeys.asScala.filter(key => metadata.containsKey(key)).map(key => key -> metadata.get(key)).toMap.asJava) - put(AssessmentConstants.COPY_TYPE, copyType) - } - } - } + def getOriginData(metadata: util.Map[String, AnyRef], copyType: String): java.util.Map[String, AnyRef] = (Map(AssessmentConstants.COPY_TYPE -> copyType) ++ originMetadataKeys.asScala.filter(key => metadata.containsKey(key)).map(key => key -> metadata.get(key)).toMap).asJava def validateRequest(request: Request)(implicit ec: ExecutionContext, oec: OntologyEngineContext): Unit = { val keysNotPresent = AssessmentConstants.REQUIRED_KEYS.filter(key => emptyCheckFilter(request.getRequest.getOrDefault(key, ""))) From c6a6916e6fd0068773735c57ede3ca7903cb8b58 Mon Sep 17 00:00:00 2001 From: joffinjoy <35325730+joffinjoy@users.noreply.github.com> Date: Wed, 11 May 2022 12:21:26 +0530 Subject: [PATCH 7/9] Issue #SB-29145 refactor: Moved Branching Functions To BranchingUtil --- .../org/sunbird/managers/CopyManager.scala | 137 ++---------------- .../org/sunbird/utils/BranchingUtil.scala | 122 ++++++++++++++++ .../sunbird/actors/QuestionSetActorTest.scala | 6 +- 3 files changed, 135 insertions(+), 130 deletions(-) create mode 100644 assessment-api/assessment-actors/src/main/scala/org/sunbird/utils/BranchingUtil.scala diff --git a/assessment-api/assessment-actors/src/main/scala/org/sunbird/managers/CopyManager.scala b/assessment-api/assessment-actors/src/main/scala/org/sunbird/managers/CopyManager.scala index 269b53040..dc73939e9 100644 --- a/assessment-api/assessment-actors/src/main/scala/org/sunbird/managers/CopyManager.scala +++ b/assessment-api/assessment-actors/src/main/scala/org/sunbird/managers/CopyManager.scala @@ -1,7 +1,5 @@ package org.sunbird.managers -import com.google.gson.{Gson, GsonBuilder} -import com.google.gson.reflect.TypeToken import org.apache.commons.collections.CollectionUtils import org.apache.commons.collections4.MapUtils import org.apache.commons.lang.StringUtils @@ -15,16 +13,14 @@ import org.sunbird.graph.nodes.DataNode import org.sunbird.graph.schema.DefinitionNode import org.sunbird.graph.utils.{NodeUtil, ScalaJsonUtils} import org.sunbird.telemetry.logger.TelemetryManager -import org.sunbird.utils.{AssessmentConstants, HierarchyConstants} +import org.sunbird.utils.{AssessmentConstants, BranchingUtil, HierarchyConstants} import java.util -import java.util.concurrent.{CompletionException, TimeUnit} -import java.util.stream.Collectors +import java.util.concurrent.{CompletionException} import java.util.{Optional, UUID} -import scala.collection.JavaConversions.{asScalaBuffer, mapAsScalaMap} +import scala.collection.JavaConversions.{mapAsScalaMap} import scala.collection.JavaConverters._ -import scala.concurrent.duration.{Duration, FiniteDuration} -import scala.concurrent.{Await, ExecutionContext, Future} +import scala.concurrent.{ExecutionContext, Future} object CopyManager { @@ -93,129 +89,16 @@ object CopyManager { }).flatMap(f => f) } - - def generateNodeBLRecord(nodesModified: util.HashMap[String, AnyRef]): util.HashMap[String, AnyRef] = { - val idSet = nodesModified.keySet().asScala.toList - val nodeBLRecord = new util.HashMap[String, AnyRef]() - idSet.map(id => { - val nodeMetaData = nodesModified.getOrDefault(id, new util.HashMap()).asInstanceOf[util.Map[String, AnyRef]].getOrDefault(AssessmentConstants.METADATA, new util.HashMap()).asInstanceOf[util.Map[String, AnyRef]] - val containsBL = nodeMetaData.containsKey(AssessmentConstants.BRANCHING_LOGIC) - nodeBLRecord.put(id, new util.HashMap[String, AnyRef]() { - { - if (containsBL) put(AssessmentConstants.BRANCHING_LOGIC, nodeMetaData.get(AssessmentConstants.BRANCHING_LOGIC)) - put(AssessmentConstants.CONTAINS_BL, containsBL.asInstanceOf[AnyRef]) - put(AssessmentConstants.COPY_OF, nodeMetaData.get(AssessmentConstants.COPY_OF).asInstanceOf[String]) - } - }) - if (containsBL) nodeMetaData.remove(AssessmentConstants.BRANCHING_LOGIC) - nodeMetaData.remove(AssessmentConstants.COPY_OF) - }) - nodeBLRecord - } - - def branchingLogicArrayHandler(nodeBL: util.HashMap[String, AnyRef], name: String, oldToNewIdMap: util.Map[String, String]) = { - val array = nodeBL.getOrDefault(name, new util.ArrayList[String]).asInstanceOf[util.ArrayList[String]] - val newArray = new util.ArrayList[String]() - array.map(id => { - if (oldToNewIdMap.containsKey(id)) { - newArray.add(oldToNewIdMap.get(id)) - } else newArray.add(id) - }) - nodeBL.remove(name) - nodeBL.put(name, newArray) - } - - def preConditionHandler(nodeBL: util.HashMap[String, AnyRef], oldToNewIdMap: util.Map[String, String]): Unit = { - val preCondition = nodeBL.get(AssessmentConstants.PRE_CONDITION).asInstanceOf[util.HashMap[String, AnyRef]] - preCondition.keySet().asScala.toList.map(key => { - val conjunctionArray = preCondition.get(key).asInstanceOf[util.ArrayList[String]] - val condition = conjunctionArray.get(0).asInstanceOf[util.HashMap[String, AnyRef]] - condition.keySet().asScala.toList.map(logicOp => { - val conditionArray = condition.get(logicOp).asInstanceOf[util.ArrayList[String]] - val sourceQuestionRecord = conditionArray.get(0).asInstanceOf[util.HashMap[String, AnyRef]] - val preConditionVar = sourceQuestionRecord.get(AssessmentConstants.PRE_CONDITION_VAR).asInstanceOf[String] - val stringArray = preConditionVar.split("\\.") - if (oldToNewIdMap.containsKey(stringArray(0))) { - val newString = oldToNewIdMap.get(stringArray(0)) + "." + stringArray.drop(1).mkString(".") - sourceQuestionRecord.remove(AssessmentConstants.PRE_CONDITION_VAR) - sourceQuestionRecord.put(AssessmentConstants.PRE_CONDITION_VAR, newString) - } - }) - }) - } - - def branchingLogicModifier(branchingLogic: util.HashMap[String, AnyRef], oldToNewIdMap: util.Map[String, String]): Unit = { - branchingLogic.keySet().asScala.toList.map(identifier => { - val nodeBL = branchingLogic.get(identifier).asInstanceOf[util.HashMap[String, AnyRef]] - nodeBL.keySet().asScala.toList.map(key => { - if (StringUtils.equalsIgnoreCase(key, AssessmentConstants.TARGET)) branchingLogicArrayHandler(nodeBL, AssessmentConstants.TARGET, oldToNewIdMap) - else if (StringUtils.equalsIgnoreCase(key, AssessmentConstants.PRE_CONDITION)) preConditionHandler(nodeBL, oldToNewIdMap) - else if (StringUtils.equalsIgnoreCase(key, AssessmentConstants.SOURCE)) branchingLogicArrayHandler(nodeBL, AssessmentConstants.SOURCE, oldToNewIdMap) - }) - if (oldToNewIdMap.containsKey(identifier)) { - branchingLogic.put(oldToNewIdMap.get(identifier), nodeBL) - branchingLogic.remove(identifier) - } - }) - } - - def generateOldToNewIdMap(nodeBLRecord: util.HashMap[String, AnyRef], identifiers: util.Map[String, String]): util.Map[String, String] = { - val oldToNewIdMap = new util.HashMap[String, String]() - nodeBLRecord.keySet().asScala.toList.map(id => { - val nodeInfo = nodeBLRecord.get(id).asInstanceOf[util.HashMap[String, AnyRef]] - val newId = identifiers.get(id) - val oldId = nodeInfo.get(AssessmentConstants.COPY_OF).asInstanceOf[String] - oldToNewIdMap.put(oldId, newId) - }) - oldToNewIdMap - } - - def hierarchyRequestModifier(request: Request, nodeBLRecord: util.HashMap[String, AnyRef], identifiers: util.Map[String, String]) = { - val nodesModified: java.util.HashMap[String, AnyRef] = request.getRequest.get(HierarchyConstants.NODES_MODIFIED).asInstanceOf[java.util.HashMap[String, AnyRef]] - val hierarchy: java.util.HashMap[String, AnyRef] = request.getRequest.get(HierarchyConstants.HIERARCHY).asInstanceOf[java.util.HashMap[String, AnyRef]] - val oldToNewIdMap = generateOldToNewIdMap(nodeBLRecord, identifiers) - nodeBLRecord.keySet().asScala.toList.map(id => { - val nodeInfo = nodeBLRecord.get(id).asInstanceOf[util.HashMap[String, AnyRef]] - val node = nodesModified.get(id).asInstanceOf[util.HashMap[String, AnyRef]] - val nodeMetaData = node.get(AssessmentConstants.METADATA).asInstanceOf[util.HashMap[String, AnyRef]] - val newId = identifiers.get(id) - if (nodeInfo.get(AssessmentConstants.CONTAINS_BL).asInstanceOf[Boolean]) { - val branchingLogic = nodeInfo.get(AssessmentConstants.BRANCHING_LOGIC).asInstanceOf[util.HashMap[String, AnyRef]] - branchingLogicModifier(branchingLogic, oldToNewIdMap) - nodeMetaData.put(AssessmentConstants.BRANCHING_LOGIC, branchingLogic) - } - node.remove(AssessmentConstants.IS_NEW) - node.put(AssessmentConstants.IS_NEW, false.asInstanceOf[AnyRef]) - nodesModified.remove(id) - nodesModified.put(newId, node) - }) - hierarchy.keySet().asScala.toList.map(id => { - val nodeHierarchy = hierarchy.get(id).asInstanceOf[util.HashMap[String, AnyRef]] - val children = nodeHierarchy.get(AssessmentConstants.CHILDREN).asInstanceOf[util.ArrayList[String]] - val newChildrenList = new util.ArrayList[String] - children.map(identifier => { - if (identifiers.containsKey(identifier)) newChildrenList.add(identifiers.get(identifier)) else newChildrenList.add(identifier) - }) - nodeHierarchy.remove(AssessmentConstants.CHILDREN) - nodeHierarchy.put(AssessmentConstants.CHILDREN, newChildrenList) - if (identifiers.containsKey(id)) { - hierarchy.remove(id) - hierarchy.put(identifiers.get(id), nodeHierarchy) - } - }) - request - } - def updateHierarchy(request: Request, node: Node, originNode: Node, originHierarchy: util.Map[String, AnyRef], copyType: String)(implicit ec: ExecutionContext, oec: OntologyEngineContext): Future[Node] = { prepareHierarchyRequest(originHierarchy, originNode, node, copyType, request).map(req => { val hierarchyRequest = new Request(request) hierarchyRequest.putAll(req) val nodesModified: java.util.HashMap[String, AnyRef] = hierarchyRequest.getRequest.get(HierarchyConstants.NODES_MODIFIED).asInstanceOf[java.util.HashMap[String, AnyRef]] - val nodeBLRecord = generateNodeBLRecord(nodesModified) + val branchingRecord = BranchingUtil.generateBranchingRecord(nodesModified) val originalRequest = JsonUtils.deserialize(ScalaJsonUtils.serialize(hierarchyRequest), classOf[Request]) - val BLExists = nodeBLRecord.exists(BLRecord => BLRecord._2.asInstanceOf[util.HashMap[String, AnyRef]].get(AssessmentConstants.CONTAINS_BL) == true) - val (firstUpdateRequest, secondUpdateRequest) = if (BLExists) (hierarchyRequest, JsonUtils.deserialize(ScalaJsonUtils.serialize(hierarchyRequest), classOf[Request])) else (originalRequest, new Request()) - UpdateHierarchyManager.updateHierarchy(firstUpdateRequest).map(response => { + val BLExists = branchingRecord.exists(BLRecord => BLRecord._2.asInstanceOf[util.HashMap[String, AnyRef]].get(AssessmentConstants.CONTAINS_BL) == true) + val (updateHierarchyReq, branchingUpdateReq) = if (BLExists) (hierarchyRequest, JsonUtils.deserialize(ScalaJsonUtils.serialize(hierarchyRequest), classOf[Request])) else (originalRequest, new Request()) + UpdateHierarchyManager.updateHierarchy(updateHierarchyReq).map(response => { if (!ResponseHandler.checkError(response)) response else { TelemetryManager.info(s"Update Hierarchy Failed For Copy Question Set Having Identifier: ${node.getIdentifier} | Response is: " + response) @@ -223,8 +106,8 @@ object CopyManager { } }).map(response => { if (BLExists) { - hierarchyRequestModifier(secondUpdateRequest, nodeBLRecord, response.getResult.get(AssessmentConstants.IDENTIFIERS).asInstanceOf[util.Map[String, String]]) - UpdateHierarchyManager.updateHierarchy(secondUpdateRequest).map(response_ => { + BranchingUtil.hierarchyRequestModifier(branchingUpdateReq, branchingRecord, response.getResult.get(AssessmentConstants.IDENTIFIERS).asInstanceOf[util.Map[String, String]]) + UpdateHierarchyManager.updateHierarchy(branchingUpdateReq).map(response_ => { if (!ResponseHandler.checkError(response_)) node else { TelemetryManager.info(s"Update Hierarchy Failed For Copy Question Set Having Identifier: ${node.getIdentifier} | Response is: " + response) diff --git a/assessment-api/assessment-actors/src/main/scala/org/sunbird/utils/BranchingUtil.scala b/assessment-api/assessment-actors/src/main/scala/org/sunbird/utils/BranchingUtil.scala new file mode 100644 index 000000000..17e5d4a8b --- /dev/null +++ b/assessment-api/assessment-actors/src/main/scala/org/sunbird/utils/BranchingUtil.scala @@ -0,0 +1,122 @@ +package org.sunbird.utils + +import org.apache.commons.lang.StringUtils +import org.sunbird.common.dto.Request + +import java.util +import scala.collection.JavaConverters._ +import scala.collection.JavaConversions.{asScalaBuffer} + +object BranchingUtil { + + def generateBranchingRecord(nodesModified: util.HashMap[String, AnyRef]): util.HashMap[String, AnyRef] = { + val idSet = nodesModified.keySet().asScala.toList + val branchingRecord = new util.HashMap[String, AnyRef]() + idSet.map(id => { + val nodeMetaData = nodesModified.getOrDefault(id, new util.HashMap()).asInstanceOf[util.Map[String, AnyRef]].getOrDefault(AssessmentConstants.METADATA, new util.HashMap()).asInstanceOf[util.Map[String, AnyRef]] + val containsBL = nodeMetaData.containsKey(AssessmentConstants.BRANCHING_LOGIC) + branchingRecord.put(id, new util.HashMap[String, AnyRef]() { + { + if (containsBL) put(AssessmentConstants.BRANCHING_LOGIC, nodeMetaData.get(AssessmentConstants.BRANCHING_LOGIC)) + put(AssessmentConstants.CONTAINS_BL, containsBL.asInstanceOf[AnyRef]) + put(AssessmentConstants.COPY_OF, nodeMetaData.get(AssessmentConstants.COPY_OF).asInstanceOf[String]) + } + }) + if (containsBL) nodeMetaData.remove(AssessmentConstants.BRANCHING_LOGIC) + nodeMetaData.remove(AssessmentConstants.COPY_OF) + }) + branchingRecord + } + + def hierarchyRequestModifier(request: Request, branchingRecord: util.HashMap[String, AnyRef], identifiers: util.Map[String, String]) = { + val nodesModified: java.util.HashMap[String, AnyRef] = request.getRequest.get(HierarchyConstants.NODES_MODIFIED).asInstanceOf[java.util.HashMap[String, AnyRef]] + val hierarchy: java.util.HashMap[String, AnyRef] = request.getRequest.get(HierarchyConstants.HIERARCHY).asInstanceOf[java.util.HashMap[String, AnyRef]] + val oldToNewIdMap = generateOldToNewIdMap(branchingRecord, identifiers) + branchingRecord.keySet().asScala.toList.map(id => { + val nodeInfo = branchingRecord.get(id).asInstanceOf[util.HashMap[String, AnyRef]] + val node = nodesModified.get(id).asInstanceOf[util.HashMap[String, AnyRef]] + val nodeMetaData = node.get(AssessmentConstants.METADATA).asInstanceOf[util.HashMap[String, AnyRef]] + val newId = identifiers.get(id) + if (nodeInfo.get(AssessmentConstants.CONTAINS_BL).asInstanceOf[Boolean]) { + val branchingLogic = nodeInfo.get(AssessmentConstants.BRANCHING_LOGIC).asInstanceOf[util.HashMap[String, AnyRef]] + branchingLogicModifier(branchingLogic, oldToNewIdMap) + nodeMetaData.put(AssessmentConstants.BRANCHING_LOGIC, branchingLogic) + } + node.put(AssessmentConstants.IS_NEW, false.asInstanceOf[AnyRef]) + nodesModified.remove(id) + nodesModified.put(newId, node) + }) + hierarchy.keySet().asScala.toList.map(id => { + val nodeHierarchy = hierarchy.get(id).asInstanceOf[util.HashMap[String, AnyRef]] + val children = nodeHierarchy.get(AssessmentConstants.CHILDREN).asInstanceOf[util.ArrayList[String]] + val newChildrenList = new util.ArrayList[String] + children.map(identifier => { + if (identifiers.containsKey(identifier)) newChildrenList.add(identifiers.get(identifier)) else newChildrenList.add(identifier) + }) + nodeHierarchy.remove(AssessmentConstants.CHILDREN) + nodeHierarchy.put(AssessmentConstants.CHILDREN, newChildrenList) + if (identifiers.containsKey(id)) { + hierarchy.remove(id) + hierarchy.put(identifiers.get(id), nodeHierarchy) + } + }) + request + } + + def branchingLogicModifier(branchingLogic: util.HashMap[String, AnyRef], oldToNewIdMap: util.Map[String, String]): Unit = { + branchingLogic.keySet().asScala.toList.map(identifier => { + val nodeBL = branchingLogic.get(identifier).asInstanceOf[util.HashMap[String, AnyRef]] + nodeBL.keySet().asScala.toList.map(key => { + if (StringUtils.equalsIgnoreCase(key, AssessmentConstants.TARGET)) branchingLogicArrayHandler(nodeBL, AssessmentConstants.TARGET, oldToNewIdMap) + else if (StringUtils.equalsIgnoreCase(key, AssessmentConstants.PRE_CONDITION)) preConditionHandler(nodeBL, oldToNewIdMap) + else if (StringUtils.equalsIgnoreCase(key, AssessmentConstants.SOURCE)) branchingLogicArrayHandler(nodeBL, AssessmentConstants.SOURCE, oldToNewIdMap) + }) + if (oldToNewIdMap.containsKey(identifier)) { + branchingLogic.put(oldToNewIdMap.get(identifier), nodeBL) + branchingLogic.remove(identifier) + } + }) + } + + def generateOldToNewIdMap(branchingRecord: util.HashMap[String, AnyRef], identifiers: util.Map[String, String]): util.Map[String, String] = { + val oldToNewIdMap = new util.HashMap[String, String]() + branchingRecord.keySet().asScala.toList.map(id => { + val nodeInfo = branchingRecord.get(id).asInstanceOf[util.HashMap[String, AnyRef]] + val newId = identifiers.get(id) + val oldId = nodeInfo.get(AssessmentConstants.COPY_OF).asInstanceOf[String] + oldToNewIdMap.put(oldId, newId) + }) + oldToNewIdMap + } + + def branchingLogicArrayHandler(nodeBL: util.HashMap[String, AnyRef], name: String, oldToNewIdMap: util.Map[String, String]) = { + val branchingLogicArray = nodeBL.getOrDefault(name, new util.ArrayList[String]).asInstanceOf[util.ArrayList[String]] + val newBranchingLogicArray = new util.ArrayList[String]() + branchingLogicArray.map(id => { + if (oldToNewIdMap.containsKey(id)) { + newBranchingLogicArray.add(oldToNewIdMap.get(id)) + } else newBranchingLogicArray.add(id) + }) + nodeBL.remove(name) + nodeBL.put(name, newBranchingLogicArray) + } + + def preConditionHandler(nodeBL: util.HashMap[String, AnyRef], oldToNewIdMap: util.Map[String, String]): Unit = { + val preCondition = nodeBL.get(AssessmentConstants.PRE_CONDITION).asInstanceOf[util.HashMap[String, AnyRef]] + preCondition.keySet().asScala.toList.map(key => { + val conjunctionArray = preCondition.get(key).asInstanceOf[util.ArrayList[String]] + val condition = conjunctionArray.get(0).asInstanceOf[util.HashMap[String, AnyRef]] + condition.keySet().asScala.toList.map(logicOp => { + val conditionArray = condition.get(logicOp).asInstanceOf[util.ArrayList[String]] + val sourceQuestionRecord = conditionArray.get(0).asInstanceOf[util.HashMap[String, AnyRef]] + val preConditionVar = sourceQuestionRecord.get(AssessmentConstants.PRE_CONDITION_VAR).asInstanceOf[String] + val stringArray = preConditionVar.split("\\.") + if (oldToNewIdMap.containsKey(stringArray(0))) { + val newString = oldToNewIdMap.get(stringArray(0)) + "." + stringArray.drop(1).mkString(".") + sourceQuestionRecord.remove(AssessmentConstants.PRE_CONDITION_VAR) + sourceQuestionRecord.put(AssessmentConstants.PRE_CONDITION_VAR, newString) + } + }) + }) + } +} diff --git a/assessment-api/assessment-actors/src/test/scala/org/sunbird/actors/QuestionSetActorTest.scala b/assessment-api/assessment-actors/src/test/scala/org/sunbird/actors/QuestionSetActorTest.scala index a984f460a..25df70a3b 100644 --- a/assessment-api/assessment-actors/src/test/scala/org/sunbird/actors/QuestionSetActorTest.scala +++ b/assessment-api/assessment-actors/src/test/scala/org/sunbird/actors/QuestionSetActorTest.scala @@ -12,7 +12,7 @@ import org.sunbird.graph.utils.ScalaJsonUtils import org.sunbird.graph.{GraphService, OntologyEngineContext} import org.sunbird.kafka.client.KafkaClient import org.sunbird.managers.CopyManager -import org.sunbird.utils.{AssessmentConstants, JavaJsonUtils} +import org.sunbird.utils.{AssessmentConstants, BranchingUtil, JavaJsonUtils} import java.util import scala.collection.JavaConversions._ @@ -597,12 +597,12 @@ class QuestionSetActorTest extends BaseSpec with MockFactory with copyTrait { } it should "return expected result for 'generateNodeBLRecord'" in { - val result = CopyManager.generateNodeBLRecord(generateNodesModified("afa2bef1-b5db-45d9-b0d7-aeea757906c3", true)) + val result = BranchingUtil.generateBranchingRecord(generateNodesModified("afa2bef1-b5db-45d9-b0d7-aeea757906c3", true)) assert(result == generateNodeBLRecord) } it should "return expected result for 'hierarchyRequestModifier'" in { - val result = CopyManager.hierarchyRequestModifier(generateUpdateRequest(false, "afa2bef1-b5db-45d9-b0d7-aeea757906c3"), generateNodeBLRecord(), generateIdentifiers()) + val result = BranchingUtil.hierarchyRequestModifier(generateUpdateRequest(false, "afa2bef1-b5db-45d9-b0d7-aeea757906c3"), generateNodeBLRecord(), generateIdentifiers()) val expectedResult = generateUpdateRequest(true, "do_11351201604857856013") assert(result.getRequest.get(AssessmentConstants.NODES_MODIFIED) == expectedResult.getRequest.get(AssessmentConstants.NODES_MODIFIED)) assert(result.getRequest.get(AssessmentConstants.HIERARCHY) == expectedResult.getRequest.get(AssessmentConstants.HIERARCHY)) From f4666a09e9b45c6b42c3f48eacc7eadc6f2c9473 Mon Sep 17 00:00:00 2001 From: joffinjoy <35325730+joffinjoy@users.noreply.github.com> Date: Wed, 11 May 2022 18:07:02 +0530 Subject: [PATCH 8/9] Issue #SB-29145 refactor: Minor Optimization & Renamed Functions --- .../scala/org/sunbird/utils/BranchingUtil.scala | 13 ++++++------- 1 file changed, 6 insertions(+), 7 deletions(-) diff --git a/assessment-api/assessment-actors/src/main/scala/org/sunbird/utils/BranchingUtil.scala b/assessment-api/assessment-actors/src/main/scala/org/sunbird/utils/BranchingUtil.scala index 17e5d4a8b..fa9098816 100644 --- a/assessment-api/assessment-actors/src/main/scala/org/sunbird/utils/BranchingUtil.scala +++ b/assessment-api/assessment-actors/src/main/scala/org/sunbird/utils/BranchingUtil.scala @@ -17,12 +17,14 @@ object BranchingUtil { val containsBL = nodeMetaData.containsKey(AssessmentConstants.BRANCHING_LOGIC) branchingRecord.put(id, new util.HashMap[String, AnyRef]() { { - if (containsBL) put(AssessmentConstants.BRANCHING_LOGIC, nodeMetaData.get(AssessmentConstants.BRANCHING_LOGIC)) + if (containsBL) { + put(AssessmentConstants.BRANCHING_LOGIC, nodeMetaData.get(AssessmentConstants.BRANCHING_LOGIC)) + nodeMetaData.remove(AssessmentConstants.BRANCHING_LOGIC) + } put(AssessmentConstants.CONTAINS_BL, containsBL.asInstanceOf[AnyRef]) put(AssessmentConstants.COPY_OF, nodeMetaData.get(AssessmentConstants.COPY_OF).asInstanceOf[String]) } }) - if (containsBL) nodeMetaData.remove(AssessmentConstants.BRANCHING_LOGIC) nodeMetaData.remove(AssessmentConstants.COPY_OF) }) branchingRecord @@ -31,7 +33,7 @@ object BranchingUtil { def hierarchyRequestModifier(request: Request, branchingRecord: util.HashMap[String, AnyRef], identifiers: util.Map[String, String]) = { val nodesModified: java.util.HashMap[String, AnyRef] = request.getRequest.get(HierarchyConstants.NODES_MODIFIED).asInstanceOf[java.util.HashMap[String, AnyRef]] val hierarchy: java.util.HashMap[String, AnyRef] = request.getRequest.get(HierarchyConstants.HIERARCHY).asInstanceOf[java.util.HashMap[String, AnyRef]] - val oldToNewIdMap = generateOldToNewIdMap(branchingRecord, identifiers) + val oldToNewIdMap = getIdentifierMapping(branchingRecord, identifiers) branchingRecord.keySet().asScala.toList.map(id => { val nodeInfo = branchingRecord.get(id).asInstanceOf[util.HashMap[String, AnyRef]] val node = nodesModified.get(id).asInstanceOf[util.HashMap[String, AnyRef]] @@ -53,7 +55,6 @@ object BranchingUtil { children.map(identifier => { if (identifiers.containsKey(identifier)) newChildrenList.add(identifiers.get(identifier)) else newChildrenList.add(identifier) }) - nodeHierarchy.remove(AssessmentConstants.CHILDREN) nodeHierarchy.put(AssessmentConstants.CHILDREN, newChildrenList) if (identifiers.containsKey(id)) { hierarchy.remove(id) @@ -78,7 +79,7 @@ object BranchingUtil { }) } - def generateOldToNewIdMap(branchingRecord: util.HashMap[String, AnyRef], identifiers: util.Map[String, String]): util.Map[String, String] = { + def getIdentifierMapping(branchingRecord: util.HashMap[String, AnyRef], identifiers: util.Map[String, String]): util.Map[String, String] = { val oldToNewIdMap = new util.HashMap[String, String]() branchingRecord.keySet().asScala.toList.map(id => { val nodeInfo = branchingRecord.get(id).asInstanceOf[util.HashMap[String, AnyRef]] @@ -97,7 +98,6 @@ object BranchingUtil { newBranchingLogicArray.add(oldToNewIdMap.get(id)) } else newBranchingLogicArray.add(id) }) - nodeBL.remove(name) nodeBL.put(name, newBranchingLogicArray) } @@ -113,7 +113,6 @@ object BranchingUtil { val stringArray = preConditionVar.split("\\.") if (oldToNewIdMap.containsKey(stringArray(0))) { val newString = oldToNewIdMap.get(stringArray(0)) + "." + stringArray.drop(1).mkString(".") - sourceQuestionRecord.remove(AssessmentConstants.PRE_CONDITION_VAR) sourceQuestionRecord.put(AssessmentConstants.PRE_CONDITION_VAR, newString) } }) From e52aedcc2981ac61b91cfd431ef734781dbb45a2 Mon Sep 17 00:00:00 2001 From: joffinjoy <35325730+joffinjoy@users.noreply.github.com> Date: Mon, 16 May 2022 15:49:38 +0530 Subject: [PATCH 9/9] Issue #SB-29145 test: Modified Test-Cases --- .../{copyTrait.scala => CopySpec.scala} | 4 +- .../sunbird/actors/QuestionActorTest.scala | 20 +++---- .../sunbird/actors/QuestionSetActorTest.scala | 56 +++++++++---------- 3 files changed, 40 insertions(+), 40 deletions(-) rename assessment-api/assessment-actors/src/test/scala/org/sunbird/actors/{copyTrait.scala => CopySpec.scala} (99%) diff --git a/assessment-api/assessment-actors/src/test/scala/org/sunbird/actors/copyTrait.scala b/assessment-api/assessment-actors/src/test/scala/org/sunbird/actors/CopySpec.scala similarity index 99% rename from assessment-api/assessment-actors/src/test/scala/org/sunbird/actors/copyTrait.scala rename to assessment-api/assessment-actors/src/test/scala/org/sunbird/actors/CopySpec.scala index 8ee744cd7..772a23d23 100644 --- a/assessment-api/assessment-actors/src/test/scala/org/sunbird/actors/copyTrait.scala +++ b/assessment-api/assessment-actors/src/test/scala/org/sunbird/actors/CopySpec.scala @@ -11,7 +11,7 @@ import scala.collection.JavaConversions.mapAsJavaMap import scala.collection.JavaConverters.asJavaIterableConverter import scala.collection.mutable -trait copyTrait { +object CopySpec { private def getQuestionSetRequest(): Request = { val request = new Request() @@ -327,7 +327,7 @@ trait copyTrait { nodesModified } - def generateNodeBLRecord(): util.HashMap[String, AnyRef] = { + def generateBranchingRecord(): util.HashMap[String, AnyRef] = { val nodeBLRecord = new util.HashMap[String, AnyRef]() nodeBLRecord.put("afa2bef1-b5db-45d9-b0d7-aeea757906c3", new util.HashMap[String, AnyRef]() { { diff --git a/assessment-api/assessment-actors/src/test/scala/org/sunbird/actors/QuestionActorTest.scala b/assessment-api/assessment-actors/src/test/scala/org/sunbird/actors/QuestionActorTest.scala index e0a2d4139..349bc916a 100644 --- a/assessment-api/assessment-actors/src/test/scala/org/sunbird/actors/QuestionActorTest.scala +++ b/assessment-api/assessment-actors/src/test/scala/org/sunbird/actors/QuestionActorTest.scala @@ -16,7 +16,7 @@ import scala.collection.JavaConverters._ import scala.concurrent.Future import scala.concurrent.ExecutionContext.Implicits.global -class QuestionActorTest extends BaseSpec with MockFactory with copyTrait { +class QuestionActorTest extends BaseSpec with MockFactory { "questionActor" should "return failed response for 'unknown' operation" in { implicit val oec: OntologyEngineContext = new OntologyEngineContext @@ -321,12 +321,12 @@ class QuestionActorTest extends BaseSpec with MockFactory with copyTrait { (oec.graphService _).expects().returns(graphDB).anyNumberOfTimes() val nodes: util.List[Node] = getCategoryNode() (graphDB.getNodeByUniqueIds(_: String, _: SearchCriteria)).expects(*, *).returns(Future(nodes)).anyNumberOfTimes() - (graphDB.getNodeByUniqueId(_: String, _: String, _: Boolean, _: Request)).expects("domain", "do_1234", false, *).returns(Future(getExistingQuestionNode())).anyNumberOfTimes() - (graphDB.readExternalProps(_: Request, _: List[String])).expects(*, List("objectMetadata")).returns(Future(getSuccessfulResponse())).anyNumberOfTimes() - (graphDB.readExternalProps(_: Request, _: List[String])).expects(*, *).returns(Future(getReadPropsResponseForQuestion())).anyNumberOfTimes() - (graphDB.addNode(_: String, _: Node)).expects(*, *).returns(Future(getNewQuestionNode())) - (graphDB.saveExternalProps(_: Request)).expects(*).returns(Future(getSuccessfulResponse())).anyNumberOfTimes - val request = getQuestionCopyRequest() + (graphDB.getNodeByUniqueId(_: String, _: String, _: Boolean, _: Request)).expects("domain", "do_1234", false, *).returns(Future(CopySpec.getExistingQuestionNode())).anyNumberOfTimes() + (graphDB.readExternalProps(_: Request, _: List[String])).expects(*, List("objectMetadata")).returns(Future(CopySpec.getSuccessfulResponse())).anyNumberOfTimes() + (graphDB.readExternalProps(_: Request, _: List[String])).expects(*, *).returns(Future(CopySpec.getReadPropsResponseForQuestion())).anyNumberOfTimes() + (graphDB.addNode(_: String, _: Node)).expects(*, *).returns(Future(CopySpec.getNewQuestionNode())) + (graphDB.saveExternalProps(_: Request)).expects(*).returns(Future(CopySpec.getSuccessfulResponse())).anyNumberOfTimes + val request = CopySpec.getQuestionCopyRequest() request.putAll(mapAsJavaMap(Map("identifier" -> "do_1234", "mode" -> "", "copyType"-> "deep"))) request.setOperation("copyQuestion") val response = callActor(request, Props(new QuestionActor())) @@ -335,7 +335,7 @@ class QuestionActorTest extends BaseSpec with MockFactory with copyTrait { it should "return error response for 'copyQuestion' when createdFor & createdBy is missing" in { implicit val oec: OntologyEngineContext = mock[OntologyEngineContext] - val request = getInvalidQuestionSetCopyRequest() + val request = CopySpec.getInvalidQuestionSetCopyRequest() request.putAll(mapAsJavaMap(Map("identifier" -> "do_1234", "mode" -> "", "copyType"-> "deep"))) request.setOperation("copyQuestion") val response = callActor(request, Props(new QuestionActor())) @@ -346,8 +346,8 @@ class QuestionActorTest extends BaseSpec with MockFactory with copyTrait { implicit val oec: OntologyEngineContext = mock[OntologyEngineContext] val graphDB = mock[GraphService] (oec.graphService _).expects().returns(graphDB).anyNumberOfTimes() - val request = getQuestionCopyRequest() - (graphDB.getNodeByUniqueId(_: String, _: String, _: Boolean, _: Request)).expects("domain", "do_1234", false, *).returns(Future(getQuestionNode())).anyNumberOfTimes() + val request = CopySpec.getQuestionCopyRequest() + (graphDB.getNodeByUniqueId(_: String, _: String, _: Boolean, _: Request)).expects("domain", "do_1234", false, *).returns(Future(CopySpec.getQuestionNode())).anyNumberOfTimes() request.putAll(mapAsJavaMap(Map("identifier" -> "do_1234", "mode" -> "", "copyType"-> "deep"))) request.setOperation("copyQuestion") val response = callActor(request, Props(new QuestionActor())) diff --git a/assessment-api/assessment-actors/src/test/scala/org/sunbird/actors/QuestionSetActorTest.scala b/assessment-api/assessment-actors/src/test/scala/org/sunbird/actors/QuestionSetActorTest.scala index 25df70a3b..25591cd35 100644 --- a/assessment-api/assessment-actors/src/test/scala/org/sunbird/actors/QuestionSetActorTest.scala +++ b/assessment-api/assessment-actors/src/test/scala/org/sunbird/actors/QuestionSetActorTest.scala @@ -20,7 +20,7 @@ import scala.collection.JavaConverters._ import scala.concurrent.ExecutionContext.Implicits.global import scala.concurrent.Future -class QuestionSetActorTest extends BaseSpec with MockFactory with copyTrait { +class QuestionSetActorTest extends BaseSpec with MockFactory { "questionSetActor" should "return failed response for 'unknown' operation" in { implicit val oec: OntologyEngineContext = new OntologyEngineContext @@ -546,20 +546,20 @@ class QuestionSetActorTest extends BaseSpec with MockFactory with copyTrait { (oec.graphService _).expects().returns(graphDB).anyNumberOfTimes() val nodes: util.List[Node] = getCategoryNode() (graphDB.getNodeByUniqueIds(_: String, _: SearchCriteria)).expects(*, *).returns(Future(nodes)).anyNumberOfTimes() - (graphDB.getNodeByUniqueId(_: String, _: String, _: Boolean, _: Request)).expects("domain", "do_1234", false, *).returns(Future(getExistingRootNode())).anyNumberOfTimes() - (graphDB.getNodeByUniqueId(_: String, _: String, _: Boolean, _: Request)).expects("domain", "do_9876", false, *).returns(Future(getNewRootNode())).anyNumberOfTimes() - (graphDB.getNodeByUniqueId(_: String, _: String, _: Boolean, _: Request)).expects("domain", "do_9876.img", false, *).returns(Future(getNewRootNode())).anyNumberOfTimes() - (graphDB.getNodeByUniqueId(_: String, _: String, _: Boolean, _: Request)).expects("domain", *, false, *).returns(Future(getQuestionNode())).anyNumberOfTimes() - (graphDB.readExternalProps(_: Request, _: List[String])).expects(*, List("objectMetadata")).returns(Future(getSuccessfulResponse())).anyNumberOfTimes() - (graphDB.readExternalProps(_: Request, _: List[String])).expects(*, *).returns(Future(getExternalPropsResponseWithData())).anyNumberOfTimes() - (graphDB.updateExternalProps(_: Request)).expects(*).returns(Future(getSuccessfulResponse())).anyNumberOfTimes - (graphDB.saveExternalProps(_: Request)).expects(*).returns(Future(getSuccessfulResponse())).anyNumberOfTimes - (graphDB.upsertNode(_: String, _: Node, _: Request)).expects(*, *, *).returns(Future(getUpsertNode())).anyNumberOfTimes() + (graphDB.getNodeByUniqueId(_: String, _: String, _: Boolean, _: Request)).expects("domain", "do_1234", false, *).returns(Future(CopySpec.getExistingRootNode())).anyNumberOfTimes() + (graphDB.getNodeByUniqueId(_: String, _: String, _: Boolean, _: Request)).expects("domain", "do_9876", false, *).returns(Future(CopySpec.getNewRootNode())).anyNumberOfTimes() + (graphDB.getNodeByUniqueId(_: String, _: String, _: Boolean, _: Request)).expects("domain", "do_9876.img", false, *).returns(Future(CopySpec.getNewRootNode())).anyNumberOfTimes() + (graphDB.getNodeByUniqueId(_: String, _: String, _: Boolean, _: Request)).expects("domain", *, false, *).returns(Future(CopySpec.getQuestionNode())).anyNumberOfTimes() + (graphDB.readExternalProps(_: Request, _: List[String])).expects(*, List("objectMetadata")).returns(Future(CopySpec.getSuccessfulResponse())).anyNumberOfTimes() + (graphDB.readExternalProps(_: Request, _: List[String])).expects(*, *).returns(Future(CopySpec.getExternalPropsResponseWithData())).anyNumberOfTimes() + (graphDB.updateExternalProps(_: Request)).expects(*).returns(Future(CopySpec.getSuccessfulResponse())).anyNumberOfTimes + (graphDB.saveExternalProps(_: Request)).expects(*).returns(Future(CopySpec.getSuccessfulResponse())).anyNumberOfTimes + (graphDB.upsertNode(_: String, _: Node, _: Request)).expects(*, *, *).returns(Future(CopySpec.getUpsertNode())).anyNumberOfTimes() inSequence { - (graphDB.addNode(_: String, _: Node)).expects(*, *).returns(Future(getNewRootNode())) - (graphDB.addNode(_: String, _: Node)).expects(*, *).returns(Future(getQuestionNode())) + (graphDB.addNode(_: String, _: Node)).expects(*, *).returns(Future(CopySpec.getNewRootNode())) + (graphDB.addNode(_: String, _: Node)).expects(*, *).returns(Future(CopySpec.getQuestionNode())) } - val request = getQuestionSetCopyRequest() + val request = CopySpec.getQuestionSetCopyRequest() request.putAll(mapAsJavaMap(Map("identifier" -> "do_1234", "mode" -> "", "copyType"-> "deep"))) request.setOperation("copyQuestionSet") val response = callActor(request, Props(new QuestionSetActor())) @@ -572,15 +572,15 @@ class QuestionSetActorTest extends BaseSpec with MockFactory with copyTrait { (oec.graphService _).expects().returns(graphDB).anyNumberOfTimes() val nodes: util.List[Node] = getCategoryNode() (graphDB.getNodeByUniqueIds(_: String, _: SearchCriteria)).expects(*, *).returns(Future(nodes)).anyNumberOfTimes() - (graphDB.getNodeByUniqueId(_: String, _: String, _: Boolean, _: Request)).expects("domain", "do_1234", false, *).returns(Future(getExistingRootNode())).anyNumberOfTimes() - (graphDB.getNodeByUniqueId(_: String, _: String, _: Boolean, _: Request)).expects("domain", "do_5678", false, *).returns(Future(getNewRootNode())).anyNumberOfTimes() - (graphDB.readExternalProps(_: Request, _: List[String])).expects(*, List("objectMetadata")).returns(Future(getSuccessfulResponse())).anyNumberOfTimes() - (graphDB.readExternalProps(_: Request, _: List[String])).expects(*, *).returns(Future(getExternalPropsResponseWithData())).anyNumberOfTimes() - (graphDB.addNode(_: String, _: Node)).expects(*, *).returns(Future(getQuestionNode())) - (graphDB.saveExternalProps(_: Request)).expects(*).returns(Future(getSuccessfulResponse())).anyNumberOfTimes - (graphDB.upsertNode(_: String, _: Node, _: Request)).expects(*, *, *).returns(Future(getUpsertNode())).anyNumberOfTimes() - (graphDB.updateExternalProps(_: Request)).expects(*).returns(Future(getSuccessfulResponse())).anyNumberOfTimes - val request = getQuestionSetCopyRequest() + (graphDB.getNodeByUniqueId(_: String, _: String, _: Boolean, _: Request)).expects("domain", "do_1234", false, *).returns(Future(CopySpec.getExistingRootNode())).anyNumberOfTimes() + (graphDB.getNodeByUniqueId(_: String, _: String, _: Boolean, _: Request)).expects("domain", "do_5678", false, *).returns(Future(CopySpec.getNewRootNode())).anyNumberOfTimes() + (graphDB.readExternalProps(_: Request, _: List[String])).expects(*, List("objectMetadata")).returns(Future(CopySpec.getSuccessfulResponse())).anyNumberOfTimes() + (graphDB.readExternalProps(_: Request, _: List[String])).expects(*, *).returns(Future(CopySpec.getExternalPropsResponseWithData())).anyNumberOfTimes() + (graphDB.addNode(_: String, _: Node)).expects(*, *).returns(Future(CopySpec.getQuestionNode())) + (graphDB.saveExternalProps(_: Request)).expects(*).returns(Future(CopySpec.getSuccessfulResponse())).anyNumberOfTimes + (graphDB.upsertNode(_: String, _: Node, _: Request)).expects(*, *, *).returns(Future(CopySpec.getUpsertNode())).anyNumberOfTimes() + (graphDB.updateExternalProps(_: Request)).expects(*).returns(Future(CopySpec.getSuccessfulResponse())).anyNumberOfTimes + val request = CopySpec.getQuestionSetCopyRequest() request.putAll(mapAsJavaMap(Map("identifier" -> "do_1234", "mode" -> "", "copyType"-> "shallow"))) request.setOperation("copyQuestionSet") val response = callActor(request, Props(new QuestionSetActor())) @@ -589,21 +589,21 @@ class QuestionSetActorTest extends BaseSpec with MockFactory with copyTrait { it should "return error response for 'copyQuestionSet' when createdFor & createdBy is missing" in { implicit val oec: OntologyEngineContext = mock[OntologyEngineContext] - val request = getInvalidQuestionCopyRequest() + val request = CopySpec.getInvalidQuestionCopyRequest() request.putAll(mapAsJavaMap(Map("identifier" -> "do_1234", "mode" -> "", "copyType"-> "deep"))) request.setOperation("copyQuestionSet") val response = callActor(request, Props(new QuestionSetActor())) assert("failed".equals(response.getParams.getStatus)) } - it should "return expected result for 'generateNodeBLRecord'" in { - val result = BranchingUtil.generateBranchingRecord(generateNodesModified("afa2bef1-b5db-45d9-b0d7-aeea757906c3", true)) - assert(result == generateNodeBLRecord) + it should "return expected result for 'generateBranchingRecord'" in { + val result = BranchingUtil.generateBranchingRecord(CopySpec.generateNodesModified("afa2bef1-b5db-45d9-b0d7-aeea757906c3", true)) + assert(result == CopySpec.generateBranchingRecord) } it should "return expected result for 'hierarchyRequestModifier'" in { - val result = BranchingUtil.hierarchyRequestModifier(generateUpdateRequest(false, "afa2bef1-b5db-45d9-b0d7-aeea757906c3"), generateNodeBLRecord(), generateIdentifiers()) - val expectedResult = generateUpdateRequest(true, "do_11351201604857856013") + val result = BranchingUtil.hierarchyRequestModifier(CopySpec.generateUpdateRequest(false, "afa2bef1-b5db-45d9-b0d7-aeea757906c3"), CopySpec.generateBranchingRecord(), CopySpec.generateIdentifiers()) + val expectedResult = CopySpec.generateUpdateRequest(true, "do_11351201604857856013") assert(result.getRequest.get(AssessmentConstants.NODES_MODIFIED) == expectedResult.getRequest.get(AssessmentConstants.NODES_MODIFIED)) assert(result.getRequest.get(AssessmentConstants.HIERARCHY) == expectedResult.getRequest.get(AssessmentConstants.HIERARCHY)) }