Source code for vfb_connect.cross_server_tools

import warnings

from .owl.owlery_query_tools import OWLeryConnect
from .neo.neo4j_tools import Neo4jConnect, re, dict_cursor
from .neo.query_wrapper import QueryWrapper
from .default_servers import get_default_servers
import pandas as pd


[docs]def gen_short_form(iri): """Generate short_form (string) from an iri string iri: An iri string""" return re.split('/|#', iri)[-1]
[docs]def dequote(string): qm = re.match("^'(.+)'$", string) if qm: return qm.group(1) else: return string
[docs]class VfbConnect: """API wrapper class. By default this wraps connections to the more basal API endpoints (OWL, Neo4j). Top level methods combined semantic queries that range across VFB content with neo4j queries, returning detailed metadata about anatomical classes and individuals that fulfill these queries. Methods allowing direct queries cypher queries of the production Neo4j are available under `nc` Methods for querying Neo4j with arbitrary lists of identifiers to return rich metadata or mappings to external IDs are available under `neo_query_wrapper`. Direct access OWL queries, returning identifiers only, are available via methods under `oc` Example semantic queries (OWL class expressions). Note quoting scheme (outer `"` + single quotes for entities). "'GABAergic neuron'" "'GABAeric neuron' that 'overlaps' some 'antennal lobe'" """ def __init__(self, neo_endpoint=get_default_servers()['neo_endpoint'], neo_credentials=get_default_servers()['neo_credentials'], owlery_endpoint=get_default_servers()['owlery_endpoint'], lookup_prefixes=('FBbt', 'VFBexp', 'VFBext')): """ VFB connect constructor. All args optional. With no args wraps connectsions to default public servers. :neo_endpoint: Specify a neo4j REST endpoint. :neo_credentials: Specify credential for Neo4j Rest endpoint. :owlery_endpoint: specify owlery server REST endpoint. :lookup_prefixes: A list of id prefixes to use for rolling name:ID lookups.""" connections = { 'neo': { "endpoint": neo_endpoint, "usr": neo_credentials[0], "pwd": neo_credentials[1] } } self.nc = Neo4jConnect(**connections['neo']) self.neo_query_wrapper = QueryWrapper(**connections['neo']) self.lookup = self.nc.get_lookup(limit_by_prefix=lookup_prefixes) self.oc = OWLeryConnect(endpoint=owlery_endpoint, lookup=self.lookup) self.vfb_base = "https://v2.virtualflybrain.org/org.geppetto.frontend/geppetto?id="
[docs] def lookup_id(self, key, return_short_form=True): if key in self.lookup.keys(): out = self.lookup[key] if return_short_form: return out.replace(':', '_') else: return out else: raise ValueError("Unrecognised value: %s" % str(key))
# """[Summary] # # :param [ParamName]: [ParamDescription], defaults to [DefaultParamVal] # :type [ParamName]: [ParamType](, optional) # ... # :raises [ErrorType]: [ErrorDescription] # ... # :return: [ReturnDescription] # :rtype: [ReturnType] # """
[docs] def get_terms_by_region(self, region, cells_only=False, verbose=False, query_by_label=True, summary=False): """Generate TermInfo reports for all terms relevant to annotating some specific region, optionally limited to cells. :param region: The name (rdfs:label) of brain region (or CURIE style ID if query_by_label is False) :param cells_only: Optional. Limits query to cell type if `True`. Defaults to `False` :param verbose: Optional. :param query_by_label: Optional (see region). Default `True` :param summary: Optional. Returns summary reports if true. Default `False` :return: Returns a list of terms as nested python data structures following VFB_json or a summary_report_json :rtype: list of VFB_json or summary_report_json """ preq = '' if cells_only: preq = "'cell' that " owl_query = preq + "'overlaps' some '%s'" % region if verbose: print("Running query: %s" % owl_query) terms = self.oc.get_subclasses(owl_query, query_by_label=query_by_label) if verbose: print("Found: %d terms" % len(terms)) return self.neo_query_wrapper.get_type_TermInfo(list(map(gen_short_form, terms)), summary=summary)
[docs] def get_subclasses(self, class_expression, query_by_label=True, direct=False, summary=False): """Generate JSON report of all subclasses of class_expression. :param class_expression: A valid OWL class expression, e.g. the name of a class. :param query_by_label: Optional. If false, class_expression takes IDs instead of labels Default `True` :param direct: Return direct subclasses only. Default `False` :param summary: Optional. Returns summary reports if true. Default `False` :return: Returns a list of terms as nested python data structures following VFB_json or a summary_report_json :rtype: list of VFB_json or summary_report_json """ if not re.search("'", class_expression): class_expression = "'" + class_expression + "'" terms = self.oc.get_subclasses("%s" % class_expression, direct=direct, query_by_label=query_by_label) return self.neo_query_wrapper.get_type_TermInfo(list(map(gen_short_form, terms)), summary=summary)
[docs] def get_superclasses(self, class_expression, query_by_label=True, direct=False, summary=False): """Generate JSON report of all superclasses of class_expression. :param class_expression: A valid OWL class expression, e.g. the name of a class. :param query_by_label: Optional. If false, class_expression takes IDs instead of labels Default `True` :param direct: Return direct superclass only. Default `False` :param summary: Optional. Returns summary reports if true. Default `False` :return: Returns a list of terms as nested python data structures following VFB_json or a summary_report_json :rtype: list of VFB_json or summary_report_json """ if not re.search("'", class_expression): class_expression = "'" + class_expression + "'" terms = self.oc.get_superclasses("%s" % class_expression, query_by_label=query_by_label, direct=direct) return self.neo_query_wrapper.get_type_TermInfo(list(map(gen_short_form, terms)), summary=summary)
[docs] def get_instances(self, class_expression, query_by_label=True, summary=False): """Generate JSON report of all instances of class_expression. Instances are specific examples of a type/class, e.g. a neuron of type DA1 adPN from the FAFB_catmaid database. :param class_expression: A valid OWL class expression, e.g. the name of a class. :param query_by_label: Optional. If false, class_expression takes IDs instead of labels Default `True` :param summary: Optional. Returns summary reports if true. Default `False` :return: Returns a list of terms as nested python data structures following VFB_json or a summary_report_json :rtype: list of VFB_json or summary_report_json """ if not re.search("'", class_expression): if query_by_label: class_expression = self.lookup[class_expression].replace(':', '_') out = self.neo_query_wrapper._get_anatomical_individual_TermInfo_by_type(class_expression, summary=summary) else: terms = self.oc.get_instances("%s" % class_expression, query_by_label=query_by_label) out = self.neo_query_wrapper.get_anatomical_individual_TermInfo(list(map(gen_short_form, terms)), summary=summary) return out
def _get_neurons_connected_to(self, neuron, weight, direction, classification=None, query_by_label=True, return_dataframe=True): instances = [] directions = ['upstream', 'downstream'] if not (direction in directions): raise Exception(ValueError) # Needs improving if classification: instances = self.oc.get_instances(classification, query_by_label=query_by_label) cypher_query = 'MATCH (upstream:Neuron)-[r:synapsed_to]->(downstream:Neuron) ' \ 'WHERE r.weight[0] >= %d ' % weight if query_by_label: cypher_query += 'AND %s.label = "%s" ' % (direction, neuron) else: cypher_query += 'AND %s.short_form = "%s" ' % (direction, neuron) if classification and instances: directions.remove(direction) cypher_query += "AND %s.iri IN %s " % (directions[0], str(instances)) cypher_query += "RETURN upstream.short_form as query_neuron_id, upstream.label as query_neuron_name, " \ "r.weight[0] as weight, " \ "downstream.short_form as target_neuron_id, downstream.label as target_neuron_name" r = self.nc.commit_list([cypher_query]) dc = dict_cursor(r) if return_dataframe: return pd.DataFrame.from_records(dc) else: return dc
[docs] def get_similar_neurons(self, neuron, similarity_score='NBLAST_score', return_dataframe=True): """Get JSON report of individual neurons similar to input neuron :param neuron: :param similarity_score: Optionally specify similarity score to chose :param return_dataframe: Returns pandas dataframe if true, otherwise returns list of dicts. :return: list of similar neurons (id, label, tags, source (db) id, accession_in_source) + similarity score. :rtype: pandas.DataFrame or list of dicts """ query = "MATCH (c1:Class)<-[:INSTANCEOF]-(n1)-[r:has_similar_morphology_to]-(n2)-[:INSTANCEOF]->(c2:Class) " \ "WHERE n1.short_form = '%s' " \ "WITH c1, n1, r, n2, c2 " \ "OPTIONAL MATCH (n1)-[dbx1:database_cross_reference]->(s1:Site), " \ "(n2)-[dbx2:database_cross_reference]->(s2:Site) " \ "WHERE s1.is_data_source and s2.is_data_source " \ "RETURN DISTINCT n2.short_form AS id, r.NBLAST_score[0] AS NBLAST_score, n2.label AS label, " \ "COLLECT(c2.label) AS tags, s2.short_form AS source_id, dbx2.accession[0] AS accession_in_source " \ "ORDER BY %s DESC""" % (neuron, similarity_score) dc = self.neo_query_wrapper._query(query) if return_dataframe: return pd.DataFrame.from_records(dc) else: return dc
[docs] def get_neurons_downstream_of(self, neuron, weight, classification=None, query_by_label=True, return_dataframe=True): """Get all neurons downstream of individual `neuron` :param neuron: the name or id of a particular neuron (dependent on query_by_label setting) :param weight: limit returned neurons to those with connected by >= weight synapses :classification: optionally specify classification of downstream neuron using a class expression e.g. `MBON` :param query_by_label: query neuron may be specified with a label if true :param return_dataframe: Returns pandas dataframe if true, otherwise returns list of dicts. """ return self._get_neurons_connected_to(neuron=neuron, weight=weight, direction='upstream', classification=classification, query_by_label=query_by_label, return_dataframe=return_dataframe)
[docs] def get_neurons_upstream_of(self, neuron, weight, classification=None, query_by_label=True, return_dataframe=True): """"Get all neurons upstream of individual `neuron` :param neuron: the name or id of a particular neuron (dependent on query_by_label setting) :param weight: limit returned neurons to those with connected by >= weight synapses :classification: optionally specify classification of upstream neuron using a class expression e.g. `MBON` :param query_by_label: query neuron may be specified with a label if true :param return_dataframe: Returns pandas dataframe if true, otherwise returns list of dicts. """ return self._get_neurons_connected_to(neuron=neuron, weight=weight, direction='downstream', classification=classification, query_by_label=query_by_label, return_dataframe=return_dataframe)
[docs] def get_connected_neurons_by_type(self, upstream_type, downstream_type, weight, query_by_label=True, return_dataframe=True): """Get all synaptic connections between individual neurons of `upstream_type` and `downstream_type` where synapse count >= `weight`. Warning: Does not support Class Expressions. :parm upstream_type: The upstream neuron type (e.g. 'GABAeric neuron'). :param downstream_type: The upstream neuron type (e.g. 'Descending neuron'). :param query_by_label: specify neuron type by label if `True` (default) or by short_form id if `False` :param return_dataframe: Returns pandas dataframe if true, otherwise returns list of dicts. """ # Note - chose not to do this with class expressions to avoid poor performance and blowing up results. # This might be confusing tough, given behavior of other, similar methods. # Might be better to refactor to work out if query is class expression or class & funnel query method # accordingly. if query_by_label: upstream_type = self.lookup_id(dequote(upstream_type)) downstream_type = self.lookup_id(dequote(downstream_type)) # upstream_instances = self.oc.get_instances(upstream_type, query_by_label=query_by_label) cypher_query = "MATCH (up:Class:Neuron)<-[:SUBCLASSOF|INSTANCEOF*..]-(n1:Neuron:Individual) " \ 'WHERE up.short_form = "%s" ' % upstream_type cypher_query += "MATCH (n1)-[r:synapsed_to]->(n2:Neuron:Individual) " \ "WHERE r.weight[0] >= %d " % weight cypher_query += "MATCH (n2)-[:SUBCLASSOF|INSTANCEOF*..]->(down:Class:Neuron) " \ 'WHERE down.short_form = "%s" ' % downstream_type cypher_query += "MATCH (c1:Class)<-[:INSTANCEOF]-(n1), (c2:Class)<-[:INSTANCEOF]-(n2) " \ "OPTIONAL MATCH (n1)-[r1:database_cross_reference]->(s1:Site) " \ "WHERE exists(s1.is_data_source) AND s1.is_data_source = True " \ "OPTIONAL MATCH (n2)-[r2:database_cross_reference]->(s2:Site) " \ "WHERE exists(s1.is_data_source) AND s2.is_data_source = True " \ "RETURN n1.short_form as upstream_neuron_id, n1.label as upstream_neuron_name, " \ "r.weight[0] as weight, n2.short_form as downstream_neuron_id, " \ "n2.label as downstream_neuron_name, " \ "apoc.text.join(collect(distinct c1.label),'|') AS upstream_class, " \ "apoc.text.join(collect(distinct c2.label),'|') as downstream_class, " \ "s1.short_form AS up_data_source, r1.accession[0] as up_accession," \ "s2.short_form AS down_source, r2.accession[0] AS down_accession" # print(cypher_query) r = self.nc.commit_list([cypher_query]) dc = dict_cursor(r) if return_dataframe: return pd.DataFrame.from_records(dc) else: return dc
[docs] def get_instances_by_dataset(self, dataset, summary=False): """Get JSON report of all individuals in a dataset :param dataset: dataset ID :param summary: Optional. Returns summary reports if true. Default `False` :return: Returns a list of terms as nested python data structures following VFB_json or a summary_report_json :rtype: list of VFB_json or summary_report_json """ if dataset: query = "MATCH (ds:DataSet)<-[:has_source]-(i:Individual) " \ "WHERE ds.short_form = '%s' " \ "RETURN collect(i.short_form) as inds" % dataset dc = self.neo_query_wrapper._query(query) # Would better to use the original column oriented return! return self.neo_query_wrapper.get_anatomical_individual_TermInfo(dc[0]['inds'], summary=summary)
[docs] def get_images_by_type(self, class_expression, template, image_folder, image_type='swc', query_by_label=True, direct=False, stomp=False): """Download all images of individuals specified by a class expression, e.g. all images of the nodulus or of MBON01. :param class_expression: A valid OWL class expression, e.g. the name or symbol of a type of neuron (MBON01) :param template: template name :param image_folder: folder to save image files & manifest to. :param image_type: image type (file extension) :param stomp: Overwrite image_folder if already exists. :return: Manifest as Pandas DataFrame """ if not re.search("'", class_expression): class_expression = "'" + class_expression + "'" instances = self.oc.get_instances(class_expression, query_by_label=query_by_label, direct=direct) return self.neo_query_wrapper.get_images([gen_short_form(i) for i in instances], template=template, image_folder=image_folder, image_type=image_type, stomp=stomp)
[docs] def get_gene_function_filters(self): """Get list of all gene function labels. :return: List of unique gene function labels in alphabetical order. :rtype: list """ query = ("MATCH (g:Gene) RETURN DISTINCT apoc.coll.subtract(labels(g), " "['Class', 'Entity', 'hasScRNAseq', 'Feature', 'Gene']) AS gene_labels") result = self.neo_query_wrapper._query(query) labels = [] for r in result: labels.extend(r['gene_labels']) labels = sorted(list(set(labels))) return labels
[docs] def get_transcriptomic_profile(self, cell_type, gene_type=False, return_dataframe=True): """Get gene expression data for a given cell_type. Returns a DataFrame of gene expression data for clusters of cells annotated as cell_type (or subtypes). Can optionally restrict to a gene_type - these can be retrieved by running get_gene_function_filters. If no data is found, returns False. :param cell_type: The ID, name or symbol of a class in the Drosophila Anatomy Ontology (FBbt). :param gene_type: Optional. A gene function label - these can be retrieved by running get_gene_function_filters(). :return: DataFrame with gene expression data for clusters of cells annotated as cell_type (or subtypes). :rtype: DataFrame """ try: cell_type_short_form = self.lookup[cell_type].replace(':', '_') except KeyError: if cell_type.replace('_', ':') in self.lookup.values(): cell_type_short_form = cell_type.replace(':', '_') else: raise KeyError("cell_type must be a valid ID, label or symbol from the Drosophila Anatomy Ontology") if not cell_type_short_form.startswith('FBbt'): raise KeyError("cell_type must be a valid ID, label or symbol from the Drosophila Anatomy Ontology") if gene_type: if gene_type not in self.get_gene_function_filters(): raise KeyError("gene_type must be a valid gene function label, try running get_gene_function_filters()") else: gene_label = ':' + gene_type else: gene_label = '' query = ("MATCH (g:Gene:Class%s)<-[e:expresses]-(clus:Cluster:Individual)-" "[:composed_primarily_of]->(c2:Class)-[:SUBCLASSOF*0..]->(c1:Neuron:Class) " "WHERE c1.short_form = '%s' " "MATCH (clus)-[:part_of]->()-[:has_part]->(sa:Sample:Individual) " "OPTIONAL MATCH (sa)-[:part_of]->(sex:Class) " "WHERE sex.short_form IN ['FBbt_00007011', 'FBbt_00007004'] " "OPTIONAL MATCH (sa)-[:overlaps]->(tis:Class:Anatomy) " "OPTIONAL MATCH (clus)-[:has_source]->(ds:DataSet:Individual) " "OPTIONAL MATCH (ds)-[:has_reference]->(p:pub:Individual) " "OPTIONAL MATCH (ds)-[dbx:database_cross_reference]->(s:Site:Individual) " "RETURN DISTINCT c2.label AS cell_type, c2.short_form AS cell_type_id, " "sex.label AS sample_sex, COLLECT(tis.label) AS sample_tissue, " "p.miniref[0] as ref, g.label AS gene, g.short_form AS gene_id, " "apoc.coll.subtract(labels(g), ['Class', 'Entity', 'hasScRNAseq', 'Feature', 'Gene']) AS function, " "e.expression_extent[0] as extent, toFloat(e.expression_level[0]) as level order by cell_type, g.label" % (gene_label, cell_type_short_form)) r = self.nc.commit_list([query]) dc = dict_cursor(r) if return_dataframe: return pd.DataFrame.from_records(dc) else: return dc