mirror of
https://github.com/apache/cloudstack.git
synced 2025-12-16 10:32:34 +01:00
108 lines
4.1 KiB
Python
108 lines
4.1 KiB
Python
# Copyright (C) 2006-2007 Robey Pointer <robeypointer@gmail.com>
|
|
# Copyright 2012 Citrix Systems, Inc. Licensed under the
|
|
# Apache License, Version 2.0 (the "License"); you may not use this
|
|
# file except in compliance with the License. Citrix Systems, Inc.
|
|
# reserves all rights not expressly granted by 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.
|
|
#
|
|
# Automatically generated by addcopyright.py at 04/03/2012
|
|
# 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA.
|
|
|
|
"""
|
|
L{SSHConfig}.
|
|
"""
|
|
|
|
import fnmatch
|
|
|
|
|
|
class SSHConfig (object):
|
|
"""
|
|
Representation of config information as stored in the format used by
|
|
OpenSSH. Queries can be made via L{lookup}. The format is described in
|
|
OpenSSH's C{ssh_config} man page. This class is provided primarily as a
|
|
convenience to posix users (since the OpenSSH format is a de-facto
|
|
standard on posix) but should work fine on Windows too.
|
|
|
|
@since: 1.6
|
|
"""
|
|
|
|
def __init__(self):
|
|
"""
|
|
Create a new OpenSSH config object.
|
|
"""
|
|
self._config = [ { 'host': '*' } ]
|
|
|
|
def parse(self, file_obj):
|
|
"""
|
|
Read an OpenSSH config from the given file object.
|
|
|
|
@param file_obj: a file-like object to read the config file from
|
|
@type file_obj: file
|
|
"""
|
|
configs = [self._config[0]]
|
|
for line in file_obj:
|
|
line = line.rstrip('\n').lstrip()
|
|
if (line == '') or (line[0] == '#'):
|
|
continue
|
|
if '=' in line:
|
|
key, value = line.split('=', 1)
|
|
key = key.strip().lower()
|
|
else:
|
|
# find first whitespace, and split there
|
|
i = 0
|
|
while (i < len(line)) and not line[i].isspace():
|
|
i += 1
|
|
if i == len(line):
|
|
raise Exception('Unparsable line: %r' % line)
|
|
key = line[:i].lower()
|
|
value = line[i:].lstrip()
|
|
|
|
if key == 'host':
|
|
del configs[:]
|
|
# the value may be multiple hosts, space-delimited
|
|
for host in value.split():
|
|
# do we have a pre-existing host config to append to?
|
|
matches = [c for c in self._config if c['host'] == host]
|
|
if len(matches) > 0:
|
|
configs.append(matches[0])
|
|
else:
|
|
config = { 'host': host }
|
|
self._config.append(config)
|
|
configs.append(config)
|
|
else:
|
|
for config in configs:
|
|
config[key] = value
|
|
|
|
def lookup(self, hostname):
|
|
"""
|
|
Return a dict of config options for a given hostname.
|
|
|
|
The host-matching rules of OpenSSH's C{ssh_config} man page are used,
|
|
which means that all configuration options from matching host
|
|
specifications are merged, with more specific hostmasks taking
|
|
precedence. In other words, if C{"Port"} is set under C{"Host *"}
|
|
and also C{"Host *.example.com"}, and the lookup is for
|
|
C{"ssh.example.com"}, then the port entry for C{"Host *.example.com"}
|
|
will win out.
|
|
|
|
The keys in the returned dict are all normalized to lowercase (look for
|
|
C{"port"}, not C{"Port"}. No other processing is done to the keys or
|
|
values.
|
|
|
|
@param hostname: the hostname to lookup
|
|
@type hostname: str
|
|
"""
|
|
matches = [x for x in self._config if fnmatch.fnmatch(hostname, x['host'])]
|
|
# sort in order of shortest match (usually '*') to longest
|
|
matches.sort(lambda x,y: cmp(len(x['host']), len(y['host'])))
|
|
ret = {}
|
|
for m in matches:
|
|
ret.update(m)
|
|
del ret['host']
|
|
return ret
|