blob: 946946d5cd1380b81efbacd8740060552065aa2f [file] [log] [blame]
# Copyright 2017-present Adtran, Inc.
#
# 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
import json
from enum import Enum
from voltha.protos.bbf_fiber_tcont_body_pb2 import TcontsConfigData
from voltha.protos.bbf_fiber_traffic_descriptor_profile_body_pb2 import TrafficDescriptorProfileData
from twisted.internet.defer import succeed, inlineCallbacks, returnValue
log = structlog.get_logger()
class TCont(object):
"""
Class to wrap TCont capabilities
"""
def __init__(self, alloc_id, traffic_descriptor, best_effort=None,
name=None, ident=None, vont_ani=None):
self.alloc_id = alloc_id
self.traffic_descriptor = traffic_descriptor
self.best_effort = best_effort
self.name = name
self.id = ident
self.vont_ani = vont_ani # (string) reference
def __str__(self):
return "TCont: {}, alloc-id: {}".format(self.name, self.alloc_id)
@staticmethod
def create(data, td):
assert isinstance(data, TcontsConfigData)
assert isinstance(td, TrafficDescriptor)
return TCont(data.alloc_id, td, best_effort=td.best_effort,
name=data.name, ident=data.id, vont_ani=data.interface_reference)
@inlineCallbacks
def add_to_hardware(self, session, pon_id, onu_id, operation='POST'):
from adtran_olt_handler import AdtranOltHandler
uri = AdtranOltHandler.GPON_TCONT_CONFIG_LIST_URI.format(pon_id, onu_id)
data = json.dumps({'alloc-id': self.alloc_id})
name = 'tcont-create-{}-{}: {}'.format(pon_id, onu_id, self.alloc_id)
what = 'tcont'
try:
# For TCONT, only leaf is the key. So only post needed
if operation == 'POST':
results = yield session.request('POST', uri, data=data, name=name)
else:
results = succeed('nop')
if self.traffic_descriptor is not None:
what = 'traffic-descriptor'
results = yield self.traffic_descriptor.add_to_hardware(session,
pon_id, onu_id,
self.alloc_id,
self.best_effort)
except Exception as e:
log.exception(what, tcont=self, td=self.traffic_descriptor, e=e)
raise
returnValue(results)
def remove_from_hardware(self, session, pon_id, onu_id):
from adtran_olt_handler import AdtranOltHandler
uri = AdtranOltHandler.GPON_TCONT_CONFIG_URI.format(pon_id, onu_id, self.alloc_id)
name = 'tcont-delete-{}-{}: {}'.format(pon_id, onu_id, self.alloc_id)
return succeed(session.request('DELETE', uri, name=name))
def _get_onu(self, olt):
onu = None
try:
vont_ani = olt.v_ont_anis.get(self.vont_ani)
ch_pair = olt.channel_pairs.get(vont_ani['preferred-channel-pair'])
ch_term = next((term for term in olt.channel_terminations.itervalues()
if term['channel-pair'] == ch_pair['name']), None)
pon = olt.pon(ch_term['xgs-ponid'])
onu = pon.onu(vont_ani['onu-id'])
except Exception:
pass
return onu
def xpon_create(self, olt):
# Look up any associated ONU. May be None if pre-provisioning
onu = self._get_onu(olt)
if onu is not None:
onu.add_tcont(self)
def xpon_update(self, olt):
# Look up any associated ONU. May be None if pre-provisioning
onu = self._get_onu(olt)
if onu is not None:
pass # TODO: Not yet supported
def xpon_delete(self, olt):
# Look up any associated ONU. May be None if pre-provisioning
onu = self._get_onu(olt)
if onu is not None:
onu.remove_tcont(self.alloc_id)
class TrafficDescriptor(object):
"""
Class to wrap the uplink traffic descriptor.
"""
class AdditionalBwEligibility(Enum):
NONE = 0
BEST_EFFORT_SHARING = 1
NON_ASSURED_SHARING = 2 # Should match xpon.py values
DEFAULT = NONE
@staticmethod
def to_string(value):
return {
TrafficDescriptor.AdditionalBwEligibility.NON_ASSURED_SHARING: "non-assured-sharing",
TrafficDescriptor.AdditionalBwEligibility.BEST_EFFORT_SHARING: "best-effort-sharing",
TrafficDescriptor.AdditionalBwEligibility.NONE: "none"
}.get(value, "unknown")
@staticmethod
def from_value(value):
"""
Matches both Adtran and xPON values
:param value:
:return:
"""
return {
0: TrafficDescriptor.AdditionalBwEligibility.NONE,
1: TrafficDescriptor.AdditionalBwEligibility.BEST_EFFORT_SHARING,
2: TrafficDescriptor.AdditionalBwEligibility.NON_ASSURED_SHARING,
}.get(value, TrafficDescriptor.AdditionalBwEligibility.DEFAULT)
def __init__(self, fixed, assured, maximum,
additional=AdditionalBwEligibility.DEFAULT,
best_effort=None,
name=None,
ident=None):
self.name = name
self.id = ident
self.fixed_bandwidth = fixed # bps
self.assured_bandwidth = assured # bps
self.maximum_bandwidth = maximum # bps
self.additional_bandwidth_eligibility = additional
self.best_effort = best_effort\
if additional == TrafficDescriptor.AdditionalBwEligibility.BEST_EFFORT_SHARING\
else None
def __str__(self):
return "TrafficDescriptor: {}, {}/{}/{}".format(self.name,
self.fixed_bandwidth,
self.assured_bandwidth,
self.maximum_bandwidth)
@staticmethod
def create(data):
assert isinstance(data, TrafficDescriptorProfileData)
additional = TrafficDescriptor.AdditionalBwEligibility.from_value(
data.additional_bw_eligibility_indicator)
if additional == TrafficDescriptor.AdditionalBwEligibility.BEST_EFFORT_SHARING:
best_effort = BestEffort(data.maximum_bandwidth,
data.priority,
data.weight)
else:
best_effort = None
return TrafficDescriptor(data.fixed_bandwidth, data.assured_bandwidth,
data.maximum_bandwidth,
name=data.name,
ident=data.id,
best_effort=best_effort,
additional=additional)
def to_dict(self):
val = {
'fixed-bandwidth': self.fixed_bandwidth,
'assured-bandwidth': self.assured_bandwidth,
'maximum-bandwidth': self.maximum_bandwidth,
'additional-bandwidth-eligibility':
TrafficDescriptor.AdditionalBwEligibility.to_string(
self.additional_bandwidth_eligibility)
}
return val
@inlineCallbacks
def add_to_hardware(self, session, pon_id, onu_id, alloc_id, best_effort):
from adtran_olt_handler import AdtranOltHandler
uri = AdtranOltHandler.GPON_TCONT_CONFIG_URI.format(pon_id, onu_id, alloc_id)
data = json.dumps({'traffic-descriptor': self.to_dict()})
name = 'tcont-td-{}-{}: {}'.format(pon_id, onu_id, alloc_id)
try:
results = yield session.request('PATCH', uri, data=data, name=name)
except Exception as e:
log.exception('traffic-descriptor', td=self, e=e)
raise
if self.additional_bandwidth_eligibility == \
TrafficDescriptor.AdditionalBwEligibility.BEST_EFFORT_SHARING:
if best_effort is None:
raise ValueError('TCONT is best-effort but does not define best effort sharing')
try:
results = yield best_effort.add_to_hardware(session, pon_id, onu_id, alloc_id)
except Exception as e:
log.exception('best-effort', best_effort=best_effort, e=e)
raise
returnValue(results)
def xpon_create(self, olt, tcont):
# Look up any associated ONU. May be None if pre-provisioning
pass # TODO
def xpon_update(self, olt, tcont):
# Look up any associated ONU. May be None if pre-provisioning
pass # TODO: Not yet supported
class BestEffort(object):
def __init__(self, bandwidth, priority, weight):
self.bandwidth = bandwidth # bps
self.priority = priority # 0.255
self.weight = weight # 0..100
def __str__(self):
return "BestEffort: {}/p-{}/w-{}".format(self.bandwidth,
self.priority,
self.weight)
def to_dict(self):
val = {
'bandwidth': self.bandwidth,
'priority': self.priority,
'weight': self.weight
}
return val
def add_to_hardware(self, session, pon_id, onu_id, alloc_id, best_effort):
from adtran_olt_handler import AdtranOltHandler
uri = AdtranOltHandler.GPON_TCONT_CONFIG_URI.format(pon_id, onu_id, alloc_id)
data = json.dumps({'best-effort': best_effort.to_dict()})
name = 'tcont-best-effort-{}-{}: {}'.format(pon_id, onu_id, alloc_id)
return session.request('PATCH', uri, data=data, name=name)