Skip to main content

A Pythonic API for accessing YottaDB databases.

Project description

YDBPython

YDBPython provides a Pythonic API for accessing YottaDB databases.

Requirements

  1. Ubuntu Server 20.04 (or similar)
  2. Python > 3.6 (f-string and type annotation used), including the python3-dev package that contains Python.h
  3. libffi
  4. YottaDB 1.34 or later

Installation

  1. Install Ubuntu Server 20.04
  2. Install YottaDB per the Quick Start guide instructions or from source
  3. Install prerequisites:
    • Ubuntu/Debian: sudo apt install python3-dev python3-setuptools libffi-dev pkg-config
    • RHEL/CentOS: yum install gcc python3 python3-setuptools python3-devel libffi-devel pkg-config
    • Arch Linux: sudo yay -Sy python-{setuptools,pip} libffi pkg-config
  4. Set environment variables
    1. Set YottaDB environment variables: source $(pkg-config --variable=prefix yottadb)/ydb_env_set
    2. Optional: If YottaDB is built with Address Sanitization (ASAN) enabled, LD_PRELOAD and ASAN_OPTIONS must be set:
      • export ASAN_OPTIONS="detect_leaks=0:disable_coredump=0:unmap_shadow_on_exit=1:abort_on_error=1"
      • export LD_PRELOAD=$(gcc -print-file-name=libasan.so)
  5. Install:
    • Option 1: From PyPI:
      • Option 1: Install in venv:
        1. Enter directory where install is desired, e.g. cd my-python-project
        2. Install the python3-venv package:
          • Ubuntu/Debian: sudo apt install python3-venv
          • RHEL/CentOS: sudo yum install python3-virtualenv
          • Arch Linux: sudo yay -Sy install python3-virtualenv
        3. Create venv: python3 -m venv .venv
        4. Activate venv: source .venv/bin/activate
        5. Install into venv: pip install yottadb
      • Option 2: Install to user:
        • pip3 install yottadb --user
      • Option 3: Install globally:
        • sudo -E pip3 install yottadb
    • Option 2: From source:
      1. Get the code: git clone https://gitlab.com/YottaDB/Lang/YDBPython.git
      2. Enter code directory cd YDBPython/
      3. Run setup.py to install:
        • Option 1: Install in venv:
          1. Install the python3-venv package:
            • Ubuntu/Debian: sudo apt install python3-venv
            • RHEL/CentOS: sudo yum install python3-virtualenv
            • Arch Linux: sudo yay -Sy install python3-virtualenv
          2. Create venv: python3 -m venv .venv
          3. Activate venv: source .venv/bin/activate
          4. Install into venv: python setup.py install
        • Option 2: Install to user:
          • python3 setup.py install --user
        • Option 3: Install globally (not suggested):
          • sudo -E python3 setup.py install

Testing

To run YDBPython's test suite with a YDBPython source installation:

  1. Enter YDBPython directory containing code repository, e.g. cd YDBPython/
  2. Install pytest, pytest-order, and psutil
    1. If pip for python3 is not installed do so:
      • Ubuntu/Debian: sudo apt install python3-pip
      • RHEL/CentOS: sudo yum install python3-pip
      • Arch Linux: sudo yay -Sy install python3-pip
    2. Use pip to install pytest, pytest-order, psutil
      • Option 1: install into venv
        1. Activate venv if it is not already: source .venv/bin/activate
        2. Install: pip install pytest pytest-order psutil
      • Option 2: install for user: pip3 install --user pytest pytest-order psutil
      • Option 3: install globally (not suggested): sudo pip3 install pytest pytest-order psutil
  3. Run the tests:
    • Option 1: in venv: python -m pytest
    • Option 2: with user or global installation: python3 -m pytest
  4. Optional: Cleanup between tests:
    • When making changes to code between test runs, some cleanup may be needed to prevent new changes being ignored due to Python caching. To clean up these files: for artifact in $(cat .gitignore); do rm -rf $artifact; done. Note that this will delete all files listed in .gitignore, including core files. If these or any other such files need to be retained, move or rename them before running the aforementioned command.

Basic Example Usage

import yottadb

# Create Key objects for conveniently accessing and manipulating database nodes
key1 = yottadb.Key('^hello')  # Create a key referencing the global variable '^hello'

print(f"{key1}: {key1.value}")  # Display current value of '^hello'
key1.value = b'Hello world!'  # Set '^hello' to 'Hello world!'
print(f"{key1}: {key1.value}")

key2 = yottadb.Key('^hello')['cowboy']  # Add a 'cowboy' subscript to the global variable '^hello', creating a new key
key2.value = 'Howdy partner!'  # Set '^hello('cowboy') to 'Howdy partner!'
print(f"{key2}: {key2.value}")

key3 = yottadb.Key('^hello')['chinese']  # Add a second subscript to '^hello', creating a third key
key3.value = bytes('你好世界!', encoding="utf-8")  # The value can be set to anything that can be encoded to `bytes`
print(key3, str(key3.value, encoding="utf-8"))  # Returned values are `bytes` objects, and so may need to be encoded

for subscript in key1.subscripts:  # Loop through all the subscripts of a key
    sub_key = key1[subscript]
    print(f"{sub_key}: {sub_key.value}")

key1.delete_node()  # Delete the value of '^hello', but not any of its child nodes

print(f"{key1}: {key1.value}")  # No value is printed
for subscript in key1.subscripts:  # The values of the child nodes are still in the database
    sub_key = key1[subscript]
    print(f"{sub_key}: {sub_key.value}")

key1.value = 'Hello world!'   # Reset the value of '^hello'
print(f"{key1}: {key1.value}")  # Prints the value
key1.delete_tree() # Delete both the value at the '^hello' node and all of it's children
print(f"{key1}: {key1.value}")  # Prints no value
for subscript in key1.subscripts:  # Loop terminates immediately and displays no subscripts
    sub_key = key1[subscript]
    print(sub_key, sub_key.value)

# Database transactions are also available
@yottadb.transaction
def simple_transaction(value):
    # Set values directly with the set() function
    yottadb.set('test1', value=value)  # Set the local variable 'test1' to the given value
    yottadb.set('test2', value=value)  # Set the local variable 'test2' to the given value
    condition_a = False
    condition_b = False
    if condition_a:
        # When a yottadb.YDBTPRollback exception is raised YottaDB will rollback the transaction
        # and then propagate the exception to the calling code.
        raise yottadb.YDBTPRollback("reason for the rollback")
    elif condition_b:
        # When a yottadb.YDBTPRestart exception is raised YottaDB will call the transaction again.
        # Warning: This code is intentionally simplistic. An infinite loop will occur
        #           if yottadb.YDBTPRestart is continually raised
        raise yottadb.YDBTPRestart()
    else:
        return yottadb.YDB_OK  # Success, transaction will be committed


simple_transaction(b'test', db)
print(f"{db[b'test1']}: {db[b'test1'].value}")
print(f"{db[b'test2']}: {db[b'test2'].value}")

Frequently Asked Questions

Does YDBPython support multi-threading?

No, YDBPython does not support multithreading. This is due to the limitations of the Python Global Interpreter Lock for CPU-intensive multithreading. For background, see the following resources:

Accordingly, the Python threading and multithreading should be avoided when developing applications with YDBPython. However, YDBPython does support multiprocessing and may be safely used with the Python multiprocessing library for parallelism. For an example of multiprocessing usage, see tests/test_threeenp1.py.

Project details


Download files

Download the file for your platform. If you're not sure which to choose, learn more about installing packages.

Source Distribution

yottadb-1.0.0.tar.gz (39.7 kB view hashes)

Uploaded Source

Supported by

AWS AWS Cloud computing and Security Sponsor Datadog Datadog Monitoring Fastly Fastly CDN Google Google Download Analytics Microsoft Microsoft PSF Sponsor Pingdom Pingdom Monitoring Sentry Sentry Error logging StatusPage StatusPage Status page