Logo Questions Linux Laravel Mysql Ubuntu Git Menu
 

How to handle value types when embedding IronPython in C#?

There is a well known issue when it comes to using .NET value types in IronPython. This has recently caused me a headache when trying to use Python as an embedded scripting language in C#. The problem can be summed up as follows:

Given a C# struct such as:

struct Vector {
    public float x;
    public float y;
}

And a C# class such as:

class Object {
    public Vector position;
}

The following will happen in IronPython:

obj = Object()
print obj.position.x    # prints ‘0’
obj.position.x = 1
print obj.position.x    # still prints ‘0’

As the article states, this means that value types are mostly immutable. However, this is a problem as I was planning on using a vector library that is implemented as seen above. Are there any workarounds for working with existing libraries that rely on value types? Modifying the library would be the very last resort, but I'd rather avoid that.

like image 755
kloffy Avatar asked Apr 18 '10 23:04

kloffy


3 Answers

There's no need to modify the library, just use a proxy.

struct Vector {
    public float X;
    public float Y;
}

class BetterVector {
    public float X;
    public float Y;
    public Vector Optimized { get { return new Vector(X, Y); } }
}

class Object {
    public BetterVector Position { get; set; }
}

Now the Python code can set fields as normal and your code can call Optimized when it needs to feed the data to OpenGL or XNA or whatever you're using.

You can even use implicit coercion if calling Optimized seems like too much work:

class BetterVector {
   // ...
   public static implicit operator Vector(BetterVector v) {
       return v.Optimized;
   }
}
like image 114
Frank Krueger Avatar answered Oct 20 '22 18:10

Frank Krueger


When you call

obj.position.x = 1

What you get is that the obj object gets you an instance of the position struct, which it essentially makes a copy of, so, setting the X value doesn't get propogated.

What you're saying is that obj.position = Vector(1,0) is what you should be doing. Similar things happen in C#.


Edit - possible work around.

If you don't want to set up the constructor, I believe this will work:

obj = Object()
pos = obj.position; # gets the object
pos.x = 1
pos.y = 2
obj.position = pos # I'm not sure if this line is necessary
like image 43
McKay Avatar answered Oct 20 '22 19:10

McKay


The only way I've found to update structs is to make use of the fact that you can specify any public field/property when creating a struct. The syntax looks like named/optional parameters in Python.

namespace StructExample
{
    public struct MyStruct
    {
        public int x;
        public int y { get; set; }
    }

    public class MyClass
    {
        public MyStruct a;
    }
}

We can use the classes in IronPython like this:

>>> foo = MyClass()
>>> print "%d:%d" % (foo.a.x, foo.a.y)
0:0
>>> foo.a.x = 1 # doesn't work!
>>> print "%d:%d" % (foo.a.x, foo.a.y)
0:0
>>> foo.a = MyStruct(x=1,y=2)
>>> print "%d:%d" % (foo.a.x, foo.a.y)
1:2

It would be nice if Python had a syntax like the F# 'with' for creating a new struct, copying the fields from the old one. Unfortunately we have to specify all the fields when cloning a struct.

like image 1
Laurion Burchall Avatar answered Oct 20 '22 17:10

Laurion Burchall