diff --git a/dictonaryReqeust.http b/dictonaryReqeust.http index e474278..9e7ca8f 100644 --- a/dictonaryReqeust.http +++ b/dictonaryReqeust.http @@ -12,4 +12,11 @@ Content-Type: application/json POST http://localhost:80/search Content-Type: application/json -{"@type":"GeneralQueryRequest","resourceCredentials":{},"query":{"searchTerm":"breast","includedTags":[],"excludedTags":[],"returnTags":"true","offset":0,"limit":10000000},"resourceUUID":null} \ No newline at end of file +{"@type":"GeneralQueryRequest","resourceCredentials":{},"query":{"searchTerm":"breast","includedTags":[],"excludedTags":[],"returnTags":"true","offset":0,"limit":10000000},"resourceUUID":null} + +### + +POST http://localhost:80/concepts/detail +Content-Type: application/json + +["\\phs000993\\pht005015\\phv00253191\\BODY_SITE\\", "\\phs002913\\W2Q_COV_REINFEC_2_OTH\\"] \ No newline at end of file diff --git a/src/main/java/edu/harvard/dbmi/avillach/dictionary/concept/ConceptController.java b/src/main/java/edu/harvard/dbmi/avillach/dictionary/concept/ConceptController.java index d2c5c1c..154a3f1 100644 --- a/src/main/java/edu/harvard/dbmi/avillach/dictionary/concept/ConceptController.java +++ b/src/main/java/edu/harvard/dbmi/avillach/dictionary/concept/ConceptController.java @@ -58,6 +58,11 @@ public ResponseEntity conceptDetail(@PathVariable(name = "dataset") Str return conceptService.conceptDetail(dataset, conceptPath).map(ResponseEntity::ok).orElse(ResponseEntity.notFound().build()); } + @PostMapping(path = "/concepts/detail") + public ResponseEntity> conceptsDetail(@RequestBody() List conceptPaths) { + return conceptService.conceptsWithDetail(conceptPaths).map(ResponseEntity::ok).orElse(ResponseEntity.notFound().build()); + } + @PostMapping(path = "/concepts/tree/{dataset}") public ResponseEntity conceptTree( @PathVariable(name = "dataset") String dataset, @RequestBody() String conceptPath, diff --git a/src/main/java/edu/harvard/dbmi/avillach/dictionary/concept/ConceptRepository.java b/src/main/java/edu/harvard/dbmi/avillach/dictionary/concept/ConceptRepository.java index 7f449bb..8d8b69a 100644 --- a/src/main/java/edu/harvard/dbmi/avillach/dictionary/concept/ConceptRepository.java +++ b/src/main/java/edu/harvard/dbmi/avillach/dictionary/concept/ConceptRepository.java @@ -3,7 +3,6 @@ import edu.harvard.dbmi.avillach.dictionary.concept.model.Concept; import edu.harvard.dbmi.avillach.dictionary.filter.Filter; import edu.harvard.dbmi.avillach.dictionary.filter.QueryParamPair; -import edu.harvard.dbmi.avillach.dictionary.legacysearch.SearchResultRowMapper; import edu.harvard.dbmi.avillach.dictionary.util.MapExtractor; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.beans.factory.annotation.Value; @@ -27,19 +26,21 @@ public class ConceptRepository { private final ConceptFilterQueryGenerator filterGen; private final ConceptMetaExtractor conceptMetaExtractor; private final ConceptResultSetExtractor conceptResultSetExtractor; + private final ConceptRowWithMetaMapper conceptRowWithMetaMapper; private final List disallowedMetaFields; @Autowired public ConceptRepository( NamedParameterJdbcTemplate template, ConceptRowMapper mapper, ConceptFilterQueryGenerator filterGen, ConceptMetaExtractor conceptMetaExtractor, ConceptResultSetExtractor conceptResultSetExtractor, - @Value("${filtering.unfilterable_concepts}") List disallowedMetaFields + ConceptRowWithMetaMapper conceptRowWithMetaMapper, @Value("${filtering.unfilterable_concepts}") List disallowedMetaFields ) { this.template = template; this.mapper = mapper; this.filterGen = filterGen; this.conceptMetaExtractor = conceptMetaExtractor; this.conceptResultSetExtractor = conceptResultSetExtractor; + this.conceptRowWithMetaMapper = conceptRowWithMetaMapper; this.disallowedMetaFields = disallowedMetaFields; } @@ -230,4 +231,52 @@ WITH RECURSIVE nodes AS ( } + public Optional> getConceptsByPathWithMetadata(List conceptPaths) { + String sql = ALLOW_FILTERING_Q + ", " + + """ + filtered_concepts AS ( + SELECT + concept_node.* + FROM + concept_node + WHERE + concept_path IN (:conceptPaths) + ), + aggregated_meta AS ( + SELECT + concept_node_meta.concept_node_id, + json_agg(json_build_object('key', concept_node_meta.key, 'value', concept_node_meta.value)) AS metadata + FROM + concept_node_meta + WHERE + concept_node_meta.concept_node_id IN ( + SELECT concept_node_id FROM filtered_concepts + ) + GROUP BY + concept_node_meta.concept_node_id + ) + SELECT + concept_node.*, + ds.REF as dataset, + ds.abbreviation AS studyAcronym, + continuous_min.VALUE as min, continuous_max.VALUE as max, + categorical_values.VALUE as values, + allow_filtering.allowFiltering AS allowFiltering, + meta_description.VALUE AS description, + aggregated_meta.metadata AS metadata + FROM + filtered_concepts as concept_node + LEFT JOIN dataset AS ds ON concept_node.dataset_id = ds.dataset_id + LEFT JOIN concept_node_meta AS meta_description ON concept_node.concept_node_id = meta_description.concept_node_id AND meta_description.KEY = 'description' + LEFT JOIN concept_node_meta AS continuous_min ON concept_node.concept_node_id = continuous_min.concept_node_id AND continuous_min.KEY = 'min' + LEFT JOIN concept_node_meta AS continuous_max ON concept_node.concept_node_id = continuous_max.concept_node_id AND continuous_max.KEY = 'max' + LEFT JOIN concept_node_meta AS categorical_values ON concept_node.concept_node_id = categorical_values.concept_node_id AND categorical_values.KEY = 'values' + LEFT JOIN allow_filtering ON concept_node.concept_node_id = allow_filtering.concept_node_id + LEFT JOIN aggregated_meta ON concept_node.concept_node_id = aggregated_meta.concept_node_id + """; + + MapSqlParameterSource params = + new MapSqlParameterSource().addValue("conceptPaths", conceptPaths).addValue("disallowed_meta_keys", disallowedMetaFields); + return Optional.of(template.query(sql, params, conceptRowWithMetaMapper)); + } } diff --git a/src/main/java/edu/harvard/dbmi/avillach/dictionary/concept/ConceptResultSetUtil.java b/src/main/java/edu/harvard/dbmi/avillach/dictionary/concept/ConceptResultSetUtil.java index 0201691..b661c52 100644 --- a/src/main/java/edu/harvard/dbmi/avillach/dictionary/concept/ConceptResultSetUtil.java +++ b/src/main/java/edu/harvard/dbmi/avillach/dictionary/concept/ConceptResultSetUtil.java @@ -3,19 +3,17 @@ import edu.harvard.dbmi.avillach.dictionary.concept.model.CategoricalConcept; import edu.harvard.dbmi.avillach.dictionary.concept.model.ContinuousConcept; import edu.harvard.dbmi.avillach.dictionary.util.JsonBlobParser; -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.stereotype.Component; import java.sql.ResultSet; import java.sql.SQLException; import java.util.List; +import java.util.Map; @Component public class ConceptResultSetUtil { - private static final Logger log = LoggerFactory.getLogger(ConceptResultSetUtil.class); private final JsonBlobParser jsonBlobParser; @Autowired @@ -23,22 +21,38 @@ public ConceptResultSetUtil(JsonBlobParser jsonBlobParser) { this.jsonBlobParser = jsonBlobParser; } - public CategoricalConcept mapCategorical(ResultSet rs) throws SQLException { + public CategoricalConcept mapCategorical(ResultSet rs, boolean withMeta) throws SQLException { + Map metadata = null; + if (withMeta) { + metadata = jsonBlobParser.parseMetaData(rs.getString("metadata")); + } + return new CategoricalConcept( rs.getString("concept_path"), rs.getString("name"), rs.getString("display"), rs.getString("dataset"), rs.getString("description"), rs.getString("values") == null ? List.of() : jsonBlobParser.parseValues(rs.getString("values")), - rs.getBoolean("allowFiltering"), rs.getString("studyAcronym"), null, null + rs.getBoolean("allowFiltering"), rs.getString("studyAcronym"), null, metadata ); } - public ContinuousConcept mapContinuous(ResultSet rs) throws SQLException { + public ContinuousConcept mapContinuous(ResultSet rs, boolean withMeta) throws SQLException { + Map metadata = null; + if (withMeta) { + metadata = jsonBlobParser.parseMetaData(rs.getString("metadata")); + } + return new ContinuousConcept( rs.getString("concept_path"), rs.getString("name"), rs.getString("display"), rs.getString("dataset"), rs.getString("description"), rs.getBoolean("allowFiltering"), jsonBlobParser.parseMin(rs.getString("values")), - jsonBlobParser.parseMax(rs.getString("values")), rs.getString("studyAcronym"), null + jsonBlobParser.parseMax(rs.getString("values")), rs.getString("studyAcronym"), metadata ); } + public ContinuousConcept mapContinuous(ResultSet rs) throws SQLException { + return mapContinuous(rs, false); + } + public CategoricalConcept mapCategorical(ResultSet rs) throws SQLException { + return mapCategorical(rs, false); + } } diff --git a/src/main/java/edu/harvard/dbmi/avillach/dictionary/concept/ConceptRowWithMetaMapper.java b/src/main/java/edu/harvard/dbmi/avillach/dictionary/concept/ConceptRowWithMetaMapper.java new file mode 100644 index 0000000..8a2b4e2 --- /dev/null +++ b/src/main/java/edu/harvard/dbmi/avillach/dictionary/concept/ConceptRowWithMetaMapper.java @@ -0,0 +1,30 @@ +package edu.harvard.dbmi.avillach.dictionary.concept; + +import edu.harvard.dbmi.avillach.dictionary.concept.model.Concept; +import edu.harvard.dbmi.avillach.dictionary.concept.model.ConceptType; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.jdbc.core.RowMapper; +import org.springframework.stereotype.Component; + +import java.sql.ResultSet; +import java.sql.SQLException; + +@Component +public class ConceptRowWithMetaMapper implements RowMapper { + + private final ConceptResultSetUtil conceptResultSetUtil; + + @Autowired + public ConceptRowWithMetaMapper(ConceptResultSetUtil conceptResultSetUtil) { + this.conceptResultSetUtil = conceptResultSetUtil; + } + + @Override + public Concept mapRow(ResultSet rs, int rowNum) throws SQLException { + return switch (ConceptType.toConcept(rs.getString("concept_type"))) { + case Categorical -> conceptResultSetUtil.mapCategorical(rs, true); + case Continuous -> conceptResultSetUtil.mapContinuous(rs, true); + }; + } + +} diff --git a/src/main/java/edu/harvard/dbmi/avillach/dictionary/concept/ConceptService.java b/src/main/java/edu/harvard/dbmi/avillach/dictionary/concept/ConceptService.java index dc35254..85554c6 100644 --- a/src/main/java/edu/harvard/dbmi/avillach/dictionary/concept/ConceptService.java +++ b/src/main/java/edu/harvard/dbmi/avillach/dictionary/concept/ConceptService.java @@ -71,4 +71,11 @@ public Optional conceptDetailWithoutAncestors(String dataset, String co return getConcept(dataset, conceptPath, false); } + public Optional> conceptsWithDetail(List conceptPaths) { + if (conceptPaths.isEmpty()) { + return Optional.empty(); + } + + return this.conceptRepository.getConceptsByPathWithMetadata(conceptPaths); + } } diff --git a/src/main/java/edu/harvard/dbmi/avillach/dictionary/util/JsonBlobParser.java b/src/main/java/edu/harvard/dbmi/avillach/dictionary/util/JsonBlobParser.java index f976e7c..22f0ff0 100644 --- a/src/main/java/edu/harvard/dbmi/avillach/dictionary/util/JsonBlobParser.java +++ b/src/main/java/edu/harvard/dbmi/avillach/dictionary/util/JsonBlobParser.java @@ -1,21 +1,31 @@ package edu.harvard.dbmi.avillach.dictionary.util; +import com.fasterxml.jackson.core.JsonProcessingException; +import com.fasterxml.jackson.core.type.TypeReference; +import com.fasterxml.jackson.databind.ObjectMapper; import org.json.JSONArray; import org.json.JSONException; import org.slf4j.Logger; import org.slf4j.LoggerFactory; import org.springframework.stereotype.Component; +import org.springframework.util.StringUtils; import java.math.BigDecimal; import java.math.BigInteger; import java.util.ArrayList; +import java.util.HashMap; import java.util.List; +import java.util.Map; +import java.util.stream.Collectors; @Component public class JsonBlobParser { private final static Logger log = LoggerFactory.getLogger(JsonBlobParser.class); + private final ObjectMapper objectMapper = new ObjectMapper(); + + public JsonBlobParser() {} public List parseValues(String valuesArr) { try { @@ -62,4 +72,17 @@ public Float parseMax(String valuesArr) { return parseFromIndex(valuesArr, 1); } + public Map parseMetaData(String jsonMetaData) { + Map metadata; + + try { + List> maps = objectMapper.readValue(jsonMetaData, new TypeReference>>() {}); + // convert the list to a flat map + metadata = maps.stream().collect(Collectors.toMap(entry -> entry.get("key"), entry -> entry.get("value"))); + } catch (JsonProcessingException e) { + throw new RuntimeException(e); + } + + return metadata; + } } diff --git a/src/test/java/edu/harvard/dbmi/avillach/dictionary/concept/ConceptControllerTest.java b/src/test/java/edu/harvard/dbmi/avillach/dictionary/concept/ConceptControllerTest.java index 363c180..b3591f5 100644 --- a/src/test/java/edu/harvard/dbmi/avillach/dictionary/concept/ConceptControllerTest.java +++ b/src/test/java/edu/harvard/dbmi/avillach/dictionary/concept/ConceptControllerTest.java @@ -148,4 +148,25 @@ void shouldDumpConcepts() { Assertions.assertEquals(concepts, actual.getBody().getContent()); Assertions.assertEquals(HttpStatus.OK, actual.getStatusCode()); } + + @Test + void shouldReturnNotFound() { + ResponseEntity> listResponseEntity = subject.conceptsDetail(List.of()); + Assertions.assertEquals(HttpStatus.NOT_FOUND, listResponseEntity.getStatusCode()); + } + + @Test + void shouldReturnConceptsWithMeta() { + CategoricalConcept fooBar = new CategoricalConcept( + "/foo//bar", "bar", "Bar", "my_dataset", "foo!", List.of("a", "b"), true, "", List.of(), Map.of("key", "value") + ); + Concept fooBaz = new ContinuousConcept("/foo//baz", "baz", "Baz", "my_dataset", "foo!", true, 0F, 100F, "", Map.of("key", "value")); + Optional> concepts = Optional.of(List.of(fooBar, fooBaz)); + List conceptPaths = List.of("/foo//bar", "/foo//bar"); + Mockito.when(conceptService.conceptsWithDetail(conceptPaths)).thenReturn(concepts); + + ResponseEntity> listResponseEntity = subject.conceptsDetail(conceptPaths); + Assertions.assertEquals(HttpStatus.OK, listResponseEntity.getStatusCode()); + Assertions.assertEquals(concepts.get(), listResponseEntity.getBody()); + } } diff --git a/src/test/java/edu/harvard/dbmi/avillach/dictionary/concept/ConceptRepositoryTest.java b/src/test/java/edu/harvard/dbmi/avillach/dictionary/concept/ConceptRepositoryTest.java index 8d20516..df0a93f 100644 --- a/src/test/java/edu/harvard/dbmi/avillach/dictionary/concept/ConceptRepositoryTest.java +++ b/src/test/java/edu/harvard/dbmi/avillach/dictionary/concept/ConceptRepositoryTest.java @@ -324,5 +324,28 @@ void shouldGetContConceptWithDecimalNotation() { Assertions.assertEquals(6.77f, concept.max()); } + @Test + void shouldGetConceptsByConceptPath() { + List conceptPaths = + List.of("\\phs002385\\TXNUM\\", "\\phs000284\\pht001902\\phv00122507\\age\\", "\\phs000007\\pht000022\\phv00004260\\FM219\\"); + Optional> conceptsByPath = subject.getConceptsByPathWithMetadata(conceptPaths); + Assertions.assertTrue(conceptsByPath.isPresent()); + Assertions.assertEquals(3, conceptsByPath.get().size()); + } + + @Test + void shouldGetSameConceptMetaAsConceptDetails() { + List conceptPaths = List.of("\\phs002385\\TXNUM\\", "\\phs000284\\pht001902\\phv00122507\\age\\"); + Optional> conceptsByPath = subject.getConceptsByPathWithMetadata(conceptPaths); + Assertions.assertTrue(conceptsByPath.isPresent()); + + // Verify the meta data is correctly retrieve by comparing against known good query. + Concept concept = conceptsByPath.get().get(0); + Map expectedMeta = subject.getConceptMeta(concept.dataset(), concept.conceptPath()); + + // compare the maps to each other. + Map actualMeta = concept.meta(); + Assertions.assertEquals(actualMeta, expectedMeta); + } }