Skip to content

Latest commit

 

History

History
289 lines (221 loc) · 8.95 KB

README.md

File metadata and controls

289 lines (221 loc) · 8.95 KB

clamped: Demonstrating the clamp package

The clamp package supports functionality like the following in Java code:

import bar.clamped.BarClamp;  // yes, you can now just import Python classes!

public class UseClamped {

    public static void main(String[] args) {
        BarClamp barclamp = new BarClamp();
	try {
	    System.out.println("BarClamp: " + barclamp.call());
	} catch (Exception ex) {
	    System.err.println("Exception: " + ex);
	}
    }

}

In addition, clamp also integrates with setuptools, in order to build jars in site-packages, and then to package up Jython, other dependent jars, the Python standard library, and installed packages into a single jar suitable for running on containers.

There are alternatives to clamp, such as object factories (or more generally dependency injection). However, object factories require that using Java code support a factory approach, vs simple constructors. This is not true of such projects as Storm. In part to address such needs as Storm integration with Python, this project was created.

Writing a Python class to use clamp

To write a clampable Python class, here's what it looks like. Note that such classes need to extend a Java class and/or Java interfaces. This Python class extends two Java interfaces, Serializable and Callable. By doing so, Java code knows how to work with your Python classes. Note this part is no different than writing a Python callback for Java. However, the key extra step is the use of ClampProxyMaker, which also indicates that the package for this class should be bar, or fully qualified, bar.clamped.BarClamp.

from java.io import Serializable
from java.util.concurrent import Callable

from clamp import clamp_base


BarBase = clamp_base("bar")  # metaclass to map to Java packages


class BarClamp(BarBase, Callable, Serializable):

    def __init__(self):
        print "Being init-ed", self

    def call(self):
        print "foo"
        return 42

Next, a key piece of clamp is that it supports setuptools and soon the PyPI ecosystem. This is all that is required to clamp a module, you simply need to specify it with the clamp keyword and require the clamp package:

import ez_setup
ez_setup.use_setuptools()

from setuptools import setup, find_packages


setup(
    name = "clamped",
    version = "0.1",
    packages = find_packages(),
    install_requires = ["clamp>=0.2"],
    clamp = ["clamped"],
)

To use this example project, you need to install install both clamp and a fork of Jython that supports SSL:

  1. hg clone https://bitbucket.org/jimbaker/jython-ssl. For this presentation, use ~jythondev/jython-ssl as the place you put this.
  2. run ant in ~jythondev/jython-ssl directory and do something like alias jython-ssl=~/jythondev/jython-ssl/dist/bin/jython
  3. git clone https://github.com/jythontools/clamp.git
  4. jython-ssl setup.py install

Both requirements will go away as soon as clamp is in PyPI and jython-ssl is merged against trunk (with some more work on SSL support, of course).

To then install this example package, which uses clamp:

$ jython-ssl setup.py install
$ jython-ssl setup.py build_jar

The build_jar step constructs a jar in site-packages/jars/clamped-0.1.jar. It also ensures that this jar is automatically added to sys.path through the use of site-pacakges/jar.pth.

You can use this built jar for Java integration, but more likely you will need to build a single jar of your project, including all other clamped jars. To do this:

$ jython-ssl setup.py singlejar

which will construct a single jar, in our case clamped-0.1-single.jar. You can make this runnable if your toplevel directory specifies a __run__.py file:

from clamped import BarClamp

x = BarClamp()
x.call()

To run, simply do the following:

$ java -jar clamped-0.1-single.jar

There you have it: Python code using a Java class to call Python, all packaged up in a single Java jar. Boggles the mind!

With this single jar, you are now ready to directly integrate with Java. Let's say you have this class:

export CLASSPATH=`pwd`/clamped-0.1-single.jar:.
cd testjava
javac UseClamped.java
java UseClamped

and then you should expect to see output like the following, mostly debugging so we know it's still working :)

ClampProxyMaker: bar None array(java.lang.Class, [<type 'java.util.concurrent.Callable'>, <type 'java.io.Serializable'>]) BarClamp clamped org.python.proxies.clamped$BarClamp$1 {'__init__': <function __init__ at 0x2>, '__module__': 'clamped', 'call': <function call at 0x3>, '__proxymaker__': <clamp.ClampProxyMaker object at 0x4>}
superclass=None, interfaces=array(java.lang.Class, [<type 'java.util.concurrent.Callable'>, <type 'java.io.Serializable'>]), className=BarClamp, pythonModuleName=clamped, fullProxyName=bar.clamped.BarClamp, mapping={'__init__': <function __init__ at 0x2>, '__module__': 'clamped', 'call': <function call at 0x3>, '__proxymaker__': <clamp.ClampProxyMaker object at 0x4>}, package=bar, kwargs={}
Entering makeClass org.python.proxies.clamp$SerializableProxyMaker$0@76ef1d4c
Looked up proxy bar.clamped.BarClamp
Being init-ed bar.clamped.BarClamp@5e476e34
BarClamp bar.clamped.BarClamp@23944847
BarClamp: 42

You can decompile the proxy class to see exactly what's going on with these steps. First, download the Procyon decompiler. I'm using 0.5.21, but the most recent when you look should be just fine.

Then unpack the jar and decompile with Procyon. You should do this in some unpacking directory, since jar unpacking will explode nicely at toplevel.

mkdir unpacked && cd unpacked
jar xf ../clamped-0.1.jar
java -jar /path/to/procyon-decompiler-0.5.21.jar bar/clamped/BarClamp.class

This will result in output like the following:

package bar.clamped;

import java.util.concurrent.*;
import java.io.*;
import org.python.core.*;
import org.python.compiler.*;

@APIVersion(33)
@MTime(-1L)
public class BarClamp implements PyProxy, Callable, Serializable, ClassDictInit
{
    protected PyObject __proxy;
    protected transient PySystemState __systemState;
    public static final long serialVersionUID;

    public void _setPyInstance(final PyObject _proxy) {
        this.__proxy = _proxy;
    }

    public PyObject _getPyInstance() {
        return this.__proxy;
    }

    public void _setPySystemState(final PySystemState _systemState) {
        this.__systemState = _systemState;
    }

    public PySystemState _getPySystemState() {
        return this.__systemState;
    }

    public void __initProxy__(final Object[] array) {
        Py.initProxy((PyProxy)this, "clamped", "BarClamp", array);
    }

    public BarClamp() {
        super();
        this.__initProxy__(Py.EmptyObjects);
    }

    public void finalize() {
        super.finalize();
    }

    public Object clone() {
        return super.clone();
    }

    public Object call() throws Exception {
        final PyObject findPython = ProxyMaker.findPython((PyProxy)this, "call");
        if (findPython != null) {
            final PyObject pyObject = findPython;
            try {
                return Py.tojava(pyObject._jcallexc((Object[])Py.EmptyObjects), (Class)Class.forName("java.lang.Object"));
            }
            catch (Exception ex) {
                throw ex;
            }
            catch (Throwable t) {
                pyObject._jthrow(t);
                return null;
            }
        }
        return null;
    }

    static {
        serialVersionUID = 1L;
    }

    public static void classDictInit(final PyObject pyObject) {
        pyObject.__setitem__("__supernames__", Py.java2py((Object)new String[] { "clone", "finalize" }));
    }
}

Known issues

clamp is still very much pre-alpha at this point; see the TODO list at the project page.

Also, it's not feasible to use __new__ in your Python classes that are clamped. Why not? Java expects that constructing an object for a given class returns an object of that class! The solution is simple: call a factory function, in Python or Java, to return arbitrary objects. This is just a simple, but fundamental, mismatch between Python and Java in its object model.

Credits

Clamp is a project that has been discussed for a long time in the Jython community as a way to replace the functionality of jythonc, which is no longer supported. Darjus Loktevic did much of the latest work to get this working, and I have added a few critical bits, including setuptools integration.