diff --git a/commons/protocol/src/main/scala/net/shrine/protocol/QueryResult.scala b/commons/protocol/src/main/scala/net/shrine/protocol/QueryResult.scala index a4aae7fa2..6a257940b 100644 --- a/commons/protocol/src/main/scala/net/shrine/protocol/QueryResult.scala +++ b/commons/protocol/src/main/scala/net/shrine/protocol/QueryResult.scala @@ -1,326 +1,328 @@ package net.shrine.protocol import javax.xml.datatype.XMLGregorianCalendar import net.shrine.problem.ProblemDigest import scala.xml.NodeSeq import net.shrine.util.{Tries, XmlUtil, NodeSeqEnrichments, SEnum, XmlDateHelper, OptionEnrichments} import net.shrine.serialization.{ I2b2Marshaller, XmlMarshaller } import scala.util.Try /** * @author Bill Simons * @since 4/15/11 * @see http://cbmi.med.harvard.edu * @see http://chip.org * <p/> * NOTICE: This software comes with NO guarantees whatsoever and is * licensed as Lgpl Open Source * @see http://www.gnu.org/licenses/lgpl.html * * NB: this is a case class to get a structural equality contract in hashCode and equals, mostly for testing */ final case class QueryResult( resultId: Long, instanceId: Long, resultType: Option[ResultOutputType], //this won't be present in the case of an error result setSize: Long, startDate: Option[XMLGregorianCalendar], endDate: Option[XMLGregorianCalendar], description: Option[String], statusType: QueryResult.StatusType, statusMessage: Option[String], breakdowns: Map[ResultOutputType, I2b2ResultEnvelope] = Map.empty, problemDigest: Option[ProblemDigest] = None ) extends XmlMarshaller with I2b2Marshaller { def this( resultId: Long, instanceId: Long, resultType: ResultOutputType, setSize: Long, startDate: XMLGregorianCalendar, endDate: XMLGregorianCalendar, statusType: QueryResult.StatusType) = { this( resultId, instanceId, Option(resultType), setSize, Option(startDate), Option(endDate), None, //description statusType, None) //statusMessage } def this( resultId: Long, instanceId: Long, resultType: ResultOutputType, setSize: Long, startDate: XMLGregorianCalendar, endDate: XMLGregorianCalendar, description: String, statusType: QueryResult.StatusType) = { this( resultId, instanceId, Option(resultType), setSize, Option(startDate), Option(endDate), Option(description), statusType, None) //statusMessage } def resultTypeIs(testedResultType: ResultOutputType): Boolean = resultType match { case Some(rt) => rt == testedResultType case _ => false } import QueryResult._ //NB: Fragile, non-type-safe == def isError = statusType == StatusType.Error def elapsed: Option[Long] = { def inMillis(xmlGc: XMLGregorianCalendar) = xmlGc.toGregorianCalendar.getTimeInMillis for { start <- startDate end <- endDate } yield inMillis(end) - inMillis(start) } //Sorting isn't strictly necessary, but makes deterministic unit testing easier. //The number of breakdowns will be at most 4, so performance should not be an issue. private def sortedBreakdowns: Seq[I2b2ResultEnvelope] = { breakdowns.values.toSeq.sortBy(_.resultType.name) } override def toI2b2: NodeSeq = { import OptionEnrichments._ XmlUtil.stripWhitespace { <query_result_instance> <result_instance_id>{ resultId }</result_instance_id> <query_instance_id>{ instanceId }</query_instance_id> { description.toXml(<description/>) } { resultType match { case Some(rt) if !rt.isError => //noinspection RedundantBlock { if (rt.isBreakdown) { rt.toI2b2NameOnly() } else { rt.toI2b2 } } case _ => ResultOutputType.ERROR.toI2b2NameOnly("") } } <set_size>{ setSize }</set_size> { startDate.toXml(<start_date/>) } { endDate.toXml(<end_date/>) } <query_status_type> <name>{ statusType }</name> { statusType.toI2b2(this) } </query_status_type> { //NB: Deliberately use Shrine XML format instead of the i2b2 one. Adding breakdowns to i2b2-format XML here is deviating from the i2b2 XSD schema in any case, //so if we're going to do that, let's produce saner XML. sortedBreakdowns.map(_.toXml.head).map(XmlUtil.renameRootTag("breakdown_data")) } </query_result_instance> } } override def toXml: NodeSeq = XmlUtil.stripWhitespace { import OptionEnrichments._ <queryResult> <resultId>{ resultId }</resultId> <instanceId>{ instanceId }</instanceId> { resultType.toXml(_.toXml) } <setSize>{ setSize }</setSize> { startDate.toXml(<startDate/>) } { endDate.toXml(<endDate/>) } { description.toXml(<description/>) } <status>{ statusType }</status> { statusMessage.toXml(<statusMessage/>) } { //Sorting isn't strictly necessary, but makes deterministic unit testing easier. //The number of breakdowns will be at most 4, so performance should not be an issue. sortedBreakdowns.map(_.toXml) } - { problemDigest.map(_.toXml).orNull } + { problemDigest.map(_.toXml).getOrElse("") } </queryResult> } def withId(id: Long): QueryResult = copy(resultId = id) def withInstanceId(id: Long): QueryResult = copy(instanceId = id) def modifySetSize(f: Long => Long): QueryResult = withSetSize(f(setSize)) def withSetSize(size: Long): QueryResult = copy(setSize = size) def withDescription(desc: String): QueryResult = copy(description = Option(desc)) def withResultType(resType: ResultOutputType): QueryResult = copy(resultType = Option(resType)) def withBreakdown(breakdownData: I2b2ResultEnvelope) = copy(breakdowns = breakdowns + (breakdownData.resultType -> breakdownData)) def withBreakdowns(newBreakdowns: Map[ResultOutputType, I2b2ResultEnvelope]) = copy(breakdowns = newBreakdowns) } object QueryResult { final case class StatusType( name: String, isDone: Boolean, i2b2Id: Option[Int] = Some(-1), private val doToI2b2:(QueryResult => NodeSeq) = StatusType.defaultToI2b2) extends StatusType.Value { def isError = this == StatusType.Error def toI2b2(queryResult: QueryResult): NodeSeq = doToI2b2(queryResult) } object StatusType extends SEnum[StatusType] { private val defaultToI2b2: QueryResult => NodeSeq = { queryResult => val i2b2Id: Int = queryResult.statusType.i2b2Id.getOrElse{ throw new IllegalStateException(s"queryResult.statusType ${queryResult.statusType} has no i2b2Id") } <status_type_id>{ i2b2Id }</status_type_id><description>{ queryResult.statusType.name }</description> } val noMessage:NodeSeq = null val Error = StatusType("ERROR", isDone = true, None, { queryResult => (queryResult.statusMessage, queryResult.problemDigest) match { case (Some(msg),Some(pd)) => <description>{ msg }</description> ++ pd.toXml case (Some(msg),None) => <description>{ msg }</description> case (None,Some(pd)) => pd.toXml case (None, None) => noMessage } }) /* msg => <codec>net.shrine.something.is.Broken</codec> <summary>Something is borked</summary> <description>{ msg }</description> <details>Herein is a stack trace, multiple lines</details> )) */ val Finished = StatusType("FINISHED", isDone = true, Some(3)) //TODO: Can we use the same <status_type_id> for Queued, Processing, and Incomplete? val Processing = StatusType("PROCESSING", isDone = false, Some(2)) val Queued = StatusType("QUEUED", isDone = false, Some(2)) val Incomplete = StatusType("INCOMPLETE", isDone = false, Some(2)) //TODO: What <status_type_id>s should these have? Does anyone care? val Held = StatusType("HELD", isDone = false) val SmallQueue = StatusType("SMALL_QUEUE", isDone = false) val MediumQueue = StatusType("MEDIUM_QUEUE", isDone = false) val LargeQueue = StatusType("LARGE_QUEUE", isDone = false) val NoMoreQueue = StatusType("NO_MORE_QUEUE", isDone = false) } def extractLong(nodeSeq: NodeSeq)(elemName: String): Long = (nodeSeq \ elemName).text.toLong private def parseDate(lexicalRep: String): Option[XMLGregorianCalendar] = XmlDateHelper.parseXmlTime(lexicalRep).toOption def elemAt(path: String*)(xml: NodeSeq): NodeSeq = path.foldLeft(xml)(_ \ _) def asText(path: String*)(xml: NodeSeq): String = elemAt(path: _*)(xml).text.trim def asResultOutputTypeOption(elemNames: String*)(breakdownTypes: Set[ResultOutputType], xml: NodeSeq): Option[ResultOutputType] = { import ResultOutputType.valueOf val typeName = asText(elemNames: _*)(xml) valueOf(typeName) orElse valueOf(breakdownTypes)(typeName) } def extractResultOutputType(xml: NodeSeq)(parse: NodeSeq => Try[ResultOutputType]): Option[ResultOutputType] = { val attempt = parse(xml) attempt.toOption } + def extractProblemDigest(xml: NodeSeq):Option[ProblemDigest] = { + + val subXml = xml \ "problem" + if(subXml.nonEmpty) Some(ProblemDigest.fromXml(subXml)) + else None + } + def fromXml(breakdownTypes: Set[ResultOutputType])(xml: NodeSeq): QueryResult = { def extract(elemName: String): Option[String] = { Option((xml \ elemName).text.trim).filter(!_.isEmpty) } def extractDate(elemName: String): Option[XMLGregorianCalendar] = extract(elemName).flatMap(parseDate) val asLong = extractLong(xml) _ import NodeSeqEnrichments.Strictness._ import Tries.sequence def extractBreakdowns(elemName: String): Map[ResultOutputType, I2b2ResultEnvelope] = { //noinspection ScalaUnnecessaryParentheses val mapAttempt = for { subXml <- xml.withChild(elemName) envelopes <- sequence(subXml.map(I2b2ResultEnvelope.fromXml(breakdownTypes))) mappings = envelopes.map(envelope => (envelope.resultType -> envelope)) } yield Map.empty ++ mappings mapAttempt.getOrElse(Map.empty) } - def extractProblemDigest():Option[ProblemDigest] = { - val subXml = xml \ "problemDigest" - if(subXml.nonEmpty) Some(ProblemDigest.fromXml(subXml)) - else None - } - QueryResult( asLong("resultId"), asLong("instanceId"), extractResultOutputType(xml \ "resultType")(ResultOutputType.fromXml), asLong("setSize"), extractDate("startDate"), extractDate("endDate"), extract("description"), StatusType.valueOf(asText("status")(xml)).get, //TODO: Avoid fragile .get call extract("statusMessage"), extractBreakdowns("resultEnvelope"), - extractProblemDigest() + extractProblemDigest(xml) ) } def fromI2b2(breakdownTypes: Set[ResultOutputType])(xml: NodeSeq): QueryResult = { def asLong = extractLong(xml) _ def asTextOption(path: String*): Option[String] = elemAt(path: _*)(xml).headOption.map(_.text.trim) def asXmlGcOption(path: String): Option[XMLGregorianCalendar] = asTextOption(path).filter(!_.isEmpty).flatMap(parseDate) QueryResult( resultId = asLong("result_instance_id"), instanceId = asLong("query_instance_id"), resultType = extractResultOutputType(xml \ "query_result_type")(ResultOutputType.fromI2b2), setSize = asLong("set_size"), startDate = asXmlGcOption("start_date"), endDate = asXmlGcOption("end_date"), description = asTextOption("description"), statusType = StatusType.valueOf(asText("query_status_type", "name")(xml)).get, //TODO: Avoid fragile .get call - statusMessage = asTextOption("query_status_type", "description")) + statusMessage = asTextOption("query_status_type", "description"), + problemDigest = extractProblemDigest(xml \ "query_status_type")) } //todo default problem description goes here def errorResult(description: Option[String], statusMessage: String,problemDigest:Option[ProblemDigest] = None) = { QueryResult( resultId = 0L, instanceId = 0L, resultType = None, setSize = 0L, startDate = None, endDate = None, description = description, statusType = StatusType.Error, statusMessage = Option(statusMessage), problemDigest = problemDigest) } } diff --git a/commons/protocol/src/test/scala/net/shrine/protocol/QueryResultTest.scala b/commons/protocol/src/test/scala/net/shrine/protocol/QueryResultTest.scala index 023a75ad7..079e93f27 100644 --- a/commons/protocol/src/test/scala/net/shrine/protocol/QueryResultTest.scala +++ b/commons/protocol/src/test/scala/net/shrine/protocol/QueryResultTest.scala @@ -1,494 +1,510 @@ package net.shrine.protocol import net.shrine.problem.ProblemDigest import net.shrine.util.ShouldMatchersForJUnit import org.junit.Test import net.shrine.util.XmlUtil import net.shrine.util.XmlDateHelper import net.shrine.util.XmlGcEnrichments /** * @author Bill Simons * @author clint * @since 8/19/11 * @see http://cbmi.med.harvard.edu * @see http://chip.org * <p/> * NOTICE: This software comes with NO guarantees whatsoever and is * licensed as Lgpl Open Source * @see http://www.gnu.org/licenses/lgpl.html */ //noinspection EmptyParenMethodAccessedAsParameterless,NameBooleanParameters final class QueryResultTest extends ShouldMatchersForJUnit with XmlRoundTripper[QueryResult] with I2b2SerializableValidator { private val date = XmlDateHelper.now private val resultId = 1L private val instanceId = 2L private val resultType = ResultOutputType.PATIENTSET private val setSize = 12L private val statusType = QueryResult.StatusType.Finished private val description = "description" private val statusMessage = "lakjdalsjd" private val problemCodec = "problem.codec" private val problemSummary = "test problem" private val problemDescription = "problem for testing" private val problemDetails = """Details of the problem |sometimes take |multiple lines. """.stripMargin private val queryResult = QueryResult(resultId, instanceId, Some(resultType), setSize, Option(date), Option(date), Option(description), statusType, Option(statusType.name)) import DefaultBreakdownResultOutputTypes.{ values => breakdownTypes, _ } private val resultWithBreakDowns = queryResult.copy( statusMessage = Some(statusMessage), breakdowns = Map(PATIENT_AGE_COUNT_XML -> I2b2ResultEnvelope(PATIENT_AGE_COUNT_XML, Map("foo" -> 1L, "bar" -> 2L)), PATIENT_RACE_COUNT_XML -> I2b2ResultEnvelope(PATIENT_RACE_COUNT_XML, Map("nuh" -> 3L, "zuh" -> 4L)), PATIENT_VITALSTATUS_COUNT_XML -> I2b2ResultEnvelope(PATIENT_VITALSTATUS_COUNT_XML, Map("blarg" -> 5L, "glarg" -> 6L)), PATIENT_GENDER_COUNT_XML -> I2b2ResultEnvelope(PATIENT_GENDER_COUNT_XML, Map("huh" -> 7L, "yeah" -> 8)))) private val expectedWhenBreakdownsArePresent = XmlUtil.stripWhitespace { <queryResult> <resultId>{ resultId }</resultId> <instanceId>{ instanceId }</instanceId> { resultType.toXml } <setSize>{ setSize }</setSize> <startDate>{ date }</startDate> <endDate>{ date }</endDate> <description>{ description }</description> <status>{ statusType }</status> <statusMessage>{ statusMessage }</statusMessage> <resultEnvelope> <resultType>{ PATIENT_AGE_COUNT_XML }</resultType> <column> <name>bar</name> <value>2</value> </column> <column> <name>foo</name> <value>1</value> </column> </resultEnvelope> <resultEnvelope> <resultType>{ PATIENT_GENDER_COUNT_XML }</resultType> <column> <name>huh</name> <value>7</value> </column> <column> <name>yeah</name> <value>8</value> </column> </resultEnvelope> <resultEnvelope> <resultType>{ PATIENT_RACE_COUNT_XML }</resultType> <column> <name>nuh</name> <value>3</value> </column> <column> <name>zuh</name> <value>4</value> </column> </resultEnvelope> <resultEnvelope> <resultType>{ PATIENT_VITALSTATUS_COUNT_XML }</resultType> <column> <name>blarg</name> <value>5</value> </column> <column> <name>glarg</name> <value>6</value> </column> </resultEnvelope> </queryResult> }.toString private val expectedI2b2Xml = XmlUtil.stripWhitespace { <query_result_instance> <result_instance_id>{ resultId }</result_instance_id> <query_instance_id>{ instanceId }</query_instance_id> <description>{ description }</description> <query_result_type> <result_type_id>1</result_type_id> <name>{ resultType }</name> <display_type>LIST</display_type><visual_attribute_type>LA</visual_attribute_type><description>Patient set</description> </query_result_type> <set_size>{ setSize }</set_size> <start_date>{ date }</start_date> <end_date>{ date }</end_date> <query_status_type> <name>{ statusType }</name> <status_type_id>3</status_type_id><description>FINISHED</description> </query_status_type> </query_result_instance> }.toString private val expectedI2b2XmlWithBreakdowns = XmlUtil.stripWhitespace { <query_result_instance> <result_instance_id>{ resultId }</result_instance_id> <query_instance_id>{ instanceId }</query_instance_id> <description>{ description }</description> { resultType.toI2b2 } <set_size>{ setSize }</set_size> <start_date>{ date }</start_date> <end_date>{ date }</end_date> <query_status_type> <name>{ statusType }</name> <status_type_id>3</status_type_id><description>FINISHED</description> </query_status_type> <breakdown_data> <resultType>{ PATIENT_AGE_COUNT_XML }</resultType> <column> <name>bar</name> <value>2</value> </column> <column> <name>foo</name> <value>1</value> </column> </breakdown_data> <breakdown_data> <resultType>{ PATIENT_GENDER_COUNT_XML }</resultType> <column> <name>huh</name> <value>7</value> </column> <column> <name>yeah</name> <value>8</value> </column> </breakdown_data> <breakdown_data> <resultType>{ PATIENT_RACE_COUNT_XML }</resultType> <column> <name>nuh</name> <value>3</value> </column> <column> <name>zuh</name> <value>4</value> </column> </breakdown_data> <breakdown_data> <resultType>{ PATIENT_VITALSTATUS_COUNT_XML }</resultType> <column> <name>blarg</name> <value>5</value> </column> <column> <name>glarg</name> <value>6</value> </column> </breakdown_data> </query_result_instance> }.toString private val expectedI2b2ErrorXml = XmlUtil.stripWhitespace { <query_result_instance> <result_instance_id>0</result_instance_id> <query_instance_id>0</query_instance_id> <description>{ description }</description> <query_result_type> <name></name> </query_result_type> <set_size>0</set_size> <query_status_type> <name>ERROR</name> <description>{ statusMessage }</description> </query_status_type> </query_result_instance> }.toString private val expectedI2b2ErrorWithProblemDigestXml = XmlUtil.stripWhitespace { <query_result_instance> <result_instance_id>0</result_instance_id> <query_instance_id>0</query_instance_id> <description>{ description }</description> <query_result_type> <name></name> </query_result_type> <set_size>0</set_size> <query_status_type> <name>ERROR</name> <description>{ statusMessage }</description> <problem> <codec>{ problemCodec }</codec> <summary>{ problemSummary }</summary> <description>{ problemDescription }</description> <details>{ problemDetails }</details> </problem> </query_status_type> </query_result_instance> }.toString //NB: See https://open.med.harvard.edu/jira/browse/SHRINE-745 private val expectedI2b2IncompleteXml = XmlUtil.stripWhitespace { <query_result_instance> <result_instance_id>0</result_instance_id> <query_instance_id>0</query_instance_id> <description>{ description }</description> <query_result_type> <name></name> </query_result_type> <set_size>0</set_size> <query_status_type> <name>INCOMPLETE</name> <description>{ statusMessage }</description> </query_status_type> </query_result_instance> }.toString import scala.xml.XML.loadString //NB: See https://open.med.harvard.edu/jira/browse/SHRINE-745 @Test def testParseIncomplete() { val qr = QueryResult.fromI2b2(breakdownTypes.toSet)(loadString(expectedI2b2IncompleteXml)) qr.statusType should be(QueryResult.StatusType.Incomplete) } @Test def testElapsed() { queryResult.copy(startDate = None).elapsed should be(None) queryResult.copy(endDate = None).elapsed should be(None) queryResult.copy(startDate = None, endDate = None).elapsed should be(None) { val now = XmlDateHelper.now queryResult.copy(startDate = Some(now), endDate = Some(now)).elapsed should equal(Some(0L)) } { val start = XmlDateHelper.now val delta = 123L import XmlGcEnrichments._ import scala.concurrent.duration._ val end = start + delta.milliseconds queryResult.copy(startDate = Some(start), endDate = Some(end)).elapsed should equal(Some(delta)) } } @Test def testIsError() { queryResult.isError should be(false) queryResult.copy(statusType = QueryResult.StatusType.Processing).isError should be(false) queryResult.copy(statusType = QueryResult.StatusType.Finished).isError should be(false) queryResult.copy(statusType = QueryResult.StatusType.Queued).isError should be(false) queryResult.copy(statusType = QueryResult.StatusType.Incomplete).isError should be(false) queryResult.copy(statusType = QueryResult.StatusType.Error).isError should be(true) } @Test def testToXml() { val queryResultForShrine = queryResult.copy(statusMessage = Some(statusMessage)) val expectedWhenNoBreakdowns = XmlUtil.stripWhitespace { <queryResult> <resultId>{ resultId }</resultId> <instanceId>{ instanceId }</instanceId> { resultType.toXml } <setSize>{ setSize }</setSize> <startDate>{ date }</startDate> <endDate>{ date }</endDate> <description>{ description }</description> <status>{ statusType }</status> <statusMessage>{ statusMessage }</statusMessage> </queryResult> }.toString queryResultForShrine.copy(statusMessage = Some(statusMessage)).toXmlString should equal(expectedWhenNoBreakdowns) val expectedWhenNoStartDate = XmlUtil.stripWhitespace { <queryResult> <resultId>{ resultId }</resultId> <instanceId>{ instanceId }</instanceId> { resultType.toXml } <setSize>{ setSize }</setSize> <endDate>{ date }</endDate> <description>{ description }</description> <status>{ statusType }</status> <statusMessage>{ statusMessage }</statusMessage> </queryResult> }.toString queryResultForShrine.copy(startDate = None).toXmlString should equal(expectedWhenNoStartDate) val expectedWhenNoEndDate = XmlUtil.stripWhitespace { <queryResult> <resultId>{ resultId }</resultId> <instanceId>{ instanceId }</instanceId> { resultType.toXml } <setSize>{ setSize }</setSize> <startDate>{ date }</startDate> <description>{ description }</description> <status>{ statusType }</status> <statusMessage>{ statusMessage }</statusMessage> </queryResult> }.toString queryResultForShrine.copy(endDate = None).toXmlString should equal(expectedWhenNoEndDate) val expectedWhenNoDescription = XmlUtil.stripWhitespace { <queryResult> <resultId>{ resultId }</resultId> <instanceId>{ instanceId }</instanceId> { resultType.toXml } <setSize>{ setSize }</setSize> <startDate>{ date }</startDate> <endDate>{ date }</endDate> <status>{ statusType }</status> <statusMessage>{ statusMessage }</statusMessage> </queryResult> }.toString queryResultForShrine.copy(description = None).toXmlString should equal(expectedWhenNoDescription) val expectedWhenNoStatusMessage = XmlUtil.stripWhitespace { <queryResult> <resultId>{ resultId }</resultId> <instanceId>{ instanceId }</instanceId> { resultType.toXml } <setSize>{ setSize }</setSize> <startDate>{ date }</startDate> <endDate>{ date }</endDate> <description>{ description }</description> <status>{ statusType }</status> </queryResult> }.toString queryResult.copy(statusMessage = None).toXmlString should equal(expectedWhenNoStatusMessage) resultWithBreakDowns.toXmlString should equal(expectedWhenBreakdownsArePresent) } @Test def testFromXml() { QueryResult.fromXml(breakdownTypes.toSet)(loadString(expectedWhenBreakdownsArePresent)) should equal(resultWithBreakDowns) } @Test def testShrineRoundTrip() = { QueryResult.fromXml(breakdownTypes.toSet)(resultWithBreakDowns.toXml) should equal(resultWithBreakDowns) } private def compareIgnoringBreakdowns(actual: QueryResult, expected: QueryResult) { //Ignore breakdowns field, since this can't be serialized to i2b2 format as part of a <query_result_instance> actual.breakdowns should equal(Map.empty) actual.description should equal(expected.description) actual.endDate should equal(expected.endDate) actual.instanceId should equal(expected.instanceId) actual.resultId should equal(expected.resultId) actual.resultType should equal(expected.resultType) actual.setSize should equal(expected.setSize) actual.startDate should equal(expected.startDate) actual.statusMessage should equal(expected.statusMessage) actual.statusType should equal(expected.statusType) } @Test def testI2b2RoundTrip() = { //NB: Needed because i2b2 handles status messages differently. In the error case, statusMessage is //descriptive; otherwise, it's the all-caps name of the status type. This is different from how //Shrine creates and parses statusMessage XML, so we need a new QueryResult here. (Previously, we //could use the same one, since we were ignoring statusMessage and description when unmarshalling //from i2b2 format.) val newStatusMessage = Some(resultWithBreakDowns.statusType.name) val resultWithBreakDownsForI2b2 = resultWithBreakDowns.copy(statusMessage = newStatusMessage) val unmarshalled = QueryResult.fromI2b2(breakdownTypes.toSet)(resultWithBreakDownsForI2b2.toI2b2) compareIgnoringBreakdowns(unmarshalled, resultWithBreakDownsForI2b2) } @Test def testFromI2b2() { compareIgnoringBreakdowns(QueryResult.fromI2b2(breakdownTypes.toSet)(loadString(expectedI2b2Xml)), queryResult) } @Test def testFromI2b2WithErrors() { val errorResult = QueryResult.errorResult(Some(description), statusMessage) val actual = QueryResult.fromI2b2(breakdownTypes.toSet)(loadString(expectedI2b2ErrorXml)) compareIgnoringBreakdowns(actual, errorResult) } @Test def testToI2b2() { queryResult.toI2b2String should equal(expectedI2b2Xml) } @Test def testToI2b2WithBreakdowns() { resultWithBreakDowns.toI2b2String should equal(expectedI2b2XmlWithBreakdowns) } @Test def testToI2b2AllStatusTypes(): Unit = { def doTest(statusType: QueryResult.StatusType) { val expectedI2b2Xml = XmlUtil.stripWhitespace { <query_result_instance> <result_instance_id>{ resultId }</result_instance_id> <query_instance_id>{ instanceId }</query_instance_id> <description>{ description }</description> { resultType.toI2b2 } <set_size>{ setSize }</set_size> <start_date>{ date }</start_date> <end_date>{ date }</end_date> <query_status_type> <name>{ statusType }</name> <status_type_id>{ statusType.i2b2Id.get }</status_type_id><description>{ statusType }</description> </query_status_type> </query_result_instance> }.toString val result = queryResult.copy(statusType = statusType) result.toI2b2String should equal(expectedI2b2Xml) } import QueryResult.StatusType //NB: Error is tested by testToI2b2WithErrors() val nonErrorStatuses = StatusType.values.toSet - StatusType.Error for (statusType <- nonErrorStatuses) { doTest(statusType) } } @Test def testToI2b2WithErrors(): Unit = { val actual = QueryResult.errorResult(Some(description), statusMessage).toI2b2String actual should equal(expectedI2b2ErrorXml) } @Test - def testToI2B2WithErrorsAndProblemDigest():Unit = { + def testWithErrorsAndProblemDigest():Unit = { + val actual = QueryResult.errorResult( Some(description), statusMessage, - Option(ProblemDigest(problemCodec,problemSummary,problemDescription,problemDetails))).toI2b2String + Option(ProblemDigest(problemCodec,problemSummary,problemDescription,problemDetails))) + + val i2b2String = actual.toI2b2String + + i2b2String should equal(expectedI2b2ErrorWithProblemDigestXml) + + val i2b2 = actual.toI2b2 + val fromI2b2 = QueryResult.fromI2b2(Set.empty)(i2b2) + + println(i2b2) + + println(fromI2b2) + + fromI2b2 should equal(actual) - actual should equal(expectedI2b2ErrorWithProblemDigestXml) + val xml = actual.toXml + val fromXml = QueryResult.fromXml(Set.empty)(xml) + fromXml should equal(actual) } } \ No newline at end of file diff --git a/commons/util/src/main/scala/net/shrine/serialization/I2b2Marshaller.scala b/commons/util/src/main/scala/net/shrine/serialization/I2b2Marshaller.scala index eddd713f8..7beb32be0 100644 --- a/commons/util/src/main/scala/net/shrine/serialization/I2b2Marshaller.scala +++ b/commons/util/src/main/scala/net/shrine/serialization/I2b2Marshaller.scala @@ -1,19 +1,19 @@ package net.shrine.serialization import xml.NodeSeq /** * @author Bill Simons - * @date 3/24/11 - * @link http://cbmi.med.harvard.edu - * @link http://chip.org + * @since 3/24/11 + * @see http://cbmi.med.harvard.edu + * @see http://chip.org * <p/> * NOTICE: This software comes with NO guarantees whatsoever and is * licensed as Lgpl Open Source - * @link http://www.gnu.org/licenses/lgpl.html + * @see http://www.gnu.org/licenses/lgpl.html */ trait I2b2Marshaller { def toI2b2: NodeSeq - def toI2b2String: String = toI2b2.toString + def toI2b2String: String = toI2b2.toString() } \ No newline at end of file