Skip to content

Commit

Permalink
Add integration tests
Browse files Browse the repository at this point in the history
  • Loading branch information
bonicim committed Dec 23, 2022
1 parent 4ed2c18 commit 2bc6bc1
Show file tree
Hide file tree
Showing 6 changed files with 179 additions and 22 deletions.
2 changes: 1 addition & 1 deletion .github/workflows/run-tests.yml
Original file line number Diff line number Diff line change
Expand Up @@ -6,7 +6,7 @@ on:
paths: ['**.py', '**.yml']
pull_request:
types: [opened, synchronize, reopened]
paths: ['**.py']
paths: ['**.py', '**.yml', '**.yaml', '**.toml']
workflow_dispatch:
inputs:
triggeredBy:
Expand Down
6 changes: 0 additions & 6 deletions .pre-commit-config.yaml
Original file line number Diff line number Diff line change
Expand Up @@ -43,12 +43,6 @@ repos:
pass_filenames: false
files: pyproject.toml

- repo: https://github.com/pycqa/isort
rev: 5.6.4
hooks:
- id: isort
args: [--profile, black, --filter-files]

# For more information about YAPF, see https://github.com/google/yapf
- repo: https://github.com/craig8/mirrors-yapf
rev: b84f670025671a341d0afd2b06b877b195d65c0f # Use the sha / tag you want to point at
Expand Down
2 changes: 1 addition & 1 deletion README.md
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
# volttron-lib-bacnet-driver

![Passing?](https://github.com/VOLTTRON/volttron-lib-bacnet-driver/actions/workflows/run-tests.yml/badge.svg)
[![Passing?](https://github.com/VOLTTRON/volttron-lib-bacnet-driver/actions/workflows/run-tests.yml/badge.svg)](https://github.com/VOLTTRON/volttron-lib-bacnet-driver/actions/workflows/run-tests.yml)
[![pypi version](https://img.shields.io/pypi/v/volttron-lib-bacnet-driver.svg)](https://pypi.org/project/volttron-lib-bacnet-driver/)

# Prerequisites
Expand Down
5 changes: 1 addition & 4 deletions pyproject.toml
Original file line number Diff line number Diff line change
Expand Up @@ -2,9 +2,6 @@
requires = ["poetry-core>=1.2.2"]
build-backend = "poetry.core.masonry.api"

[tool.isort]
profile = "black"

[tool.poetry]
name = "volttron-lib-bacnet-driver"
version = "0.1.0"
Expand Down Expand Up @@ -43,7 +40,7 @@ coverage = "^6.3.2"
isort = "^5.10.1"
pytest-sugar = "^0.9.6"

[tool.poetry.group.documentation.dependencies]
[tool.poetry.group.docs.dependencies]
Sphinx = "^4.5.0"
sphinx-rtd-theme = "^1.0.0"

Expand Down
6 changes: 1 addition & 5 deletions tests/conftest.py
Original file line number Diff line number Diff line change
@@ -1,7 +1,3 @@
"""Configuration for the pytest test suite."""

from pathlib import Path

TESTS_DIR = Path(__file__).parent
TMP_DIR = TESTS_DIR / "tmp"
FIXTURES_DIR = TESTS_DIR / "fixtures"
from volttrontesting.fixtures.volttron_platform_fixtures import *
180 changes: 175 additions & 5 deletions tests/test_driver.py
Original file line number Diff line number Diff line change
@@ -1,8 +1,178 @@
"""Unit tests for volttron-lib-bacnet-driver"""
"""Integration tests for volttron-lib-bacnet-driver"""
import logging
import os
import socket

from volttron.driver.interfaces.bacnet.bacnet import BACnet, BaseInterface
import gevent
import pytest
from mock import MagicMock # type: ignore
from volttron.client.known_identities import CONFIGURATION_STORE, PLATFORM_DRIVER
from volttron.utils import setup_logging
from volttrontesting.platformwrapper import PlatformWrapper

setup_logging()
logger = logging.getLogger(__name__)

def test_driver():
driver = BACnet()
assert isinstance(driver, BaseInterface)
BACNET_DEVICE_TOPIC = "devices/bacnet"
BACNET_TEST_IP = "BACNET_TEST_IP"
skip_msg = f"Env var {BACNET_TEST_IP} not set. Please set the env var to the proper IP to run this integration test."

# apply skipif to all tests
pytestmark = pytest.mark.skipif(os.environ.get(BACNET_TEST_IP) is None, reason=skip_msg)


def test_scrape_all_should_succeed(bacnet_test_agent):
register_values = [
"3820a/Field Bus.3820A CHILLER.AHU-COIL-CHWR-T",
"3820a/Field Bus.3820A CHILLER.CHW-FLOW",
]
actual_values = bacnet_test_agent.vip.rpc.call(PLATFORM_DRIVER, "scrape_all",
"bacnet").get(timeout=10)
logger.info(f"Result of scrape_all: {actual_values}")

for register in register_values:
assert register in actual_values


def test_get_point_should_succeed(bacnet_test_agent):
register_values = [
"3820a/Field Bus.3820A CHILLER.AHU-COIL-CHWR-T",
"3820a/Field Bus.3820A CHILLER.CHW-FLOW",
]
for register in register_values:
async_res = bacnet_test_agent.vip.rpc.call(PLATFORM_DRIVER, "get_point", "bacnet",
register)
value = async_res.get()
logger.info(f"Value for point {register}: {value}")
assert isinstance(value, float)


@pytest.fixture(scope="module")
def bacnet_proxy_agent(volttron_instance: PlatformWrapper):
device_address = socket.gethostbyname(socket.gethostname() + ".local")
print(f"Device address for proxy agent for testing: {device_address}")
bacnet_proxy_agent_config = {
"device_address": device_address,
# below are optional; values are set to show configuration options; values use the default values
"max_apdu_length": 1024,
"object_id": 599,
"object_name": "Volttron BACnet driver",
"vendor_id": 5,
"segmentation_supported": "segmentedBoth",
}

# Installs volttron-bacnet-proxy from PyPi
bacnet_proxy_agent_uuid = volttron_instance.install_agent(
agent_dir="volttron-bacnet-proxy",
config_file=bacnet_proxy_agent_config,
)
gevent.sleep(1)
volttron_instance.start_agent(bacnet_proxy_agent_uuid)
assert volttron_instance.is_agent_running(bacnet_proxy_agent_uuid)

yield bacnet_proxy_agent_uuid

print("Teardown of bacnet_proxy_agent")
volttron_instance.stop_agent(bacnet_proxy_agent_uuid)


@pytest.fixture(scope="module")
def config_store_connection(volttron_instance: PlatformWrapper):
capabilities = [{"edit_config_store": {"identity": PLATFORM_DRIVER}}]
connection = volttron_instance.build_connection(peer=CONFIGURATION_STORE,
capabilities=capabilities)
gevent.sleep(1)

# Installs volttron-platform-driver from PyPi
# Start the platform driver agent which would in turn start the bacnet driver
platform_uuid = volttron_instance.install_agent(
agent_dir="volttron-platform-driver",
config_file={
"publish_breadth_first_all": False,
"publish_depth_first": False,
"publish_breadth_first": False,
},
start=True,
)
gevent.sleep(2) # wait for the agent to start and start the devices

yield connection

volttron_instance.stop_agent(platform_uuid)
volttron_instance.remove_agent(platform_uuid)
connection.kill()


@pytest.fixture(scope="module")
def config_store(config_store_connection):
# this fixture will setup a the BACnet driver that will communicate with a live BACnet device located at PNNL campus in Richland at the given device_address
device_address = os.environ.get(BACNET_TEST_IP)
if os.system("ping -c 1 " + device_address) != 0:
pytest.skip(f"BACnet device cannot be reached at {device_address} ")

registry_config = "bacnet_test.csv"
registry_string = f"""Reference Point Name,Volttron Point Name,Units,Unit Details,BACnet Object Type,Property,Writable,Index,Write Priority,Notes
3820a/Field Bus.3820A CHILLER.AHU-COIL-CHWR-T,3820a/Field Bus.3820A CHILLER.AHU-COIL-CHWR-T,degreesFahrenheit,-50.00 to 250.00,analogInput,presentValue,FALSE,3000741,,Primary CHW Return Temp
3820a/Field Bus.3820A CHILLER.CHW-FLOW,3820a/Field Bus.3820A CHILLER.CHW-FLOW,usGallonsPerMinute,-50.00 to 250.00,analogInput,presentValue,FALSE,3000744,,Chiller 1 CHW Flow"""

# registry config
config_store_connection.call(
"manage_store",
PLATFORM_DRIVER,
registry_config,
registry_string,
config_type="csv",
)

# driver config
driver_config = {
"driver_config": {
"device_address": device_address,
"device_id": 506892
},
"driver_type": "bacnet",
"registry_config": f"config://{registry_config}",
"timezone": "US/Pacific",
"interval": 15,
}

config_store_connection.call(
"manage_store",
PLATFORM_DRIVER,
BACNET_DEVICE_TOPIC,
driver_config,
)

yield config_store_connection

print("Wiping out store.")
config_store_connection.call("manage_delete_store", PLATFORM_DRIVER)
gevent.sleep(0.1)


@pytest.fixture(scope="module")
def bacnet_test_agent(bacnet_proxy_agent, config_store, volttron_instance):
test_agent = volttron_instance.build_agent(identity="test-agent")
assert test_agent.core.identity

# create a mock callback to use with a subscription to the driver's publish publishes
test_agent.poll_callback = MagicMock(name="poll_callback")

# subscribe to device topic results
test_agent.vip.pubsub.subscribe(
peer="pubsub",
prefix=BACNET_DEVICE_TOPIC,
callback=test_agent.poll_callback,
).get()

# give the test agent the capability to modify the platform_driver's config store
capabilities = {"edit_config_store": {"identity": PLATFORM_DRIVER}}
volttron_instance.add_capabilities(test_agent.core.publickey, capabilities)

# A sleep was required here to get the platform to consistently add the edit config store capability
gevent.sleep(1)

yield test_agent

print("In teardown method of query_agent")
test_agent.core.stop()

0 comments on commit 2bc6bc1

Please sign in to comment.