#!/usr/bin/env python3
-from dataclasses import dataclass
+"""Location/site dependent data."""
+
import logging
import platform
-from typing import Callable
+from dataclasses import dataclass
+from typing import Callable, Optional
# Note: this module is fairly early loaded. Be aware of dependencies.
import config
-import presence
+from type.locations import Location
logger = logging.getLogger(__name__)
+
args = config.add_commandline_args(
- f'({__file__})',
- 'Args related to __file__'
+ f'Global Site Config ({__file__})',
+ 'Args related to global site-specific configuration',
)
args.add_argument(
'--site_config_override_location',
default='NONE',
const='NONE',
nargs='?',
- choices=('HOUSE', 'CABIN', 'NONE'),
- help='Where are we, HOUSE, CABIN?',
+ choices=['HOUSE', 'CABIN', 'NONE'],
+ help='Where are we, HOUSE, CABIN? Overrides standard detection code.',
)
@dataclass
class SiteConfig(object):
- location: str
+ """The set of information specific to where the program is running."""
+
+ location_name: str
+ location: Location
network: str
network_netmask: str
network_router_ip: str
- presence_location: presence.Location
- is_anyone_present: Callable[None, bool]
+ presence_location: Location
+ is_anyone_present: Callable
arper_minimum_device_count: int
+ arper_cache_file: str
-def get_location():
+def get_location_name():
"""
Where are we?
- >>> location = get_location()
+ >>> location = get_location_name()
>>> location == 'HOUSE' or location == 'CABIN'
True
+ """
+ return get_config().location_name
+
+
+def get_location():
+ """
+ Returns location as an enum instead of a string.
+
+ >>> from type.locations import Location
+ >>> location = get_location()
+ >>> location == Location.HOUSE or location == Location.CABIN
+ True
+
"""
return get_config().location
-def is_anyone_present_wrapper(location: presence.Location):
- p = presence.PresenceDetection()
+def is_anyone_present_wrapper(location: Location):
+ import base_presence
+
+ p = base_presence.PresenceDetection()
return p.is_anyone_in_location_now(location)
-def get_config():
+def other_location() -> str:
+ """
+ Returns the location where this program is _NOT_ running.
+
+ >>> x = other_location()
+ >>> x in set(['HOUSE', 'CABIN'])
+ True
+
+ """
+ hostname = platform.node()
+ if '.house' in hostname:
+ location = 'CABIN'
+ elif '.cabin' in hostname:
+ location = 'HOUSE'
+ else:
+ raise Exception(f"{hostname} doesn't help me know where I'm running?!")
+ return location
+
+
+def this_location() -> str:
+ """
+ Returns the location where this program _IS_ running.
+
+ >>> x = this_location()
+ >>> x in set(['HOUSE', 'CABIN'])
+ True
+
+ """
+ hostname = platform.node()
+ if '.house' in hostname:
+ location = 'HOUSE'
+ elif '.cabin' in hostname:
+ location = 'CABIN'
+ else:
+ raise Exception(f"{hostname} doesn't help me know where I'm running?!")
+ return location
+
+
+def effective_location(location_override: Optional[str] = None) -> str:
+ """Detects and returns a location taking into account two override
+ mechanisms.
+
+ >>> x = effective_location()
+ >>> x in set(['HOUSE', 'CABIN'])
+ True
+
+ >>> effective_location('HOUSE')
+ 'HOUSE'
+
+ """
+ if location_override is None:
+ try:
+ location_override = config.config['site_config_override_location']
+ except KeyError:
+ location_override = None
+
+ if location_override is None or location_override == 'NONE':
+ location = this_location()
+ else:
+ logger.debug('site_config\'s location_override was set to: %s', location_override)
+ location = location_override
+ return location
+
+
+def get_config(location_override: Optional[str] = None):
"""
Get a configuration dataclass with information that is
site-specific including the current running location.
>>> cfg = get_config()
- >>> cfg.location == 'HOUSE' or cfg.location == 'CABIN'
+ >>> cfg.location_name == 'HOUSE' or cfg.location_name == 'CABIN'
True
"""
- hostname = platform.node()
- try:
- location_override = config.config['site_config_override_location']
- except KeyError:
- location_override = 'NONE'
- if location_override == 'NONE':
- if '.house' in hostname:
- location = 'HOUSE'
- elif '.cabin' in hostname:
- location = 'CABIN'
+ location = effective_location(location_override)
if location == 'HOUSE':
return SiteConfig(
- location = 'HOUSE',
- network = '10.0.0.0/24',
- network_netmask = '255.255.255.0',
- network_router_ip = '10.0.0.1',
- presence_location = presence.Location.HOUSE,
- is_anyone_present = lambda x=presence.Location.HOUSE: is_anyone_present_wrapper(x),
- arper_minimum_device_count = 50,
+ location_name='HOUSE',
+ location=Location.HOUSE,
+ network='10.0.0.0/24',
+ network_netmask='255.255.255.0',
+ network_router_ip='10.0.0.1',
+ presence_location=Location.HOUSE,
+ is_anyone_present=lambda x=Location.HOUSE: is_anyone_present_wrapper(x),
+ arper_minimum_device_count=50,
+ arper_cache_file='/home/scott/cache/.arp_table_cache_house',
)
elif location == 'CABIN':
return SiteConfig(
- location = 'CABIN',
- network = '192.168.0.0/24',
- network_netmask = '255.255.255.0',
- network_router_ip = '192.168.0.1',
- presence_location = presence.Location.CABIN,
- is_anyone_present = lambda x=presence.Location.CABIN: is_anyone_present_wrapper(x),
- arper_minimum_device_count = 15,
+ location_name='CABIN',
+ location=Location.CABIN,
+ network='192.168.0.0/24',
+ network_netmask='255.255.255.0',
+ network_router_ip='192.168.0.1',
+ presence_location=Location.CABIN,
+ is_anyone_present=lambda x=Location.CABIN: is_anyone_present_wrapper(x),
+ arper_minimum_device_count=15,
+ arper_cache_file='/home/scott/cache/.arp_table_cache_cabin',
)
else:
raise Exception(f'Unknown site location: {location}')
if __name__ == '__main__':
import doctest
+
doctest.testmod()