########################### Connecting to a AF Database ########################### To retrieve data from the PI Asset Framework, the :class:`~.AF.AFDatabase` object should be used. The following code connects to the default database on the default server, and prints its server name: .. code-block:: python import PIconnect as PI PI.load_SDK() with PI.AFDatabase() as database: print(database.server_name) The Asset Framework represents a hierarchy of elements, with attributes on the elements. The database has a dictionary of `children`, which you can loop over as follows: .. code-block:: python import PIconnect as PI PI.load_SDK() with PI.AFDatabase() as database: for root in database.children.values(): print("Root element: {r}".format(r=root)) The keys of the dictionary are the names of the elements inside. The following snippet first gets the first key in the dictionary, and the uses that to get the corresponding :class:`~.Asset.AFElement` from the dictionary. Then from this element its :class:`~.Asset.AFAttribute`'s are extracted: .. code-block:: python import PIconnect as PI PI.load_SDK() with PI.AFDatabase() as database: key = next(iter(database.children)) element = database.children[key] for attr in element.attributes: print(element.attributes[attr]) To get the data for the last 48 hours from a given attribute you need the :meth:`.AFAttribute.recorded_values` method: .. code-block:: python import PIconnect as PI PI.load_SDK() with PI.AFDatabase() as database: key = next(iter(database.children)) element = database.children[key] attribute = next(iter(element.attributes.values())) data = attribute.recorded_values("*-48h", "*") print(data) .. note:: Attributes on root elements within the database might not have meaningful summaries. To get a better result take a look at :ref:`finding_descendants` below. .. _finding_descendants: ************************************ Finding descendants in the hierarchy ************************************ Whilst it is possible to traverse the hierarchy one at a time, by using the :attr:`.AFElement.children` dictionaries, it is also possible to get a further descendant using the :meth:`.AFElement.descendant` method. Assuming the database has a root element called `Plant1` with a child element `Outlet`, the latter element could be accessed directly as follows: .. code-block:: python import PIconnect as PI PI.load_SDK() with PI.AFDatabase() as database: element = database.descendant(r"Plant1\Outlet") .. note:: Elements in the hierarchy are separated by a single backslash `\\`, use either raw strings (using the `r` prefix, as in the example above) or escape each backslash as `\\\\`. For a more flexible search mechanism, see :doc:`search`. .. _connect_piaf_database: **************************************** Connecting to other servers or databases **************************************** When no arguments are passed to the :class:`~.AF.AFDatabase` constructor, a connection is returned to the default database on the default server. It is possible to connect to other servers or databases, by passing the name of the server and database as arguments to the :class:`~.AF.AFDatabase` constructor. .. code-block:: python import PIconnect as PI PI.load_SDK() with PI.AFDatabase(server="ServerName", database="DatabaseName") as database: print(database.server_name) .. note:: It is also possible to specify only server or database. When only server is specified, a connection to the default database on that server is returned. Similarly, when only a database is specified, the connection is made to that database on the default server. A list of the available servers can be found using the :meth:`~.AF.AFDatabase.servers` classmethod. This is a dictionary, where the keys are the server names. To get the list of server names you can use the following code. .. code-block:: python import PIconnect as PI PI.load_SDK() print(list(PI.AFDatabase.servers().keys())) A list of the databases on a given server can be retrieved from the same :meth:`~.AF.AFDatabase.servers` attribute. Each item in the dictionary of servers is a dictionary with two items, ``server`` and ``databases``. The first contains the raw server object from the SDK, while the ``databases`` item is a dictionary of ``{name: object}`` pairs. So to get the databases for a given server you can use the following code: .. code-block:: python import PIconnect as PI PI.load_SDK() print(list(PI.AFDatabase.servers["ServerName"]["databases"].keys())) .. _piaf_tables: ********************************* Accessing tables in the PI AF SDK ********************************* It is possible to define custom SQL like tables in the PI AF SDK. These tables can be accessed using the :attr:`~.AF.AFDatabase.tables` attribute. This attribute is a dictionary of ``{name: table}`` pairs. The table can be loaded into a :class:`pandas.DataFrame` using the :attr:`AFTable.data` property: .. code-block:: python import PIconnect as PI PI.load_SDK() with PI.AFDatabase() as database: table = database.tables["MyTable"] df = table.data print(df)