1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495 |
- # SPDX-License-Identifier: Apache-2.0
- # Copyright 2020 Contributors to OpenLEADR
- # Licensed under the Apache License, Version 2.0 (the "License");
- # you may not use this file except in compliance with the License.
- # You may obtain a copy of the License at
- # http://www.apache.org/licenses/LICENSE-2.0
- # Unless required by applicable law or agreed to in writing, software
- # distributed under the License is distributed on an "AS IS" BASIS,
- # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- # See the License for the specific language governing permissions and
- # limitations under the License.
- from aiohttp import web
- from openleadr.service import EventService, PollService, RegistrationService, ReportService, OptService, VTNService
- from openleadr.messaging import create_message, parse_message
- from openleadr.utils import certificate_fingerprint
- from functools import partial
- class OpenADRServer:
- _MAP = {'on_created_event': EventService,
- 'on_request_event': EventService,
- 'on_register_report': ReportService,
- 'on_create_report': ReportService,
- 'on_created_report': ReportService,
- 'on_request_report': ReportService,
- 'on_update_report': ReportService,
- 'on_poll': PollService,
- 'on_query_registration': RegistrationService,
- 'on_create_party_registration': RegistrationService,
- 'on_cancel_party_registration': RegistrationService}
- def __init__(self, vtn_id, cert=None, key=None, passphrase=None, fingerprint_lookup=None):
- """
- Create a new OpenADR VTN (Server).
- :param vtn_id string: An identifier string for this VTN. This is how you identify yourself to the VENs that talk to you.
- :param cert string: Path to the PEM-formatted certificate file that is used to sign outgoing messages
- :param key string: Path to the PEM-formatted private key file that is used to sign outgoing messages
- :param passphrase string: The passphrase used to decrypt the private key file
- :param fingerprint_lookup callable: A callable that receives a ven_id and should return the registered fingerprint for that VEN.
- You should receive these fingerprints outside of OpenADR and configure them manually.
- """
- self.app = web.Application()
- self.services = {'event_service': EventService(vtn_id),
- 'report_service': ReportService(vtn_id),
- 'poll_service': PollService(vtn_id),
- 'opt_service': OptService(vtn_id),
- 'registration_service': RegistrationService(vtn_id)}
- self.app.add_routes([web.post(f"/OpenADR2/Simple/2.0b/{s.__service_name__}", s.handler) for s in self.services.values()])
- # Configure message signing
- if cert and key:
- with open(cert, "rb") as file:
- cert = file.read()
- with open(key, "rb") as file:
- key = file.read()
- print("*" * 80)
- print("Your VTN Certificate Fingerprint is", certificate_fingerprint(cert))
- print("Please deliver this fingerprint to the VTN you are connecting to.")
- print("You do not need to keep this a secret.")
- print("*" * 80)
- VTNService._create_message = partial(create_message, cert=cert, key=key, passphrase=passphrase)
- VTNService._parse_message = partial(parse_message, fingerprint_lookup=fingerprint_lookup)
- self.__setattr__ = self.add_handler
- def run(self):
- """
- Starts the asyncio-loop and runs the server in it. This function is
- blocking. For other ways to run the server in a more flexible context,
- please refer to the `aiohttp documentation
- <https://docs.aiohttp.org/en/stable/web_advanced.html#aiohttp-web-app-runners>`_.
- """
- web.run_app(self.app)
- def add_handler(self, name, func):
- """
- Add a handler to the OpenADRServer.
- :param name string: The name for this handler. Should be one of: on_created_event, on_request_event, on_register_report, on_create_report, on_created_report, on_request_report, on_update_report, on_poll, on_query_registration, on_create_party_registration, on_cancel_party_registration.
- :param func coroutine: A coroutine that handles this event. It receives the message, and should return the contents of a response.
- """
- print("Called add_handler", name, func)
- if name in self._MAP:
- setattr(self._MAP[name], name, staticmethod(func))
- else:
- raise NameError(f"Unknown handler {name}. Correct handler names are: {self._MAP.keys()}")
|