Skip to content

Commit

Permalink
Add new endpoint that excepts a list of concepts
Browse files Browse the repository at this point in the history
Returns a list of concepts with their metadata included.
  • Loading branch information
Gcolon021 committed Nov 18, 2024
1 parent 89d5728 commit 79af16f
Show file tree
Hide file tree
Showing 9 changed files with 189 additions and 10 deletions.
9 changes: 8 additions & 1 deletion dictonaryReqeust.http
Original file line number Diff line number Diff line change
Expand Up @@ -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}
{"@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\\"]
Original file line number Diff line number Diff line change
Expand Up @@ -58,6 +58,11 @@ public ResponseEntity<Concept> conceptDetail(@PathVariable(name = "dataset") Str
return conceptService.conceptDetail(dataset, conceptPath).map(ResponseEntity::ok).orElse(ResponseEntity.notFound().build());
}

@PostMapping(path = "/concepts/detail")
public ResponseEntity<List<Concept>> conceptsDetail(@RequestBody() List<String> conceptPaths) {
return conceptService.conceptsWithDetail(conceptPaths).map(ResponseEntity::ok).orElse(ResponseEntity.notFound().build());
}

@PostMapping(path = "/concepts/tree/{dataset}")
public ResponseEntity<Concept> conceptTree(
@PathVariable(name = "dataset") String dataset, @RequestBody() String conceptPath,
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -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;
Expand All @@ -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<String> disallowedMetaFields;

@Autowired
public ConceptRepository(
NamedParameterJdbcTemplate template, ConceptRowMapper mapper, ConceptFilterQueryGenerator filterGen,
ConceptMetaExtractor conceptMetaExtractor, ConceptResultSetExtractor conceptResultSetExtractor,
@Value("${filtering.unfilterable_concepts}") List<String> disallowedMetaFields
ConceptRowWithMetaMapper conceptRowWithMetaMapper, @Value("${filtering.unfilterable_concepts}") List<String> disallowedMetaFields
) {
this.template = template;
this.mapper = mapper;
this.filterGen = filterGen;
this.conceptMetaExtractor = conceptMetaExtractor;
this.conceptResultSetExtractor = conceptResultSetExtractor;
this.conceptRowWithMetaMapper = conceptRowWithMetaMapper;
this.disallowedMetaFields = disallowedMetaFields;
}

Expand Down Expand Up @@ -230,4 +231,52 @@ WITH RECURSIVE nodes AS (
}


public Optional<List<Concept>> getConceptsByPathWithMetadata(List<String> 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));
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -3,42 +3,56 @@
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
public ConceptResultSetUtil(JsonBlobParser jsonBlobParser) {
this.jsonBlobParser = jsonBlobParser;
}

public CategoricalConcept mapCategorical(ResultSet rs) throws SQLException {
public CategoricalConcept mapCategorical(ResultSet rs, boolean withMeta) throws SQLException {
Map<String, String> 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<String, String> 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);
}

}
Original file line number Diff line number Diff line change
@@ -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<Concept> {

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);
};
}

}
Original file line number Diff line number Diff line change
Expand Up @@ -71,4 +71,11 @@ public Optional<Concept> conceptDetailWithoutAncestors(String dataset, String co
return getConcept(dataset, conceptPath, false);
}

public Optional<List<Concept>> conceptsWithDetail(List<String> conceptPaths) {
if (conceptPaths.isEmpty()) {
return Optional.empty();
}

return this.conceptRepository.getConceptsByPathWithMetadata(conceptPaths);
}
}
Original file line number Diff line number Diff line change
@@ -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<String> parseValues(String valuesArr) {
try {
Expand Down Expand Up @@ -62,4 +72,17 @@ public Float parseMax(String valuesArr) {
return parseFromIndex(valuesArr, 1);
}

public Map<String, String> parseMetaData(String jsonMetaData) {
Map<String, String> metadata;

try {
List<Map<String, String>> maps = objectMapper.readValue(jsonMetaData, new TypeReference<List<Map<String, String>>>() {});
// 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;
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -148,4 +148,25 @@ void shouldDumpConcepts() {
Assertions.assertEquals(concepts, actual.getBody().getContent());
Assertions.assertEquals(HttpStatus.OK, actual.getStatusCode());
}

@Test
void shouldReturnNotFound() {
ResponseEntity<List<Concept>> 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<List<Concept>> concepts = Optional.of(List.of(fooBar, fooBaz));
List<String> conceptPaths = List.of("/foo//bar", "/foo//bar");
Mockito.when(conceptService.conceptsWithDetail(conceptPaths)).thenReturn(concepts);

ResponseEntity<List<Concept>> listResponseEntity = subject.conceptsDetail(conceptPaths);
Assertions.assertEquals(HttpStatus.OK, listResponseEntity.getStatusCode());
Assertions.assertEquals(concepts.get(), listResponseEntity.getBody());
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -324,5 +324,28 @@ void shouldGetContConceptWithDecimalNotation() {
Assertions.assertEquals(6.77f, concept.max());
}

@Test
void shouldGetConceptsByConceptPath() {
List<String> conceptPaths =
List.of("\\phs002385\\TXNUM\\", "\\phs000284\\pht001902\\phv00122507\\age\\", "\\phs000007\\pht000022\\phv00004260\\FM219\\");
Optional<List<Concept>> conceptsByPath = subject.getConceptsByPathWithMetadata(conceptPaths);
Assertions.assertTrue(conceptsByPath.isPresent());
Assertions.assertEquals(3, conceptsByPath.get().size());
}

@Test
void shouldGetSameConceptMetaAsConceptDetails() {
List<String> conceptPaths = List.of("\\phs002385\\TXNUM\\", "\\phs000284\\pht001902\\phv00122507\\age\\");
Optional<List<Concept>> 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<String, String> expectedMeta = subject.getConceptMeta(concept.dataset(), concept.conceptPath());

// compare the maps to each other.
Map<String, String> actualMeta = concept.meta();
Assertions.assertEquals(actualMeta, expectedMeta);
}

}

0 comments on commit 79af16f

Please sign in to comment.