slsDetectorPackage/pyctbgui/tests/unit/test_npy_writer.py
Dhanya Thattil 5b61ff24bb
Dev/pyctbgui merge (#960)
* added empty c extension

* added rotation to the decoding

* added color map, options and findex

* minor

* move checks to before acquisition

* added pixel map based decoder

* cleanup

* no thread creation for single thread processing

* added rotation and test to compare

* allow high and low water mark for zmq (also buffer size) for fast readouts

* removed roatation during decoding

* added Transpose to image and invert Y False to invert it

* retains the zoomed state after the first image of gui, catch and display exception if no detector connected

* moved start frame to dockable widget, removed extra frame number label, moved current measurement also to dockable widget, hide frame plot entirely when showing patternviewer

* first image dependin on which plot

* remember settings of main window size and position, dockewidget if docked, its size and posisiotn as well, then update it next time the gui is opened

* change in comment

* using c decoder for moench 04 and matterhorn

* catch exception from invalid image from decoder

* clean up

* update row and col when choosing image type, neeeded to show values

* fix for previous PR

* disable resetting colormap values
keep the range selected for every new acquisition

* fix typos + tested on virtual matterhorn

* minor print

* refactored Slow ADCs Tab

* refactored DAC tab

* refactored power supplies

* refactored signals tab

* refactored transceiver tab

* fix typo

* fix typo2

* remove commented code

* delete commented code

* delete commented code

* delete commented signals code

* remove commented code for transceiver tab

* refactor adc tab

* refactor Pattern Tab

* Refactor transceivers tab (PR#5) (#118)

* refactored transceiver tab

* remove commented code for transceiver tab

---------

Co-authored-by: Erik Frojdh <erik.frojdh@gmail.com>

* refactor adc tab (PR#6) (#119)


* refactor adc tab

* refactored Plot and Acquisition Tabs

* fix the regression issue

* restructure project files

* applying singleton and renaming tabs to services

* working install using pip

* applies singleton to tab classes and remove CI erros

* added pyzmq and pillow

* remove the singleton implementation and keep changes

* fix merge errors in mainWindow

* moved misplaced init file

* rename service to tab

* reorganize imports

* iterate over tabs

* reorder tabs

* add slowadc to the list

* saving changes (buggy)
power supply ui not showing in the gui

* split power supply tab

* fixed tests

* add hardcoded values to defines file

* fix error

* separate power supply

* fix errors for powerSuppliesTab

* split dacs

* split slow adcs

* split signals tab

* added tests for bit_utils

* add slowAdc class to defines

* split transceiver ui file

* split adc.ui

* split pattern ui file

* split plot and acquisition ui file

* added basic test for parsing bit names

* removed redundant code in read_alias_file

* fix dacs ui position

* testing for correct exception

* cmd and args at split

* group radio buttons

* fix comments from PR#1

* show legend

* added python version and dev requirements to setup.py

* fix dac issue

* moved _decoder into pkg

* added inplace build

* removed clear

* fixed dependencies

* make tests run without slsdet

* updated name of action

* define colcount

* fixed loading of alias file

* add yapf and ruff

* apply formatting

* fix E and F rules

* add more ruff rules

* change variable name

* squashing gh debugging commits and add pre-commit

* update label values to mv units

* add hook for yapf

* reconfigure yapf precommit hook

* add format and check_format to makefile

* change gh actions

* update readme

* added check_format

* WIP

* added linting in github action

* updated readme]

* add more control for color choice

* remove useless file

* fix un-updated line after refactoring Defines
BIT0_31_MASK is not found in Defines.signals

* visually improve the interface

* fix last commit

* add only selected plots for legend

* add hide legend button

* change hide legend to show legend
checkbox show legend is checked by default

* add support for saving in numpy

* solve conversations

* fix acq index offset

* fix browse button in pattern error

* fix other browse button errors

* finish tests and add usage.md

* remove buffer

* add file,numpy-like interface and tests

* remove useless .npy files

* subscriptible npz files

* remove useless files

* remove repetetive tests

* save changes

* add mode r+, add with support,remove logging

* remove offset of acqIndex between raw and numpy saving

* fix only saving last frame

* save signals of multiple devices

* add comments and move condition for clearer code

* fix bug when vieweing pattern file

* iterate over enabled and plotted plots

* add padestal substraction to transceiver and analog data

* init pedestal frames to detector.frames

* restore old exception

* add pedestal substraction for digital signals

* remove frames spinbox from plotTab

* remove comments and use str instead of Path

* avoid saving all frames

* correct exception and log error's trace

* add gui tests

* add waveform test

* add pedestal test

* refactor by using fixtures

* add tests for moench analog and pattern

* add pytest-qt to dependencies

* add save and load gui parameters

* remove nohup file

* fix old bug IndexError

* save plot type

* a

* handle canceling load, loading matterhorn pedestal for moench

* remove comparing .png files for pattern test

* save plot type

* red error on status bar when shape mismatch for loaded pedestal

* fix makefile and docstrings

* fix PRs conversation

* move code into different function

* fix wrong function names for power supply

* removed old ctbgui

* removed unnecessary files

---------

Co-authored-by: Erik Frojdh <erik.frojdh@gmail.com>
Co-authored-by: Braham Bechir <braham_b@pc11979.psi.ch>
Co-authored-by: Bechir <bechir.braham@psi.ch>
Co-authored-by: Bechir <bechir.brahem420@gmail.com>
2024-09-10 16:00:04 +02:00

161 lines
6.2 KiB
Python

import filecmp
from pathlib import Path
import pytest
from pyctbgui.utils.numpyWriter.npy_writer import NumpyFileManager
import numpy as np
def test_create_new_file(tmp_path):
npw = NumpyFileManager(tmp_path / 'tmp.npy', 'w', (400, 400), np.int32)
npw.writeOneFrame(np.ones([400, 400], dtype=np.int32))
npw.writeOneFrame(np.ones([400, 400], dtype=np.int32))
npw.writeOneFrame(np.ones([400, 400], dtype=np.int32))
npw.writeOneFrame(np.ones([400, 400], dtype=np.int32))
npw.close()
arr = np.load(tmp_path / 'tmp.npy')
assert arr.dtype == np.int32
assert arr.shape == (4, 400, 400)
assert np.array_equal(arr, np.ones([4, 400, 400], dtype=np.int32))
np.save(tmp_path / 'tmp2.npy', np.ones([4, 400, 400], dtype=np.int32))
assert filecmp.cmp(tmp_path / 'tmp.npy', tmp_path / 'tmp2.npy')
def test_open_old_file(tmp_path):
npw = NumpyFileManager(tmp_path / 'tmp.npy', 'w', (4000, ), np.float32)
npw.writeOneFrame(np.ones(4000, dtype=np.float32))
npw.writeOneFrame(np.ones(4000, dtype=np.float32))
npw.close()
npw2 = NumpyFileManager(tmp_path / 'tmp.npy', 'r+')
assert npw2.frameCount == 2
assert npw2.frameShape == (4000, )
assert npw2.dtype == np.float32
npw2.writeOneFrame(np.ones(4000, dtype=np.float32))
del npw2
np.save(tmp_path / 'tmp2.npy', np.ones([3, 4000], dtype=np.float32))
assert filecmp.cmp(tmp_path / 'tmp.npy', tmp_path / 'tmp2.npy')
@pytest.mark.parametrize('mode', ['w', 'x'])
def test_init_parameters2(mode, tmp_path):
# test opening files with missing parameters for write
with pytest.raises(AssertionError):
NumpyFileManager(tmp_path / 'abaababababa.npyx', mode)
with pytest.raises(AssertionError):
NumpyFileManager(tmp_path / 'abaababababa.npyx2', mode, frameShape=(12, 34))
with pytest.raises(AssertionError):
NumpyFileManager(tmp_path / 'abaababababa.npyx3', mode, dtype=np.int64)
# opening new file with required parameters (this should work)
NumpyFileManager(tmp_path / 'abaababababa.npyx3', mode, dtype=np.int64, frameShape=(6, 6))
assert Path.is_file(tmp_path / 'abaababababa.npyx3')
def test_init_parameters(tmp_path):
with pytest.raises(TypeError):
NumpyFileManager()
# test opening file that does not exist
with pytest.raises(FileNotFoundError):
NumpyFileManager(tmp_path / 'abaababababa.npyx')
with pytest.raises(FileNotFoundError):
NumpyFileManager(tmp_path / 'abaababababa.npyx2', frameShape=(12, 34))
with pytest.raises(FileNotFoundError):
NumpyFileManager(tmp_path / 'abaababababa.npyx3', dtype=np.int64)
with pytest.raises(FileNotFoundError):
NumpyFileManager(tmp_path / 'abaababababa.npyx3', dtype=np.int64, frameShape=(6, 6))
# re-opening the same file
NumpyFileManager(tmp_path / 'abaababababa.npyx3', 'w', dtype=np.int64, frameShape=(6, 6))
NumpyFileManager(tmp_path / 'abaababababa.npyx3')
# re-opening the file with wrong parameters
with pytest.raises(AssertionError):
NumpyFileManager(tmp_path / 'abaababababa.npyx3', frameShape=(6, 2))
with pytest.raises(AssertionError):
NumpyFileManager(tmp_path / 'abaababababa.npyx3', dtype=np.int32)
with pytest.raises(AssertionError):
NumpyFileManager(tmp_path / 'abaababababa.npyx3', dtype=np.float32, frameShape=(5, 5))
# test resetting an existing file
npw = NumpyFileManager(tmp_path / 'tmp4.npy', 'w', dtype=np.float32, frameShape=(5, 5))
npw.writeOneFrame(np.ones((5, 5), dtype=np.float32))
npw.close()
assert np.load(tmp_path / 'tmp4.npy').shape == (1, 5, 5)
npw = NumpyFileManager(tmp_path / 'tmp4.npy', 'w', dtype=np.int64, frameShape=(7, 7))
npw.flush()
assert np.load(tmp_path / 'tmp4.npy').shape == (0, 7, 7)
# test adding frames with the wrong shape to an existing file
with pytest.raises(ValueError, match=r'frame shape given \(9, 4, 4\) '):
npw.writeOneFrame(np.ones((9, 4, 4)))
def test_get_item(tmp_path):
rng = np.random.default_rng(seed=42)
arr = rng.random((1000, 20, 20))
npw = NumpyFileManager(tmp_path / 'tmp.npy', 'w', frameShape=(20, 20), dtype=arr.dtype)
for frame in arr:
npw.writeOneFrame(frame)
assert np.array_equal(npw[50:100], arr[50:100])
assert np.array_equal(npw[0:1], arr[0:1])
assert np.array_equal(npw[0:10000], arr)
assert np.array_equal(npw[999:1000], arr[999:1000])
assert np.array_equal(npw[999:1005], arr[999:1000])
assert np.array_equal(npw[49:300], arr[49:300])
assert np.array_equal(npw[88:88], arr[88:88])
assert np.array_equal(npw[0:0], arr[0:0])
assert np.array_equal(npw[0:77], arr[0:77])
assert np.array_equal(npw[77:0], arr[77:0])
with pytest.raises(NotImplementedError):
npw[-1:-3]
with pytest.raises(NotImplementedError):
npw[10:20:2]
with pytest.raises(NotImplementedError):
npw[-5:-87:5]
with pytest.raises(NotImplementedError):
npw[-5:-87:-5]
with pytest.raises(NotImplementedError):
npw.readFrames(-1, -4)
with pytest.raises(NotImplementedError):
npw.readFrames(0, -77)
with pytest.raises(NotImplementedError):
npw.readFrames(-5, -5)
def test_file_functions(tmp_path):
rng = np.random.default_rng(seed=42)
arr = rng.random((1000, 20, 20))
npw = NumpyFileManager(tmp_path / 'tmp.npy', 'w', frameShape=(20, 20), dtype=arr.dtype)
for frame in arr:
npw.writeOneFrame(frame)
assert np.array_equal(npw.read(10), arr[:10])
assert np.array_equal(npw.read(10), arr[10:20])
assert np.array_equal(npw.read(10), arr[20:30])
npw.readFrames(500, 600)
assert np.array_equal(npw.read(10), arr[30:40])
npw.readFrames(0, 2)
assert np.array_equal(npw.read(100), arr[40:140])
npw.writeOneFrame(arr[700])
assert np.array_equal(npw.read(5), arr[140:145])
npw.seek(900)
assert np.array_equal(npw.read(20), arr[900:920])
npw.seek(5)
npw.readFrames(500, 600)
assert np.array_equal(npw.read(10), arr[5:15])
def test_with_statement(tmp_path):
arr = np.ones((5, 5))
with NumpyFileManager(tmp_path / 'tmp.npy', 'w', (5, 5), arr.dtype) as npw:
npw.writeOneFrame(arr)
np.save(tmp_path / 'tmp2.npy', np.expand_dims(arr, 0))
assert filecmp.cmp(tmp_path / 'tmp2.npy', tmp_path / 'tmp.npy')