Search code examples
djangoapache-flexsessioncookiespyamf

Flash + pyAMF + Django session cookie security


First off, if there is a true, official way of having flash/flex's NetConnections usurp the session/cookie state of the surrounding web page, so that if the user has already logged in, they don't need to provide credentials again just to set up an AMF connection, please stop me now and post the official answer.

Barring that, I'm assuming there is not, as I have searched and it seems to not exist. I've concocted a means of doing this, but want some feedback as to whether it is secure.

  1. Accessing a wrapper-page for a flash object will always go to secure https due to django middleware
  2. When the page view is loaded in Django, it creates a "session alias" object with a unique key that points to the current session in play (in which someone ostensibly logged in)
  3. That session alias model is saved, and that key is placed into a cookie whose key is another random string, call it randomcookie
  4. That randomcookie key name is passed as a context variable and written into the html as a flashvar to the swf
  5. The swf is also loaded only via https
  6. The flash application uses ExternalInterface to call java to grab the value at that randomcookie location, and also deletes the cookie
  7. It then creates a NetConnection to a secure server https location, passing that randomcookie as an argument (data, not in the url) to a login-using-cookie rpc
  8. At the gateway side, pyamf looks up the session alias and gets the session it points to, and logs in the user based on that (and deletes the alias, so it can't be reused)
  9. (And the gateway request could also set the session cookie and session.session_key to the known session ID, but I could let it make a whole new session key... I'm assuming that doing so should affect the response properly so that it contains the correct session key)
  10. At this point, the returned cookie values on the flash side should stick to the NetConnection so that further calls are authenticated (if a connection is authenticated using username and password the normal way, this definitely works, so I think this is a safe bet, testing will soon prove or disprove this)

So, is this unsafe, or will this work properly? As far as I know, since the html page is guaranteed to be over ssl, the key and cookie data should be encrypted and not steal-able. Then, the info therein should be safe to use one-time as basically a temporary password, sent again over ssl because the gateway is also https. After that, it's using the normal pyAMF system over https and not doing anything out of the ordinary.


Solution

  • No responses on this so far, so the best I can do is confirm that it does in fact physically work. For details on how to set up Flex Builder to write html-wrappers that communicate with Django pages templates, see my other post. The above was accomplished using a combination of the aforementioned, plus:

    Made a SessionAlias model:

    class SessionAlias(models.Model):
      alias   = models.CharField( max_length=40, primary_key=True )
      session = models.ForeignKey( Session )
      created = models.DateTimeField( auto_now_add=True )
    

    Flex points to a Django page that loads via a view containing:

    s = SessionAlias()
    s.alias = SessionStore().session_key // generates new 40-char random
    s.session = Session.objects.get( session_key=request.session.session_key )
    s.save();
    randomcookie = SessionStore().session_key // generates new 40-char random
    kwargs['extra_context']['randomcookie'] = randomcookie
    response = direct_to_template( request, **kwargs )
    response.set_cookie( randomcookie, value=alias )
    

    In the flex html-wrapper, where randomcookie is the location to look for the alias:

    <param name="flashVars" value="randomcookie={{randomcookie}}" />
    

    In applicationComplete, where we get randomcookie and find the alias, and log on using that:

    var randomcookie:String = this.parameters["randomcookie"];
    // randomcookie is something like "abc123"
    var js:String = "function get_cookie(){return document.cookie;}";
    var cookies:String = ExternalInterface.call(js).toString();
    // cookies looks like "abc123=def456; sessionid=ghi789; ..."
    var alias:String = // strip out the "def456"
    mynetconnection.call( "loginByAlias", alias, successFunc, failureFunc );
    

    Which in turn access this pyamf gateway rpc:

    from django.contrib.auth import SESSION_KEY, load_backend
    from django.contrib.auth.models import User
    from django.contrib import auth
    from django.conf import settings
    def loginByAlias( request, alias ):
      a = SessionAlias.objects.get( alias=alias )
      session_engine = __import__( settings.SESSION_ENGINE, {}, {}, [''] )
      session_wrapper = session_engine.SessionStore( a.session.session_key )
      user_id = session_wrapper.get( SESSION_KEY )
      user = User.objects.get( id=user_id )
      user.backend='django.contrib.auth.backends.ModelBackend'
      auth.login( request, user )
      a.delete()
      return whateverToFlash
    

    And at that point, on the flash/flex side, that particular mynetconnection retains the session cookie state that can make future calls such that, inside the gateway, request.user is the properly-authenticated user that logged onto the webpage in the first place.

    Note again that the run/debug settings for flex must use https, as well as the gateway settings for NetConnection. And when releasing this, I have to make sure that authenticated users stay on https.

    Any further info from people would be appreciated, especially if there's real feedback on the security aspects of this...