Source code for pybel.io.bel_commons_client

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

"""Transport functions for `BEL Commons <https://github.com/bel-commons/bel-commons>`_.

BEL Commons is a free, open-source platform for hosting BEL content. Because it was originally
developed and published in an academic capacity at Fraunhofer SCAI, a public instance can be
found at https://bel-commons-dev.scai.fraunhofer.de. However, this instance is only supported
out of posterity and will not be updated. If you would like to host your own instance of
BEL Commons, there are instructions on its GitHub page.
"""

import logging
from typing import Optional

import pystow
import requests

from .nodelink import from_nodelink, to_nodelink
from ..struct.graph import BELGraph
from ..version import get_version

__all__ = [
    "to_bel_commons",
    "from_bel_commons",
]

logger = logging.getLogger(__name__)

RECIEVE_ENDPOINT = "/api/receive/"
GET_ENDPOINT = "/api/network/{}/export/nodelink"


def _get_host() -> Optional[str]:
    """Find the host with :func:`pystow.get_config`.

    Has two possibilities:

    1. The PyBEL config entry ``PYBEL_REMOTE_HOST``, loaded in :mod:`pybel.constants`
    2. The environment variable ``PYBEL_REMOTE_HOST``
    """
    return pystow.get_config("pybel", "remote_host")


def _get_user() -> Optional[str]:
    return pystow.get_config("pybel", "remote_user")


def _get_password() -> Optional[str]:
    return pystow.get_config("pybel", "remote_password")


[docs]def to_bel_commons( graph: BELGraph, host: Optional[str] = None, user: Optional[str] = None, password: Optional[str] = None, public: bool = True, ) -> requests.Response: """Send a graph to the receiver service and returns the :mod:`requests` response object. :param graph: A BEL graph :param host: The location of the BEL Commons server. Alternatively, looks up in PyBEL config with ``PYBEL_REMOTE_HOST`` or the environment as ``PYBEL_REMOTE_HOST``. :param user: Username for BEL Commons. Alternatively, looks up in PyBEL config with ``PYBEL_REMOTE_USER`` or the environment as ``PYBEL_REMOTE_USER`` :param password: Password for BEL Commons. Alternatively, looks up in PyBEL config with ``PYBEL_REMOTE_PASSWORD`` or the environment as ``PYBEL_REMOTE_PASSWORD`` :param public: Should the network be made public? :return: The response object from :mod:`requests` """ if host is None: host = _get_host() logger.debug("using host: %s", host) if user is None: user = _get_user() if user is None: raise ValueError("no user found") if password is None: password = _get_password() if password is None: raise ValueError("no password found") url = host.rstrip("/") + RECIEVE_ENDPOINT response = requests.post( url, json=to_nodelink(graph), headers={ "content-type": "application/json", "User-Agent": "PyBEL v{}".format(get_version()), "bel-commons-public": "true" if public else "false", }, auth=(user, password), ) logger.debug("received response: %s", response) return response
[docs]def from_bel_commons(network_id: int, host: Optional[str] = None) -> BELGraph: """Retrieve a public network from BEL Commons. In the future, this function may be extended to support authentication. :param network_id: The BEL Commons network identifier :param host: The location of the BEL Commons server. Alternatively, looks up in PyBEL config with ``PYBEL_REMOTE_HOST`` or the environment as ``PYBEL_REMOTE_HOST``. :raises: ValueError if host configuration can not be found """ if host is None: host = _get_host() if host is None: raise ValueError("host not specified in arguments, PyBEL configuration, or environment.") url = host + GET_ENDPOINT.format(network_id) res = requests.get(url) graph_json = res.json() graph = from_nodelink(graph_json) return graph