Tuesday, October 31, 2006

p2p -> q2q

I've been playing around with divmod.org quotiont 2 quotient protocol implementation called vertex.

The q2q protocol is a p2p protocol where you can basically connect to a thing called a "resource".

The Protocol offers resource look up (Where is X?) and authentication (Is it X I'm talking to). The most interesting bit for me is that they implemented STUN and NAT traversal.

Vertex is a demo and proof-of-concept type implementation of this protocol which uses this protocol to authenticate users and send files through the connection.

Wednesday, October 25, 2006

Success!

Yeee-haaa!

I've just connected to Skype using Python and the Skype API Framework. I get notifications, and can send commands:


seletz@yeti:~/develop/python/pyskype/test $python skypeapi-test-osx.py
2006-10-25 01:05:55.934 Python[14094] running event loop
2006-10-25 01:05:56.935 Python[14094] connecting to skype
2006-10-25 01:05:56.941 Python[14094] waiting ....
2006-10-25 01:06:00.547 Python[14094] skypeAttachResponse: 1
2006-10-25 01:06:00.576 Python[14094] skypeNotificationReceived: CONNSTATUS ONLINE
2006-10-25 01:06:00.578 Python[14094] skypeNotificationReceived: CURRENTUSERHANDLE seletz29
2006-10-25 01:06:00.578 Python[14094] skypeNotificationReceived: USERSTATUS ONLINE
2006-10-25 01:06:06.942 Python[14094] sending
2006-10-25 01:06:06.942 Python[14094] got answer: None
2006-10-25 01:06:06.943 Python[14094] skypeNotificationReceived: PROTOCOL 1



I've started a new project at Google Code, where I try to create a platform-independent Python wrapper for the Skype API.

The trick was, of course, RTFM. I just needed to use PyObjC to create a informal protocol for the delegate class I use to connect with the Skype API. It's rather byzantine how Apple encodes type information.

Tuesday, October 24, 2006

WiKi entry on Skype Devzone

I've created a WiKi entry over at the Skype Devzone WiKi about code snippets. Unfortunately, you need a Skype Account to use that WiKi. Doh.

I plan to investigate all available OS Versions of the Skype API and build a common Python wrapper for it.

Monday, October 23, 2006

Skype API on OS X

Well here's my code so far. It uses pyobjc to fetch the Skype Public API framework and then defines a class to be used as a delegate for the Skype API. Skype is supposed to call the skypeAttachResponse() callback, but that does not happen.

BTW, note the magic in the objc.loadBundle() call ....

Sending messages to the Skype API then results in a log message stating that no client is attached. Doh.

Calling isSkypeRunning() et al works, so I'm actually pretty sure that I'm talking to the Skype API ;)

Any pointers would be greatly appreciated ;)

Update: I noticed that the signature of the callback methods of my Python Delegator class is probably not correct. I'll investigate further. There are pointers to that inside the PyObjC docs.

import objc
from Foundation import *

objc.loadBundle( "SkypeAPI", globals(),
bundle_path=objc.pathForFramework( \
"
~/Library/Frameworks/Skype.framework" ) )

#objc.setSignatureForSelector( "SkypeAPI",
# "sendSkypeCommand:", "v@N^@:@" )

SkypeAPIDelegate = objc.protocolNamed( "SkypeAPIDelegate" )

class Delegate(NSObject, SkypeAPIDelegate):
def init(self, app_name):
self = super( Delegate, self).init()
if self is None:
return None

self._appname = app_name

return self

def clientApplicationName( self ):
return self._appname

def skypeAttachResponse( self, code ):
NSLog( "skypeAttachResponse: %d" % ( code ) )

def skypeNotificationReceived( self, nsstring ):
NSLog( "skypeNotificationReceivest: %s" % ( nsstring ) )

def skypeBecameAvailable( self, notification ):
NSLog( "skypeBecameAvailable: %s" % ( notification ) )

def skypeBecameUnavailable( self, notification ):
NSLog( "skypeBecameUnavailable: %s" % ( notification ) )

def do_connect():
delegate = Delegate.alloc().init( "testapp" )

NSLog( "connecting to skype" )
SkypeAPI.setSkypeDelegate_( delegate )
SkypeAPI.connect()

NSLog( "sending" )
SkypeAPI.sendSkypeCommand_( u"PROTOCOL 1" )

def main():
import sys
from PyObjCTools import AppHelper

NSLog( "running event loop" )
AppHelper.callLater( 5, do_connect )
AppHelper.runConsoleEventLoop( installInterrupt=True )

if __name__ == "__main__":
main()

# vim: set ts=4 sw=4 expandtab :


Note: I love Vim ;)

Embedding Python and Multithreading

Just found this post on comp.lang.python about an article detailing the hoops one needs to jump through when it comes to embedding python and you have a multithreaded application.

Note that all that still means that you can't have one Python interpreter running in multiple threads. If I understood things as they are, Python currently needs a 1:1 relationship between Python interpreters and threads. That is, you can have more than one Python interpreter per process but only one per thread. You can have more than one interpreter per process if you really want, but these threads then compete against the same GIL.

This is due to the fact that the standard CPython implementation uses a so-called global interpreter lock (GIL) to serialize access to the internal Python API. There's one GIL per interpreter.

Also note that there's hidden pitfall here: The interpreter periodically (every n byte code instructions) yields execution and releases the GIL. That means if other threads pend on the GIL they might get runnable, which is perhaps not what one might expect.

Calling C extension is another common way to release the GIL, such that a expensive calculation in does not block other threads.

Saturday, October 21, 2006

Python Skype API

Hrmph. My Python wrapper using the ctypes module which comes with Python 2.5 segfaults. I can check if Skype is running, using the IsSkypeRunning() call, but as soon as I try to use SetSkypeDelegate() it segfaults. Now I'm downloading ctypes 1.0.1, perhaps there's more docs and examples there. Especially on how to use the Structure class with Python callbacks.

Update: No avail. Either SF is down or I don't know what. I'm unable to download the ctypes stuff.

Skype API

I just discovered the Skype API. I know, Skype is not open-source, but hey ...

The API is quite cool. Apart from the pretty obvious API access to the voice call and chat stuff, there's an API for transparently sending raw streams between programs. So, one could use the Skype API to use a pretty nifty transparent channel to send data between applications. They call this the Application2Application Protocol.

This could be the last building block I need for a pet project I have, the Personal Information Server. There I need a method to communicate with different PIS in a P2P manner, no DNS name resolution and stuff. Transparent "just works" type of communication.

Using the PIS one would mark certain types of information managed by the PIS as "public", and other PIS (those from your friends, co-workers, company , ...) would
query any "new" stuff and incorporate this new stuff into their own information store.

Using the Skype API, this could work out pretty nifty:

  • You start up your PIS instance
  • the PIS instance checks whether or not you're online with Skype
  • if you're online:
  • check if my buddies are online and have their PIS started
  • fetch the new stuff (contact info's, phone numbers, events, blogs, ...)

So, in theory, if you start up your PIS, it begins to sync with your peer PIS from your friends. That way, contacts and events stay synchronized, even if you're not online all the time, and even if you're behind a firewall.

You could also have more than one PIS, one on the road on your laptop, one at home, etc.

Nifty.

I think I'll play around with this stuff a bit. I've found no python bindings yet, but hey, at last we all have ctypes now with python 2.5, don't we ;)

Horrible week.

Phew, this was a horrible week.

Really.

First, I had to boot our old SGI Octane machine because a customer wanted a new release of an CAD Model analysis tool we wrote. Thanks god it booted. And, thanks god again I didn't threw that pile of old, heavy, bulk and green hardware on the junkyard. Pure laziness, oh my.

Actually the recompile of our internal C based libs ran very smooth, and the python part of the program ran immediately. Testing took some time, though. That machine is no speed demon, thus analyzing a mere 24MiB worth of 3d CAD data took well over an hour. It produced tons of trace, using my hackery of sys.settrace() and my C tracing library. Well, I shipped it.

Then another customer filled their trac with tickets, all assigned to me. They're using a web application we wrote using Plone as an base to manage CAD drawings. They have lots of CAD drawings. Unfortunately they use Plone 2.1.1, which we migrated to from 2.0.5, which would well be worth a story of its own. Plone 2.1.1 seems to be a bad performer. We ran the profiling tools and killed the performance hogs, which turned out to have nothing to do with my application at all. Now they report that the command line tools I wrote to post/update/revoke drawings, which follow some sort of meta work flow on their own, does not delete drawings which became invalid. Oh well. Of course, on my test system the stuff works. Grrr.

On top of that I try to wrap my head around the new (?) and shiny (??) new python web application frameworks. Currently I'm investigating the possibilities of twisted, zope 3, turbogears and the new WSGI shipped with python 2.5. Much to learn. It seems I've missed most of the new development in that area which happened the last 1 1/2 years or so.

Google's GData API

I'm currently investigating the use of the GData API when it comes to create a command line blog posting tool written in pure python.

Well, it seems that there's no Python API for the pure Google Data stuff yet (I know about the Google Base Data API).

But from what I've understood this is all pure HTML POST and GET using an extended version of the ATOM XML schema. Using urllib and ElementTree in python its just plain simple to actually get your blog feeds. They've even built in some sort of query language.

Unfortunately, to authenticate with Google using your Google account, there's a bit more involved. The Java guys have it better here: There's a nice high-level API. Doh.

Authentication is, as far as I understood, needed to post blog entries.

Note to myself:
  • Investigate Google Authentication wrt. Plone PAS
  • provide a pure python script which posts to a blog
  • hook script into a plone work flow to publish atricles from plone to blogs. Create a new content type for that, track blog post IDs we got back from google inside plone. Use Five and notifiers to update blog if Plone document changes. This would be a PIS module.

Tuesday, October 17, 2006

My own blog. I'm getting old and weird probably.

Oh well. Just playing around. I've started a blog, oh my.

Well, I've read seome very good posts on blogs recently, specially when it comes
to my current interest in python web development.

For starters you might try effbot.org, a very extensive blog from Fredrik Lundh

Black.

Yes, This is all black. I love black.

And courier. I love terminals.

Hmmm, email-to-blog?

Nifty. But I'll have to remind myself to remove my signature from emails.