Source code for biokeen.convert.io

# -*- coding: utf-8 -*-

"""Input and output for BEL conversion."""

import itertools as itt
import json
import logging
from collections import Counter
from typing import Dict, Optional, Tuple

import pandas as pd
from tqdm import tqdm

from pybel import BELGraph
from pybel.dsl import BaseEntity
from .converters import (
    AssociationConverter, CorrelationConverter, DecreasesAmountConverter, DrugIndicationConverter,
    DrugSideEffectConverter, EquivalenceConverter, IncreasesAmountConverter, IsAConverter,
    ListComplexHasComponentConverter, MiRNADecreasesExpressionConverter, MiRNADirectlyDecreasesExpressionConverter,
    NamedComplexHasComponentConverter, PartOfNamedComplexConverter, ProteinPartOfBiologicalProcess,
    RegulatesActivityConverter, RegulatesAmountConverter, SubprocessPartOfBiologicalProcess,
)
from ..constants import EMOJI

__all__ = [
    'to_pykeen_path',
    'to_pykeen_df',
    'get_pykeen_summary',
    'to_pykeen_summary_path',
    'get_triple',
]

logger = logging.getLogger(__name__)


[docs]def to_pykeen_path(df: pd.DataFrame, path: str) -> bool: """Write the relationships in the BEL graph to a KEEN TSV file. If you have a BEL graph, first do: >>> from biokeen.convert import to_pykeen_df, to_pykeen_path >>> graph = ... # Something from PyBEL >>> df = to_pykeen_df(graph) >>> to_pykeen_path(df, 'graph.keen.tsv') """ if len(df.index) == 0: return False df.to_csv(path, sep='\t', index=None, header=None) return True
def get_pykeen_summary(df: pd.DataFrame) -> Dict: """Summarize a KEEN dataframe.""" entity_count = Counter(itt.chain(df[df.columns[0]], df[df.columns[2]])) return { 'namespaces': Counter( element.split(':')[0] for element in itt.chain(df[df.columns[0]], df[df.columns[2]]) ), 'entities': len(entity_count), 'relations': len(df.index), } def to_pykeen_summary_path(df: pd.DataFrame, path: str, indent=2, **kwargs): """Write the summary of a KEEN dataframe to a file.""" with open(path, 'w') as file: json.dump(get_pykeen_summary(df), file, indent=indent, **kwargs)
[docs]def to_pykeen_df(graph: BELGraph, use_tqdm: bool = True) -> pd.DataFrame: """Get a DataFrame representing the triples.""" it = graph.edges(keys=True) if use_tqdm: it = tqdm(it, total=graph.number_of_edges(), desc=f'{EMOJI} preparing TSV') triples = ( get_triple(graph, u, v, key) for u, v, key in it ) # clean duplicates and Nones triples = list(sorted({triple for triple in triples if triple is not None})) return pd.DataFrame(triples, columns=['subject', 'predicate', 'object'])
def get_triple(graph: BELGraph, u: BaseEntity, v: BaseEntity, key: str) -> Optional[Tuple[str, str, str]]: # noqa: C901 """Get the triples' strings that should be written to the file.""" data = graph[u][v][key] # order is important converters = [ NamedComplexHasComponentConverter, ListComplexHasComponentConverter, PartOfNamedComplexConverter, SubprocessPartOfBiologicalProcess, ProteinPartOfBiologicalProcess, RegulatesActivityConverter, MiRNADecreasesExpressionConverter, MiRNADirectlyDecreasesExpressionConverter, IsAConverter, EquivalenceConverter, CorrelationConverter, AssociationConverter, DrugIndicationConverter, DrugSideEffectConverter, RegulatesAmountConverter, IncreasesAmountConverter, DecreasesAmountConverter, ] for converter in converters: if converter.predicate(u, v, key, data): return converter.convert(u, v, key, data) logger.warning(f'{EMOJI} unhandled: {graph.edge_to_bel(u, v, data)}') return None