From 76a37da4032657d6cf9c97f7985b5f982505a4bc Mon Sep 17 00:00:00 2001 From: Ismail Ugur Bayindir Date: Mon, 20 May 2024 14:13:13 +0100 Subject: [PATCH] Add method to track cell contribution percentages by cxg obs fields (#73) * Updated version to 0.1.13 * Added dataset object to rdf representation * Disabled supercluster_of removal to fix missing nodes in graph generation * Added add_metadata_nodes method * Rounded up percentages in add_metadata_nodes * Refactored add_metadata_nodes to utilize OWL reification for improved visualisation in neo4j. --- pandasaurus_cxg/anndata_analyzer.py | 4 +- .../graph_generator/graph_generator.py | 61 +++++++++++++++++++ 2 files changed, 63 insertions(+), 2 deletions(-) diff --git a/pandasaurus_cxg/anndata_analyzer.py b/pandasaurus_cxg/anndata_analyzer.py index 2eb4b05..5692d3f 100644 --- a/pandasaurus_cxg/anndata_analyzer.py +++ b/pandasaurus_cxg/anndata_analyzer.py @@ -142,9 +142,9 @@ def co_annotation_report(self, disease: Optional[str] = None, enrich: bool = Fal [item for sublist in [[k, v] for k, v in record.items()] for item in sublist] for record in temp_result ] - unique_result = AnndataAnalyzer._remove_duplicates(result) + # unique_result = AnndataAnalyzer._remove_duplicates(result) self.report_df = pd.DataFrame( - [inner_list[:2] + inner_list[5:6] + inner_list[2:4] for inner_list in unique_result], + [inner_list[:2] + inner_list[5:6] + inner_list[2:4] for inner_list in result], columns=["field_name1", "value1", "predicate", "field_name2", "value2"], ) return self.report_df diff --git a/pandasaurus_cxg/graph_generator/graph_generator.py b/pandasaurus_cxg/graph_generator/graph_generator.py index 0bb33f4..5bf1baf 100644 --- a/pandasaurus_cxg/graph_generator/graph_generator.py +++ b/pandasaurus_cxg/graph_generator/graph_generator.py @@ -193,6 +193,67 @@ def enrich_rdf_graph(self): # add enrichment graph, subClassOf relations self.graph += self.ea.enricher_manager.enricher.graph + def add_metadata_nodes(self, metadata_fields: List[str]): + """ + Add metadata nodes to an RDF graph based on the specified metadata fields. Each node represents a metadata + attribute, and edges connecting these metadata nodes to cell clusters indicate the percentage contribution + of each metadata to the cluster. + + This function modifies the internal state of the RDF graph by adding new nodes and edges. + + Args: + metadata_fields (List[str]): A list of metadata field names that exist in the schema and should be added + to the RDF graph as nodes. + + Returns: + + """ + obs = self.ea.enricher_manager.anndata.obs + # metadata field validation + # TODO schema should be involved + missing_fields = [field for field in metadata_fields if field not in obs.keys()] + if missing_fields: + raise KeyError(f"Missing metadata fields: {', '.join(missing_fields)}") + + author_cell_types = list(self.ea.analyzer_manager.all_cell_type_identifiers) + # remove 'cell_type' from all_cell_type_identifiers + author_cell_types.pop(-1) + # add an annotation property for percentage + percentage_annotation_property = self.ns["percentage"] + self.graph.add((percentage_annotation_property, RDF.type, OWL.AnnotationProperty)) + for metadata in metadata_fields: + for s, _, _ in self.graph.triples((None, RDF.type, URIRef(CLUSTER.get("iri")))): + for a_cell_type in author_cell_types: + literal = self.graph.value(subject=s, predicate=self.ns[a_cell_type]) + if literal is None: + continue + percentages = ( + obs[obs[a_cell_type] == str(literal)][metadata].value_counts(normalize=True) + * 100 + ).loc[lambda x: x != 0.0] + for label, percentage in percentages.items(): + annotated_target = self.graph.value( + predicate=RDFS.label, object=Literal(label) + ) + if annotated_target is None: + annotated_target = URIRef(self.ns[str(uuid.uuid4())]) + self.graph.add((annotated_target, RDF.type, self.ns[metadata])) + self.graph.add((annotated_target, RDFS.label, Literal(label))) + bnode_axiom = BNode() + self.graph.add((bnode_axiom, RDF.type, OWL.Axiom)) + self.graph.add((bnode_axiom, OWL.annotatedSource, s)) + self.graph.add( + (bnode_axiom, OWL.annotatedProperty, self.ns["has_" + metadata]) + ) + self.graph.add((bnode_axiom, OWL.annotatedTarget, annotated_target)) + self.graph.add( + ( + bnode_axiom, + percentage_annotation_property, + Literal("{:.2f}".format(percentage)), + ) + ) + def save_rdf_graph( self, graph: Optional[Graph] = None,