- #39 Implement transparency monitoring.
- Add
observer schedule-focus-sweep
command to Overwatcher actor to schedule a focus sweep before the next tile.
- #40 Slight internal restructuring of the core classes
Gort
,GortClient
, device and remote actor classes. The main goal was to avoid any other part of the library knowing aboutGortClient
, which does not include anything not related to its AMQP client function any more. - Use
Retrier
fromlvmopstools
to handle remote command retries. - Prevent repeat notifications with the same message.
- Add retries to NPS commands.
- Prevent trying to observe while a calibration is ongoing even if it's night.
- Add
max_start_time
tobias_sequence
to prevent if from running after twilight.
- #41 Only emit an error event when the exception is actually raised.
- Report if the observer is focusing or troubleshooting.
- Stop MoTAN devices before a new move and improve error reporting.
- Allow fibre selector to rehome if a move fails.
- Fix a bug that would cause the calibration module to always add a night log comment indicating that the calibration had failed.
- Prevent the K-mirror from being homed and parked at the same time.
- Overwatcher now reports error events via notifications. If the error happens while a tile is being observed, a comment in the night log is added.
- Roll over the GORT log when the SJD changes.
- Improve the logic handling how the Overwatcher observer decides when to open or close the dome near evening or morning twilight.
- Run a clean-up first in pre-observing in case the spectrographs are not in a good state.
- Run some pre-observing checks before calling each
GortObserver.observe_tile()
in theObserverOverwatcher
. Currently only checks if the spectrographs have an error state and resets them. - Handle
SPECTROGRAPH_NOT_IDLE
errors in the troubleshooter. - Disable the Overwatcher and cancel observations if the dome fails to move.
- Add retries for safe enclosure operations.
- Fixed a bug that would prevent a new SJD to trigger an update of the ephemeris and calibrations.
- Fix a bug that would leave the Overwatcher in cancelling mode if
start_observing
failed.
- Take AG darks during the pre-observing task.
- Add
retry_without_parking
option to the shutdown recipe. - Emit events for dome opening and closing and report them as notifications.
- Modified
emergency_shutdown()
to close the dome if the shutdown recipe fails.
- Prevent the calibrations module from trying to close the dome when a calibration is retrying.
- Prevent a case in which failing to park the telescopes could have caused the dome to not be closed even if
retry_without_parking
was set toTrue
.
- #38 Add a post-observing daily task that runs 15 minutes after morning twilight and will do a few check (make sure the dome is closed, park the telescopes, etc.) and retry safe calibrations that failed during the normal sequence.
- Add a comment to the night log when a calibration fails.
- Do not start exposure if we are within 10 minutes of twilight.
- Prevent the Overwatcher observer from opening the dome while calibrations are ongoing.
- Fixed a bug in the twilight flats recipe related to the extra exposures.
- Use API to send notifications.
GortObserver.observe_tile
now default toasync_readout=False
. This will block until the exposure is done, which is a more natural behaviour for an external user that is not trying to over-optimise things. The code that usesobserve_tile
in GORT (Gort.observe()
andObserverOverwatcher.observe_loop_task()
) have been updated to explicitly useasync_readout=True
.
- #37 Basic implementation of the
Troubleshooter
class for the Overwatcher. Currently only very broad troubleshooting checks and recipes are implemented.
- Removed morning twilight flats.
- Temporary fix in the cleanup recipe for a bug in
lvmscp
caused by a quick reset after reading out a pending exposure.
- Added a very basic test to confirm the Overwatcher can be initialised.
- #34 Adds a
safety
module to the Overwatcher that will monitor the alerts independently and close the dome at a very low level if the main task fails to do it after 5 minutes.
- #35 Refactor dither observing to allow finer control of when to reacquire a tile and when to keep observing and adjust the science telescope dither position.
- #36 Modity the evening twilight recipe to continue cycling the standards mask and taking flats until the exposure time reaches 100s.
- GORT will fail to initialise if the Overwatcher is running. This can be overridden by passing
override_overwatcher=True
to theGort
constructor or--override-overwatcher
in the CLI. - Rearranged the Overwatcher helpers and make the pre- and post-observing scripts recipes.
- Added a framework for run daily tasks and a pre-observing task.
- Add option to disable the overwatcher to the shutdown recipe.
- Overwatcher: cancel the current cancellation if conditions are unsafe.
- Overwatcher: lock the dome for 30 minutes if it's closed due to unsafe conditions.
- Always reset the spectrographs before an exposure. Add some checks to ensure the shutters are closed during a cleanup.
- Overwatcher: rename allow dome calibrations to allow calibrations, which enabled/disables all calibration (not only in-dome calibrations).
- Overwatcher: fixes to daytime logic.
- Add the current GORT version to the Overwatcher actor.
- Set the correct dither position in
GortObserver
.
- #32 Initial complete implementation of the overwatcher.
- #33 Fixed an issue in which the first standard would not be reacquired during dithered observations. Also forces a rehoming of the fibre selector mask before each observation.
- Removed the websocket code and CLI. All this functionality is now part of
lvmapi
.
- Wait for previous exposure to finish reading out in
Observer.expose()
. While this was already happening when callingExposure.expose()
, we are now blocking until the exposure finishes a bit earlier which prevents the standard loop to begin too early. - Upgraded to
polars
1.0.0.
- Replace all instances of Pandas with Polars.
- In
Gort.observe()
, retry after a wait period if the scheduler cannot find a valid tile to observe. - Output a message after the shutdown recipe instructing observers to check the dome and confirm it's fully closed.
- Manually refresh the progress bar only once a second, and only after an update. Hopefully this helps with the event loop getting sluggish after a while.
- Turn off spectrograph room lights on startup.
- Format code using
ruff
and updated dependencies and workflows.
- #23 Added observer calibration scripts as recipes
quick_cals
,bias_sequence
, andtwilight_flats
.
- Fixed a serious issue in which if the slew fails, the retry did not send the PA and it would default to PA=0.
- After the initial slew, subsequent slews of the spec telescope do not check for whether the enclosure state is remote.
- #20 Added
GuiderSet.monitor()
which can be used to guide at zenith with low cadence and monitor transparency and FWHM while conditions are bad.
- #19 Improvemts to focusing. After a focus sweep, the focus is adjusted based on the delta temperature before each exposure.
- #21 Added
GuiderMonitor
which replacesbuild_guider_reply_list
. All Pandas dataframes now usepyarrow
dtypes, and some deprecations and errors have been addressed.
- Tile standard coordinates can now be defined as a list of Gaia DR3 source IDs. If the standard coordinates is a single integer, it will be assumed to be a source ID and the coordinates will be retrieved from the database.
- Added code for telescope pointing models.
- Break out of `Gort.observe()`` on keyboard interrupt
- Added
utilities_room
light to enclosure. - Update astropy to 6.0.0 and avoid it requiring to download orientation data from the internet.
- Fixed exposure loop not restarting after a keyboard interrupt.
- #13 Adds a signal handler for
SIGINT
andSIGTERM
. WhenGortObserver
is running and an interrupt is received the signal handler will run the cleanup routine. - Added
spectrograph_room
light to enclosure.
SpectrographSet.reset
now acceptsfull=True
which performs additional checks and opens/closes the shutter and hartmann doors as needed.reset
withfull=True
is run automatically during thecleanup
recipe.- The GORT websocket now uses the CLU model for
lvmecp
instead of asking the PLC to report its status on each call toWebsocketServer.enclosure_status()
.
- #14 Turn off dome lights as part of the
startup
andcleanup
recipes. - Use `lvmnps all-off`` to turn off all lamps faster.
- Disable marking tiles as bad on error for now. This could cause confusion if an exposure failed and the tile was marked bad, but then the images were recovered using lockfiles.
- Skip the calibration sequence on startup by default.
- Added
selfie
position.
- Added NPS devices for telescopes and MOCON.
- Added spectrograph exposure and enclosure timeouts.
- Support exposing only some spectrographs.
- Verify the number of files writtent to disk by the spectrographs and their MD5s.
- Mark exposure bad in post-readout if it fails.
- Relaxed spec and sky telescope guiding tolerances to 3 arcsec.
- Updated
NPS
to handlelvmnps
1.0.0 (including upgrade tounclick
0.1.0).
- File logger logs to
logging.path
(defaults to/data/logs/lvmgort/<SJD>.log
).
- Disable guider corrections for
spec
telescope after acquisition. This prevents field rotation affecting the exposure at the cost of tracking error. - Pass
-PA
to the guider due to different handiness for PA between the guider and the tiling database.
- Fixed a bug in the recording of the standards acquisition and exposure times.
- Pass
force
to_prepare_telescopes()
.
- Do not run cleanup after each iteration in
Gort.observe()
- Set
stop_degs_before.sci
to 1.5.
- #11 Support asynchronous readout during
GortObserved
-enabled observations and implement dithering:- Remove calls to
lvmguider stop --now
. The newstop
command always cancels the guider task. - Added
pre-readout
hook forExposure
. - Allow
GortObserver
to read out an exposure asynchronously. Gort.observe_tile()
will by default read the last exposure asynchronously to allow for the next slew and acquisition to happen during readout.- Added a context manager
GortObserver.register_overhead()
that measured the elapsed time in different parts of the slew and acquisition process and records the values to the database. - The dither positions associated with a
Tile
are now a list of positions that the scheduler wants us to observe.Gort.observe_tile()
will try to observe all dither positions without stopping the guider. - Added
Gort.observe()
with a simple loop for observing tiles.
- Remove calls to
- Add dither information to
Tile
and support exposing multiple dither positions in the same observation.
- Pass the observed sky and standard pks to the scheduler during registration.
- The K-mirror offset angle is now wrapped between -180 and 180 degrees. Hopefully this prevents some cases in which offsets angles very close to 360 degrees cause the K-mirror to fail because of a software limit.
- Fixed deprecations in Pandas
fillna()
withmethod='bfill'
.
- Added support for telescope enclosure lights.
- #10
GortObserver.slew()
and the standard iteration task now calculate the adjusted target coordinates for the spec telescope so that the slew puts the star on top of the desired fibre to be observed. Most of this relies on code copied fromlvmtipo
used to calculate the field angle through the siderostat. SkyCoordinates
now accepts aname
attribute that is passed to theSKYENAME
andSKYWNAME
header keywords.- IFU PA angles passed to the headers.
- Pass exposure time when registering an exposure.
- Support new outputs from
lvmguider 0.4.0b1
. - Get actor versions on init.
- Support guiding in PA (disabled for now).
- Reconnect AGs during
startup
recipe. - Add
tile_id
as `OBJECT`` if no object string user-defined.
- Set
GR{telescope}FR{0|N}
with the range of guider frames in which we were guiding (as opposed to the entire guide loop including acquisition frames). - Changed default guide tolerance to 1 arcsec.
- Fixed some additional issues with the exception hooks that cause a recursion loop in IPython in some cases.
- Fixed a bug that would accumulate the range of guider frames for an spectrograph exposure if
GortObserver.expose()
was called multiple times. - Fixed a slicing issue with newer versions of Pandas.
- Fixed a bug that caused the exposure flavour not being passed to the headers.
- Deal with cases when the scheduler cannot find a good tile to observe.
- Guider frames for each telescope taken during an exposure are added to the spectrograph headers.
- Standards observed are added to the spectrograph headers. The standard information can be accessed as
GortObserver.standards
. - Added a
cleanup
recipe. - Added exposure frame to the progress bar.
- Add
PA
parameter to tiles and propagate it to the k-mirror slew commands. - Use
rich
tracebacks. This can be disabled by callingGort
withuse_rich_output=False
. Tracebacks generated in an interactive IPython session are now saved to the file log. - Log
Gort
output to a temporary file by default.Gort
can also be called with an argumentlog_file_path
to indicate where to save the log, or withlog_file_path=False
to disable logging. The current file log path can be retrieved asGort.get_log_path()
.
- #9 Refactoring of
Exposure
andSpectrographSet.expose()
. Most of the code in the latter has been moved to the former. Added a system of hooks for the exposure process. For now, only apre-readout
hook that is called with the header before the exposure is read. - Concurrently stop spec guiding and reslew.
- Pass
seg_time
andseg_min_num
toslewStart
. - The code used to build the spectrograph header has been cleaned up and consolidated in
GortObserver
.
- Explicitely define site coordinates to avoid astropy needing to download files.
- #5 Add option to restart deployments associated with a device set as
GortDeviceSet.restart()
. - #7 If guiding during an
Exposure
, a Pandas DataFrame is created with a summary of the guider data. The guider outputs summary guider information as INFO level messages. - #8 Added framework for recipes along recipes for startup, shutdown, and calibration.
- Add
TelescopeSet.restart_lvmtan()
to restart Twice-As-Nice controller. - TAN devices will timeout when homing or moving.
- Add pointing information to the exposure.
- Add
TelemetrySet
andTelemetry
devices. - Use the focus-temperature relationship to determine the initial guess for focusing.
- Stop the guiders with
now=True
after 60 seconds. - The object keyword values is set as an attribute in
Exposure
and shown in the__repr__
. Coordinates.set_mf_pixel()
is now available for all coordinates.- Allow to wait only for some telescopes to converge guiding.
- Check if telescopes are already parked before opening/closing the dome.
- Report summary of best focus in
focus()
command as INFO level. - Print last guider measurement before removing NaNs.
- Better control of which TAN devices to home during
Telescope.home()
orTelescopeSet.home()
. The previoushome_subdevices
argument has been replaced byhome_kms
,home_telescopes
,home_focusers
, andhome_fibsel
. - Restore previous focuser positions after homing.
- Allow multiple exposures in
GortObserver.expose()
- Focus initial guesses are now included in the configuration. If
Gort.guiders.focus()
is called without a guess, the configuration values are used. The guess can now also be a dictionary of telescope name to guess value. set_mf_pixel()
can now be used with all coordinates.- Better logging of remote actor command warnings and errors.
- Spec telescope will now slew to coordinates such that the target falls on the selected fibre.
- Check if TAN devices are reachable.
- Improvements to the documentation.
- Removed the options
min_skies
andrequire_spec
forGortObserver.acquire()
andGort.observe_tile()
. If no skies or standards are defined a warning is issued but the code will not fail.
- Lint using
ruff
. - Improve style of the documentation.
- Move
Exposure
to its own file.
- Added the
testcal
calibration sequence.
- Updated the calibrations section in the documentation.
- Prevent multiple devices reconnecting to RabbitMQ at the same time.
- Set the
OBJECT
header keyword in spectrograph exposures. - Telescopes keep track of whether they have been homed.
- Telescopes will move to park automatically when trying to open or close the dome.
- Fixed an issue that would cause object exposures to be taken as biases.
- Fixed progress bar being affected by stdout and logs by setting the same
rich
Console
object for logs and progress bars. - Fixed accessing the configuration from an
Exposure
object.
SpectrographSet.expose()
now accepts acount
parameters to expose multiple frames.- The parameter
flavour
forSpectrographSet.expose()
is not explicitely exposed and documented. SpectrographSet.calibrate()
accepts aslew_telescopes
parameters. Whether the telescopes are slewed is also determined by the type of calibrations to take.- Various improvements to the calibration schema.
flavour
,warmup
, and thefibsel
parameters now have defaults in the configuration file to which the calibration routine will revert to. Iffibsel: true
, a default mask iteration will be performed. Both darks and lamps now acceptexposure_time
which can be a float or a list of floats.
SpectrographSet.calibrate()
does not park the telescope by default.
- Progress bars are now transient.
- Added support for spectrograph IEBs.
- Added
SpectrographSet.get_calibration_sequence()
. - Added
initialise
(init) andabort
methods for the spectrographs.
- #4 Use the
rich
library for status bars. It provides better style and works better with stdout and logger outputs. - Expose various parameters in
Gort.observe_tile()
. - Various fixes and additional routes for the websockets server.
- Allow to pass a calibration sequence as a dictionary, and validate against JSON schema.
- Avoid having all MoTan devices moving at once by introducing a delay.
- Update
unclick
to 0.1.0b5. - Use the
rich
library logger but customise it to look like the usual console formatter. - Allow to define the configuration file to use as
$GORT_CONFIG_FILE
.
- Made the
ScienceCoordinates.set_mf_pixel()
method public. - Allow to set master frame pixel in
ScienceCoordinates
withxz
tuple. - Added
fibre_to_master_frame()
transformation function. - Improve
__repr__
forTile
andGortObserver
. - Complete documentation for tiles, observing, and offsets.
- Initial version. Bugs are likely but most features are functional.
DeviceSet
classes for telescopes, AGs, guiders, enclosure, spectrographs, and NPS.Tile
,GortObserver
classes to perform observations.Kubernetes
class to interact with the Kubernetes cluster.- Various tools and transformation functions.
- Very preliminary websocket server.