Skip to content

Commit

Permalink
Update to newer Python and dependencies versions (#58)
Browse files Browse the repository at this point in the history
* Update of minimal pyAT and NumPy versions
    pyAT >=0.6.1
    numpy >=2.0.0
    dropping Python 3.8
    testing with Python 3.12

* Switching to pyproject.toml instead of setup.py

* Updating github actions
    switching to hatchling due to only __editable__pySC...created in environments

* Switched to new interface of AT tracking functions

* Reducing the amount of single element NumPy arrays throughout  the package
  • Loading branch information
lmalina authored Sep 8, 2024
1 parent c382cbc commit 23bd8e5
Show file tree
Hide file tree
Showing 18 changed files with 122 additions and 138 deletions.
10 changes: 5 additions & 5 deletions .github/workflows/coverage.yml
Original file line number Diff line number Diff line change
Expand Up @@ -17,18 +17,18 @@ jobs:
runs-on: ${{ matrix.os }}
strategy:
matrix:
os: [ubuntu-22.04]
python-version: [3.11]
os: [ubuntu-24.04]
python-version: [3.12]

steps:
- uses: actions/checkout@v3
- uses: actions/checkout@v4

- name: Set up Python ${{ matrix.python-version }}
uses: actions/setup-python@v4
uses: actions/setup-python@v5
with:
python-version: ${{ matrix.python-version }}
cache: 'pip'
cache-dependency-path: '**/setup.py'
cache-dependency-path: '**/pyproject.toml'

- name: Upgrade pip, setuptools and wheel
run: python -m pip install --upgrade pip setuptools wheel
Expand Down
8 changes: 4 additions & 4 deletions .github/workflows/documentation.yml
Original file line number Diff line number Diff line change
Expand Up @@ -20,17 +20,17 @@ jobs:
strategy:
matrix: # only lowest supported Python on latest ubuntu
os: [ubuntu-latest]
python-version: [3.8]
python-version: [3.9]

steps:
- uses: actions/checkout@v3
- uses: actions/checkout@v4

- name: Set up Python ${{ matrix.python-version }}
uses: actions/setup-python@v4
uses: actions/setup-python@v5
with:
python-version: ${{ matrix.python-version }}
cache: 'pip'
cache-dependency-path: '**/setup.py'
cache-dependency-path: '**/pyproject.toml'

- name: Get full Python version
id: full-python-version
Expand Down
25 changes: 6 additions & 19 deletions .github/workflows/tests.yml
Original file line number Diff line number Diff line change
Expand Up @@ -14,31 +14,18 @@ jobs:
runs-on: ${{ matrix.os }}
strategy:
matrix:
os: [ubuntu-20.04, ubuntu-22.04, windows-latest, macos-latest]
python-version: [3.8, 3.9, "3.10", "3.11"]
os: [ubuntu-22.04, ubuntu-24.04, windows-latest, macos-latest]
python-version: [3.9, "3.10", 3.11, 3.12]

steps:
- uses: actions/checkout@v3
- uses: actions/checkout@v4

- name: Set up Python ${{ matrix.python-version }}
uses: actions/setup-python@v4
uses: actions/setup-python@v5
with:
python-version: ${{ matrix.python-version }}

- name: Get full Python version
id: full-python-version
run: echo ::set-output name=version::$(python -c "import sys; print('-'.join(str(v) for v in sys.version_info))")

- name: Set up cache
uses: actions/cache@v2
id: cache
with:
path: .venv
key: venv-${{ runner.os }}-${{ steps.full-python-version.outputs.version }}

- name: Ensure cache is healthy
if: steps.cache.outputs.cache-hit == 'true'
run: pip --version >/dev/null 2>&1 || rm -rf .venv
cache: 'pip'
cache-dependency-path: '**/pyproject.toml'

- name: Upgrade pip, setuptools and wheel
run: |
Expand Down
2 changes: 1 addition & 1 deletion pySC/__init__.py
Original file line number Diff line number Diff line change
Expand Up @@ -9,7 +9,7 @@
__title__ = "pySC"
__description__ = "Python version of Simulated Commissioning toolkit for synchrotrons (https://github.com/ThorstenHellert/SC) "
__url__ = "https://github.com/lmalina/pySC"
__version__ = "0.1.0"
__version__ = "0.2.0"
__author__ = "lmalina"
__author_email__ = "lukas.malina@desy.de"

Expand Down
9 changes: 5 additions & 4 deletions pySC/core/beam.py
Original file line number Diff line number Diff line change
Expand Up @@ -61,7 +61,7 @@ def bpm_reading(SC: SimulatedCommissioning, bpm_ords: ndarray = None, calculate_
mean_bpm_sums_3d = np.nansum(mean_bpm_sums_3d, axis=2, keepdims=True) / SC.INJ.nTurns
if calculate_errors and SC.INJ.trackMode == TRACK_TBT:
bpm_orbits_4d[np.sum(np.isnan(bpm_orbits_4d), axis=3) > 0, :] = np.nan
squared_orbit_diffs = np.square(bpm_orbits_4d - mean_bpm_orbits_3d)
squared_orbit_diffs = np.square(bpm_orbits_4d - mean_bpm_orbits_3d[:, :, :, np.newaxis])
err_bpm_orbits_3d = np.sqrt(np.average(np.ma.array(squared_orbit_diffs, mask=np.isnan(bpm_orbits_4d)),
weights=np.ma.array(bpm_sums_4d, mask=np.isnan(bpm_orbits_4d)), axis=3)).filled(np.nan)
# Organising the array 2 x (nturns x nbpms) sorted by "arrival time"
Expand Down Expand Up @@ -128,7 +128,8 @@ def beam_transmission(SC: SimulatedCommissioning, nParticles: int = None, nTurns
if nTurns is None:
nTurns = SC.INJ.nTurns
LOGGER.debug(f'Calculating maximum beam transmission for {nParticles} particles and {nTurns} turns: ')
T = at_wrapper.patpass(SC.RING, generate_bunches(SC, nParticles=nParticles), nTurns, np.array([len(SC.RING)]), keep_lattice=False)
T = at_wrapper.lattice_track(SC.RING, generate_bunches(SC, nParticles=nParticles), nTurns, np.array([len(SC.RING)]),
keep_lattice=False, use_mp=True)
fraction_survived = np.mean(~np.isnan(T[0, :, :, :]), axis=(0, 1))
max_turns = np.sum(fraction_survived > 1 - SC.INJ.beamLostAt)
if plot:
Expand Down Expand Up @@ -185,7 +186,7 @@ def _real_bpm_reading(SC, track_mat, bpm_inds=None): # track_mat should be only
bpm_noise = bpm_noise[:, :, np.newaxis] * sc_tools.randnc(2, (2, n_bpms, nTurns))
bpm_offset = np.transpose(at_wrapper.atgetfieldvalues(SC.RING, bpm_ords, 'Offset') + at_wrapper.atgetfieldvalues(SC.RING, bpm_ords, 'SupportOffset'))
bpm_cal_error = np.transpose(at_wrapper.atgetfieldvalues(SC.RING, bpm_ords, 'CalError'))
bpm_roll = np.squeeze(at_wrapper.atgetfieldvalues(SC.RING, bpm_ords, 'Roll') + at_wrapper.atgetfieldvalues(SC.RING, bpm_ords, 'SupportRoll'), axis=1)
bpm_roll = np.ravel(at_wrapper.atgetfieldvalues(SC.RING, bpm_ords, 'Roll')) + np.ravel(at_wrapper.atgetfieldvalues(SC.RING, bpm_ords, 'SupportRoll'))
bpm_sum_error = np.transpose(at_wrapper.atgetfieldvalues(SC.RING, bpm_ords, 'SumError'))[:, np.newaxis] * sc_tools.randnc(2, (n_bpms, nTurns))
# averaging the X and Y positions at BPMs over particles
mean_orbit = np.nanmean(track_mat, axis=1)
Expand All @@ -211,7 +212,7 @@ def _tracking(SC: SimulatedCommissioning, refs: ndarray) -> ndarray:
if SC.INJ.trackMode == TRACK_ORB:
pos = np.transpose(at_wrapper.findorbit6(SC.RING, refs, keep_lattice=False)[1])[[0, 2], :].reshape(2, 1, len(refs), 1)
else:
pos = at_wrapper.atpass(SC.RING, generate_bunches(SC), SC.INJ.nTurns, refs, keep_lattice=False)[[0, 2], :, :, :]
pos = at_wrapper.lattice_track(SC.RING, generate_bunches(SC), SC.INJ.nTurns, refs, keep_lattice=False)[[0, 2], :, :, :]
pos[1, np.isnan(pos[0, :, :, :])] = np.nan
return pos

Expand Down
8 changes: 4 additions & 4 deletions pySC/core/simulated_commissioning.py
Original file line number Diff line number Diff line change
Expand Up @@ -35,7 +35,7 @@ class SimulatedCommissioning:
Examples:
>>> import at
>>> SC = SimulatedCommissioning(at.Lattice(at.Monitor('BPM'))
>>> SC = SimulatedCommissioning(at.Lattice(at.Monitor('BPM')))
>>> SC.register_bpms(np.array(0), Offset=2e-6*np.ones(2))
>>> print(SC.ORD)
0
Expand Down Expand Up @@ -560,7 +560,7 @@ def apply_errors(self, nsigmas: float = 2):
self.INJ.randomInjectionZ = 1 * self.SIG.randomInjectionZ
# Circumference
if 'Circumference' in self.SIG.keys():
circScaling = 1 + self.SIG.Circumference * sc_tools.randnc(nsigmas, (1, 1))
circScaling = 1 + self.SIG.Circumference * sc_tools.randnc(nsigmas, ())
self.RING = sc_tools.scale_circumference(self.RING, circScaling, 'rel')
LOGGER.info('Circumference error applied.')
# Misalignments
Expand All @@ -586,7 +586,7 @@ def _apply_support_alignment_error(self, nsigmas):
classdef_tools.randn_cutoff(value, nsigmas) if field in self.SIG.Support[ordPair[1]].keys()
else getattr(self.RING[ordPair[0]], field))

struct_length = np.remainder(np.diff(s_pos[ordPair]), s_pos[-1])
struct_length = np.remainder(s_pos[ordPair][1] - s_pos[ordPair][0], s_pos[-1])
rolls0 = copy.deepcopy(getattr(self.RING[ordPair[0]], f"{support_type}Roll")) # Twisted supports are not considered
offsets0 = copy.deepcopy(getattr(self.RING[ordPair[0]], f"{support_type}Offset"))
offsets1 = copy.deepcopy(getattr(self.RING[ordPair[1]], f"{support_type}Offset"))
Expand Down Expand Up @@ -696,7 +696,7 @@ def update_supports(self, offset_bpms: bool = True, offset_magnets: bool = True)
for i, ind in enumerate(self.ORD.BPM):
setattr(self.RING[ind], "SupportOffset", offsets[0:2, i]) # No longitudinal BPM offsets implemented
setattr(self.RING[ind], "SupportRoll",
np.array([rolls[0, i]])) # BPM pitch and yaw angles not implemented
rolls[0, i]) # BPM pitch and yaw angles not implemented
else:
LOGGER.warning('SC: No BPMs have been registered!')

Expand Down
2 changes: 1 addition & 1 deletion pySC/correction/bba.py
Original file line number Diff line number Diff line change
Expand Up @@ -94,7 +94,7 @@ def fake_bba(SC, bpm_ords, mag_ords, errors=None, fake_offset=None):
fake_bpm_offset = (SC.RING[mag_ords[inds[0], inds[1]]].MagnetOffset[inds[0]]
+ SC.RING[mag_ords[inds[0], inds[1]]].SupportOffset[inds[0]]
- SC.RING[bpm_ords[inds[0], inds[1]]].SupportOffset[inds[0]]
+ fake_offset[inds[0]] * sc_tools.randnc())
+ fake_offset[inds[0]] * sc_tools.randnc(2, ()))
if not np.isnan(fake_bpm_offset):
SC.RING[bpm_ords[inds[0], inds[1]]].Offset[inds[0]] = fake_bpm_offset
else:
Expand Down
2 changes: 1 addition & 1 deletion pySC/correction/injection_fit.py
Original file line number Diff line number Diff line change
Expand Up @@ -68,7 +68,7 @@ def _fit_bpm_data(s_bpm, bref):


def _merit_function(x, SC, bref, ords_used):
t = at_wrapper.atpass(SC.IDEALRING, np.concatenate((x, np.zeros(2))), 1, ords_used, keep_lattice=False)[[0, 2], 0, :, 0]
t = at_wrapper.lattice_track(SC.IDEALRING, np.concatenate((x, np.zeros(2))), 1, ords_used, keep_lattice=False)[[0, 2], 0, :, 0]
return np.sqrt(np.mean(bref - t) ** 2)


Expand Down
10 changes: 5 additions & 5 deletions pySC/lattice_properties/apertures.py
Original file line number Diff line number Diff line change
Expand Up @@ -26,7 +26,7 @@ def SCdynamicAperture(RING, dE, bounds=np.array([0, 1e-3]), nturns=1000, thetas=
for cntt in range(len(thetas)): # Loop over angles
theta = thetas[cntt]
limits = inibounds
at_wrapper.atpass(RING, np.full(6, np.nan), 1, [1]) # Fake Track to initialize lattice
at_wrapper.lattice_track(RING, np.full(6, np.nan), 1, [1]) # Fake Track to initialize lattice
scales = 0
while scales < 16:
if _check_bounds(RING, ZCO, nturns, theta, limits):
Expand Down Expand Up @@ -102,7 +102,7 @@ def _check_bounds(RING, ZCO, nturns, theta, boundsIn):
Zmax = ZCO[:]
Zmax[0] = boundsIn[1] * np.cos(theta)
Zmax[2] = boundsIn[1] * np.sin(theta)
ROUT = at_wrapper.atpass(RING, [Zmin, Zmax], nturns, len(RING), keep_lattice=True) # Track
ROUT = at_wrapper.lattice_track(RING, [Zmin, Zmax], nturns, len(RING), keep_lattice=True) # Track
RLAST = ROUT[:, len(ROUT) - 1:len(ROUT)] # Get positions after last turn
return ~np.isnan(RLAST[0, 0]) and np.isnan(RLAST[0, 1])

Expand All @@ -114,7 +114,7 @@ def _refine_bounds(RING, ZCO, nturns, theta, boundsIn):
Z = ZCO[:]
Z[0] = rmid * np.cos(theta)
Z[2] = rmid * np.sin(theta)
ROUT = at_wrapper.atpass(RING, Z, nturns, len(RING), keep_lattice=True) # Track
ROUT = at_wrapper.lattice_track(RING, Z, nturns, len(RING), keep_lattice=True) # Track
RLAST = ROUT[:, len(ROUT) - 1] # Get positions after last turn
return [rmin, rmid] if np.isnan(RLAST[0]) else [rmid, rmax] # Midpoint is outside or inside DA

Expand Down Expand Up @@ -146,7 +146,7 @@ def refine_bounds(local_bounds, RING, ZCO, nturns):
dmean = np.mean(local_bounds)
Z0 = ZCO
Z0[4] = Z0[4] + dmean
ROUT = at_wrapper.atpass(RING, Z0, nturns, len(RING))
ROUT = at_wrapper.lattice_track(RING, Z0, nturns, len(RING))
if np.isnan(ROUT[0]): # Particle dead :(
local_bounds[1] = dmean # Set abs-upper bound to midpoint
else: # Particle alive :)
Expand All @@ -157,7 +157,7 @@ def refine_bounds(local_bounds, RING, ZCO, nturns):
def check_bounds(local_bounds, RING, ZCO, nturns):
Z = np.array([ZCO, ZCO])
Z[4, :] = Z[4, :] + local_bounds[:]
ROUT = at_wrapper.atpass(RING, Z, nturns, len(RING))
ROUT = at_wrapper.lattice_track(RING, Z, nturns, len(RING))
if np.isnan(ROUT[0, 0]) and not np.isnan(ROUT[0, 1]):
LOGGER.warning('Closer-to-momentum particle is unstable. This shouldnt be!')
return not np.isnan(ROUT[0, 0]) and np.isnan(ROUT[0, 1])
Expand Down
4 changes: 2 additions & 2 deletions pySC/lattice_properties/response_model.py
Original file line number Diff line number Diff line change
Expand Up @@ -49,7 +49,7 @@ def SCgetModelRM(SC, BPMords, CMords, trackMode='TBT', Z0=np.zeros(6), nTurns=1,
"""
LOGGER.info('Calculating model response matrix')
track_methods = dict(TBT=at_wrapper.atpass, ORB=orbpass)
track_methods = dict(TBT=at_wrapper.lattice_track, ORB=orbpass)
if trackMode not in track_methods.keys():
ValueError(f'Unknown track mode {trackMode}. Valid values are {track_methods.keys()}')
ring = SC.IDEALRING.deepcopy() if useIdealRing else SCgetModelRING(SC)
Expand Down Expand Up @@ -121,7 +121,7 @@ def SCgetModelDispersion(SC, BPMords, CAVords, trackMode='ORB', Z0=np.zeros(6),
"""
LOGGER.info('Calculating model dispersion')
track_methods = dict(TBT=at_wrapper.atpass, ORB=orbpass)
track_methods = dict(TBT=at_wrapper.lattice_track, ORB=orbpass)
if trackMode not in track_methods.keys():
ValueError(f'Unknown track mode {trackMode}. Valid values are {track_methods.keys()}')
ring = SC.IDEALRING.deepcopy() if useIdealRing else SCgetModelRING(SC)
Expand Down
2 changes: 1 addition & 1 deletion pySC/plotting/plot_phase_space.py
Original file line number Diff line number Diff line change
Expand Up @@ -10,7 +10,7 @@ def plot_phase_space(SC, ords=np.zeros(1, dtype=int), custom_bunch=None, nPartic
init_font = plt.rcParams["font.size"]
plt.rcParams.update({'font.size': 18})
z_in, n_particles, n_turns = _check_input(SC, custom_bunch, nParticles, nTurns)
T = at_wrapper.atpass(SC.RING, z_in, n_turns, ords, keep_lattice=False)
T = at_wrapper.lattice_track(SC.RING, z_in, n_turns, ords, keep_lattice=False)
T[:, np.isnan(T[0, :])] = np.nan
label_str = [r'$\Delta x$ [$\mu$m]', r"$\Delta x'$ [$\mu$rad]", r'$\Delta y$ [$\mu$m]', r"$\Delta y'$ [$\mu$rad]",
r'$\Delta S$ [m]', r'$\delta E$ $[\%]$']
Expand Down
4 changes: 2 additions & 2 deletions pySC/plotting/plot_support.py
Original file line number Diff line number Diff line change
Expand Up @@ -64,9 +64,9 @@ def plot_support(SC: SimulatedCommissioning, font_size: int = 8, x_lim: Tuple[fl
# Longitudinal offsets and Pitch and Yaw angles not supported for BPMs
pad_off, pad_roll = ((0, 0), (0, 1)), ((0, 0), (0, 2))
off_bpm = np.pad(at_wrapper.atgetfieldvalues(SC.RING, SC.ORD.BPM, "Offset"), pad_off)
roll_bpm = np.pad(at_wrapper.atgetfieldvalues(SC.RING, SC.ORD.BPM, "Roll"), pad_roll)
roll_bpm = np.pad(np.ravel(at_wrapper.atgetfieldvalues(SC.RING, SC.ORD.BPM, "Roll"))[:, np.newaxis], pad_roll)
off_bpm_support = np.pad(at_wrapper.atgetfieldvalues(SC.RING, SC.ORD.BPM, "SupportOffset"), pad_off)
roll_bpm_support = np.pad(at_wrapper.atgetfieldvalues(SC.RING, SC.ORD.BPM, "SupportRoll"), pad_roll)
roll_bpm_support = np.pad(np.ravel(at_wrapper.atgetfieldvalues(SC.RING, SC.ORD.BPM, "SupportRoll"))[:, np.newaxis], pad_roll)

# create figure
fig, ax = plt.subplots(nrows=9, ncols=2, num=1213, sharex="all", figsize=(10, 15))
Expand Down
14 changes: 5 additions & 9 deletions pySC/utils/at_wrapper.py
Original file line number Diff line number Diff line change
Expand Up @@ -6,7 +6,7 @@
which are not member functions of used ``pyAT`` objects.
This is due to observed side effects, such as modification of input parameters.
Tracking fuctions ``latice_pass``, ``find_orbit4``, ``find_orbit6``
Tracking fuctions ``latice_track``, ``find_orbit4``, ``find_orbit6``
index the result the same way as ``get_s_pos``,
i.e. 0 means entrance of the first element, len(elements) means end of the last element.
Function ``get_value_refpts`` indexes elements as usual.
Expand All @@ -21,14 +21,10 @@
from at import Lattice


def atpass(ring: Lattice, init_pos: ndarray, nturns: int, refpts: ndarray, keep_lattice: bool = False):
return at.lattice_pass(lattice=ring.copy(), r_in=init_pos.copy(), nturns=nturns, refpts=refpts,
keep_lattice=keep_lattice)


def patpass(ring: Lattice, init_pos: ndarray, nturns: int, refpts: ndarray, keep_lattice: bool = False):
return at.patpass(lattice=ring.copy(), r_in=init_pos.copy(), nturns=nturns, refpts=refpts,
keep_lattice=keep_lattice)
def lattice_track(ring: Lattice, init_pos: ndarray, nturns: int, refpts: ndarray, keep_lattice: bool = False,
use_mp: bool = False):
return at.lattice_track(lattice=ring.copy(), r_in=init_pos.copy(), nturns=nturns, refpts=refpts,
keep_lattice=keep_lattice, in_place=False, use_mp=use_mp)[0]


def atgetfieldvalues(ring: Lattice, refpts: ndarray, attrname: str, index: int = None):
Expand Down
18 changes: 12 additions & 6 deletions pySC/utils/sc_tools.py
Original file line number Diff line number Diff line change
Expand Up @@ -2,18 +2,20 @@

import numpy as np
from matplotlib import pyplot as plt
from numpy import ndarray
from numpy import ndarray, float64
from at import Lattice
from typing import Union

from pySC.utils import at_wrapper, logging_tools


LOGGER = logging_tools.get_logger(__name__)


def randnc(cut_off: float = 2, shape: tuple = (1,)) -> ndarray:
def randnc(cut_off: float = 2, shape: tuple = (1,)) -> Union[ndarray, float64]:
"""
Generates an array of random number(s) from normal distribution with a cut-off.
If shape = () a single float value is returned.
Parameters
----------
Expand All @@ -24,15 +26,19 @@ def randnc(cut_off: float = 2, shape: tuple = (1,)) -> ndarray:
Returns
-------
out : ndarray
The output array.
out : Union[ndarray, float64]
The output array (If shape = () a single float value)
"""
if np.any([s < 0 for s in shape]):
raise ValueError('The shape of the output array must be positive.')
out_shape = (1,) if np.sum(shape) < 1 else shape
out = np.random.randn(np.prod(out_shape))
outindex = np.abs(out) > np.abs(cut_off)
while np.sum(outindex):
out[outindex] = np.random.randn(np.sum(outindex))
outindex = np.abs(out) > np.abs(cut_off)
if np.sum(shape) < 1:
return out[0]
return out.reshape(out_shape)


Expand Down Expand Up @@ -165,10 +171,10 @@ def _plot_singular_values(s_mat, d_mat):
fig, ax = plt.subplots(nrows=1, ncols=2, figsize=(12, 4), dpi=100, facecolor="w")
ax[0].semilogy(np.diag(s_mat) / np.max(np.diag(s_mat)), 'o--')
ax[0].set_xlabel('Number of SV')
ax[0].set_ylabel('$\sigma/\sigma_0$')
ax[0].set_ylabel(r'$\sigma/\sigma_0$')
ax[1].plot(s_mat * d_mat, 'o--')
ax[1].set_xlabel('Number of SV')
ax[1].set_ylabel('$\sigma * \sigma^+$')
ax[1].set_ylabel(r'$\sigma * \sigma^+$')
ax[1].yaxis.tick_right()
ax[1].yaxis.set_label_position("right")
fig.show()
Expand Down
Loading

0 comments on commit 23bd8e5

Please sign in to comment.