mirror of
https://github.com/Telecominfraproject/oopt-gnpy.git
synced 2025-10-29 09:12:37 +00:00
This will be used to compute the max total power for design per OMS. Signed-off-by: EstherLerouzic <esther.lerouzic@orange.com> Change-Id: I392f06c792af9f32d4a14324c683bd3fae655de8
826 lines
28 KiB
Python
826 lines
28 KiB
Python
#!/usr/bin/env python3
|
|
# -*- coding: utf-8 -*-
|
|
|
|
# SPDX-License-Identifier: BSD-3-Clause
|
|
# gnpy.core.utils: utility functions that are used with gnpy
|
|
# Copyright (C) 2025 Telecom Infra Project and GNPy contributors
|
|
# see AUTHORS.rst for a list of contributors
|
|
|
|
"""
|
|
gnpy.core.utils
|
|
===============
|
|
|
|
This module contains utility functions that are used with gnpy.
|
|
"""
|
|
|
|
from csv import writer
|
|
from numpy import pi, cos, sqrt, log10, linspace, zeros, shape, where, logical_and, mean, array
|
|
from scipy import constants
|
|
from copy import deepcopy
|
|
from typing import List, Union, Dict
|
|
|
|
from gnpy.core.exceptions import ConfigurationError
|
|
|
|
|
|
def write_csv(obj, filename):
|
|
"""
|
|
Convert dictionary items to a CSV file the dictionary format:
|
|
::
|
|
|
|
{'result category 1':
|
|
[
|
|
# 1st line of results
|
|
{'header 1' : value_xxx,
|
|
'header 2' : value_yyy},
|
|
# 2nd line of results: same headers, different results
|
|
{'header 1' : value_www,
|
|
'header 2' : value_zzz}
|
|
],
|
|
'result_category 2':
|
|
[
|
|
{},{}
|
|
]
|
|
}
|
|
|
|
The generated csv file will be:
|
|
::
|
|
|
|
result_category 1
|
|
header 1 header 2
|
|
value_xxx value_yyy
|
|
value_www value_zzz
|
|
result_category 2
|
|
...
|
|
"""
|
|
with open(filename, 'w', encoding='utf-8') as f:
|
|
w = writer(f)
|
|
for data_key, data_list in obj.items():
|
|
# main header
|
|
w.writerow([data_key])
|
|
# sub headers:
|
|
headers = [_ for _ in data_list[0].keys()]
|
|
w.writerow(headers)
|
|
for data_dict in data_list:
|
|
w.writerow([_ for _ in data_dict.values()])
|
|
|
|
|
|
def arrange_frequencies(length, start, stop):
|
|
"""Create an array of frequencies
|
|
|
|
:param length: number of elements
|
|
:param start: Start frequency in THz
|
|
:param stop: Stop frequency in THz
|
|
:type length: integer
|
|
:type start: float
|
|
:type stop: float
|
|
:return: an array of frequencies determined by the spacing parameter
|
|
:rtype: numpy.ndarray
|
|
"""
|
|
return linspace(start, stop, length)
|
|
|
|
|
|
def lin2db(value):
|
|
"""Convert linear unit to logarithmic (dB)
|
|
|
|
>>> lin2db(0.001)
|
|
-30.0
|
|
>>> round(lin2db(1.0), 2)
|
|
0.0
|
|
>>> round(lin2db(1.26), 2)
|
|
1.0
|
|
>>> round(lin2db(10.0), 2)
|
|
10.0
|
|
>>> round(lin2db(100.0), 2)
|
|
20.0
|
|
"""
|
|
return 10 * log10(value)
|
|
|
|
|
|
def db2lin(value):
|
|
"""Convert logarithimic units to linear
|
|
|
|
>>> round(db2lin(10.0), 2)
|
|
10.0
|
|
>>> round(db2lin(20.0), 2)
|
|
100.0
|
|
>>> round(db2lin(1.0), 2)
|
|
1.26
|
|
>>> round(db2lin(0.0), 2)
|
|
1.0
|
|
>>> round(db2lin(-10.0), 2)
|
|
0.1
|
|
"""
|
|
return 10**(value / 10)
|
|
|
|
|
|
def watt2dbm(value):
|
|
"""Convert Watt units to dBm
|
|
|
|
>>> round(watt2dbm(0.001), 1)
|
|
0.0
|
|
>>> round(watt2dbm(0.02), 1)
|
|
13.0
|
|
"""
|
|
return lin2db(value * 1e3)
|
|
|
|
|
|
def dbm2watt(value):
|
|
"""Convert dBm units to Watt
|
|
|
|
>>> round(dbm2watt(0), 4)
|
|
0.001
|
|
>>> round(dbm2watt(-3), 4)
|
|
0.0005
|
|
>>> round(dbm2watt(13), 4)
|
|
0.02
|
|
"""
|
|
return db2lin(value) * 1e-3
|
|
|
|
|
|
def psd2powerdbm(psd_mwperghz, baudrate_baud):
|
|
"""computes power in dBm based on baudrate in bauds and psd in mW/GHz
|
|
|
|
>>> round(psd2powerdbm(0.031176, 64e9),3)
|
|
3.0
|
|
>>> round(psd2powerdbm(0.062352, 32e9),3)
|
|
3.0
|
|
>>> round(psd2powerdbm(0.015625, 64e9),3)
|
|
0.0
|
|
"""
|
|
return lin2db(baudrate_baud * psd_mwperghz * 1e-9)
|
|
|
|
|
|
def power_dbm_to_psd_mw_ghz(power_dbm, baudrate_baud):
|
|
"""computes power spectral density in mW/GHz based on baudrate in bauds and power in dBm
|
|
|
|
>>> power_dbm_to_psd_mw_ghz(0, 64e9)
|
|
0.015625
|
|
>>> round(power_dbm_to_psd_mw_ghz(3, 64e9), 6)
|
|
0.031176
|
|
>>> round(power_dbm_to_psd_mw_ghz(3, 32e9), 6)
|
|
0.062352
|
|
"""
|
|
return db2lin(power_dbm) / (baudrate_baud * 1e-9)
|
|
|
|
|
|
def psd_mw_per_ghz(power_watt, baudrate_baud):
|
|
"""computes power spectral density in mW/GHz based on baudrate in bauds and power in W
|
|
|
|
>>> psd_mw_per_ghz(2e-3, 32e9)
|
|
0.0625
|
|
>>> psd_mw_per_ghz(1e-3, 64e9)
|
|
0.015625
|
|
>>> psd_mw_per_ghz(0.5e-3, 32e9)
|
|
0.015625
|
|
"""
|
|
return power_watt * 1e3 / (baudrate_baud * 1e-9)
|
|
|
|
|
|
def round2float(number, step):
|
|
"""Round a floating point number so that its "resolution" is not bigger than 'step'
|
|
|
|
The finest step is fixed at 0.01; smaller values are silently changed to 0.01.
|
|
|
|
>>> round2float(123.456, 1000)
|
|
0.0
|
|
>>> round2float(123.456, 100)
|
|
100.0
|
|
>>> round2float(123.456, 10)
|
|
120.0
|
|
>>> round2float(123.456, 1)
|
|
123.0
|
|
>>> round2float(123.456, 0.1)
|
|
123.5
|
|
>>> round2float(123.456, 0.01)
|
|
123.46
|
|
>>> round2float(123.456, 0.001)
|
|
123.46
|
|
>>> round2float(123.249, 0.5)
|
|
123.0
|
|
>>> round2float(123.250, 0.5)
|
|
123.0
|
|
>>> round2float(123.251, 0.5)
|
|
123.5
|
|
>>> round2float(123.300, 0.2)
|
|
123.2
|
|
>>> round2float(123.301, 0.2)
|
|
123.4
|
|
"""
|
|
step = round(step, 1)
|
|
if step >= 0.01:
|
|
number = round(number / step, 0)
|
|
number = round(number * step, 1)
|
|
else:
|
|
number = round(number, 2)
|
|
return number
|
|
|
|
|
|
wavelength2freq = constants.lambda2nu
|
|
freq2wavelength = constants.nu2lambda
|
|
|
|
|
|
def snr_sum(snr, bw, snr_added, bw_added=12.5e9):
|
|
snr_added = snr_added - lin2db(bw / bw_added)
|
|
snr = -lin2db(db2lin(-snr) + db2lin(-snr_added))
|
|
return snr
|
|
|
|
|
|
def per_label_average(values, labels):
|
|
"""computes the average per defined spectrum band, using labels
|
|
|
|
>>> labels = ['A', 'A', 'A', 'A', 'A', 'B', 'B', 'B', 'B', 'B', 'B', 'B', 'B', 'C', 'D', 'D', 'D', 'D']
|
|
>>> values = [28.51, 28.23, 28.15, 28.17, 28.36, 28.53, 28.64, 28.68, 28.7, 28.71, 28.72, 28.73, 28.74, 28.91, 27.96, 27.85, 27.87, 28.02]
|
|
>>> per_label_average(values, labels)
|
|
{'A': 28.28, 'B': 28.68, 'C': 28.91, 'D': 27.92}
|
|
"""
|
|
|
|
label_set = sorted(set(labels))
|
|
summary = {}
|
|
for label in label_set:
|
|
vals = [val for val, lab in zip(values, labels) if lab == label]
|
|
summary[label] = round(mean(vals), 2)
|
|
return summary
|
|
|
|
|
|
def pretty_summary_print(summary):
|
|
"""Build a prettty string that shows the summary dict values per label with 2 digits"""
|
|
if len(summary) == 1:
|
|
return f'{list(summary.values())[0]:.2f}'
|
|
text = ', '.join([f'{label}: {value:.2f}' for label, value in summary.items()])
|
|
return text
|
|
|
|
|
|
def deltawl2deltaf(delta_wl, wavelength):
|
|
"""deltawl2deltaf(delta_wl, wavelength):
|
|
delta_wl is BW in wavelength units
|
|
wavelength is the center wl
|
|
units for delta_wl and wavelength must be same
|
|
|
|
:param delta_wl: delta wavelength BW in same units as wavelength
|
|
:param wavelength: wavelength BW is relevant for
|
|
:type delta_wl: float or numpy.ndarray
|
|
:type wavelength: float
|
|
:return: The BW in frequency units
|
|
:rtype: float or ndarray
|
|
|
|
"""
|
|
f = wavelength2freq(wavelength)
|
|
return delta_wl * f / wavelength
|
|
|
|
|
|
def deltaf2deltawl(delta_f, frequency):
|
|
"""convert delta frequency to delta wavelength
|
|
|
|
Units for delta_wl and wavelength must be same.
|
|
|
|
:param delta_f: delta frequency in same units as frequency
|
|
:param frequency: frequency BW is relevant for
|
|
:type delta_f: float or numpy.ndarray
|
|
:type frequency: float
|
|
:return: The BW in wavelength units
|
|
:rtype: float or ndarray
|
|
|
|
"""
|
|
wl = freq2wavelength(frequency)
|
|
return delta_f * wl / frequency
|
|
|
|
|
|
def rrc(ffs, baud_rate, alpha):
|
|
"""compute the root-raised cosine filter function
|
|
|
|
:param ffs: A numpy array of frequencies
|
|
:param baud_rate: The Baud Rate of the System
|
|
:param alpha: The roll-off factor of the filter
|
|
:type ffs: numpy.ndarray
|
|
:type baud_rate: float
|
|
:type alpha: float
|
|
:return: hf a numpy array of the filter shape
|
|
:rtype: numpy.ndarray
|
|
|
|
"""
|
|
Ts = 1 / baud_rate
|
|
l_lim = (1 - alpha) / (2 * Ts)
|
|
r_lim = (1 + alpha) / (2 * Ts)
|
|
hf = zeros(shape(ffs))
|
|
slope_inds = where(
|
|
logical_and(abs(ffs) > l_lim, abs(ffs) < r_lim))
|
|
hf[slope_inds] = 0.5 * (1 + cos((pi * Ts / alpha) *
|
|
(abs(ffs[slope_inds]) - l_lim)))
|
|
p_inds = where(logical_and(abs(ffs) > 0, abs(ffs) < l_lim))
|
|
hf[p_inds] = 1
|
|
return sqrt(hf)
|
|
|
|
|
|
def merge_amplifier_restrictions(dict1, dict2):
|
|
"""Update contents of dicts recursively
|
|
|
|
>>> d1 = {'params': {'restrictions': {'preamp_variety_list': [], 'booster_variety_list': []}}}
|
|
>>> d2 = {'params': {'target_pch_out_db': -20}}
|
|
>>> merge_amplifier_restrictions(d1, d2)
|
|
{'params': {'restrictions': {'preamp_variety_list': [], 'booster_variety_list': []}, 'target_pch_out_db': -20}}
|
|
|
|
>>> d3 = {'params': {'restrictions': {'preamp_variety_list': ['foo'], 'booster_variety_list': ['bar']}}}
|
|
>>> merge_amplifier_restrictions(d1, d3)
|
|
{'params': {'restrictions': {'preamp_variety_list': [], 'booster_variety_list': []}}}
|
|
"""
|
|
|
|
copy_dict1 = dict1.copy()
|
|
for key in dict2:
|
|
if key in dict1:
|
|
if isinstance(dict1[key], dict):
|
|
copy_dict1[key] = merge_amplifier_restrictions(copy_dict1[key], dict2[key])
|
|
else:
|
|
copy_dict1[key] = dict2[key]
|
|
return copy_dict1
|
|
|
|
|
|
def use_pmd_coef(dict1: dict, dict2: dict):
|
|
"""If Fiber dict1 is missing the pmd_coef value then use the one of dict2.
|
|
In addition records in "pmd_coef_defined" key the pmd_coef if is was defined in dict1.
|
|
|
|
:param dict1: A dictionnary that contains "pmd_coef" key.
|
|
:type dict1: dict
|
|
:param dict2: Another dictionnary that contains "pmd_coef" key.
|
|
:type dict2: dict
|
|
|
|
>>> dict1 = {'a': 1, 'pmd_coef': 1.5e-15}
|
|
>>> dict2 = {'a': 2, 'pmd_coef': 2e-15}
|
|
>>> use_pmd_coef(dict1, dict2)
|
|
>>> dict1
|
|
{'a': 1, 'pmd_coef': 1.5e-15, 'pmd_coef_defined': True}
|
|
|
|
>>> dict1 = {'a': 1}
|
|
>>> use_pmd_coef(dict1, dict2)
|
|
>>> dict1
|
|
{'a': 1, 'pmd_coef_defined': False, 'pmd_coef': 2e-15}
|
|
"""
|
|
if 'pmd_coef' in dict1 and not dict1['pmd_coef'] \
|
|
or ('pmd_coef' not in dict1 and 'pmd_coef' in dict2):
|
|
dict1['pmd_coef_defined'] = False
|
|
dict1['pmd_coef'] = dict2['pmd_coef']
|
|
elif 'pmd_coef' in dict1 and dict1['pmd_coef']:
|
|
dict1['pmd_coef_defined'] = True
|
|
# all other case do not need any change
|
|
|
|
|
|
def silent_remove(this_list, elem):
|
|
"""Remove matching elements from a list without raising ValueError
|
|
|
|
>>> li = [0, 1]
|
|
>>> li = silent_remove(li, 1)
|
|
>>> li
|
|
[0]
|
|
>>> li = silent_remove(li, 1)
|
|
>>> li
|
|
[0]
|
|
"""
|
|
|
|
try:
|
|
this_list.remove(elem)
|
|
except ValueError:
|
|
pass
|
|
return this_list
|
|
|
|
|
|
def automatic_nch(f_min, f_max, spacing):
|
|
"""How many channels are available in the spectrum
|
|
|
|
:param f_min Lowest frequenecy [Hz]
|
|
:param f_max Highest frequency [Hz]
|
|
:param spacing Channel width [Hz]
|
|
:return Number of uniform channels
|
|
|
|
>>> automatic_nch(191.325e12, 196.125e12, 50e9)
|
|
96
|
|
>>> automatic_nch(193.475e12, 193.525e12, 50e9)
|
|
1
|
|
"""
|
|
return int((f_max - f_min) // spacing)
|
|
|
|
|
|
def automatic_fmax(f_min, spacing, nch):
|
|
"""Find the high-frequenecy boundary of a spectrum
|
|
|
|
:param f_min Start of the spectrum (lowest frequency edge) [Hz]
|
|
:param spacing Grid/channel spacing [Hz]
|
|
:param nch Number of channels
|
|
:return End of the spectrum (highest frequency) [Hz]
|
|
|
|
>>> automatic_fmax(191.325e12, 50e9, 96)
|
|
196125000000000.0
|
|
"""
|
|
return f_min + spacing * nch
|
|
|
|
|
|
def convert_length(value, units):
|
|
"""Convert length into basic SI units
|
|
|
|
>>> convert_length(1, 'km')
|
|
1000.0
|
|
>>> convert_length(2.0, 'km')
|
|
2000.0
|
|
>>> convert_length(123, 'm')
|
|
123.0
|
|
>>> convert_length(123.0, 'm')
|
|
123.0
|
|
>>> convert_length(42.1, 'km')
|
|
42100.0
|
|
>>> convert_length(666, 'yards')
|
|
Traceback (most recent call last):
|
|
...
|
|
gnpy.core.exceptions.ConfigurationError: Cannot convert length in "yards" into meters
|
|
"""
|
|
if units == 'm':
|
|
return value * 1e0
|
|
elif units == 'km':
|
|
return value * 1e3
|
|
else:
|
|
raise ConfigurationError(f'Cannot convert length in "{units}" into meters')
|
|
|
|
|
|
def replace_none(dictionary):
|
|
""" Replaces None with inf values in a frequency slots dict
|
|
|
|
>>> replace_none({'N': 3, 'M': None})
|
|
{'N': 3, 'M': inf}
|
|
|
|
"""
|
|
for key, val in dictionary.items():
|
|
if val is None:
|
|
dictionary[key] = float('inf')
|
|
if val == float('inf'):
|
|
dictionary[key] = None
|
|
return dictionary
|
|
|
|
|
|
def order_slots(slots):
|
|
""" Order frequency slots from larger slots to smaller ones up to None
|
|
|
|
>>> l = [{'N': 3, 'M': None}, {'N': 2, 'M': 1}, {'N': None, 'M': None},{'N': 7, 'M': 2},{'N': None, 'M': 1} , {'N': None, 'M': 0}]
|
|
>>> order_slots(l)
|
|
([7, 2, None, None, 3, None], [2, 1, 1, 0, None, None], [3, 1, 4, 5, 0, 2])
|
|
"""
|
|
slots_list = deepcopy(slots)
|
|
slots_list = [replace_none(e) for e in slots_list]
|
|
for i, e in enumerate(slots_list):
|
|
e['i'] = i
|
|
slots_list = sorted(slots_list, key=lambda x: (-x['M'], x['N']) if x['M'] != float('inf') else (x['M'], x['N']))
|
|
slots_list = [replace_none(e) for e in slots_list]
|
|
return [e['N'] for e in slots_list], [e['M'] for e in slots_list], [e['i'] for e in slots_list]
|
|
|
|
|
|
def restore_order(elements, order):
|
|
""" Use order to re-order the element of the list, and ignore None values
|
|
|
|
>>> restore_order([7, 2, None, None, 3, None], [3, 1, 4, 5, 0, 2])
|
|
[3, 2, 7]
|
|
"""
|
|
return [elements[i[0]] for i in sorted(enumerate(order), key=lambda x:x[1]) if elements[i[0]] is not None]
|
|
|
|
|
|
def unique_ordered(elements):
|
|
"""
|
|
"""
|
|
unique_elements = []
|
|
for element in elements:
|
|
if element not in unique_elements:
|
|
unique_elements.append(element)
|
|
return unique_elements
|
|
|
|
|
|
def convert_empty_to_none(json_data: Union[list, dict]) -> dict:
|
|
"""Convert all instances of "a": [None] into "a": None
|
|
|
|
:param json_data: the input data.
|
|
:type json_data: dict
|
|
:return: the converted data.
|
|
:rtype: dict
|
|
|
|
>>> json_data = {
|
|
... "uid": "[east edfa in Lannion",
|
|
... "type_variety": "multiband_booster",
|
|
... "metadata": {
|
|
... "location": {
|
|
... "latitude": 0.000000,
|
|
... "longitude": 0.000000,
|
|
... "city": "Zion",
|
|
... "region": ""
|
|
... }
|
|
... },
|
|
... "type": "Multiband_amplifier",
|
|
... "amplifiers": [{
|
|
... "type_variety": "multiband_booster_LOW_C",
|
|
... "operational": {
|
|
... "gain_target": 12.22,
|
|
... "delta_p": 4.19,
|
|
... "out_voa": [None],
|
|
... "tilt_target": 0.00,
|
|
... "f_min": 191.3,
|
|
... "f_max": 196.1
|
|
... }
|
|
... }, {
|
|
... "type_variety": "multiband_booster_LOW_L",
|
|
... "operational": {
|
|
... "gain_target": 12.05,
|
|
... "delta_p": 4.19,
|
|
... "out_voa": [None],
|
|
... "tilt_target": 0.00,
|
|
... "f_min": 186.1,
|
|
... "f_max": 190.9
|
|
... }
|
|
... }
|
|
... ]
|
|
... }
|
|
>>> convert_empty_to_none(json_data)
|
|
{'uid': '[east edfa in Lannion', 'type_variety': 'multiband_booster', \
|
|
'metadata': {'location': {'latitude': 0.0, 'longitude': 0.0, 'city': 'Zion', 'region': ''}}, \
|
|
'type': 'Multiband_amplifier', 'amplifiers': [{'type_variety': 'multiband_booster_LOW_C', \
|
|
'operational': {'gain_target': 12.22, 'delta_p': 4.19, 'out_voa': None, 'tilt_target': 0.0, \
|
|
'f_min': 191.3, 'f_max': 196.1}}, {'type_variety': 'multiband_booster_LOW_L', \
|
|
'operational': {'gain_target': 12.05, 'delta_p': 4.19, 'out_voa': None, 'tilt_target': 0.0, \
|
|
'f_min': 186.1, 'f_max': 190.9}}]}
|
|
|
|
"""
|
|
if isinstance(json_data, dict):
|
|
for key, value in json_data.items():
|
|
json_data[key] = convert_empty_to_none(value)
|
|
elif isinstance(json_data, list):
|
|
if len(json_data) == 1 and json_data[0] is None:
|
|
return None
|
|
for i, elem in enumerate(json_data):
|
|
json_data[i] = convert_empty_to_none(elem)
|
|
return json_data
|
|
|
|
|
|
def convert_none_to_empty(json_data: Union[list, dict]) -> dict:
|
|
"""Convert all instances of "a": None into "a": [None], to be compliant with RFC7951.
|
|
|
|
:param json_data: the input data.
|
|
:type json_data: dict
|
|
:return: the converted data.
|
|
:rtype: dict
|
|
|
|
>>> a = {'uid': '[east edfa in Lannion', 'type_variety': 'multiband_booster',
|
|
... 'metadata': {'location': {'latitude': 0.0, 'longitude': 0.0, 'city': 'Zion', 'region': ''}},
|
|
... 'type': 'Multiband_amplifier', 'amplifiers': [{'type_variety': 'multiband_booster_LOW_C',
|
|
... 'operational': {'gain_target': 12.22, 'delta_p': 4.19, 'out_voa': None, 'tilt_target': 0.0,
|
|
... 'f_min': 191.3, 'f_max': 196.1}}, {'type_variety': 'multiband_booster_LOW_L',
|
|
... 'operational': {'gain_target': 12.05, 'delta_p': 4.19, 'out_voa': None, 'tilt_target': 0.0,
|
|
... 'f_min': 186.1, 'f_max': 190.9}}]}
|
|
>>> convert_none_to_empty(a)
|
|
{'uid': '[east edfa in Lannion', 'type_variety': 'multiband_booster', \
|
|
'metadata': {'location': {'latitude': 0.0, 'longitude': 0.0, 'city': 'Zion', 'region': ''}}, \
|
|
'type': 'Multiband_amplifier', 'amplifiers': [{'type_variety': 'multiband_booster_LOW_C', \
|
|
'operational': {'gain_target': 12.22, 'delta_p': 4.19, 'out_voa': [None], 'tilt_target': 0.0, \
|
|
'f_min': 191.3, 'f_max': 196.1}}, {'type_variety': 'multiband_booster_LOW_L', \
|
|
'operational': {'gain_target': 12.05, 'delta_p': 4.19, 'out_voa': [None], 'tilt_target': 0.0, \
|
|
'f_min': 186.1, 'f_max': 190.9}}]}
|
|
|
|
"""
|
|
if json_data == [None]:
|
|
# already conformed
|
|
return json_data
|
|
if isinstance(json_data, dict):
|
|
for key, value in json_data.items():
|
|
json_data[key] = convert_none_to_empty(value)
|
|
elif isinstance(json_data, list):
|
|
for i, elem in enumerate(json_data):
|
|
json_data[i] = convert_none_to_empty(elem)
|
|
elif json_data is None:
|
|
return [None]
|
|
return json_data
|
|
|
|
|
|
def calculate_absolute_min_or_zero(x: array) -> array:
|
|
"""Calculates the element-wise absolute minimum between the x and zero.
|
|
|
|
Parameters:
|
|
x (array): The first input array.
|
|
|
|
Returns:
|
|
array: The element-wise absolute minimum between x and zero.
|
|
|
|
Example:
|
|
>>> x = array([-1, 2, -3])
|
|
>>> calculate_absolute_min_or_zero(x)
|
|
array([1., 0., 3.])
|
|
"""
|
|
return (abs(x) - x) / 2
|
|
|
|
|
|
def nice_column_str(data: List[List[str]], max_length: int = 30, padding: int = 1) -> str:
|
|
"""data is a list of rows, creates strings with nice alignment per colum and padding with spaces
|
|
letf justified
|
|
|
|
>>> table_data = [['aaa', 'b', 'c'], ['aaaaaaaa', 'bbb', 'c'], ['a', 'bbbbbbbbbb', 'c']]
|
|
>>> print(nice_column_str(table_data))
|
|
aaa b c
|
|
aaaaaaaa bbb c
|
|
a bbbbbbbbbb c
|
|
"""
|
|
# transpose data to determine size of columns
|
|
transposed_data = list(map(list, zip(*data)))
|
|
column_width = [max(len(word) for word in column) + padding for column in transposed_data]
|
|
nice_str = []
|
|
for row in data:
|
|
column = ''.join(word[0:max_length].ljust(min(width, max_length)) for width, word in zip(column_width, row))
|
|
nice_str.append(f'{column}')
|
|
return '\n'.join(nice_str)
|
|
|
|
|
|
def filter_valid_amp_bands(amp_bands: List[List[dict]]) -> List[List[dict]]:
|
|
"""Filter out invalid amplifier bands that lack f_min or f_max.
|
|
|
|
:param amp_bands: A list of lists containing amplifier band dictionaries.
|
|
:type amp_bands: List[List[dict]]
|
|
:return: A filtered list of amplifier bands that contain valid f_min and f_max.
|
|
:rtype: List[List[dict]]
|
|
"""
|
|
return [amp for amp in amp_bands if all(band.get('f_min') is not None and band.get('f_max') is not None
|
|
for band in amp)]
|
|
|
|
|
|
def remove_duplicates(amp_bands: List[List[dict]]) -> List[List[dict]]:
|
|
"""Remove duplicate amplifier bands.
|
|
|
|
:param amp_bands: A list of lists containing amplifier band dictionaries.
|
|
:type amp_bands: List[List[dict]]
|
|
:return: A list of unique amplifier bands.
|
|
:rtype: List[List[dict]]
|
|
"""
|
|
unique_amp_bands = []
|
|
for amp in amp_bands:
|
|
if amp not in unique_amp_bands:
|
|
unique_amp_bands.append(amp)
|
|
return unique_amp_bands
|
|
|
|
|
|
def calculate_spacing(first: dict, second: dict, default_spacing: float, default_design_bands: Union[List[Dict], None],
|
|
f_min: float, f_max: float) -> float:
|
|
"""Calculate the spacing for the given frequency range.
|
|
|
|
:param first: The first amplifier band dictionary.
|
|
:type first: dict
|
|
:param second: The second amplifier band dictionary.
|
|
:type second: dict
|
|
:param default_spacing: The default spacing to use if no specific spacing can be determined.
|
|
:type default_spacing: float
|
|
:param default_design_bands: Optional list of design bands to determine spacing from.
|
|
:type default_design_bands: Union[List[Dict], None]
|
|
:param f_min: The minimum frequency of the range.
|
|
:type f_min: float
|
|
:param f_max: The maximum frequency of the range.
|
|
:type f_max: float
|
|
:return: The calculated spacing for the given frequency range.
|
|
:rtype: float
|
|
"""
|
|
if first.get('spacing') is not None and second.get('spacing') is not None:
|
|
return max(first['spacing'], second['spacing'])
|
|
elif first.get('spacing') is not None:
|
|
return first['spacing']
|
|
elif second.get('spacing') is not None:
|
|
return second['spacing']
|
|
elif default_design_bands:
|
|
temp = get_spacing_from_band(default_design_bands, f_min, f_max)
|
|
return temp if temp is not None else default_spacing
|
|
return default_spacing
|
|
|
|
|
|
def find_common_range(amp_bands: List[List[dict]], default_band_f_min: Union[float, None],
|
|
default_band_f_max: Union[float, None], default_spacing: float,
|
|
default_design_bands: Union[List[Dict], None] = None) -> List[dict]:
|
|
"""
|
|
Find the common frequency range of amplifier bands.
|
|
|
|
If there are no amplifiers in the path, then use the default band parameters.
|
|
|
|
:param amp_bands: A list of lists containing amplifier band dictionaries, each with 'f_min', 'f_max',
|
|
and optionally 'spacing'.
|
|
:type amp_bands: List[List[dict]]
|
|
:param default_band_f_min: The minimum frequency of the default band.
|
|
:type default_band_f_min: Union[float, None]
|
|
:param default_band_f_max: The maximum frequency of the default band.
|
|
:type default_band_f_max: Union[float, None]
|
|
:param default_spacing: The default spacing to use if no specific spacing can be determined.
|
|
:type default_spacing: float
|
|
:param default_design_bands: Optional list of design bands to determine spacing from.
|
|
:type default_design_bands: Union[List[Dict], None]
|
|
:return: A list of dictionaries representing the common frequency ranges with their respective spacings.
|
|
:rtype: List[dict]
|
|
|
|
>>> amp_bands = [[{'f_min': 191e12, 'f_max' : 195e12, 'spacing': 70e9}, {'f_min': 186e12, 'f_max' : 190e12}], \
|
|
[{'f_min': 185e12, 'f_max' : 189e12}, {'f_min': 192e12, 'f_max' : 196e12}], \
|
|
[{'f_min': 186e12, 'f_max': 193e12}]]
|
|
>>> find_common_range(amp_bands, 190e12, 195e12, 50e9)
|
|
[{'f_min': 186000000000000.0, 'f_max': 189000000000000.0, 'spacing': 50000000000.0}, \
|
|
{'f_min': 192000000000000.0, 'f_max': 193000000000000.0, 'spacing': 70000000000.0}]
|
|
|
|
>>> amp_bands = [[{'f_min': 191e12, 'f_max' : 195e12}, {'f_min': 186e12, 'f_max' : 190e12}], \
|
|
[{'f_min': 185e12, 'f_max' : 189e12}, {'f_min': 192e12, 'f_max' : 196e12}], \
|
|
[{'f_min': 186e12, 'f_max': 192e12}]]
|
|
>>> find_common_range(amp_bands, 190e12, 195e12, 50e9)
|
|
[{'f_min': 186000000000000.0, 'f_max': 189000000000000.0, 'spacing': 50000000000.0}]
|
|
"""
|
|
# Step 1: Filter and sort amplifier bands
|
|
_amp_bands = [sorted(amp, key=lambda x: x['f_min']) for amp in filter_valid_amp_bands(amp_bands)]
|
|
unique_amp_bands = remove_duplicates(_amp_bands)
|
|
|
|
# Step 2: Handle cases with no valid bands
|
|
if unique_amp_bands:
|
|
common_range = unique_amp_bands[0]
|
|
else:
|
|
if default_band_f_min is None or default_band_f_max is None:
|
|
return []
|
|
return [{'f_min': default_band_f_min, 'f_max': default_band_f_max, 'spacing': None}]
|
|
|
|
# Step 3: Calculate common frequency range
|
|
for bands in unique_amp_bands:
|
|
new_common_range = []
|
|
for first in common_range:
|
|
for second in bands:
|
|
f_min = max(first['f_min'], second['f_min'])
|
|
f_max = min(first['f_max'], second['f_max'])
|
|
if f_min < f_max:
|
|
spacing = calculate_spacing(first, second, default_spacing, default_design_bands, f_min, f_max)
|
|
new_common_range.append({'f_min': f_min, 'f_max': f_max, 'spacing': spacing})
|
|
|
|
common_range = new_common_range
|
|
|
|
return sorted(common_range, key=lambda x: x['f_min'])
|
|
|
|
|
|
def transform_data(data: str) -> Union[List[int], None]:
|
|
"""Transforms a float into an list of one integer or a string separated by "|" into a list of integers.
|
|
|
|
Args:
|
|
data (float or str): The data to transform.
|
|
|
|
Returns:
|
|
list of int: The transformed data as a list of integers.
|
|
|
|
Examples:
|
|
>>> transform_data(5.0)
|
|
[5]
|
|
|
|
>>> transform_data('1 | 2 | 3')
|
|
[1, 2, 3]
|
|
"""
|
|
if isinstance(data, float):
|
|
return [int(data)]
|
|
if isinstance(data, str):
|
|
return [int(x) for x in data.split(' | ')]
|
|
return None
|
|
|
|
|
|
def convert_pmd_lineic(pmd: Union[float, None], length: float, length_unit: str) -> Union[float, None]:
|
|
"""Convert PMD value of the span in ps into pmd_lineic in s/sqrt(km)
|
|
|
|
:param pmd: value in ps
|
|
:type pmd: Union[float, None]
|
|
:param length: value in length_unit
|
|
:type length: float
|
|
:param length_unit: 'km' or 'm'
|
|
:type length_unit: str
|
|
:return: lineic PMD s/sqrt(m)
|
|
:rtype: Union[float, None]
|
|
|
|
>>> convert_pmd_lineic(10, 0.001, 'km')
|
|
1e-11
|
|
"""
|
|
if pmd:
|
|
return pmd * 1e-12 / sqrt(convert_length(length, length_unit))
|
|
return None
|
|
def get_spacing_from_band(design_bands: List[Dict], f_min, f_max):
|
|
"""Retrieve the spacing for a frequency range based on design bands.
|
|
|
|
This function checks if the midpoint of the provided frequency range (f_min, f_max)
|
|
falls within any of the design bands. If it does, the corresponding spacing is returned.
|
|
|
|
:param design_bands: A list of design band dictionaries, each containing 'f_min', 'f_max', and 'spacing'.
|
|
:type design_bands: List[Dict]
|
|
:param f_min: The minimum frequency of the range.
|
|
:type f_min: float
|
|
:param f_max: The maximum frequency of the range.
|
|
:type f_max: float
|
|
:return: The spacing corresponding to the design band that contains the midpoint of the range,
|
|
or None if no such band exists.
|
|
:rtype: Union[float, None]
|
|
"""
|
|
midpoint = (f_min + f_max) / 2
|
|
for band in design_bands:
|
|
if midpoint >= band['f_min'] and midpoint <= band['f_max']:
|
|
return band['spacing']
|
|
return None
|
|
|
|
|
|
def reorder_per_degree_design_bands(per_degree_design_bands: dict):
|
|
"""Sort the design bands for each degree by their minimum frequency (f_min).
|
|
|
|
This function modifies the input dictionary in place, sorting the design bands for each unique identifier.
|
|
|
|
:param per_degree_design_bands: A dictionary where keys are unique identifiers and values are lists of design band dictionaries.
|
|
:type per_degree_design_bands: Dict[str, List[Dict]]
|
|
"""
|
|
for uid, design_bands in per_degree_design_bands.items():
|
|
per_degree_design_bands[uid] = sorted(design_bands, key=lambda x: x['f_min'])
|