blob: c9550187e2805af13c818327b22057738effcb73 [file] [log] [blame]
#
# Copyright 2017 the original author or authors.
#
# 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.
#
import structlog
from twisted.internet import reactor
from voltha.extensions.omci.database.mib_db_dict import MibDbVolatileDict
from voltha.extensions.omci.database.mib_db_ext import MibDbExternal
from voltha.extensions.omci.state_machines.mib_sync import MibSynchronizer
from voltha.extensions.omci.tasks.mib_upload import MibUploadTask
from voltha.extensions.omci.tasks.get_mds_task import GetMdsTask
from voltha.extensions.omci.tasks.mib_resync_task import MibResyncTask
from voltha.extensions.omci.tasks.mib_reconcile_task import MibReconcileTask
from voltha.extensions.omci.tasks.sync_time_task import SyncTimeTask
from voltha.extensions.omci.state_machines.alarm_sync import AlarmSynchronizer
from voltha.extensions.omci.tasks.alarm_resync_task import AlarmResyncTask
from voltha.extensions.omci.database.alarm_db_ext import AlarmDbExternal
from voltha.extensions.omci.tasks.interval_data_task import IntervalDataTask
from voltha.extensions.omci.onu_device_entry import OnuDeviceEntry
from voltha.extensions.omci.state_machines.omci_onu_capabilities import OnuOmciCapabilities
from voltha.extensions.omci.tasks.onu_capabilities_task import OnuCapabilitiesTask
from voltha.extensions.omci.state_machines.performance_intervals import PerformanceIntervals
from voltha.extensions.omci.tasks.omci_create_pm_task import OmciCreatePMRequest
from voltha.extensions.omci.tasks.omci_delete_pm_task import OmciDeletePMRequest
from voltha.extensions.omci.state_machines.image_agent import ImageDownloadeSTM, OmciSoftwareImageDownloadSTM
from voltha.extensions.omci.tasks.file_download_task import FileDownloadTask
from voltha.extensions.omci.tasks.omci_sw_image_upgrade_task import OmciSwImageUpgradeTask
OpenOmciAgentDefaults = {
'mib-synchronizer': {
'state-machine': MibSynchronizer, # Implements the MIB synchronization state machine
'database': MibDbVolatileDict, # Implements volatile ME MIB database
# 'database': MibDbExternal, # Implements persistent ME MIB database
'advertise-events': True, # Advertise events on OpenOMCI event bus
'tasks': {
'mib-upload': MibUploadTask,
'get-mds': GetMdsTask,
'mib-audit': GetMdsTask,
'mib-resync': MibResyncTask,
'mib-reconcile': MibReconcileTask
}
},
'omci-capabilities': {
'state-machine': OnuOmciCapabilities, # Implements OMCI capabilities state machine
'advertise-events': False, # Advertise events on OpenOMCI event bus
'tasks': {
'get-capabilities': OnuCapabilitiesTask # Get supported ME and Commands
}
},
'performance-intervals': {
'state-machine': PerformanceIntervals, # Implements PM Intervals State machine
'advertise-events': False, # Advertise events on OpenOMCI event bus
'tasks': {
'sync-time': SyncTimeTask,
'collect-data': IntervalDataTask,
'create-pm': OmciCreatePMRequest,
'delete-pm': OmciDeletePMRequest,
},
},
'alarm-synchronizer': {
'state-machine': AlarmSynchronizer, # Implements the Alarm sync state machine
'database': AlarmDbExternal, # For any State storage needs
'advertise-events': True, # Advertise events on OpenOMCI event bus
'tasks': {
'alarm-resync': AlarmResyncTask
}
},
'image_downloader': {
'state-machine': ImageDownloadeSTM,
'advertise-event': True,
'tasks': {
'download-file': FileDownloadTask
}
},
'image_upgrader': {
'state-machine': OmciSoftwareImageDownloadSTM,
'advertise-event': True,
'tasks': {
'omci_upgrade_task': OmciSwImageUpgradeTask
}
}
# 'image_activator': {
# 'state-machine': OmciSoftwareImageActivateSTM,
# 'advertise-event': True,
# }
}
class OpenOMCIAgent(object):
"""
OpenOMCI for VOLTHA
This will become the primary interface into OpenOMCI for ONU Device Adapters
in VOLTHA v1.3 sprint 3 time frame.
"""
def __init__(self, core, support_classes=OpenOmciAgentDefaults, clock=None):
"""
Class initializer
:param core: (VolthaCore) VOLTHA Core
:param support_classes: (Dict) Classes to support OMCI
"""
self.log = structlog.get_logger()
self._core = core
self.reactor = clock if clock is not None else reactor
self._started = False
self._devices = dict() # device-id -> DeviceEntry
self._event_bus = None
# OMCI related databases are on a per-agent basis. State machines and tasks
# are per ONU Vendore
#
# MIB Synchronization Database
self._mib_db = None
self._mib_database_cls = support_classes['mib-synchronizer']['database']
# Alarm Synchronization Database
self._alarm_db = None
self._alarm_database_cls = support_classes['alarm-synchronizer']['database']
@property
def core(self):
""" Return a reference to the VOLTHA Core component"""
return self._core
@property
def database_class(self):
return self._mib_database_cls
@property
def database(self):
return self._mib_db
def start(self):
"""
Start OpenOMCI
"""
if self._started:
return
self.log.debug('OpenOMCIAgent.start')
self._started = True
try:
# Create all databases as needed. This should be done before
# State machines are started for the first time
if self._mib_db is None:
self._mib_db = self._mib_database_cls(self)
if self._alarm_db is None:
self._alarm_db = self._alarm_database_cls(self)
# Start/restore databases
self._mib_db.start()
self._alarm_db.start()
for device in self._devices.itervalues():
device.start()
except Exception as e:
self.log.exception('startup', e=e)
def stop(self):
"""
Shutdown OpenOMCI
"""
if not self._started:
return
self.log.debug('stop')
self._started = False
self._event_bus = None
# ONUs OMCI shutdown
for device in self._devices.itervalues():
device.stop()
# DB shutdown
self._mib_db.stop()
self._alarm_db.stop()
def mk_event_bus(self):
""" Get the event bus for OpenOMCI"""
if self._event_bus is None:
from voltha.extensions.omci.openomci_event_bus import OpenOmciEventBus
self._event_bus = OpenOmciEventBus()
return self._event_bus
def advertise(self, event_type, data):
"""
Advertise an OpenOMCU event on the kafka bus
:param event_type: (int) Event Type (enumberation from OpenOMCI protobuf definitions)
:param data: (Message, dict, ...) Associated data (will be convert to a string)
"""
if self._started:
try:
self.mk_event_bus().advertise(event_type, data)
except Exception as e:
self.log.exception('advertise-failure', e=e)
def add_device(self, device_id, adapter_agent, custom_me_map=None,
support_classes=OpenOmciAgentDefaults):
"""
Add a new ONU to be managed.
To provide vendor-specific or custom Managed Entities, create your own Entity
ID to class mapping dictionary.
Since ONU devices can be added at any time (even during Device Handler
startup), the ONU device handler is responsible for calling start()/stop()
for this object.
:param device_id: (str) Device ID of ONU to add
:param adapter_agent: (AdapterAgent) Adapter agent for ONU
:param custom_me_map: (dict) Additional/updated ME to add to class map
:param support_classes: (dict) State machines and tasks for this ONU
:return: (OnuDeviceEntry) The ONU device
"""
self.log.debug('OpenOMCIAgent.add-device', device_id=device_id)
device = self._devices.get(device_id)
if device is None:
device = OnuDeviceEntry(self, device_id, adapter_agent, custom_me_map,
self._mib_db, self._alarm_db, support_classes, clock=self.reactor)
self._devices[device_id] = device
return device
def remove_device(self, device_id, cleanup=False):
"""
Remove a managed ONU
:param device_id: (str) Device ID of ONU to remove
:param cleanup: (bool) If true, scrub any state related information
"""
self.log.debug('remove-device', device_id=device_id, cleanup=cleanup)
device = self._devices.get(device_id)
if device is not None:
device.stop()
if cleanup:
del self._devices[device_id]
def device_ids(self):
"""
Get an immutable set of device IDs managed by this OpenOMCI instance
:return: (frozenset) Set of device IDs (str)
"""
return frozenset(self._devices.keys())
def get_device(self, device_id):
"""
Get ONU device entry. For external (non-OpenOMCI users) the ONU Device
returned should be used for read-only activity.
:param device_id: (str) ONU Device ID
:return: (OnuDeviceEntry) ONU Device entry
:raises KeyError: If device does not exist
"""
return self._devices[device_id]