Project's sources are accessible from a Mercurial version control repository hosted at BitBucket.
Project development should be tracked in the TODO.txt
file.
- Exact formatting is not important as long as its content is kept formatted consistently.
- Done tasks should be marked as such and not deleted.
Testing:
pytest
testing framework needed to run unit tests.- To run the tests using Python 3 first process them and the rest of the library
sources using the Python
py2to3
conversion tool. - For more detailed information see the DEVELOPMENT & TESTING ENVIRONMENT section below.
Reproducing problematic use cases:
- Failing web service processing examples can be easily packaged as reproducible test cases using the suds library 'message & reply injection' technique.
- Some things you can achieve using this technique (for examples, see existing
project unit tests):
- Create a client object based on a fixed WSDL string.
- Have a client object send a fixed request string without having it construct one based on the loaded WSDL schema and received arguments.
- Have a client object process a fixed reply string without having it send a request to an actual external web service.
Base sources should remain Python 2.x compatible. Since the original project states aiming for Python 2.4 compatibility we should do so as well.
Python features that need to be avoided for compatibility with older Python versions:
Features introduced in Python 2.5.
any
&all
functions.with
statement.try
/except
/finally
blocks.Prior to this Python release, code like the following:
try: A except XXX: B finally: C
was considered illegal and needed to be written using nested
try
blocks as in:try: try: A except XXX: B finally: C
yield
expression inside atry
block with afinally
clause.Prior to this Python release, code like the following:
try: yield x finally: do_something()
is considered illegal, but can be replaced with legal code similar to the following:
try: yield x except: do_something() raise do_something()
Features introduced in Python 2.6.
bytes
type.- Byte literals, e.g.
b"quack"
. - String
format()
method.
Features introduced in Python 2.7.
- Dictionary & set comprehensions.
- Set literals.
Handling Python 3 related patches applicable to the original suds development project.
- Originally synchronized with the Mercurial patch queue maintained by Bernhard Leiner.
- Used to be kept synchronized with the originating queue but no longer. It became more and more difficult to keep them synchronized as our work progressed and both the original suds project and this patch queue seem dead otherwise.
- Used to be committed first to the 'Python 3 support' branch and then merged back to the trunk from there.
External documentation:
- SOAP
- http://www.w3.org/TR/soap
- Version 1.1.
- Version 1.2.
- Part0: Primer
- Part1: Messaging Framework
- Part2: Adjuncts
- Specification Assertions and Test Collection
- WS-I Basic Profile 1.1
- WSDL 1.1
- XML Schema
- Part 0: Primer Second Edition - http://www.w3.org/TR/xmlschema-0
- Non-normative document intended to provide an easily readable description of the XML Schema facilities, and is oriented towards quickly understanding how to create schemas using the XML Schema language.
- Part 1: Structures - http://www.w3.org/TR/xmlschema-1
- Part 2: Datatypes - http://www.w3.org/TR/xmlschema-2
- Part 0: Primer Second Edition - http://www.w3.org/TR/xmlschema-0
For additional design, research & development project notes see the project's
notes/
folder.
- Mercurial version control related data.
- Folders created during project setup procedure (build/install).
- Internal project design, research & development notes.
- Basic project source code.
- Project test code.
- Build system configuration file listing the files to be included in the project's source distribution packages in addition to those automatically added to those packages by the used package preparation system.
- Internal project documentation.
- Basic development script for running the full project test suite using multiple Python interpreter versions on a Windows development machine.
- Basic project Python configuration.
Standard Python project setup script.
Usage examples:
setup.py --help
show detailed usage information
setup.py build
build the project
setup.py develop
prepare the development environment (add the project folder to the Python module search path)
setup.py install
build & install the project
setup.py register
register a project release at PyPI
setup.py sdist
prepare a source distribution
setup.py test
run the project's test suite (requires
pytest
)setup.py upload
upload prepared packages to PyPI
- Document the release correctly in
README.rst
. - Test the project build with the latest available
setuptools
project and update theez_setup.py
setuptools
installation script as needed.
- Chosen
setuptools
version needs to support all the Python interpreter versions supported by our project.setuptools
version 2.0 dropped support for Python 2.4 & 2.5.
- Version identification.
- Remove the
(development)
suffix for official release builds.
- Tag in Hg.
- Name the tag like
release-<version-info>
, e.g.release-0.5
.
- Prepare official releases based only on tagged commits.
- Official releases should always be prepared based on tagged revisions with no local changes in the used sandbox.
- Prepare source distribution packages (both .zip & .tar.bz2 formats), register the new release at PyPI and upload the prepared source packages.
- Run
setup.py sdist register upload
.- Upload the prepared source package to the project site.
- Use the BitBucket project web interface.
- Next development version identification.
- Bump up the forked project version counter.
- Add back the
(development)
suffix, e.g. as in0.5 (development)
.
- Notify whomever the new release might concern.
In all command-line examples below pyX, pyXY & pyXYZ represent a Python interpreter executable for a specific Python version X, X.Y & X.Y.Z respectively.
Testing environment is generally set up as follows:
- Install Python.
- Install
setuptools
(usingsetup_ez.py
or from the source distribution). - Install
pip
usingsetuptools
(optional). - Install
pytest
usingpip
orsetuptools
.
This should hold for all Python releases except some older ones explicitly listed below.
To run all of the project unit tests with a specific interpreter without
additional configuration options run the project's setup.py
script with the
'test' parameter and an appropriate Python interpreter. E.g. run any of the
following from the top level project folder:
py243 setup.py test py27 setup.py test py3 setup.py test
To have more control over the test suite run it from the top level project
folder using pytest
, e.g.
Using a Python 2.x interpreter:
py27 -m pytest
Using a Python 3.x interpreter:
py33 setup.py build & py33 -m pytest build
This way you can specify additional pytest
options on the command-line.
In both cases, tests run using Python interpreter version 3.x will be run in the
build folder constructed by the setup.py
script running the py2to3
tool
on the project's sources. You might need to manually remove the build folder in
order to have sources in it regenerated when wanting to run the test suite using
a different Python 3.x interpreter version, as those sources are regenerated
based solely on the original & processed source file timestamp information and
not the Python version used to process them.
See the pytest
documentation for a detailed list of available command-line
options. Some interesting ones:
-l show local variable state in tracebacks --tb=short shorter traceback information for each failure -x stop on first failure
On Windows you might have a problem setting up multiple parallel Python interpreter versions in case they match their major and minor version numbers, e.g. Python 2.4.3 & 2.4.4. In those cases, standard Windows installer will automatically remove the previous installation instead of simply adding a new one. In order to achieve such parallel setup we suggest the following steps:
- Install the first version in a dummy folder, and do so for the current user only.
- Copy the dummy target folder to the desired folder for the first installation, e.g. Python243.
- Uninstall the original version.
- Set up a shortcut or a batch script (e.g. py243.cmd) for running this interpreter without having to have it added to the system path.
- Repeat the steps for the second installation.
Installing Python for the current user only is necessary in order to make Python install all of its files into the target folder and not move some of them into shared system folders.
Note that this will leave you without start menu or registry entries for these Python installations. Registry entries should be needed only if you want to run some external Python package installation tool requiring those entries in order to determine where to install its package data. In that case you can set those entries manually, e.g. by using a script similar to the one found at http://nedbatchelder.com/blog/201007/installing_python_packages_from_windows_installers_into.html.
Python 2.4.3
Does not work with HTTPS links so you can not use the Python package index directly, since it, at some point, switched to using HTTPS links only.
You could potentially work around this problem by somehow mapping its https: links to http: ones or download its link page manually, locally modify it to contain http: links and then use that download link page instead of the default downloaded one.
An alternative and tested solution is to install into Python 2.4.4 and then copy all the related site-packages entries from that installation into this one.
For
pytest
2.4.1 withpy
library version 1.4.15 the following data was copied.Folders:
_pytest argparse-1.2.1-py2.4.egg-info py py-1.4.15-py2.4.egg-info pytest-2.4.1-py2.4.egg-info
Files:
argparse.py pytest.py
Python 2.4.x
Can not run
pip
usingpython.exe -m pip
. Workaround is to use one of thepip
startup scripts found in the Python installation'sScripts
folder or to use the following invocation:py244 -c "import pip;pip.main()" <regular-pip-options>
pip
1.1 - last version supporting Python 2.4.
Install using:
py244 -m easy_install pip==1.1
Can not be run using
python.exe -m pip
.Workaround is to use one of the
pip
startup scripts found in the Python installationsScripts
folder or the following invocation:py244 -c "import pip;pip.main()" <regular-pip-options>
pytest
2.4.1 - last version supporting Python 2.4.
Install using:
py244 -c "import pip;pip.main()" install pytest==2.4.1
Depends on the
py
package library version >= 1.4.16. However those versions fail to install with Python 2.4 (tested up to and including 1.4.18).May be worked around by forcing
pytest
to use an olderpy
package library version:Run the
pytest
installation usingpip
. It will fail but it will install everything needed except thepy
package library.Install the
py
package library version 1.4.15 using:py244 -c "import pip;pip.main()" install py==1.4.15
If worked around by using the
py
1.4.15 library version,pytest
's startup scripts will not work (as they explicitly checkpytest
's package dependencies), butpytest
can still be run using:py244 -m pytest <regular-pytest-options>
There seems to be no complete standards conformance overview for the suds project. This section contains just some related notes, taken down while hacking on this project. As more related information is uncovered, it should be added here as well, and eventually this whole section should be moved to the project's user documentation.
- Each message part is interpreted as a single parameter.
- What we refer to here as a 'parameter' may not necessarily correspond 1-1 to a Python function argument passed when using the suds library's Python function interface for invoking web service operations. In some cases suds may attempt to make the Python function interfaces more intuitive to the user by automatically unwrapping a parameter as defined inside a WSDL schema into multiple Python function arguments.
- In order to achieve interoperability with existing software 'in the wild',
suds does not fully conform to the WSDL 1.1 specification with regard as to
how message parts are mapped to input data contained in SOAP XML web service
operation invocation request documents.
- WSDL 1.1 standard states:
- 2.3.1 Message Parts.
- A message may have message parts referencing either an element or a type defined in the WSDL's XSD schema.
- If a message has a message part referencing a type defined in the WSDL's XSD schema, then that must be its only message part.
- 3.5 soap:body.
- If using document/literal binding and a message has a message part referencing a type defined in the WSDL's XSD schema then that part becomes the schema type of the enclosing SOAP envelope Body element.
- 2.3.1 Message Parts.
- Suds supports multiple message parts, each of which may be related either to an element or a type.
- Suds uses message parts related to types, as if they were related to an element, using the message part name as the representing XML element name in the constructed related SOAP XML web service operation invocation request document.
- WS-I Basic Profile 1.1 standard explicitly avoids the issue by stating the
following:
- R2204 - A document/literal binding in a DESCRIPTION MUST refer, in each of its soapbind:body element(s), only to wsdl:part element(s) that have been defined using the element attribute.
- Rationale.
- No other software has been encountered implementing the exact functionality specified in the WSDL 1.1 standard.
- Already done in the original suds implementation.
- Example software whose implementation matches our own.
- SoapUI.
- Tested with version 4.6.1.
- WSDL analyzer & invoker at http://www.validwsdl.com.
- SoapUI.
- WSDL 1.1 standard states:
minOccurs
/maxOccurs
attributes onall
,choice
&sequence
schema elements are ignored.- Rationale.
- Already done in the original suds implementation.
- Extra notes.
- SoapUI (tested with version 4.6.1).
- For
all
,choice
&sequence
schema elements with theirminOccurs
attribute set to "0", does not explicitly mark elements found in such containers as optional.
- For
- SoapUI (tested with version 4.6.1).
- Rationale.
- Supports sending multiple same-named web service operation parameters, but
only if they are specified next to each other in the constructed web service
operation invocation request document.
- Done by passing a list or tuple of such values to the suds constructed Python function representing the web service operation in question.
- Rationale.
- Already done in the original suds implementation.
- Extra notes.
- Such same-named values break other web service related tools as well, e.g. WSDL analyzer & invoker at http://www.validwsdl.com.