-
Notifications
You must be signed in to change notification settings - Fork 49
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Remove camera creation from Observatory #612
Changes from 8 commits
85ffef8
bdd105e
7c07fc2
b9a2e1e
0088074
ea99f15
19e81df
8d5a9ac
ed5e115
e56465a
0d5efcb
46d36be
df3d577
81cc3b3
4cf2034
2e8780e
da3f59c
56a1797
d47bcfa
bc8fca3
867cb08
831c04b
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1,2 +1,157 @@ | ||
from collections import OrderedDict | ||
|
||
from pocs.utils import error | ||
from pocs.utils import load_module | ||
from pocs.utils.config import load_config | ||
|
||
from pocs.camera.camera import AbstractCamera # pragma: no flakes | ||
from pocs.camera.camera import AbstractGPhotoCamera # pragma: no flakes | ||
|
||
from pocs.utils import list_connected_cameras | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. FWIW, seems like the kind of function that could be moved to this file. |
||
from pocs.utils import logger as logger_module | ||
|
||
|
||
def create_cameras_from_config(config=None, logger=None, **kwargs): | ||
"""Creates a camera object(s) | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Creates camera object(s) based on the config. |
||
|
||
Loads the cameras via the configuration. | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Drop this line. |
||
|
||
Creates a camera for each camera item listed in the config. Ensures the | ||
appropriate camera module is loaded. | ||
|
||
Note: | ||
This does not actually make a connection to the camera. To do so, | ||
call 'camera.connect()' explicitly. | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. ... on each camera. |
||
|
||
Args: | ||
**kwargs (dict): Can pass a `cameras` object that overrides the info in | ||
the configuration file. Can also pass `auto_detect`(bool) to try and | ||
automatically discover the ports. | ||
|
||
Returns: | ||
OrderedDict: An ordered dictionary of created camera objects. | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. What is the key? And why is it ordered? There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Or None if there are no cameras declared in the config. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Added docs to describe. Changed so that an empty OrderedDict is returned instead of None and updated docs. |
||
|
||
Raises: | ||
error.CameraNotFound: Description | ||
error.PanError: Description | ||
""" | ||
if not logger: | ||
logger = logger_module.get_root_logger() | ||
|
||
if not config: | ||
config = load_config(**kwargs) | ||
|
||
if 'cameras' not in config: | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Why is this not after the line: camera_info = kwargs_or_config('cameras') |
||
logger.info('No camera information in config.') | ||
return None | ||
|
||
# Helper method to first check kwargs then config | ||
def kwargs_or_config(item, default=None): | ||
return kwargs.get(item, config.get(item, default)) | ||
|
||
camera_info = kwargs_or_config('cameras') | ||
a_simulator = 'camera' in kwargs_or_config('simulator', default=list()) | ||
auto_detect = kwargs_or_config('auto_detect', default=False) | ||
|
||
logger.debug("Camera config: {}".format(camera_info)) | ||
|
||
ports = list() | ||
|
||
# Lookup the connected ports if not using a simulator | ||
if not a_simulator and auto_detect: | ||
logger.debug("Auto-detecting ports for cameras") | ||
try: | ||
ports = list_connected_cameras() | ||
except Exception as e: | ||
logger.warning(e) | ||
|
||
if len(ports) == 0: | ||
raise error.PanError( | ||
msg="No cameras detected. Use --simulator=camera for simulator.") | ||
else: | ||
logger.debug("Detected Ports: {}".format(ports)) | ||
|
||
cameras = OrderedDict() | ||
primary_camera = None | ||
|
||
device_info = camera_info['devices'] | ||
for cam_num, device_config in enumerate(device_info): | ||
cam_name = 'Cam{:02d}'.format(cam_num) | ||
|
||
if not a_simulator: | ||
camera_model = device_config.get('model') | ||
|
||
# Assign an auto-detected port. If none are left, skip | ||
if auto_detect: | ||
try: | ||
camera_port = ports.pop() | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. I'm perplexed by this. If the config file lists two canon cameras with the prefixes of their serial numbers, how do we know the order in which to pop the ports? There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. This is only if auto_detect is true, which means that it is ignoring the config. Serial number is set when camera object is initialized by lookup (via gphoto2) There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Ah. Then I may have misunderstood how to auto_detect and primary work. I included the serial number prefixes and primary:True in pocs_local.yaml on PAN006. My goal is to make sure that a specific camera is treated as primary every time. And ideally we'd get a warning if a specified camera is not found, or if another camera is found. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. I think #95 is still true unfortunately. This function could certainly be cleaned up some. Happy to do it here if we want just to make sure it gets done soon but I imagine it will all need to be revisited anyway if we set up separate process controls for the camera. Which hopefully happens soon. :) |
||
except IndexError: | ||
logger.warning( | ||
"No ports left for {}, skipping.".format(cam_name)) | ||
continue | ||
else: | ||
try: | ||
camera_port = device_config['port'] | ||
except KeyError: | ||
raise error.CameraNotFound( | ||
msg="No port specified and auto_detect=False") | ||
|
||
camera_focuser = device_config.get('focuser', None) | ||
camera_readout = device_config.get('readout_time', 6.0) | ||
|
||
else: | ||
logger.debug('Using camera simulator.') | ||
# Set up a simulated camera with fully configured simulated | ||
# focuser | ||
camera_model = 'simulator' | ||
camera_port = '/dev/camera/simulator' | ||
camera_focuser = {'model': 'simulator', | ||
'focus_port': '/dev/ttyFAKE', | ||
'initial_position': 20000, | ||
'autofocus_range': (40, 80), | ||
'autofocus_step': (10, 20), | ||
'autofocus_seconds': 0.1, | ||
'autofocus_size': 500} | ||
camera_readout = 0.5 | ||
|
||
camera_set_point = device_config.get('set_point', None) | ||
camera_filter = device_config.get('filter_type', None) | ||
|
||
logger.debug('Creating camera: {}'.format(camera_model)) | ||
|
||
try: | ||
module = load_module('pocs.camera.{}'.format(camera_model)) | ||
logger.debug('Camera module: {}'.format(module)) | ||
except ImportError: | ||
raise error.CameraNotFound(msg=camera_model) | ||
else: | ||
# Create the camera object | ||
cam = module.Camera(name=cam_name, | ||
model=camera_model, | ||
port=camera_port, | ||
set_point=camera_set_point, | ||
filter_type=camera_filter, | ||
focuser=camera_focuser, | ||
readout_time=camera_readout) | ||
|
||
is_primary = '' | ||
if camera_info.get('primary', '') == cam.uid: | ||
primary_camera = cam | ||
is_primary = ' [Primary]' | ||
|
||
logger.debug("Camera created: {} {} {}".format( | ||
cam.name, cam.uid, is_primary)) | ||
|
||
cameras[cam_name] = cam | ||
|
||
if len(cameras) == 0: | ||
raise error.CameraNotFound( | ||
msg="No cameras available. Exiting.", exit=True) | ||
|
||
# If no camera was specified as primary use the first | ||
if primary_camera is None: | ||
primary_camera = cameras['Cam00'] | ||
|
||
logger.debug("Finished creating cameras from config") | ||
|
||
return cameras |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
FWIW, while thinking about adding a supervisor and better messaging, and also while reviewing Anthony's distributed cameras PR (which includes a nameserver), I wondered about introducing some kind of device registry. Something we can discuss in the future.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Yes, I think we will definitely want to change how the Observatory interacts with and discovers devices. Hopefully this gets us closer to doing that.