Skip to content

Commit

Permalink
python3: version neutral API
Browse files Browse the repository at this point in the history
Using the six library we make version neutral API calls to
dictionary methods and to unicode methods.

partial: vmware#55
  • Loading branch information
hartsock committed Jul 28, 2014
1 parent e88f67a commit ad73d76
Show file tree
Hide file tree
Showing 4 changed files with 50 additions and 25 deletions.
5 changes: 4 additions & 1 deletion pyVim/connect.py
Original file line number Diff line number Diff line change
Expand Up @@ -314,7 +314,10 @@ def __Login(host, port, user, pwd, service, adapter, version, path,
# library the fault occurred. Without the traceback we have no idea
# why the connection failed beyond the message string.
(type, value, traceback) = sys.exc_info()
reraise(vim.fault.HostConnectFault(msg=str(e)), None, traceback)
if traceback:
reraise(vim.fault.HostConnectFault(msg=str(e)), None, traceback)
else:
raise vim.fault.HostConnectFault(msg=str(e))

# Get a ticket if we're connecting to localhost and password is not specified
if host == 'localhost' and not pwd:
Expand Down
3 changes: 2 additions & 1 deletion pyVmomi/Iso8601.py
Original file line number Diff line number Diff line change
Expand Up @@ -21,6 +21,7 @@
"""
__author__ = 'VMware, Inc.'

from six import iteritems
import time
from datetime import datetime, timedelta, tzinfo
import re
Expand Down Expand Up @@ -119,7 +120,7 @@ def ParseISO8601(datetimeStr):
if match:
try:
dt = {}
for key, defaultVal in _dtExprKeyDefValMap.iteritems():
for key, defaultVal in iteritems(_dtExprKeyDefValMap):
val = match.group(key)
if val:
if key == 'microsecond':
Expand Down
37 changes: 26 additions & 11 deletions pyVmomi/SoapAdapter.py
Original file line number Diff line number Diff line change
Expand Up @@ -14,21 +14,34 @@
# limitations under the License.
from __future__ import absolute_import
from six.moves import http_client
from six import iteritems
from six import PY2
from six import PY3
if PY3:
long = int
basestring = str
from six import text_type
from six import u

import sys
import os
import socket
import subprocess
import thread
try:
import thread
except ImportError:
import _thread
import time
import urlparse

from six.moves import urllib
from datetime import datetime
from xml.parsers.expat import ParserCreate
# We have our own escape functionality.
# from xml.sax.saxutils import escape
from cStringIO import StringIO
try:
from cStringIO import StringIO
except ImportError:
from io import StringIO
from pyVmomi.VmomiSupport import *
from pyVmomi.StubAdapterAccessorImpl import StubAdapterAccessorMixin
import pyVmomi.Iso8601
Expand Down Expand Up @@ -68,7 +81,7 @@

SOAP_ENVELOPE_START = '<{0} '.format(SOAP_ENVELOPE_TAG) + \
' '.join(['xmlns:' + prefix + '="' + urn + '"' \
for urn, prefix in SOAP_NSMAP.iteritems()]) + \
for urn, prefix in iteritems(SOAP_NSMAP)]) + \
'>\n'
SOAP_ENVELOPE_END = "\n</{0}>".format(SOAP_ENVELOPE_TAG)
SOAP_HEADER_START = "<{0}>".format(SOAP_HEADER_TAG)
Expand Down Expand Up @@ -181,7 +194,7 @@ def __init__(self, writer, version, nsMap, encoding):
self.version = version
self.nsMap = nsMap and nsMap or {}
self.encoding = encoding and encoding or XML_ENCODING
for ns, prefix in self.nsMap.iteritems():
for ns, prefix in iteritems(self.nsMap):
if prefix == '':
self.defaultNS = ns
break
Expand Down Expand Up @@ -385,7 +398,9 @@ def _Serialize(self, val, info, defNS):
# it means that if you emit output in other encoding than UTF-8,
# you cannot serialize it again once more. That's feature, not
# a bug.
val = str(val).decode('UTF-8')
val = str(val)
if PY2:
val = val.decode('UTF-8')
result = XmlEscape(val)
self.writer.write('<{0}{1}>{2}</{0}>'.format(info.name, attr,
result.encode(self.encoding)))
Expand Down Expand Up @@ -821,7 +836,7 @@ def SerializeRequest(self, mo, info, args):

if reqContexts or samlToken:
result.append(SOAP_HEADER_START)
for key, val in reqContexts.iteritems():
for key, val in iteritems(reqContexts):
# Note: Support req context of string type only
if not isinstance(val, basestring):
raise TypeError("Request context key ({0}) has non-string value ({1}) of {2}".format(key, val, type(val)))
Expand Down Expand Up @@ -1126,7 +1141,7 @@ def __init__(self, host='localhost', port=443, ns=None, path='/sdk',
# the UnixSocketConnection ctor expects to find it -- see above
self.host = sock
elif url:
scheme, self.host, urlpath = urlparse.urlparse(url)[:3]
scheme, self.host, urlpath = urllib.parse(url)[:3]
# Only use the URL path if it's sensible, otherwise use the path
# keyword argument as passed in.
if urlpath not in ('', '/'):
Expand Down Expand Up @@ -1163,7 +1178,7 @@ def __init__(self, host='localhost', port=443, ns=None, path='/sdk',
self.poolSize = poolSize
self.pool = []
self.connectionPoolTimeout = connectionPoolTimeout
self.lock = thread.allocate_lock()
self.lock = _thread.allocate_lock()
self.schemeArgs = {}
if certKeyFile:
self.schemeArgs['key_file'] = certKeyFile
Expand Down Expand Up @@ -1490,7 +1505,7 @@ def InvokeMethod(self, mo, info, args):
raise obj

# Raise any socket/httplib errors caught above.
raise
raise SystemError()

## Retrieve a managed property
#
Expand Down Expand Up @@ -1520,7 +1535,7 @@ def InvokeAccessor(self, mo, info):
raise e
return obj
# Raise any socket/httplib errors caught above.
raise
raise SystemError()

## Handle the login method call
#
Expand Down
30 changes: 18 additions & 12 deletions pyVmomi/VmomiSupport.py
Original file line number Diff line number Diff line change
Expand Up @@ -17,8 +17,14 @@
from __future__ import absolute_import
from __future__ import with_statement # 2.5 only

from six import iteritems
from six import iterkeys
from six import itervalues
from six import text_type
from six import u
from six import PY3
if PY3:
long = int
basestring = str

from datetime import datetime
import pyVmomi.Iso8601
Expand Down Expand Up @@ -1030,7 +1036,7 @@ def GetWsdlTypes():
with _lazyLock:
for ns, name in _wsdlDefMap:
GetWsdlType(ns, name)
return _wsdlTypeMap.itervalues()
return itervalues(_wsdlTypeMap)

## Get the qualified XML schema name (ns, name) of a type
def GetQualifiedWsdlName(type):
Expand Down Expand Up @@ -1117,21 +1123,21 @@ def GetServiceVersions(namespace):
by compatibility (i.e. any version in the list that is compatible with some version
v in the list will preceed v)
"""
versions = dict((v, True) for (v, n) in serviceNsMap.iteritems() if n == namespace)
versions = dict((v, True) for (v, n) in iteritems(serviceNsMap) if n == namespace)
mappings = {}
for v in versions.iterkeys():
mappings[v] = set(parent for parent in parentMap[v].iterkeys()
if parent != v and versions.has_key(parent))
for v in iterkeys(versions):
mappings[v] = set(parent for parent in iterkeys(parentMap[v])
if parent != v and parent in versions.keys())
res = []
while True:
el = [ k for (k, v) in mappings.iteritems() if len(v) == 0 ]
el = [ k for (k, v) in iteritems(mappings) if len(v) == 0 ]
if len(el) == 0:
return res
el.sort()
for k in el:
res.insert(0, k)
del mappings[k]
for values in mappings.itervalues():
for values in itervalues(mappings):
values.discard(k)


Expand Down Expand Up @@ -1222,7 +1228,7 @@ def GetCompatibleType(type, version):

## Invert an injective mapping
def InverseMap(map):
return dict([ (v, k) for (k, v) in map.iteritems() ])
return dict([ (v, k) for (k, v) in iteritems(map) ])

types = Object()
nsMap = {}
Expand Down Expand Up @@ -1267,7 +1273,7 @@ def InverseMap(map):
}
_wsdlNameMap = InverseMap(_wsdlTypeMap)

for ((ns, name), typ) in _wsdlTypeMap.items():
for ((ns, name), typ) in iteritems(dict(_wsdlTypeMap)):
if typ is not NoneType:
setattr(types, typ.__name__, typ)
_wsdlTypeMapNSs.add(ns)
Expand Down Expand Up @@ -1326,7 +1332,7 @@ def InverseMap(map):
vmodlNames = {}

## Add array type into special names
for name, typ in vmodlTypes.copy().iteritems():
for name, typ in iteritems(vmodlTypes.copy()):
if typ is not NoneType:
try:
arrayType = typ.Array
Expand Down Expand Up @@ -1458,7 +1464,7 @@ def __init__(self, *args, **kwargs):

# Same as dict setdefault, except this will call through our __setitem__
def update(self, *args, **kwargs):
for k, v in dict(*args, **kwargs).iteritems():
for k, v in iteritems(dict(*args, **kwargs)):
self[k] = v

# Same as dict setdefault, except this will call through our __setitem__
Expand Down

0 comments on commit ad73d76

Please sign in to comment.