Skip to content

Commit

Permalink
portconfig option to configure Tx power and laser frequency of ZR tra…
Browse files Browse the repository at this point in the history
…nsceiver module (#2197) (#2330)

Cherry-pick of PR #2197

Added new configuration CLI options to configure Tx power and laser frequency of ZR transceiver module
  • Loading branch information
prgeor authored Aug 25, 2022
1 parent 4bacc1c commit e0166a0
Show file tree
Hide file tree
Showing 3 changed files with 142 additions and 2 deletions.
63 changes: 62 additions & 1 deletion config/main.py
Original file line number Diff line number Diff line change
Expand Up @@ -1692,7 +1692,7 @@ def load_minigraph(db, no_service_restart, traffic_shift_away):
cfggen_namespace_option = " -n {}".format(namespace)
clicommon.run_command(db_migrator + ' -o set_version' + cfggen_namespace_option)

# Keep device isolated with TSA
# Keep device isolated with TSA
if traffic_shift_away:
clicommon.run_command("TSA", display_cmd=True)
if os.path.isfile(DEFAULT_GOLDEN_CONFIG_DB_FILE):
Expand Down Expand Up @@ -4631,6 +4631,67 @@ def transceiver(ctx):
"""SFP transceiver configuration"""
pass

#
# 'frequency' subcommand ('config interface transceiver frequency ...')
#
@transceiver.command()
@click.pass_context
@click.argument('interface_name', metavar='<interface_name>', required=True)
@click.argument('frequency', metavar='<frequency>', required=True, type=int)
def frequency(ctx, interface_name, frequency):
"""Set transciever (only for 400G-ZR) frequency"""
# Get the config_db connector
config_db = ctx.obj['config_db']

if clicommon.get_interface_naming_mode() == "alias":
interface_name = interface_alias_to_name(config_db, interface_name)
if interface_name is None:
ctx.fail("'interface_name' is None!")

if interface_name_is_valid(config_db, interface_name) is False:
ctx.fail("Interface name is invalid. Please enter a valid interface name!!")

log.log_info("{} Setting transceiver frequency {} GHz".format(interface_name, frequency))

if ctx.obj['namespace'] is DEFAULT_NAMESPACE:
command = "portconfig -p {} -F {}".format(interface_name, frequency)
else:
command = "portconfig -p {} -F {} -n {}".format(interface_name, frequency, ctx.obj['namespace'])

clicommon.run_command(command)


#
# 'tx_power' subcommand ('config interface transceiver tx_power ...')
# For negative float use:-
# config interface transceiver tx_power Ethernet0 -- -27.4"
#
@transceiver.command('tx_power')
@click.pass_context
@click.argument('interface_name', metavar='<interface_name>', required=True)
@click.argument('tx-power', metavar='<tx-power>', required=True, type=float)
def tx_power(ctx, interface_name, tx_power):
"""Set transciever (only for 400G-ZR) Tx laser power"""
# Get the config_db connector
config_db = ctx.obj['config_db']

if clicommon.get_interface_naming_mode() == "alias":
interface_name = interface_alias_to_name(config_db, interface_name)
if interface_name is None:
ctx.fail("'interface_name' is None!")

if interface_name_is_valid(config_db, interface_name) is False:
ctx.fail("Interface name is invalid. Please enter a valid interface name!!")

log.log_info("{} Setting transceiver power {} dBm".format(interface_name, tx_power))

if ctx.obj['namespace'] is DEFAULT_NAMESPACE:
command = "portconfig -p {} -P {}".format(interface_name, tx_power)
else:
command = "portconfig -p {} -P {} -n {}".format(interface_name, tx_power, ctx.obj['namespace'])

clicommon.run_command(command)

#
# 'lpmode' subcommand ('config interface transceiver lpmode ...')
#
Expand Down
32 changes: 31 additions & 1 deletion scripts/portconfig
Original file line number Diff line number Diff line change
Expand Up @@ -22,9 +22,12 @@ optional arguments:
-t --interface-type port interface type
-T --adv-interface-types port advertised interface types
-lt --link-training port link training mode
-P --tx-power 400G ZR modulet target Tx output power (dBm)
-F --laser-freq 400G ZR module 75GHz grid frequency (GHz)
"""
import os
import sys
import decimal
import argparse

# mock the redis for unit test purposes #
Expand All @@ -51,6 +54,8 @@ PORT_ADV_SPEEDS_CONFIG_FIELD_NAME = "adv_speeds"
PORT_INTERFACE_TYPE_CONFIG_FIELD_NAME = "interface_type"
PORT_ADV_INTERFACE_TYPES_CONFIG_FIELD_NAME = "adv_interface_types"
PORT_LINK_TRAINING_CONFIG_FIELD_NAME = "link_training"
PORT_XCVR_LASER_FREQ_FIELD_NAME = "laser_freq"
PORT_XCVR_TX_POWER_FIELD_NAME = "tx_power"
PORT_CHANNEL_TABLE_NAME = "PORTCHANNEL"
PORT_CHANNEL_MBR_TABLE_NAME = "PORTCHANNEL_MEMBER"
TPID_CONFIG_FIELD_NAME = "tpid"
Expand Down Expand Up @@ -159,6 +164,14 @@ class portconfig(object):
mode = 'on' if mode == 'enabled' else 'off'
self.db.mod_entry(PORT_TABLE_NAME, port, {PORT_AUTONEG_CONFIG_FIELD_NAME: mode})

def set_tx_power(self, port, tx_power):
print("Setting target Tx output power to %s dBm on port %s" % (tx_power, port))
self.db.mod_entry(PORT_TABLE_NAME, port, {PORT_XCVR_TX_POWER_FIELD_NAME: tx_power})

def set_laser_freq(self, port, laser_freq):
print("Setting laser frequency to %s GHz on port %s" % (laser_freq, port))
self.db.mod_entry(PORT_TABLE_NAME, port, {PORT_XCVR_LASER_FREQ_FIELD_NAME: laser_freq})

def set_adv_speeds(self, port, adv_speeds):
if self.verbose:
print("Setting adv_speeds %s on port %s" % (adv_speeds, port))
Expand Down Expand Up @@ -307,6 +320,10 @@ def main():
help = 'port advertised interface types', default=None)
parser.add_argument('-lt', '--link-training', type = str, required = False,
help = 'port link training mode', default=None)
parser.add_argument('-P', '--tx-power', type=float, required=False,
help='Tx output power(dBm)', default=None)
parser.add_argument('-F', '--laser-freq', type=int, required=False,
help='Laser frequency(GHz)', default=None)
args = parser.parse_args()

# Load database config files
Expand All @@ -315,7 +332,9 @@ def main():
port = portconfig(args.verbose, args.port, args.namespace)
if args.list:
port.list_params(args.port)
elif args.speed or args.fec or args.mtu or args.link_training or args.autoneg or args.adv_speeds or args.interface_type or args.adv_interface_types or args.tpid:
elif args.speed or args.fec or args.mtu or args.link_training or args.autoneg or args.adv_speeds or \
args.interface_type or args.adv_interface_types or args.tpid or \
args.tx_power or args.laser_freq:
if args.speed:
port.set_speed(args.port, args.speed)
if args.fec:
Expand All @@ -334,6 +353,17 @@ def main():
port.set_adv_interface_types(args.port, args.adv_interface_types)
if args.tpid:
port.set_tpid(args.port, args.tpid)
if args.tx_power:
d = decimal.Decimal(str(args.tx_power))
if d.as_tuple().exponent < -1:
print("Error: tx power must be with single decimal place")
sys.exit(1)
port.set_tx_power(args.port, args.tx_power)
if args.laser_freq:
if args.laser_freq <= 0:
print("Error: Frequency must be > 0")
sys.exit(1)
port.set_laser_freq(args.port, args.laser_freq)
else:
parser.print_help()
sys.exit(1)
Expand Down
49 changes: 49 additions & 0 deletions tests/config_xcvr_test.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,49 @@
import click
import config.main as config
import operator
import os
import pytest
import sys

from click.testing import CliRunner
from utilities_common.db import Db

test_path = os.path.dirname(os.path.abspath(__file__))
modules_path = os.path.dirname(test_path)
scripts_path = os.path.join(modules_path, "scripts")
sys.path.insert(0, modules_path)


@pytest.fixture(scope='module')
def ctx(scope='module'):
db = Db()
obj = {'config_db':db.cfgdb, 'namespace': ''}
yield obj


class TestConfigXcvr(object):
@classmethod
def setup_class(cls):
print("SETUP")
os.environ["PATH"] += os.pathsep + scripts_path
os.environ["UTILITIES_UNIT_TESTING"] = "1"

def test_config_laser_frequency(self, ctx):
#self.basic_check("link-training", ["Ethernet0", "on"], ctx)
result = self.basic_check("frequency", ["Ethernet0", "191300"], ctx)
assert "Setting laser frequency" in result.output
result = self.basic_check("frequency", ["Ethernet0", "--", "-1"], ctx, op=operator.ne)
assert "Error: Frequency must be > 0" in result.output

def test_config_tx_power(self, ctx):
result = self.basic_check("tx_power", ["Ethernet0", "11.3"], ctx)
assert "Setting target Tx output power" in result.output
result = self.basic_check("tx_power", ["Ethernet0", "11.34"], ctx, op=operator.ne)
assert "Error: tx power must be with single decimal place" in result.output

def basic_check(self, command_name, para_list, ctx, op=operator.eq, expect_result=0):
runner = CliRunner()
result = runner.invoke(config.config.commands["interface"].commands["transceiver"].commands[command_name], para_list, obj = ctx)
print(result.output)
assert op(result.exit_code, expect_result)
return result

0 comments on commit e0166a0

Please sign in to comment.