2015-08-28 18:19:13 +00:00
|
|
|
# Copyright 2015 Canonical, Ltd.
|
|
|
|
#
|
|
|
|
# This program is free software: you can redistribute it and/or modify
|
|
|
|
# it under the terms of the GNU Affero General Public License as
|
|
|
|
# published by the Free Software Foundation, either version 3 of the
|
|
|
|
# License, or (at your option) any later version.
|
|
|
|
#
|
|
|
|
# This program is distributed in the hope that it will be useful,
|
|
|
|
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
# GNU Affero General Public License for more details.
|
|
|
|
#
|
|
|
|
# You should have received a copy of the GNU Affero General Public License
|
|
|
|
# along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
|
|
|
|
import logging
|
2015-10-23 15:03:04 +00:00
|
|
|
import yaml
|
2015-08-28 18:19:13 +00:00
|
|
|
import os
|
2017-11-15 01:27:24 +00:00
|
|
|
from probert.network import (StoredDataObserver, UdevObserver)
|
2015-08-28 18:19:13 +00:00
|
|
|
from probert.storage import (Storage,
|
|
|
|
StorageInfo)
|
|
|
|
|
2016-06-30 18:17:01 +00:00
|
|
|
log = logging.getLogger('subiquitycore.prober')
|
2015-08-28 18:19:13 +00:00
|
|
|
|
2015-11-02 22:48:37 +00:00
|
|
|
|
2015-10-23 15:03:04 +00:00
|
|
|
class ProberException(Exception):
|
|
|
|
'''Base Prober Exception'''
|
|
|
|
pass
|
2015-09-03 18:35:07 +00:00
|
|
|
|
2015-11-02 22:48:37 +00:00
|
|
|
|
2015-08-28 18:19:13 +00:00
|
|
|
class Prober():
|
|
|
|
def __init__(self, opts):
|
|
|
|
self.opts = opts
|
2017-11-15 01:27:24 +00:00
|
|
|
|
2015-08-28 18:19:13 +00:00
|
|
|
self.probe_data = {}
|
2017-11-15 01:27:24 +00:00
|
|
|
self.saved_config = None
|
2015-08-28 18:19:13 +00:00
|
|
|
|
|
|
|
if self.opts.machine_config:
|
|
|
|
log.debug('User specified machine_config: {}'.format(
|
2015-09-03 18:35:07 +00:00
|
|
|
self.opts.machine_config))
|
2017-11-15 01:27:24 +00:00
|
|
|
self.saved_config = \
|
|
|
|
self._load_machine_config(self.opts.machine_config)
|
|
|
|
self.probe_data = self.saved_config
|
|
|
|
log.debug('Prober() init finished, data:{}'.format(self.saved_config))
|
2015-08-28 18:19:13 +00:00
|
|
|
|
2015-10-23 15:03:04 +00:00
|
|
|
def _load_machine_config(self, machine_config):
|
|
|
|
with open(machine_config) as mc:
|
|
|
|
try:
|
|
|
|
data = yaml.safe_load(mc)
|
|
|
|
except (UnicodeDecodeError, yaml.reader.ReaderError):
|
|
|
|
err = 'Failed to parse machine config'
|
|
|
|
log.exception(err)
|
|
|
|
raise ProberException(err)
|
|
|
|
|
|
|
|
return data
|
|
|
|
|
2017-11-15 01:27:24 +00:00
|
|
|
def probe_network(self, receiver):
|
|
|
|
if self.opts.machine_config:
|
|
|
|
observer = StoredDataObserver(
|
|
|
|
self.saved_config['network'], receiver)
|
|
|
|
else:
|
|
|
|
observer = UdevObserver(receiver)
|
|
|
|
return observer, observer.start()
|
|
|
|
|
2015-08-28 18:19:13 +00:00
|
|
|
def get_storage(self):
|
|
|
|
''' Load a StorageInfo class. Probe if it's not present '''
|
|
|
|
if 'storage' not in self.probe_data:
|
|
|
|
log.debug('get_storage: no storage in probe_data, fetching')
|
|
|
|
storage = Storage()
|
|
|
|
results = storage.probe()
|
|
|
|
self.probe_data['storage'] = results
|
|
|
|
|
|
|
|
return self.probe_data['storage']
|
|
|
|
|
|
|
|
def get_storage_info(self, device):
|
|
|
|
''' Load a StorageInfo class for specified device '''
|
|
|
|
return StorageInfo({device: self.get_storage().get(device)})
|