embedded joker into the repository
This commit is contained in:
parent
373e4c5c86
commit
8a5a9a65b8
191
joker/LICENSE
Normal file
191
joker/LICENSE
Normal file
@ -0,0 +1,191 @@
|
||||
Apache License
|
||||
Version 2.0, January 2004
|
||||
http://www.apache.org/licenses/
|
||||
|
||||
TERMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION
|
||||
|
||||
1. Definitions.
|
||||
|
||||
"License" shall mean the terms and conditions for use, reproduction, and
|
||||
distribution as defined by Sections 1 through 9 of this document.
|
||||
|
||||
"Licensor" shall mean the copyright owner or entity authorized by the copyright
|
||||
owner that is granting the License.
|
||||
|
||||
"Legal Entity" shall mean the union of the acting entity and all other entities
|
||||
that control, are controlled by, or are under common control with that entity.
|
||||
For the purposes of this definition, "control" means (i) the power, direct or
|
||||
indirect, to cause the direction or management of such entity, whether by
|
||||
contract or otherwise, or (ii) ownership of fifty percent (50%) or more of the
|
||||
outstanding shares, or (iii) beneficial ownership of such entity.
|
||||
|
||||
"You" (or "Your") shall mean an individual or Legal Entity exercising
|
||||
permissions granted by this License.
|
||||
|
||||
"Source" form shall mean the preferred form for making modifications, including
|
||||
but not limited to software source code, documentation source, and configuration
|
||||
files.
|
||||
|
||||
"Object" form shall mean any form resulting from mechanical transformation or
|
||||
translation of a Source form, including but not limited to compiled object code,
|
||||
generated documentation, and conversions to other media types.
|
||||
|
||||
"Work" shall mean the work of authorship, whether in Source or Object form, made
|
||||
available under the License, as indicated by a copyright notice that is included
|
||||
in or attached to the work (an example is provided in the Appendix below).
|
||||
|
||||
"Derivative Works" shall mean any work, whether in Source or Object form, that
|
||||
is based on (or derived from) the Work and for which the editorial revisions,
|
||||
annotations, elaborations, or other modifications represent, as a whole, an
|
||||
original work of authorship. For the purposes of this License, Derivative Works
|
||||
shall not include works that remain separable from, or merely link (or bind by
|
||||
name) to the interfaces of, the Work and Derivative Works thereof.
|
||||
|
||||
"Contribution" shall mean any work of authorship, including the original version
|
||||
of the Work and any modifications or additions to that Work or Derivative Works
|
||||
thereof, that is intentionally submitted to Licensor for inclusion in the Work
|
||||
by the copyright owner or by an individual or Legal Entity authorized to submit
|
||||
on behalf of the copyright owner. For the purposes of this definition,
|
||||
"submitted" means any form of electronic, verbal, or written communication sent
|
||||
to the Licensor or its representatives, including but not limited to
|
||||
communication on electronic mailing lists, source code control systems, and
|
||||
issue tracking systems that are managed by, or on behalf of, the Licensor for
|
||||
the purpose of discussing and improving the Work, but excluding communication
|
||||
that is conspicuously marked or otherwise designated in writing by the copyright
|
||||
owner as "Not a Contribution."
|
||||
|
||||
"Contributor" shall mean Licensor and any individual or Legal Entity on behalf
|
||||
of whom a Contribution has been received by Licensor and subsequently
|
||||
incorporated within the Work.
|
||||
|
||||
2. Grant of Copyright License.
|
||||
|
||||
Subject to the terms and conditions of this License, each Contributor hereby
|
||||
grants to You a perpetual, worldwide, non-exclusive, no-charge, royalty-free,
|
||||
irrevocable copyright license to reproduce, prepare Derivative Works of,
|
||||
publicly display, publicly perform, sublicense, and distribute the Work and such
|
||||
Derivative Works in Source or Object form.
|
||||
|
||||
3. Grant of Patent License.
|
||||
|
||||
Subject to the terms and conditions of this License, each Contributor hereby
|
||||
grants to You a perpetual, worldwide, non-exclusive, no-charge, royalty-free,
|
||||
irrevocable (except as stated in this section) patent license to make, have
|
||||
made, use, offer to sell, sell, import, and otherwise transfer the Work, where
|
||||
such license applies only to those patent claims licensable by such Contributor
|
||||
that are necessarily infringed by their Contribution(s) alone or by combination
|
||||
of their Contribution(s) with the Work to which such Contribution(s) was
|
||||
submitted. If You institute patent litigation against any entity (including a
|
||||
cross-claim or counterclaim in a lawsuit) alleging that the Work or a
|
||||
Contribution incorporated within the Work constitutes direct or contributory
|
||||
patent infringement, then any patent licenses granted to You under this License
|
||||
for that Work shall terminate as of the date such litigation is filed.
|
||||
|
||||
4. Redistribution.
|
||||
|
||||
You may reproduce and distribute copies of the Work or Derivative Works thereof
|
||||
in any medium, with or without modifications, and in Source or Object form,
|
||||
provided that You meet the following conditions:
|
||||
|
||||
You must give any other recipients of the Work or Derivative Works a copy of
|
||||
this License; and
|
||||
You must cause any modified files to carry prominent notices stating that You
|
||||
changed the files; and
|
||||
You must retain, in the Source form of any Derivative Works that You distribute,
|
||||
all copyright, patent, trademark, and attribution notices from the Source form
|
||||
of the Work, excluding those notices that do not pertain to any part of the
|
||||
Derivative Works; and
|
||||
If the Work includes a "NOTICE" text file as part of its distribution, then any
|
||||
Derivative Works that You distribute must include a readable copy of the
|
||||
attribution notices contained within such NOTICE file, excluding those notices
|
||||
that do not pertain to any part of the Derivative Works, in at least one of the
|
||||
following places: within a NOTICE text file distributed as part of the
|
||||
Derivative Works; within the Source form or documentation, if provided along
|
||||
with the Derivative Works; or, within a display generated by the Derivative
|
||||
Works, if and wherever such third-party notices normally appear. The contents of
|
||||
the NOTICE file are for informational purposes only and do not modify the
|
||||
License. You may add Your own attribution notices within Derivative Works that
|
||||
You distribute, alongside or as an addendum to the NOTICE text from the Work,
|
||||
provided that such additional attribution notices cannot be construed as
|
||||
modifying the License.
|
||||
You may add Your own copyright statement to Your modifications and may provide
|
||||
additional or different license terms and conditions for use, reproduction, or
|
||||
distribution of Your modifications, or for any such Derivative Works as a whole,
|
||||
provided Your use, reproduction, and distribution of the Work otherwise complies
|
||||
with the conditions stated in this License.
|
||||
|
||||
5. Submission of Contributions.
|
||||
|
||||
Unless You explicitly state otherwise, any Contribution intentionally submitted
|
||||
for inclusion in the Work by You to the Licensor shall be under the terms and
|
||||
conditions of this License, without any additional terms or conditions.
|
||||
Notwithstanding the above, nothing herein shall supersede or modify the terms of
|
||||
any separate license agreement you may have executed with Licensor regarding
|
||||
such Contributions.
|
||||
|
||||
6. Trademarks.
|
||||
|
||||
This License does not grant permission to use the trade names, trademarks,
|
||||
service marks, or product names of the Licensor, except as required for
|
||||
reasonable and customary use in describing the origin of the Work and
|
||||
reproducing the content of the NOTICE file.
|
||||
|
||||
7. Disclaimer of Warranty.
|
||||
|
||||
Unless required by applicable law or agreed to in writing, Licensor provides the
|
||||
Work (and each Contributor provides its Contributions) on an "AS IS" BASIS,
|
||||
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied,
|
||||
including, without limitation, any warranties or conditions of TITLE,
|
||||
NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A PARTICULAR PURPOSE. You are
|
||||
solely responsible for determining the appropriateness of using or
|
||||
redistributing the Work and assume any risks associated with Your exercise of
|
||||
permissions under this License.
|
||||
|
||||
8. Limitation of Liability.
|
||||
|
||||
In no event and under no legal theory, whether in tort (including negligence),
|
||||
contract, or otherwise, unless required by applicable law (such as deliberate
|
||||
and grossly negligent acts) or agreed to in writing, shall any Contributor be
|
||||
liable to You for damages, including any direct, indirect, special, incidental,
|
||||
or consequential damages of any character arising as a result of this License or
|
||||
out of the use or inability to use the Work (including but not limited to
|
||||
damages for loss of goodwill, work stoppage, computer failure or malfunction, or
|
||||
any and all other commercial damages or losses), even if such Contributor has
|
||||
been advised of the possibility of such damages.
|
||||
|
||||
9. Accepting Warranty or Additional Liability.
|
||||
|
||||
While redistributing the Work or Derivative Works thereof, You may choose to
|
||||
offer, and charge a fee for, acceptance of support, warranty, indemnity, or
|
||||
other liability obligations and/or rights consistent with this License. However,
|
||||
in accepting such obligations, You may act only on Your own behalf and on Your
|
||||
sole responsibility, not on behalf of any other Contributor, and only if You
|
||||
agree to indemnify, defend, and hold each Contributor harmless for any liability
|
||||
incurred by, or claims asserted against, such Contributor by reason of your
|
||||
accepting any such warranty or additional liability.
|
||||
|
||||
END OF TERMS AND CONDITIONS
|
||||
|
||||
APPENDIX: How to apply the Apache License to your work
|
||||
|
||||
To apply the Apache License to your work, attach the following boilerplate
|
||||
notice, with the fields enclosed by brackets "[]" replaced with your own
|
||||
identifying information. (Don't include the brackets!) The text should be
|
||||
enclosed in the appropriate comment syntax for the file format. We also
|
||||
recommend that a file or class name and description of purpose be included on
|
||||
the same "printed page" as the copyright notice for easier identification within
|
||||
third-party archives.
|
||||
|
||||
Copyright [yyyy] [name of copyright owner]
|
||||
|
||||
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.
|
49
joker/README.md
Normal file
49
joker/README.md
Normal file
@ -0,0 +1,49 @@
|
||||
Joker
|
||||
=====
|
||||
|
||||
Joker is a configuration data collector framework for OpenStack-based clouds.
|
||||
This framework inspects hardware and software configuration of cloud components.
|
||||
The results of this inspection are used to create Architecture Data Model which
|
||||
could be exported and used by external services.
|
||||
|
||||
Installation and configuration
|
||||
------------------------------
|
||||
|
||||
Usage and integration
|
||||
---------------------
|
||||
|
||||
Architecture data model produced by Joker could be consumed by configuration
|
||||
validator tool (Dark Knight), by architecture graph (Stencil) and others.
|
||||
|
||||
At some point it should be made convertible into format accepted by deployment
|
||||
systems (e.g. Fuel or TripleO) which will allow to effectively 'clone' OpenStack
|
||||
clouds using different deployment applications.
|
||||
|
||||
This model could be reused by Rally project to compare benchmarking results for
|
||||
different architectures.
|
||||
|
||||
The model can be used to inspect existing clouds for subsequent upgrade.
|
||||
|
||||
The model suits as base for questionaire to assess existing installations for
|
||||
support contract pricing purposes.
|
||||
|
||||
Architecture Data Model
|
||||
-----------------------
|
||||
|
||||
This section proposes data model which allows to describe any OpenStack
|
||||
installation. The model includes data regarding physical infrastructure, logical
|
||||
topology of services and mapping between the two.
|
||||
|
||||
Architecture data model could be serialized as JSON or YaML document of the
|
||||
following format::
|
||||
|
||||
openstack
|
||||
nodes
|
||||
node1
|
||||
-param1: value
|
||||
-param2: value
|
||||
services
|
||||
nova
|
||||
configuration
|
||||
-param1: value
|
||||
-param2: value
|
53
joker/build/lib.linux-x86_64-2.7/joker/__init__.py
Normal file
53
joker/build/lib.linux-x86_64-2.7/joker/__init__.py
Normal file
@ -0,0 +1,53 @@
|
||||
#from nodes import NodesDict
|
||||
import sys
|
||||
|
||||
# yaml.load(sys.stdin.read());
|
||||
|
||||
VAGRANT_DEFAULT_KEY = "-----BEGIN RSA PRIVATE KEY----\
|
||||
MIIEogIBAAKCAQEA6NF8iallvQVp22WDkTkyrtvp9eWW6A8YVr+kz4TjGYe7gHzI\
|
||||
w+niNltGEFHzD8+v1I2YJ6oXevct1YeS0o9HZyN1Q9qgCgzUFtdOKLv6IedplqoP\
|
||||
kcmF0aYet2PkEDo3MlTBckFXPITAMzF8dJSIFo9D8HfdOV0IAdx4O7PtixWKn5y2\
|
||||
hMNG0zQPyUecp4pzC6kivAIhyfHilFR61RGL+GPXQ2MWZWFYbAGjyiYJnAmCP3NO\
|
||||
Td0jMZEnDkbUvxhMmBYSdETk1rRgm+R4LOzFUGaHqHDLKLX+FIPKcF96hrucXzcW\
|
||||
yLbIbEgE98OHlnVYCzRdK8jlqm8tehUc9c9WhQIBIwKCAQEA4iqWPJXtzZA68mKd\
|
||||
ELs4jJsdyky+ewdZeNds5tjcnHU5zUYE25K+ffJED9qUWICcLZDc81TGWjHyAqD1\
|
||||
Bw7XpgUwFgeUJwUlzQurAv+/ySnxiwuaGJfhFM1CaQHzfXphgVml+fZUvnJUTvzf\
|
||||
TK2Lg6EdbUE9TarUlBf/xPfuEhMSlIE5keb/Zz3/LUlRg8yDqz5w+QWVJ4utnKnK\
|
||||
iqwZN0mwpwU7YSyJhlT4YV1F3n4YjLswM5wJs2oqm0jssQu/BT0tyEXNDYBLEF4A\
|
||||
sClaWuSJ2kjq7KhrrYXzagqhnSei9ODYFShJu8UWVec3Ihb5ZXlzO6vdNQ1J9Xsf\
|
||||
4m+2ywKBgQD6qFxx/Rv9CNN96l/4rb14HKirC2o/orApiHmHDsURs5rUKDx0f9iP\
|
||||
cXN7S1uePXuJRK/5hsubaOCx3Owd2u9gD6Oq0CsMkE4CUSiJcYrMANtx54cGH7Rk\
|
||||
EjFZxK8xAv1ldELEyxrFqkbE4BKd8QOt414qjvTGyAK+OLD3M2QdCQKBgQDtx8pN\
|
||||
CAxR7yhHbIWT1AH66+XWN8bXq7l3RO/ukeaci98JfkbkxURZhtxV/HHuvUhnPLdX\
|
||||
3TwygPBYZFNo4pzVEhzWoTtnEtrFueKxyc3+LjZpuo+mBlQ6ORtfgkr9gBVphXZG\
|
||||
YEzkCD3lVdl8L4cw9BVpKrJCs1c5taGjDgdInQKBgHm/fVvv96bJxc9x1tffXAcj\
|
||||
3OVdUN0UgXNCSaf/3A/phbeBQe9xS+3mpc4r6qvx+iy69mNBeNZ0xOitIjpjBo2+\
|
||||
dBEjSBwLk5q5tJqHmy/jKMJL4n9ROlx93XS+njxgibTvU6Fp9w+NOFD/HvxB3Tcz\
|
||||
6+jJF85D5BNAG3DBMKBjAoGBAOAxZvgsKN+JuENXsST7F89Tck2iTcQIT8g5rwWC\
|
||||
P9Vt74yboe2kDT531w8+egz7nAmRBKNM751U/95P9t88EDacDI/Z2OwnuFQHCPDF\
|
||||
llYOUI+SpLJ6/vURRbHSnnn8a/XG+nzedGH5JGqEJNQsz+xT2axM0/W/CRknmGaJ\
|
||||
kda/AoGANWrLCz708y7VYgAtW2Uf1DPOIYMdvo6fxIB5i9ZfISgcJ/bbCUkFrhoH\
|
||||
+vq/5CIWxCPp0f85R4qxxQ5ihxJ0YDQT9Jpx4TMss4PSavPaBH3RXow5Ohe+bYoQ\
|
||||
NE5OgEXk2wVfZczCZpigBKbKZHNYcelXtTt/nP3rsCuGcM4h53s=\
|
||||
-----END RSA PRIVATE KEY-----"
|
||||
|
||||
class Joker():
|
||||
|
||||
def __init__(self, key = VAGRANT_DEFAULT_KEY, *args, **kwargs):
|
||||
self.default_key = key
|
||||
|
||||
def addNode(self, name, host, port, user):
|
||||
return
|
||||
|
||||
def genStub(self, hostname, ip, port, user, key):
|
||||
return {"name": hostname, "ip": ip, "user": user,
|
||||
"key": key, "port": 22}
|
||||
|
||||
def discover(self):
|
||||
return [self.genStub(
|
||||
"controller", "192.168.28.100", 22, "vagrant", self.default_key),
|
||||
self.genStub(
|
||||
"compute1", "192.168.28.101", 22, "vagrant", self.default_key),
|
||||
self.genStub(
|
||||
"compute2", "192.168.30.101", 22, "vagrant", self.default_key)
|
||||
]
|
136
joker/build/lib.linux-x86_64-2.7/joker/nodes.py
Normal file
136
joker/build/lib.linux-x86_64-2.7/joker/nodes.py
Normal file
@ -0,0 +1,136 @@
|
||||
import collections
|
||||
import paramiko
|
||||
|
||||
|
||||
class TransformedDict(collections.MutableMapping):
|
||||
|
||||
def __init__(self, *args, **kwargs):
|
||||
self.store = dict()
|
||||
self.update(dict(*args, **kwargs)) # use the free update to set keys
|
||||
|
||||
def __getitem__(self, key):
|
||||
return self.store[self.__keytransform__(key)]
|
||||
|
||||
def __setitem__(self, key, value):
|
||||
self.store[self.__keytransform__(key)] = value
|
||||
|
||||
def __delitem__(self, key):
|
||||
del self.store[self.__keytransform__(key)]
|
||||
|
||||
def __iter__(self):
|
||||
return iter(self.store)
|
||||
|
||||
def __len__(self):
|
||||
return len(self.store)
|
||||
|
||||
def __keytransform__(self, key):
|
||||
return key
|
||||
|
||||
|
||||
class NodesDict(TransformedDict):
|
||||
|
||||
def add(self, element):
|
||||
return self.__setitem__(element, element)
|
||||
|
||||
def __keytransform__(self, key):
|
||||
try:
|
||||
# now uniq for hash is only hwaddr key
|
||||
# print 'hwaddr = ' + key['hwaddr']
|
||||
return key['hwaddr']
|
||||
except KeyError:
|
||||
raise
|
||||
|
||||
|
||||
class Node():
|
||||
|
||||
def __init__(self, name, ip):
|
||||
|
||||
self.ssh = paramiko.SSHClient()
|
||||
self.ssh.set_missing_host_key_policy(paramiko.AutoAddPolicy())
|
||||
self.setHostName(ip)
|
||||
self.setName(name)
|
||||
self.connected = False
|
||||
|
||||
self.neighbours = NodesDict()
|
||||
|
||||
def prepare(self):
|
||||
self.runCommand(
|
||||
"[ ! -x arp-scan ] && sudo apt-get --force-yes install arp-scan")
|
||||
# install arp-scan on node
|
||||
return True
|
||||
|
||||
def infect(self):
|
||||
# infect node
|
||||
return True
|
||||
|
||||
def setName(self, name):
|
||||
self.name = name
|
||||
|
||||
def setHostName(self, hostname):
|
||||
self.hostName = hostname
|
||||
|
||||
def setAccessPort(self, port):
|
||||
self.accessPort = port
|
||||
|
||||
def assignCredential(self, user, password, key):
|
||||
self.user = user
|
||||
self.password = password
|
||||
self.key = key
|
||||
|
||||
def setProxyCommand(self, proxyCommand):
|
||||
self.proxyCommand = proxyCommand
|
||||
|
||||
def connect(self):
|
||||
if self.connected is True:
|
||||
raise assertionError(self.connected is True)
|
||||
try:
|
||||
self.ssh.connect(self.hostName, self.accessPort, self.user,
|
||||
key_filename=self.key)
|
||||
self.connected = True
|
||||
return True
|
||||
except paramiko.BadHostKeyException, e:
|
||||
print "Host key could not be verified: ", e
|
||||
return False
|
||||
except paramiko.AuthenticationException, e:
|
||||
print "Error unable to authenticate: ", e
|
||||
return False
|
||||
except paramiko.SSHException, e:
|
||||
print e
|
||||
return False
|
||||
|
||||
def runCommand(self, command):
|
||||
if (command == ""):
|
||||
assertionError(command == "")
|
||||
|
||||
if (self.connected is False):
|
||||
self.connect()
|
||||
|
||||
stdin, stdout, stderr = self.ssh.exec_command(command)
|
||||
|
||||
return (stdout.readlines(), stderr.readlines())
|
||||
|
||||
def discovery(self):
|
||||
self.prepare()
|
||||
|
||||
(self.discovery_data, _) = self.runCommand(
|
||||
"ip link | awk -F: '/^[0-9]+?: / {print $2}' |\
|
||||
sudo xargs -I% arp-scan -l -I % 2>&1 | grep -E '^[0-9]+?\.'")
|
||||
|
||||
for node in self.discovery_data:
|
||||
( node['ip'], node['hwAddr'] ) = node.split("\t")
|
||||
|
||||
return True
|
||||
|
||||
|
||||
# ssh -p2301 -i /home/ryabin/.vagrant.d/insecure_private_key
|
||||
# vagrant@127.0.0.1 " link | grep -B1 link/ether | awk -F: '/^[0-9]+?:
|
||||
# / {print \$2}' | sudo xargs -I% arp-scan -l -I % 2>&1 | grep -E
|
||||
# '^[0-9]+?\.'
|
||||
|
||||
n = Node("controller", "127.0.0.1")
|
||||
n.assignCredential(
|
||||
"vagrant", None, "/home/ryabin/.vagrant.d/insecure_private_key")
|
||||
n.setAccessPort(2301)
|
||||
|
||||
n.discovery()
|
||||
print n.discovery_data
|
BIN
joker/dist/joker-0.1-py2.7.egg
vendored
Normal file
BIN
joker/dist/joker-0.1-py2.7.egg
vendored
Normal file
Binary file not shown.
10
joker/joker.egg-info/PKG-INFO
Normal file
10
joker/joker.egg-info/PKG-INFO
Normal file
@ -0,0 +1,10 @@
|
||||
Metadata-Version: 1.0
|
||||
Name: joker
|
||||
Version: 0.1
|
||||
Summary: why so serious?
|
||||
Home-page: https://github.com/MirantisLabs/joker
|
||||
Author: Mirantis
|
||||
Author-email: emailme@mirantis.com
|
||||
License: Apache
|
||||
Description: UNKNOWN
|
||||
Platform: UNKNOWN
|
8
joker/joker.egg-info/SOURCES.txt
Normal file
8
joker/joker.egg-info/SOURCES.txt
Normal file
@ -0,0 +1,8 @@
|
||||
setup.py
|
||||
joker/__init__.py
|
||||
joker/nodes.py
|
||||
joker.egg-info/PKG-INFO
|
||||
joker.egg-info/SOURCES.txt
|
||||
joker.egg-info/dependency_links.txt
|
||||
joker.egg-info/not-zip-safe
|
||||
joker.egg-info/top_level.txt
|
1
joker/joker.egg-info/dependency_links.txt
Normal file
1
joker/joker.egg-info/dependency_links.txt
Normal file
@ -0,0 +1 @@
|
||||
|
1
joker/joker.egg-info/not-zip-safe
Normal file
1
joker/joker.egg-info/not-zip-safe
Normal file
@ -0,0 +1 @@
|
||||
|
1
joker/joker.egg-info/top_level.txt
Normal file
1
joker/joker.egg-info/top_level.txt
Normal file
@ -0,0 +1 @@
|
||||
joker
|
62
joker/joker.yaml
Normal file
62
joker/joker.yaml
Normal file
@ -0,0 +1,62 @@
|
||||
templates:
|
||||
tcp_socket: &tcp_socket # we accept ip4 and ipv6
|
||||
host:
|
||||
port: 22
|
||||
|
||||
proxy_command: &proxy_command
|
||||
cmd:
|
||||
|
||||
password: &password
|
||||
pass:
|
||||
|
||||
public_key: &public_key
|
||||
path:
|
||||
|
||||
connection_layer: &connection_layer
|
||||
type: "HostName"
|
||||
|
||||
access_method:
|
||||
type: # ssh || telnet
|
||||
method: j
|
||||
|
||||
node: &node
|
||||
#we suppose, mac address is unique key-property inside ethernet segment
|
||||
hwaddr:
|
||||
|
||||
ssh_entry: &ssh_entry
|
||||
name:
|
||||
connection_layer:
|
||||
#ProxyCommand, or HostName
|
||||
type:
|
||||
# connection data == ( *tcp_socket || *proxyCommand )
|
||||
data:
|
||||
authentication_layer: &authentication_layer
|
||||
user:
|
||||
type: # password || public_key
|
||||
data: # *password || *public_key
|
||||
transport_layer: # XXX stub
|
||||
|
||||
dkenv_entry: &dkenv_entry
|
||||
<<: *ssh_entry
|
||||
authentication_layer: &dkenv_authentication_layer
|
||||
<<: *authentication_layer
|
||||
user: "vagrant"
|
||||
type: "public_key"
|
||||
data:
|
||||
path: "~/.vagrant.d/insecure_private_key"
|
||||
connection_layer: &dkenv_connection_layer
|
||||
<<: *connection_layer
|
||||
type: "HostName"
|
||||
data: &dkenv_connection_layer_data
|
||||
<<: *tcp_socket
|
||||
host: "127.0.0.1"
|
||||
|
||||
controller: &controller_joker
|
||||
<<: *dkenv_entry
|
||||
name: "controller.joker"
|
||||
connection_layer:
|
||||
<<: *dkenv_connection_layer
|
||||
|
||||
|
||||
nodes:
|
||||
- *controller_joker
|
77
joker/joker/__init__.py
Normal file
77
joker/joker/__init__.py
Normal file
@ -0,0 +1,77 @@
|
||||
from nodes import NodesDict,Node
|
||||
#from os import remove
|
||||
import os
|
||||
|
||||
PETYA_ENV_KEY = "-----BEGIN RSA PRIVATE KEY-----\n\
|
||||
MIIEogIBAAKCAQEA6NF8iallvQVp22WDkTkyrtvp9eWW6A8YVr+kz4TjGYe7gHzI\n\
|
||||
w+niNltGEFHzD8+v1I2YJ6oXevct1YeS0o9HZyN1Q9qgCgzUFtdOKLv6IedplqoP\n\
|
||||
kcmF0aYet2PkEDo3MlTBckFXPITAMzF8dJSIFo9D8HfdOV0IAdx4O7PtixWKn5y2\n\
|
||||
hMNG0zQPyUecp4pzC6kivAIhyfHilFR61RGL+GPXQ2MWZWFYbAGjyiYJnAmCP3NO\n\
|
||||
Td0jMZEnDkbUvxhMmBYSdETk1rRgm+R4LOzFUGaHqHDLKLX+FIPKcF96hrucXzcW\n\
|
||||
yLbIbEgE98OHlnVYCzRdK8jlqm8tehUc9c9WhQIBIwKCAQEA4iqWPJXtzZA68mKd\n\
|
||||
ELs4jJsdyky+ewdZeNds5tjcnHU5zUYE25K+ffJED9qUWICcLZDc81TGWjHyAqD1\n\
|
||||
Bw7XpgUwFgeUJwUlzQurAv+/ySnxiwuaGJfhFM1CaQHzfXphgVml+fZUvnJUTvzf\n\
|
||||
TK2Lg6EdbUE9TarUlBf/xPfuEhMSlIE5keb/Zz3/LUlRg8yDqz5w+QWVJ4utnKnK\n\
|
||||
iqwZN0mwpwU7YSyJhlT4YV1F3n4YjLswM5wJs2oqm0jssQu/BT0tyEXNDYBLEF4A\n\
|
||||
sClaWuSJ2kjq7KhrrYXzagqhnSei9ODYFShJu8UWVec3Ihb5ZXlzO6vdNQ1J9Xsf\n\
|
||||
4m+2ywKBgQD6qFxx/Rv9CNN96l/4rb14HKirC2o/orApiHmHDsURs5rUKDx0f9iP\n\
|
||||
cXN7S1uePXuJRK/5hsubaOCx3Owd2u9gD6Oq0CsMkE4CUSiJcYrMANtx54cGH7Rk\n\
|
||||
EjFZxK8xAv1ldELEyxrFqkbE4BKd8QOt414qjvTGyAK+OLD3M2QdCQKBgQDtx8pN\n\
|
||||
CAxR7yhHbIWT1AH66+XWN8bXq7l3RO/ukeaci98JfkbkxURZhtxV/HHuvUhnPLdX\n\
|
||||
3TwygPBYZFNo4pzVEhzWoTtnEtrFueKxyc3+LjZpuo+mBlQ6ORtfgkr9gBVphXZG\n\
|
||||
YEzkCD3lVdl8L4cw9BVpKrJCs1c5taGjDgdInQKBgHm/fVvv96bJxc9x1tffXAcj\n\
|
||||
3OVdUN0UgXNCSaf/3A/phbeBQe9xS+3mpc4r6qvx+iy69mNBeNZ0xOitIjpjBo2+\n\
|
||||
dBEjSBwLk5q5tJqHmy/jKMJL4n9ROlx93XS+njxgibTvU6Fp9w+NOFD/HvxB3Tcz\n\
|
||||
6+jJF85D5BNAG3DBMKBjAoGBAOAxZvgsKN+JuENXsST7F89Tck2iTcQIT8g5rwWC\n\
|
||||
P9Vt74yboe2kDT531w8+egz7nAmRBKNM751U/95P9t88EDacDI/Z2OwnuFQHCPDF\n\
|
||||
llYOUI+SpLJ6/vURRbHSnnn8a/XG+nzedGH5JGqEJNQsz+xT2axM0/W/CRknmGaJ\n\
|
||||
kda/AoGANWrLCz708y7VYgAtW2Uf1DPOIYMdvo6fxIB5i9ZfISgcJ/bbCUkFrhoH\n\
|
||||
+vq/5CIWxCPp0f85R4qxxQ5ihxJ0YDQT9Jpx4TMss4PSavPaBH3RXow5Ohe+bYoQ\n\
|
||||
NE5OgEXk2wVfZczCZpigBKbKZHNYcelXtTt/nP3rsCuGcM4h53s=\n\
|
||||
-----END RSA PRIVATE KEY-----"
|
||||
|
||||
#TMP_PATH="/tmp/%s"
|
||||
|
||||
class Joker():
|
||||
|
||||
def __init__(self, key = PETYA_ENV_KEY, *args, **kwargs):
|
||||
|
||||
self.default_key = key
|
||||
|
||||
self.nodes = NodesDict()
|
||||
self.keyPath = None
|
||||
self.name = "EntryPoint"
|
||||
#self.setKey()
|
||||
|
||||
|
||||
|
||||
# def __del__(self):
|
||||
# if (self.keyPath):
|
||||
# # look at this.
|
||||
# # epic security hole oneliner
|
||||
# print "os.remove(self.keyPath)"
|
||||
|
||||
def addNode(self, name, host, port, user):
|
||||
self.entryPoint = Node(name, host, port);
|
||||
self.entryPoint.assignCredential(user, self.default_key, None)
|
||||
return
|
||||
|
||||
def genStub(self, hostname, ip, port, user, key, proxycommand):
|
||||
return {"name": hostname, "ip": ip, "user": user,
|
||||
"key": key, "port": port,
|
||||
"proxy_command": proxycommand
|
||||
}
|
||||
|
||||
def discover(self):
|
||||
result = []
|
||||
discoveryData = self.entryPoint.discovery()
|
||||
|
||||
|
||||
for node in discoveryData:
|
||||
result.append(self.genStub(discoveryData[node], discoveryData[node], self.default_key , "vagrant", None, "ssh -i " + "%%PATH_TO_KEY%%" + " -p " + str(self.entryPoint.accessPort) + " -q " + self.entryPoint.user + "@" + self.entryPoint.hostName + " nc -q0 " + discoveryData[node] + " 22"))
|
||||
|
||||
return result
|
||||
|
||||
joker = Joker(PETYA_ENV_KEY)
|
||||
joker.addNode("controller1", "172.18.66.112", 2301, "vagrant");
|
||||
print joker.discover()
|
174
joker/joker/nodes.py
Normal file
174
joker/joker/nodes.py
Normal file
@ -0,0 +1,174 @@
|
||||
import collections
|
||||
import paramiko
|
||||
|
||||
from StringIO import StringIO
|
||||
from paramiko.rsakey import RSAKey
|
||||
from paramiko.dsskey import DSSKey
|
||||
|
||||
|
||||
class TransformedDict(collections.MutableMapping):
|
||||
|
||||
def __init__(self, *args, **kwargs):
|
||||
self.store = dict()
|
||||
self.update(dict(*args, **kwargs)) # use the free update to set keys
|
||||
|
||||
def __getitem__(self, key):
|
||||
return self.store[self.__keytransform__(key)]
|
||||
|
||||
def __setitem__(self, key, value):
|
||||
self.store[self.__keytransform__(key)] = value
|
||||
|
||||
def __delitem__(self, key):
|
||||
del self.store[self.__keytransform__(key)]
|
||||
|
||||
def __iter__(self):
|
||||
return iter(self.store)
|
||||
|
||||
def __len__(self):
|
||||
return len(self.store)
|
||||
|
||||
def __keytransform__(self, key):
|
||||
return key
|
||||
|
||||
|
||||
class NodesDict(TransformedDict):
|
||||
|
||||
def add(self, element):
|
||||
return self.__setitem__(element, element)
|
||||
|
||||
def __keytransform__(self, key):
|
||||
try:
|
||||
# now uniq for hash is only hwaddr key
|
||||
# print 'hwaddr = ' + key['hwaddr']
|
||||
return key['hwaddr']
|
||||
except KeyError:
|
||||
raise
|
||||
|
||||
|
||||
class Node():
|
||||
|
||||
def __init__(self, name, ip, port):
|
||||
|
||||
self.ssh = paramiko.SSHClient()
|
||||
self.ssh.set_missing_host_key_policy(paramiko.AutoAddPolicy())
|
||||
self.setHostName(ip)
|
||||
self.setName(name)
|
||||
self.setAccessPort(port)
|
||||
self.connected = False
|
||||
|
||||
#self.neighbours = NodesDict()
|
||||
self.neighbours = {}
|
||||
self.debug = True
|
||||
|
||||
self.hwAddr = None
|
||||
|
||||
def discoveryHwAddr(self):
|
||||
try:
|
||||
(stdout, stderr) = self.runCommand("ip link | grep -m1 -A1 BROADCAST,MULTICAST,UP,LOWER_UP | awk -F\" \" '/link/ {print $2}'");
|
||||
except:
|
||||
raise()
|
||||
return stdout[0].strip()
|
||||
|
||||
def setUniqData(self):
|
||||
self.hwAddr = self.discoveryHwAddr()
|
||||
|
||||
def getUniqData(self):
|
||||
return self.hwAddr
|
||||
|
||||
def debugLog(self, debugData):
|
||||
if self.debug is True:
|
||||
print debugData
|
||||
|
||||
|
||||
def prepare(self):
|
||||
# install arp-scan on node
|
||||
self.runCommand(
|
||||
"[ ! -x arp-scan ] && sudo apt-get --force-yes -y install arp-scan")
|
||||
self.setUniqData()
|
||||
self.debugLog("Unit data is " + self.getUniqData());
|
||||
return True
|
||||
|
||||
def infect(self):
|
||||
# infect node
|
||||
return True
|
||||
|
||||
def setName(self, name):
|
||||
self.name = name
|
||||
|
||||
def setHostName(self, hostname):
|
||||
self.hostName = hostname
|
||||
|
||||
def setAccessPort(self, port):
|
||||
self.accessPort = port
|
||||
|
||||
def assignCredential(self, user, key, password = None):
|
||||
self.user = user
|
||||
self.password = password
|
||||
|
||||
# from paramiko?
|
||||
self.origKey = key
|
||||
|
||||
self._pkey = RSAKey.from_private_key(StringIO(self.origKey))
|
||||
|
||||
# try:
|
||||
# self._pkey = RSAKey.from_private_key(StringIO(self.origKey))
|
||||
# except SSHException:
|
||||
# try:
|
||||
# self._pkey = DSSKey.from_private_key(StringIO(self.origKey))
|
||||
# except SSHException:
|
||||
# raise "Unknown private key format"
|
||||
|
||||
|
||||
def setProxyCommand(self, proxyCommand):
|
||||
self.proxyCommand = proxyCommand
|
||||
|
||||
def connect(self):
|
||||
if self.connected is True:
|
||||
raise assertionError(self.connected is True)
|
||||
try:
|
||||
print self.hostName, " ", self.accessPort, " ", self.user, " "
|
||||
self.ssh.connect(self.hostName, self.accessPort, self.user,
|
||||
pkey=self._pkey)
|
||||
self.connected = True
|
||||
return True
|
||||
except paramiko.BadHostKeyException, e:
|
||||
print "Host key could not be verified: ", e
|
||||
return False
|
||||
except paramiko.AuthenticationException, e:
|
||||
print "Error unable to authenticate: ", e
|
||||
return False
|
||||
except paramiko.SSHException, e:
|
||||
print e
|
||||
return False
|
||||
|
||||
def runCommand(self, command):
|
||||
if (command == ""):
|
||||
assertionError(command == "")
|
||||
|
||||
if (self.connected is False):
|
||||
self.connect()
|
||||
self.debugLog("---> " + self.hostName + " " + command)
|
||||
stdin, stdout, stderr = self.ssh.exec_command(command)
|
||||
self.debugLog("OK " + self.hostName + " " + command)
|
||||
|
||||
return (stdout.readlines(), stderr.readlines())
|
||||
|
||||
def __discovery__(self):
|
||||
|
||||
# tuesday discovery
|
||||
(self.discovery_data, _) = self.runCommand(
|
||||
"ip link | awk -F: '/^[0-9]+?: eth/ {print $2}' |\
|
||||
sudo xargs -I% arp-scan -l -I % 2>&1 | grep -E '^[0-9]+?\.' | grep -E '192.168.(28|30)'.101")
|
||||
|
||||
def discovery(self):
|
||||
self.prepare()
|
||||
node = {}
|
||||
|
||||
self.__discovery__()
|
||||
for n in self.discovery_data:
|
||||
( node['ip'], node['hwaddr'], _) = n.split("\t")
|
||||
self.neighbours[node['hwaddr']] = node['ip']
|
||||
|
||||
self.neighbours[self.getUniqData()] = self.hostName
|
||||
|
||||
return self.neighbours
|
1
joker/requirements.txt
Normal file
1
joker/requirements.txt
Normal file
@ -0,0 +1 @@
|
||||
paramiko==1.11.0
|
20
joker/sandbox/bin/joker.py
Executable file
20
joker/sandbox/bin/joker.py
Executable file
@ -0,0 +1,20 @@
|
||||
#!/usr/bin/env python
|
||||
import paramiko
|
||||
import json
|
||||
import sys
|
||||
|
||||
|
||||
nodes = json.loads(sys.stdin.read())['nodes'];
|
||||
|
||||
ssh = paramiko.SSHClient()
|
||||
ssh.set_missing_host_key_policy(paramiko.AutoAddPolicy())
|
||||
|
||||
|
||||
while node = pop(nodes):
|
||||
ssh.connect(node['host']['dst'], port=node['host']['port'], username=node['user']['name'], key_filename=node['user']['private_key_path']);
|
||||
|
||||
for node in nodes:
|
||||
ssh.connect(node['host']['dst'], port=node['host']['port'], username=node['user']['name'], key_filename=node['user']['private_key_path']);
|
||||
#stdin, stdout, stderr = ssh.exec_command('arp -ad')
|
||||
#print stdout.readlines()
|
||||
ssh.close()
|
73
joker/sandbox/bin/vagrant-ssh-config2yaml.awk
Executable file
73
joker/sandbox/bin/vagrant-ssh-config2yaml.awk
Executable file
@ -0,0 +1,73 @@
|
||||
#!/bin/sh
|
||||
|
||||
#controller: &controller
|
||||
# host:
|
||||
# <<: *host
|
||||
# dst: 127.0.0.1
|
||||
# port: 2222
|
||||
# user:
|
||||
# <<: *user
|
||||
# name: 'vagrant'
|
||||
# private_key_path: '/home/ryabin/.vagrant.d/insecure_private_key'
|
||||
# l7_options:
|
||||
# - 'LogLevel FATAL'
|
||||
# - 'IdentitiesOnly yes'
|
||||
# - 'PasswordAuthentication no'
|
||||
# - 'StrictHostKeyChecking no'
|
||||
# - 'UserKnownHostsFile /dev/null'
|
||||
|
||||
# vagrant ssh-config name | awk -f this.script
|
||||
function ssh_template_yaml(env) {
|
||||
|
||||
fmt = "%s_joker: &%s_joker\n"
|
||||
printf fmt, env["Host"], env["Host"];
|
||||
printf " host:\n"
|
||||
printf " <<: *host\n"
|
||||
fmt = " %s: %s\n"
|
||||
printf fmt, "dst", env["HostName"]
|
||||
if (env["Port"]) {
|
||||
printf fmt, "port", env["Port"]
|
||||
}
|
||||
printf " user:\n"
|
||||
printf " <<: *user\n"
|
||||
printf fmt, "name", env["User"]
|
||||
printf fmt, "private_key_path", env["IdentityFile"]
|
||||
printf fmt, "l7_options", ""
|
||||
|
||||
fmt = " - \"%s\"\n"
|
||||
for (i in l7_options) {
|
||||
printf fmt, l7_options[i]
|
||||
}
|
||||
|
||||
exit
|
||||
}
|
||||
|
||||
# ssh entry is
|
||||
# "^Host foobar"
|
||||
# Key value
|
||||
# Key value
|
||||
# ....
|
||||
# Key value
|
||||
# "^Host foobar2" -- we got start point of new host entry
|
||||
# Key value
|
||||
# Key value
|
||||
# ....
|
||||
# Key value
|
||||
|
||||
BEGIN {
|
||||
# first ssh entry applies for all described after
|
||||
env["Host"] = "default"
|
||||
}
|
||||
|
||||
# entry starts
|
||||
/^Host /i {
|
||||
ssh_template_yaml(env); # dump entry
|
||||
delete env; # reset env
|
||||
}
|
||||
|
||||
env[$1] = gensub(/.*[^ ] /, "", $0)
|
||||
|
||||
END {
|
||||
# end of parsing, dumping last entry
|
||||
#ssh_template_yaml(env)
|
||||
}
|
8
joker/sandbox/ssh/config
Normal file
8
joker/sandbox/ssh/config
Normal file
@ -0,0 +1,8 @@
|
||||
#reusable tcp connections
|
||||
ControlMaster auto
|
||||
ControlPath ~/.ssh/tmp/%h_%p_%r
|
||||
|
||||
#describe tunnel
|
||||
Host node1.joker
|
||||
User joker
|
||||
ProxyCommand ssh -q joker@entryPoint nc -q0 node1.customer.int 22
|
1
joker/sandbox/worm.json
Normal file
1
joker/sandbox/worm.json
Normal file
@ -0,0 +1 @@
|
||||
{"controller_joker": {"host": {"ssh_ident": null, "name": null, "dst": "127.0.0.1", "ssh_ports": [22], "interfaces": [{"ip_addr": [], "link": null, "name": null, "type": null}], "hw": {"mac_addr": []}, "port": 2222}, "user": {"private_key_path": "/home/ryabin/.vagrant.d/insecure_private_key", "private_key": null, "password": "generatedByCustomer", "name": "vagrant", "l7_options": ["LogLevel FATAL", "UserKnownHostsFile /dev/null", "StrictHostKeyChecking no", "PasswordAuthentication no", "IdentitiesOnly yes"]}}, "customerStorage": {"dst": ["strangename.*", "goodName.*"], "user": {"password": "storagePassword", "name": "storageUser"}}, "controller": {"host": {"ssh_ident": null, "name": null, "dst": "127.0.0.1", "ssh_ports": [22], "interfaces": [{"ip_addr": [], "link": null, "name": null, "type": null}], "hw": {"mac_addr": []}, "port": 2222}, "user": {"private_key_path": "/home/ryabin/.vagrant.d/insecure_private_key", "private_key": null, "password": "generatedByCustomer", "name": "vagrant", "l7_options": ["LogLevel FATAL", "IdentitiesOnly yes", "PasswordAuthentication no", "StrictHostKeyChecking no", "UserKnownHostsFile /dev/null"]}}, "customerCompute": {"dst": ["compute-*.customer.int"], "user": {"private_key": null, "password": "compute-password", "name": "joker", "user": "compute-robot"}}, "template": {"interface": {"ip_addr": [], "link": null, "name": null, "type": null}, "host": {"ssh_ports": [22], "ssh_ident": null, "interfaces": [{"ip_addr": [], "link": null, "name": null, "type": null}], "hw": {"mac_addr": []}, "name": null}, "user": {"private_key": null, "password": "generatedByCustomer", "name": "joker"}}, "credentials": [{"dst": ["strangename.*", "goodName.*"], "user": {"password": "storagePassword", "name": "storageUser"}}, {"dst": ["compute-*.customer.int"], "user": {"private_key": null, "password": "compute-password", "name": "joker", "user": "compute-robot"}}], "compute1_joker": {"host": {"ssh_ident": null, "name": null, "dst": "127.0.0.1", "ssh_ports": [22], "interfaces": [{"ip_addr": [], "link": null, "name": null, "type": null}], "hw": {"mac_addr": []}, "port": 2200}, "user": {"private_key_path": "/home/ryabin/.vagrant.d/insecure_private_key", "private_key": null, "password": "generatedByCustomer", "name": "vagrant", "l7_options": ["LogLevel FATAL", "UserKnownHostsFile /dev/null", "StrictHostKeyChecking no", "PasswordAuthentication no", "IdentitiesOnly yes"]}}, "start_point": {"host": {"ssh_ident": null, "name": null, "dst": "compute42", "interfaces": [{"ip_addr": [], "link": null, "name": null, "type": null}], "hw": {"mac_addr": []}, "ssh_ports": [22]}, "user": {"private_key": null, "password": "customerPassword", "name": "customerUser"}}, "nodes": [{"host": {"ssh_ident": null, "name": null, "dst": "127.0.0.1", "ssh_ports": [22], "interfaces": [{"ip_addr": [], "link": null, "name": null, "type": null}], "hw": {"mac_addr": []}, "port": 2222}, "user": {"private_key_path": "/home/ryabin/.vagrant.d/insecure_private_key", "private_key": null, "password": "generatedByCustomer", "name": "vagrant", "l7_options": ["LogLevel FATAL", "UserKnownHostsFile /dev/null", "StrictHostKeyChecking no", "PasswordAuthentication no", "IdentitiesOnly yes"]}}, {"host": {"ssh_ident": null, "name": null, "dst": "127.0.0.1", "ssh_ports": [22], "interfaces": [{"ip_addr": [], "link": null, "name": null, "type": null}], "hw": {"mac_addr": []}, "port": 2200}, "user": {"private_key_path": "/home/ryabin/.vagrant.d/insecure_private_key", "private_key": null, "password": "generatedByCustomer", "name": "vagrant", "l7_options": ["LogLevel FATAL", "UserKnownHostsFile /dev/null", "StrictHostKeyChecking no", "PasswordAuthentication no", "IdentitiesOnly yes"]}}]}
|
112
joker/sandbox/worm.yaml
Normal file
112
joker/sandbox/worm.yaml
Normal file
@ -0,0 +1,112 @@
|
||||
# online yaml2json converting tool present in http://yaml-online-parser.appspot.com/
|
||||
# Worm credentials definition
|
||||
template:
|
||||
interface: &interface
|
||||
name:
|
||||
type:
|
||||
ip_addr: []
|
||||
link:
|
||||
|
||||
user: &user
|
||||
name: joker
|
||||
password: generatedByCustomer
|
||||
private_key:
|
||||
|
||||
host: &host
|
||||
name:
|
||||
hw:
|
||||
mac_addr: []
|
||||
interfaces: [
|
||||
*interface
|
||||
]
|
||||
ssh_ident:
|
||||
ssh_ports:
|
||||
- 22
|
||||
|
||||
customerCompute: &customerComputeTemplate
|
||||
dst:
|
||||
- compute-*.customer.int
|
||||
user:
|
||||
<<: *user
|
||||
user: compute-robot
|
||||
password: compute-password
|
||||
|
||||
customerStorage: &customerStorageTemplate
|
||||
dst:
|
||||
- strangename.*
|
||||
- goodName.*
|
||||
user:
|
||||
name: storageUser
|
||||
password: storagePassword
|
||||
|
||||
|
||||
credentials:
|
||||
- *customerStorageTemplate
|
||||
- *customerComputeTemplate
|
||||
|
||||
|
||||
# entry point for deploy
|
||||
start_point: &start
|
||||
host:
|
||||
<<: *host
|
||||
dst: compute42
|
||||
user:
|
||||
<<: *user
|
||||
name: 'customerUser'
|
||||
password: 'customerPassword'
|
||||
|
||||
# vagrant controller from deploy env
|
||||
|
||||
# describe controller credentials
|
||||
controller: &controller
|
||||
host:
|
||||
<<: *host
|
||||
dst: 127.0.0.1
|
||||
port: 2222
|
||||
user:
|
||||
<<: *user
|
||||
name: 'vagrant'
|
||||
private_key_path: '/home/ryabin/.vagrant.d/insecure_private_key'
|
||||
l7_options:
|
||||
- 'LogLevel FATAL'
|
||||
- 'IdentitiesOnly yes'
|
||||
- 'PasswordAuthentication no'
|
||||
- 'StrictHostKeyChecking no'
|
||||
- 'UserKnownHostsFile /dev/null'
|
||||
|
||||
compute1_joker: &compute1_joker
|
||||
host:
|
||||
<<: *host
|
||||
dst: 127.0.0.1
|
||||
port: 2200
|
||||
user:
|
||||
<<: *user
|
||||
name: vagrant
|
||||
private_key_path: /home/ryabin/.vagrant.d/insecure_private_key
|
||||
l7_options:
|
||||
- "LogLevel FATAL"
|
||||
- "UserKnownHostsFile /dev/null"
|
||||
- "StrictHostKeyChecking no"
|
||||
- "PasswordAuthentication no"
|
||||
- "IdentitiesOnly yes"
|
||||
controller_joker: &controller_joker
|
||||
host:
|
||||
<<: *host
|
||||
dst: 127.0.0.1
|
||||
port: 2222
|
||||
user:
|
||||
<<: *user
|
||||
name: vagrant
|
||||
private_key_path: /home/ryabin/.vagrant.d/insecure_private_key
|
||||
l7_options:
|
||||
- "LogLevel FATAL"
|
||||
- "UserKnownHostsFile /dev/null"
|
||||
- "StrictHostKeyChecking no"
|
||||
- "PasswordAuthentication no"
|
||||
- "IdentitiesOnly yes"
|
||||
|
||||
|
||||
#must be updated through discovering process new network nodes
|
||||
|
||||
|
||||
nodes: [ *controller_joker, *compute1_joker ]
|
11
joker/setup.cfg
Normal file
11
joker/setup.cfg
Normal file
@ -0,0 +1,11 @@
|
||||
[metadata]
|
||||
name = joker
|
||||
version = 0.0.1
|
||||
author = Mirantis Labs
|
||||
author-email = labs-all@mirantis.com
|
||||
summary = why so serious?
|
||||
#description-file = README
|
||||
license = Apache
|
||||
|
||||
[files]
|
||||
packages = joker
|
14
joker/setup.py
Normal file
14
joker/setup.py
Normal file
@ -0,0 +1,14 @@
|
||||
#!/usr/bin/env python
|
||||
|
||||
|
||||
try:
|
||||
from setuptools import setup
|
||||
except ImportError:
|
||||
from distribute_setup import use_setuptools
|
||||
use_setuptools()
|
||||
from setuptools import setup
|
||||
|
||||
setup(
|
||||
setup_requires=['d2to1'],
|
||||
d2to1=True
|
||||
)
|
2
joker/test-requirements.txt
Normal file
2
joker/test-requirements.txt
Normal file
@ -0,0 +1,2 @@
|
||||
hacking>=0.5.6,<0.8
|
||||
|
12
joker/test_joker.py
Normal file
12
joker/test_joker.py
Normal file
@ -0,0 +1,12 @@
|
||||
from joker import Joker
|
||||
import unittest
|
||||
|
||||
|
||||
class JokerTest(unittest.TestCase):
|
||||
|
||||
def setUp(self):
|
||||
self.joker = Joker()
|
||||
|
||||
def test_3nodes(self):
|
||||
self.assertEqual(3, len(self.joker.discovery()))
|
||||
return 1
|
29
joker/test_nodes.py
Normal file
29
joker/test_nodes.py
Normal file
@ -0,0 +1,29 @@
|
||||
from nodes import NodesDict
|
||||
import unittest
|
||||
|
||||
|
||||
class NodesTest(unittest.TestCase):
|
||||
|
||||
def setUp(self):
|
||||
self.nodes = NodesDict()
|
||||
|
||||
def test_uniq(self):
|
||||
compute1 = dict()
|
||||
compute1['hwaddr'] = 'compute1_hwaddr'
|
||||
|
||||
compute2 = dict()
|
||||
compute2['hwaddr'] = 'compute2_hwaddr'
|
||||
|
||||
compute3 = dict()
|
||||
compute3['hwaddr'] = 'compute2_hwaddr' # duplicate hwaddr
|
||||
|
||||
self.nodes.add(compute1)
|
||||
self.nodes.add(compute2)
|
||||
self.nodes.add(compute3)
|
||||
|
||||
self.assertEqual(2, self.nodes.__len__())
|
||||
|
||||
def test_add(self):
|
||||
compute1 = dict()
|
||||
compute1['hwa3ddr'] = 'compute1_hwaddr'
|
||||
self.assertRaises(KeyError, self.nodes.add, compute1)
|
32
joker/tox.ini
Normal file
32
joker/tox.ini
Normal file
@ -0,0 +1,32 @@
|
||||
[tox]
|
||||
minversion = 1.6
|
||||
envlist = pep8
|
||||
skipsdist = True
|
||||
|
||||
[testenv]
|
||||
sitepackages = True
|
||||
usedevelop = True
|
||||
install_command = pip install -U {opts} {packages}
|
||||
setenv = VIRTUAL_ENV={envdir}
|
||||
LANG=en_US.UTF-8
|
||||
LANGUAGE=en_US:en
|
||||
LC_ALL=C
|
||||
deps = -r{toxinidir}/requirements.txt
|
||||
-r{toxinidir}/test-requirements.txt
|
||||
commands =
|
||||
python setup.py test --slowest --testr-args='{posargs}'
|
||||
|
||||
[testenv:pep8]
|
||||
sitepackages = False
|
||||
commands =
|
||||
flake8 {posargs}
|
||||
|
||||
[testenv:venv]
|
||||
commands = {posargs}
|
||||
|
||||
[testenv:unittests]
|
||||
commands =
|
||||
python -m unittest discover {posargs}
|
||||
|
||||
[flake8]
|
||||
exclude = .venv,.git,.tox,dist,lib/python*,*egg,build
|
Loading…
x
Reference in New Issue
Block a user