Initial Python skeleton and the model
classes to match the YAML schema
This commit is contained in:
parent
04d8a5e9d9
commit
07cb34e82d
13
helm_drydock/__init__.py
Normal file
13
helm_drydock/__init__.py
Normal file
@ -0,0 +1,13 @@
|
||||
# Copyright 2017 AT&T Intellectual Property. All other rights reserved.
|
||||
#
|
||||
# Licensed under the Apache License, Version 2.0 (the "License");
|
||||
# you may not use this file except in compliance with the License.
|
||||
# You may obtain a copy of the License at
|
||||
#
|
||||
# http://www.apache.org/licenses/LICENSE-2.0
|
||||
#
|
||||
# Unless required by applicable law or agreed to in writing, software
|
||||
# distributed under the License is distributed on an "AS IS" BASIS,
|
||||
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
# See the License for the specific language governing permissions and
|
||||
# limitations under the License.
|
35
helm_drydock/config.py
Normal file
35
helm_drydock/config.py
Normal file
@ -0,0 +1,35 @@
|
||||
# Copyright 2017 AT&T Intellectual Property. All other rights reserved.
|
||||
#
|
||||
# Licensed under the Apache License, Version 2.0 (the "License");
|
||||
# you may not use this file except in compliance with the License.
|
||||
# You may obtain a copy of the License at
|
||||
#
|
||||
# http://www.apache.org/licenses/LICENSE-2.0
|
||||
#
|
||||
# Unless required by applicable law or agreed to in writing, software
|
||||
# distributed under the License is distributed on an "AS IS" BASIS,
|
||||
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
# See the License for the specific language governing permissions and
|
||||
# limitations under the License.
|
||||
#
|
||||
|
||||
#
|
||||
# Read application configuration
|
||||
#
|
||||
|
||||
# configuration map with defaults
|
||||
|
||||
class DrydockConfig(object):
|
||||
|
||||
def __init__(self):
|
||||
self.selected_server_driver = helm_drydock.drivers.server.maasdriver
|
||||
self.selected_network_driver = helm_drydock.drivers.network.noopdriver
|
||||
self.control_config = {}
|
||||
self.ingester_config = {
|
||||
plugins = [helm_drydock.ingester.plugins.aicyaml.AicYamlIngester]
|
||||
}
|
||||
self.introspection_config = {}
|
||||
self.orchestrator_config = {}
|
||||
self.statemgmt_config = {
|
||||
backend_driver = 'helm_drydock.drivers.statemgmt.etcd',
|
||||
}
|
19
helm_drydock/drivers/__init__.py
Normal file
19
helm_drydock/drivers/__init__.py
Normal file
@ -0,0 +1,19 @@
|
||||
# Copyright 2017 AT&T Intellectual Property. All other rights reserved.
|
||||
#
|
||||
# Licensed under the Apache License, Version 2.0 (the "License");
|
||||
# you may not use this file except in compliance with the License.
|
||||
# You may obtain a copy of the License at
|
||||
#
|
||||
# http://www.apache.org/licenses/LICENSE-2.0
|
||||
#
|
||||
# Unless required by applicable law or agreed to in writing, software
|
||||
# distributed under the License is distributed on an "AS IS" BASIS,
|
||||
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
# See the License for the specific language governing permissions and
|
||||
# limitations under the License.
|
||||
|
||||
class ProviderDriver(object):
|
||||
|
||||
__init__(self):
|
||||
pass
|
||||
|
18
helm_drydock/drivers/server/__init__.py
Normal file
18
helm_drydock/drivers/server/__init__.py
Normal file
@ -0,0 +1,18 @@
|
||||
# Copyright 2017 AT&T Intellectual Property. All other rights reserved.
|
||||
#
|
||||
# Licensed under the Apache License, Version 2.0 (the "License");
|
||||
# you may not use this file except in compliance with the License.
|
||||
# You may obtain a copy of the License at
|
||||
#
|
||||
# http://www.apache.org/licenses/LICENSE-2.0
|
||||
#
|
||||
# Unless required by applicable law or agreed to in writing, software
|
||||
# distributed under the License is distributed on an "AS IS" BASIS,
|
||||
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
# See the License for the specific language governing permissions and
|
||||
# limitations under the License.
|
||||
|
||||
import helm_drydock.drivers.ProviderDriver
|
||||
|
||||
class ServerDriver(ProviderDriver):
|
||||
|
17
helm_drydock/drivers/server/maasdriver/__init__.py
Normal file
17
helm_drydock/drivers/server/maasdriver/__init__.py
Normal file
@ -0,0 +1,17 @@
|
||||
# Copyright 2017 AT&T Intellectual Property. All other rights reserved.
|
||||
#
|
||||
# Licensed under the Apache License, Version 2.0 (the "License");
|
||||
# you may not use this file except in compliance with the License.
|
||||
# You may obtain a copy of the License at
|
||||
#
|
||||
# http://www.apache.org/licenses/LICENSE-2.0
|
||||
#
|
||||
# Unless required by applicable law or agreed to in writing, software
|
||||
# distributed under the License is distributed on an "AS IS" BASIS,
|
||||
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
# See the License for the specific language governing permissions and
|
||||
# limitations under the License.
|
||||
import helm_drydock.drivers.server.ServerDriver
|
||||
|
||||
class MaasServerDriver(object):
|
||||
|
72
helm_drydock/ingester/ingester.py
Normal file
72
helm_drydock/ingester/ingester.py
Normal file
@ -0,0 +1,72 @@
|
||||
# Copyright 2017 AT&T Intellectual Property. All other rights reserved.
|
||||
#
|
||||
# Licensed under the Apache License, Version 2.0 (the "License");
|
||||
# you may not use this file except in compliance with the License.
|
||||
# You may obtain a copy of the License at
|
||||
#
|
||||
# http://www.apache.org/licenses/LICENSE-2.0
|
||||
#
|
||||
# Unless required by applicable law or agreed to in writing, software
|
||||
# distributed under the License is distributed on an "AS IS" BASIS,
|
||||
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
# See the License for the specific language governing permissions and
|
||||
# limitations under the License.
|
||||
#
|
||||
# ingester - Ingest host topologies to define site design and
|
||||
# persist design to helm-drydock's statemgmt service
|
||||
|
||||
import logging
|
||||
import yaml
|
||||
|
||||
class Ingester(object):
|
||||
|
||||
registered_plugins = {}
|
||||
|
||||
def __init__(self):
|
||||
logging.basicConfig(format="%(asctime)-15s [%(levelname)] %(module)s %(process)d %(message)s")
|
||||
self.log = logging.Logger("ingester")
|
||||
|
||||
"""
|
||||
enable_plugins
|
||||
|
||||
params: plugins - A list of class objects denoting the ingester plugins to be enabled
|
||||
|
||||
Enable plugins that can be used for ingest_data calls. Each plugin should use
|
||||
helm_drydock.ingester.plugins.IngesterPlugin as its base class. As long as one
|
||||
enabled plugin successfully initializes, the call is considered successful. Otherwise
|
||||
it will throw an exception
|
||||
"""
|
||||
def enable_plugins(self, plugins=[]):
|
||||
if len(plugin) == 0:
|
||||
self.log.error("Cannot have an empty plugin list.")
|
||||
|
||||
for plugin in plugins:
|
||||
try:
|
||||
new_plugin = plugin()
|
||||
plugin_name = new_plugin.get_name()
|
||||
registered_plugins[plugin_name] = new_plugin
|
||||
except:
|
||||
self.log.error("Could not enable plugin %s" % (plugin.__name__))
|
||||
|
||||
if len(registered_plugins) == 0:
|
||||
self.log.error("Could not enable at least one plugin")
|
||||
raise Exception("Could not enable at least one plugin")
|
||||
|
||||
"""
|
||||
ingest_data
|
||||
|
||||
params: plugin_name - Which plugin should be used for ingestion
|
||||
params: params - A map of parameters that will be passed to the plugin's ingest_data method
|
||||
|
||||
Execute a data ingestion using the named plugin (assuming it is enabled)
|
||||
"""
|
||||
def ingest_data(self, plugin_name, params={}):
|
||||
if plugin_name in registered_plugins:
|
||||
design_data = registered_plugins[plugin_name].ingest_data(params)
|
||||
# Need to persist data here, but we don't yet have the statemgmt service working
|
||||
yaml.dump(design_data)
|
||||
else
|
||||
self.log.error("Could not find plugin %s to ingest data." % (plugin_name))
|
||||
raise LookupError("Could not find plugin %s" % plugin_name)
|
||||
|
||||
|
55
helm_drydock/ingester/plugins/aicyaml.py
Normal file
55
helm_drydock/ingester/plugins/aicyaml.py
Normal file
@ -0,0 +1,55 @@
|
||||
# Copyright 2017 AT&T Intellectual Property. All other rights reserved.
|
||||
#
|
||||
# Licensed under the Apache License, Version 2.0 (the "License");
|
||||
# you may not use this file except in compliance with the License.
|
||||
# You may obtain a copy of the License at
|
||||
#
|
||||
# http://www.apache.org/licenses/LICENSE-2.0
|
||||
#
|
||||
# Unless required by applicable law or agreed to in writing, software
|
||||
# distributed under the License is distributed on an "AS IS" BASIS,
|
||||
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
# See the License for the specific language governing permissions and
|
||||
# limitations under the License.
|
||||
|
||||
#
|
||||
# AIC YAML Ingester - This data ingester will consume a AIC YAML design
|
||||
# file
|
||||
#
|
||||
import yaml
|
||||
import logging
|
||||
|
||||
import helm_drydock.ingester.plugins.IngesterPlugin
|
||||
|
||||
class AicYamlIngester(IngesterPlugin):
|
||||
|
||||
def __init__(self):
|
||||
super(AicYamlIngester, self).__init__()
|
||||
|
||||
def get_name(self):
|
||||
return "aic_yaml"
|
||||
|
||||
"""
|
||||
AIC YAML ingester params
|
||||
|
||||
filename - Absolute path to the YAML file to ingest
|
||||
"""
|
||||
def ingest_data(self, **kwargs):
|
||||
if 'filename' in params:
|
||||
input_string = read_input_file(params['filename'])
|
||||
parsed_data = parse_input_data(input_string)
|
||||
processed_data = compute_effective_data(parsed_data)
|
||||
else:
|
||||
|
||||
raise Exception('Missing parameter')
|
||||
|
||||
return processed_data
|
||||
|
||||
def read_input_file(self, filename):
|
||||
try:
|
||||
file = open(filename,'rt')
|
||||
except OSError as err:
|
||||
self.log.error("Error opening input file %s for ingestion: %s" % (filename, err))
|
||||
return {}
|
||||
|
||||
|
30
helm_drydock/ingester/plugins/ingester_plugin.py
Normal file
30
helm_drydock/ingester/plugins/ingester_plugin.py
Normal file
@ -0,0 +1,30 @@
|
||||
# Copyright 2017 AT&T Intellectual Property. All other rights reserved.
|
||||
#
|
||||
# Licensed under the Apache License, Version 2.0 (the "License");
|
||||
# you may not use this file except in compliance with the License.
|
||||
# You may obtain a copy of the License at
|
||||
#
|
||||
# http://www.apache.org/licenses/LICENSE-2.0
|
||||
#
|
||||
# Unless required by applicable law or agreed to in writing, software
|
||||
# distributed under the License is distributed on an "AS IS" BASIS,
|
||||
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
# See the License for the specific language governing permissions and
|
||||
# limitations under the License.
|
||||
#
|
||||
# Plugins to parse incoming topology and translate it to helm-drydock's
|
||||
# intermediate representation
|
||||
|
||||
import logging
|
||||
|
||||
class IngesterPlugin(object):
|
||||
|
||||
def __init__(self):
|
||||
self.log = logging.Logger('ingester')
|
||||
return
|
||||
|
||||
def get_data(self):
|
||||
return "ingester_skeleton"
|
||||
|
||||
def ingest_data(self, **kwargs):
|
||||
return {}
|
286
helm_drydock/model/__init__.py
Normal file
286
helm_drydock/model/__init__.py
Normal file
@ -0,0 +1,286 @@
|
||||
# Copyright 2017 AT&T Intellectual Property. All other rights reserved.
|
||||
#
|
||||
# Licensed under the Apache License, Version 2.0 (the "License");
|
||||
# you may not use this file except in compliance with the License.
|
||||
# You may obtain a copy of the License at
|
||||
#
|
||||
# http://www.apache.org/licenses/LICENSE-2.0
|
||||
#
|
||||
# Unless required by applicable law or agreed to in writing, software
|
||||
# distributed under the License is distributed on an "AS IS" BASIS,
|
||||
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
# See the License for the specific language governing permissions and
|
||||
# limitations under the License.
|
||||
#
|
||||
# Models for helm_drydock
|
||||
#
|
||||
|
||||
|
||||
class HardwareProfile(object):
|
||||
|
||||
def __init__(self, **kwargs):
|
||||
self.api_version = kwargs.get('apiVersion', '')
|
||||
|
||||
if self.api_version == "1.0":
|
||||
metadata = kwargs.get('metadata', {})
|
||||
spec = kwargs.get('spec', {})
|
||||
|
||||
# Need to add validation logic, we'll assume the input is
|
||||
# valid for now
|
||||
self.name = metadata.get('name', '')
|
||||
self.region = metadata.get('region', '')
|
||||
self.vendor = spec.get('vendor', '')
|
||||
self.generation = spec.get('generation', '')
|
||||
self.hw_version = spec.get('hw_version', '')
|
||||
self.bios_version = spec.get('bios_version', '')
|
||||
self.boot_mode = spec.get('boot_mode', '')
|
||||
self.bootstrap_protocol = spec.get('bootstrap_protocol', '')
|
||||
self.pxe_interface = spec.get('pxe_interface', '')
|
||||
self.devices = []
|
||||
|
||||
device_aliases = spec.get('device_aliases', {})
|
||||
|
||||
pci_devices = device_aliases.get('pci', [])
|
||||
scsi_devices = device_aliases.get('scsi', [])
|
||||
|
||||
for d in pci_devices:
|
||||
d['bus_type'] = 'pci'
|
||||
self.devices.append(
|
||||
HardwareDeviceAlias(self.api_version, **d))
|
||||
|
||||
for d in scsi_devices:
|
||||
d['bus_type'] = 'scsi'
|
||||
self.devices.append(
|
||||
HardwareDeviceAlias(self.api_version, **d))
|
||||
else:
|
||||
raise ValueError('Unknown API version of object')
|
||||
|
||||
return
|
||||
|
||||
|
||||
class HardwareDeviceAlias(object):
|
||||
|
||||
def __init__(self, api_version, **kwargs):
|
||||
self.api_version = api_version
|
||||
|
||||
if self.api_version == "1.0":
|
||||
self.bus_type = kwargs.get('bus_type', '')
|
||||
self.address = kwargs.get('address', '')
|
||||
self.alias = kwargs.get('alias', '')
|
||||
self.type = kwargs.get('type', '')
|
||||
else:
|
||||
raise ValueError('Unknown API version of object')
|
||||
|
||||
return
|
||||
|
||||
|
||||
class Site(object):
|
||||
|
||||
def __init__(self, **kwargs):
|
||||
self.api_version = kwargs.get('apiVersion', '')
|
||||
|
||||
if self.api_version == "1.0":
|
||||
metadata = kwargs.get('metadata', {})
|
||||
|
||||
# Need to add validation logic, we'll assume the input is
|
||||
# valid for now
|
||||
self.name = metadata.get('name', '')
|
||||
|
||||
self.networks = []
|
||||
self.network_links = []
|
||||
self.host_profiles = []
|
||||
self.hardware_profiles = []
|
||||
self.baremetal_nodes = []
|
||||
|
||||
else:
|
||||
raise ValueError('Unknown API version of object')
|
||||
|
||||
|
||||
class NetworkLink(object):
|
||||
|
||||
def __init__(self, **kwargs):
|
||||
self.api_version = kwargs.get('apiVersion', '')
|
||||
|
||||
if self.api_version == "1.0":
|
||||
metadata = kwargs.get('metadata', {})
|
||||
spec = kwargs.get('spec', {})
|
||||
|
||||
self.name = metadata.get('name', '')
|
||||
self.region = metadata.get('region', '')
|
||||
|
||||
bonding = spec.get('bonding', {})
|
||||
self.bonding_mode = bonding.get('mode', 'none')
|
||||
|
||||
# TODO How should we define defaults for CIs not in the input?
|
||||
if self.bonding_mode == '802.3ad':
|
||||
self.bonding_xmit_hash = bonding.get('hash', 'layer3+4')
|
||||
self.bonding_peer_rate = bonding.get('peer_rate', 'fast')
|
||||
self.bonding_mon_rate = bonding.get('mon_rate', '')
|
||||
self.bonding_up_delay = bonding.get('up_delay', '')
|
||||
self.bonding_down_delay = bonding.get('down_delay', '')
|
||||
|
||||
self.mtu = spec.get('mtu', 1500)
|
||||
self.linkspeed = spec.get('linkspeed', 'auto')
|
||||
|
||||
trunking = spec.get('trunking', {})
|
||||
self.trunk_mode = trunking.get('mode', 'none')
|
||||
|
||||
self.native_network = spec.get('default_network', '')
|
||||
else:
|
||||
raise ValueError('Unknown API version of object')
|
||||
|
||||
|
||||
class Network(object):
|
||||
|
||||
def __init__(self, **kwargs):
|
||||
self.api_version = kwargs.get('apiVersion', '')
|
||||
|
||||
if self.api_version == "1.0":
|
||||
metadata = kwargs.get('metadata', {})
|
||||
spec = kwargs.get('spec', {})
|
||||
|
||||
self.name = metadata.get('name', '')
|
||||
self.region = metadata.get('region', '')
|
||||
self.cidr = spec.get('cidr', '')
|
||||
self.allocation_strategy = spec.get('allocation', 'static')
|
||||
self.vlan_id = spec.get('vlan_id', 1)
|
||||
self.mtu = spec.get('mtu', 0)
|
||||
|
||||
dns = spec.get('dns', {})
|
||||
self.dns_domain = dns.get('domain', 'local')
|
||||
self.dns_servers = dns.get('servers', '')
|
||||
|
||||
ranges = spec.get('ranges', [])
|
||||
self.ranges = []
|
||||
|
||||
for r in ranges:
|
||||
self.ranges.append(NetworkAddressRange(self.api_version, **r))
|
||||
|
||||
routes = spec.get('routes', [])
|
||||
self.routes = []
|
||||
|
||||
for r in routes:
|
||||
self.routes.append(NetworkRoute(self.api_version, **r))
|
||||
else:
|
||||
raise ValueError('Unknown API version of object')
|
||||
|
||||
|
||||
class NetworkAddressRange(object):
|
||||
|
||||
def __init__(self, api_version, **kwargs):
|
||||
self.api_version = api_version
|
||||
|
||||
if self.api_version == "1.0":
|
||||
self.type = kwargs.get('type', 'static')
|
||||
self.start = kwargs.get('start', '')
|
||||
self.end = kwargs.get('end', '')
|
||||
else:
|
||||
raise ValueError('Unknown API version of object')
|
||||
|
||||
|
||||
class NetworkRoute(object):
|
||||
|
||||
def __init__(self, api_version, **kwargs):
|
||||
self.api_version = api_version
|
||||
|
||||
if self.api_version == "1.0":
|
||||
self.type = kwargs.get('subnet', '')
|
||||
self.start = kwargs.get('gateway', '')
|
||||
self.end = kwargs.get('metric', 100)
|
||||
else:
|
||||
raise ValueError('Unknown API version of object')
|
||||
|
||||
|
||||
class HostProfile(object):
|
||||
|
||||
def __init__(self, **kwargs):
|
||||
self.api_version = kwargs.get('apiVersion', '')
|
||||
|
||||
if self.api_version == "1.0":
|
||||
metadata = kwargs.get('metadata', {})
|
||||
spec = kwargs.get('spec', {})
|
||||
|
||||
self.name = metadata.get('name', '')
|
||||
self.region = metadata.get('region', '')
|
||||
|
||||
oob = spec.get('oob', {})
|
||||
self.oob_type = oob.get('type', 'ipmi')
|
||||
self.oob_network = oob.get('network', 'oob')
|
||||
self.oob_account = oob.get('account', '')
|
||||
self.oob_credential = oob.get('credential', '')
|
||||
|
||||
storage = spec.get('storage', {})
|
||||
self.storage_layout = storage.get('layout', 'lvm')
|
||||
|
||||
bootdisk = storage.get('bootdisk', {})
|
||||
self.bootdisk_device = bootdisk.get('device', '')
|
||||
self.bootdisk_root_size = bootdisk.get('root_size', '')
|
||||
self.bootdisk_boot_size = bootdisk.get('boot_size', '')
|
||||
|
||||
partitions = storage.get('partitions', [])
|
||||
self.partitions = []
|
||||
|
||||
for p in partitions:
|
||||
self.partitions.append(HostPartition(self.api_version, **p))
|
||||
|
||||
interfaces = spec.get('interfaces', [])
|
||||
self.interfaces = []
|
||||
|
||||
for i in interfaces:
|
||||
self.interfaces.append(HostInterface(self.api_version, **i))
|
||||
|
||||
else:
|
||||
raise ValueError('Unknown API version of object')
|
||||
|
||||
|
||||
class HostInterface(object):
|
||||
|
||||
def __init__(self, api_version, **kwargs):
|
||||
self.api_version = api_version
|
||||
|
||||
if self.api_version == "1.0":
|
||||
self.device_name = kwargs.get('device_name', '')
|
||||
self.network_link = kwargs.get('device_link', '')
|
||||
|
||||
self.hardware_slaves = []
|
||||
slaves = kwargs.get('slaves', [])
|
||||
|
||||
for s in slaves:
|
||||
self.hardware_slaves.append(s)
|
||||
|
||||
self.networks = []
|
||||
networks = kwargs.get('networks', [])
|
||||
|
||||
for n in networks:
|
||||
self.networks.append(n)
|
||||
|
||||
else:
|
||||
raise ValueError('Unknown API version of object')
|
||||
|
||||
|
||||
class HostPartition(object):
|
||||
|
||||
def __init__(self, api_version, **kwargs):
|
||||
self.api_version = api_version
|
||||
|
||||
if self.api_version == "1.0":
|
||||
self.name = kwargs.get('name', '')
|
||||
self.device = kwargs.get('device', '')
|
||||
self.part_uuid = kwargs.get('part_uuid', '')
|
||||
self.size = kwargs.get('size', '')
|
||||
self.mountpoint = kwargs.get('mountpoint', '')
|
||||
self.fstype = kwargs.get('fstype', 'ext4')
|
||||
self.mount_options = kwargs.get('mount_options', 'defaults')
|
||||
self.fs_uuid = kwargs.get('fs_uuid', '')
|
||||
self.fs_label = kwargs.get('fs_label', '')
|
||||
else:
|
||||
raise ValueError('Unknown API version of object')
|
||||
|
||||
|
||||
# A BaremetalNode is really nothing more than a physical
|
||||
# instantiation of a HostProfile, so they both represent
|
||||
# the same set of CIs
|
||||
class BaremetalNode(HostProfile):
|
||||
|
||||
def __init__(self, **kwargs):
|
||||
super(BaremetalNode, self).__init__()
|
BIN
helm_drydock/model/__init__.pyc
Normal file
BIN
helm_drydock/model/__init__.pyc
Normal file
Binary file not shown.
11
helm_drydock/tox.ini
Normal file
11
helm_drydock/tox.ini
Normal file
@ -0,0 +1,11 @@
|
||||
[tox]
|
||||
envlist = py35
|
||||
|
||||
[testenv]
|
||||
deps=
|
||||
-rrequirements.txt
|
||||
setenv=
|
||||
PYTHONWARNING=all
|
||||
|
||||
[flake8]
|
||||
ignore=E302,H306
|
7
requirements.txt
Normal file
7
requirements.txt
Normal file
@ -0,0 +1,7 @@
|
||||
PyYAML
|
||||
oauth
|
||||
requests-oauthlib
|
||||
pyipmi
|
||||
netaddr
|
||||
pecan
|
||||
python-libmaas
|
56
setup.py
Normal file
56
setup.py
Normal file
@ -0,0 +1,56 @@
|
||||
# Copyright 2017 AT&T Intellectual Property. All other rights reserved.
|
||||
#
|
||||
# Licensed under the Apache License, Version 2.0 (the "License");
|
||||
# you may not use this file except in compliance with the License.
|
||||
# You may obtain a copy of the License at
|
||||
#
|
||||
# http://www.apache.org/licenses/LICENSE-2.0
|
||||
#
|
||||
# Unless required by applicable law or agreed to in writing, software
|
||||
# distributed under the License is distributed on an "AS IS" BASIS,
|
||||
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
# See the License for the specific language governing permissions and
|
||||
# limitations under the License.
|
||||
#
|
||||
# helm_drydock - A tool to consume a host topology and orchestrate
|
||||
# and monitor the provisioning of those hosts and execution of bootstrap
|
||||
# scripts
|
||||
#
|
||||
# Modular services:
|
||||
# smelter - A service to consume the host topology, will support multiple
|
||||
# input formats. Initially supports a YAML schema as demonstrated
|
||||
# in the examples folder
|
||||
# tarot - A service for persisting the host topology and orchestration state
|
||||
# and making the data available via API
|
||||
# cockpit - The entrypoint API for users to control helm-drydock and query
|
||||
# current state
|
||||
# alchemist - The core orchestrator
|
||||
# drivers - A tree with all of the plugins that alchemist uses to execute
|
||||
# orchestrated tasks
|
||||
# jabberwocky - An introspection API that newly provisioned nodes can use to
|
||||
# ingest self-data and bootstrap their application deployment process
|
||||
|
||||
from setuptools import setup
|
||||
|
||||
setup(name='helm_drydock',
|
||||
version='0.1a1',
|
||||
description='Bootstrapper for Kubernetes infrastructure',
|
||||
url='http://github.com/att-comdev/drydock',
|
||||
author='Scott Hussey - AT&T',
|
||||
author_email='sh8121@att.com',
|
||||
license='Apache 2.0',
|
||||
packages=['helm_drydock',
|
||||
'helm_drydock.model',
|
||||
'helm_drydock.ingester'],
|
||||
install_requires=[
|
||||
'PyYAML',
|
||||
'oauth',
|
||||
'requests-oauthlib',
|
||||
'pyipmi',
|
||||
'netaddr',
|
||||
'pecan'
|
||||
],
|
||||
dependency_link=[
|
||||
'git+https://github.com/maas/python-libmaas.git'
|
||||
]
|
||||
)
|
2
testrequirements.txt
Normal file
2
testrequirements.txt
Normal file
@ -0,0 +1,2 @@
|
||||
pytest
|
||||
tox
|
BIN
tests/__pycache__/test_models.cpython-35-PYTEST.pyc
Normal file
BIN
tests/__pycache__/test_models.cpython-35-PYTEST.pyc
Normal file
Binary file not shown.
69
tests/test_models.py
Normal file
69
tests/test_models.py
Normal file
@ -0,0 +1,69 @@
|
||||
# Copyright 2017 AT&T Intellectual Property. All other rights reserved.
|
||||
#
|
||||
# Licensed under the Apache License, Version 2.0 (the "License");
|
||||
# you may not use this file except in compliance with the License.
|
||||
# You may obtain a copy of the License at
|
||||
#
|
||||
# http://www.apache.org/licenses/LICENSE-2.0
|
||||
#
|
||||
# Unless required by applicable law or agreed to in writing, software
|
||||
# distributed under the License is distributed on an "AS IS" BASIS,
|
||||
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
# See the License for the specific language governing permissions and
|
||||
# limitations under the License.
|
||||
|
||||
import pytest
|
||||
import yaml
|
||||
from helm_drydock.model import HardwareProfile
|
||||
|
||||
class TestClass(object):
|
||||
|
||||
def setup_method(self, method):
|
||||
print("Running test {0}".format(method.__name__))
|
||||
|
||||
def test_hardwareprofile(self):
|
||||
yaml_snippet = ("---\n"
|
||||
"apiVersion: '1.0'\n"
|
||||
"kind: HardwareProfile\n"
|
||||
"metadata:\n"
|
||||
" name: HPGen8v3\n"
|
||||
" region: sitename\n"
|
||||
" date: 17-FEB-2017\n"
|
||||
" name: Sample hardware definition\n"
|
||||
" author: Scott Hussey\n"
|
||||
"spec:\n"
|
||||
" # Vendor of the server chassis\n"
|
||||
" vendor: HP\n"
|
||||
" # Generation of the chassis model\n"
|
||||
" generation: '8'\n"
|
||||
" # Version of the chassis model within its generation - not version of the hardware definition\n"
|
||||
" hw_version: '3'\n"
|
||||
" # The certified version of the chassis BIOS\n"
|
||||
" bios_version: '2.2.3'\n"
|
||||
" # Mode of the default boot of hardware - bios, uefi\n"
|
||||
" boot_mode: bios\n"
|
||||
" # Protocol of boot of the hardware - pxe, usb, hdd\n"
|
||||
" bootstrap_protocol: pxe\n"
|
||||
" # Which interface to use for network booting within the OOB manager, not OS device\n"
|
||||
" pxe_interface: 0\n"
|
||||
" # Map hardware addresses to aliases/roles to allow a mix of hardware configs\n"
|
||||
" # in a site to result in a consistent configuration\n"
|
||||
" device_aliases:\n"
|
||||
" pci:\n"
|
||||
" - address: pci@0000:00:03.0\n"
|
||||
" alias: prim_nic01\n"
|
||||
" # type could identify expected hardware - used for hardware manifest validation\n"
|
||||
" type: '82540EM Gigabit Ethernet Controller'\n"
|
||||
" - address: pci@0000:00:04.0\n"
|
||||
" alias: prim_nic02\n"
|
||||
" type: '82540EM Gigabit Ethernet Controller'\n"
|
||||
" scsi:\n"
|
||||
" - address: scsi@2:0.0.0\n"
|
||||
" alias: primary_boot\n"
|
||||
" type: 'VBOX HARDDISK'\n")
|
||||
|
||||
hw_profile = yaml.load(yaml_snippet)
|
||||
hw_profile_model = HardwareProfile(**hw_profile)
|
||||
|
||||
assert hasattr(hw_profile_model, 'bootstrap_protocol')
|
||||
|
BIN
tests/test_models.pyc
Normal file
BIN
tests/test_models.pyc
Normal file
Binary file not shown.
Loading…
x
Reference in New Issue
Block a user