The Game's Afoot - the indie Volity dev blog

Volity is a new up-and-coming platform that enables people to play casual games (the "old fashioned" card/board/dice kind) over the internet. This blog is a place where independent developers of Volity games can discuss their development efforts.

Saturday, June 24, 2006

A (hopefully) simple guide to getting ready to develop for volity on a windows box

This post contains the aforementioned Step 0: preparing to develop Volity games using python, and developing UI files for same, on a Windows box.

Part 1: Download all necessary software.

You will need the following software packages to develop using python:

  • A python interpreter (and if you're like me, an integrated development environment to develop in).  I went for the Activestate freely downloadable version because I knew where to find it and knew it would be a low-maintenance install process, but the canonical Windows version is at www.python.org.  Python.org's Windows version is also packaged in an installer and should not present any difficulty with respect to installation or configuration either (though I have yet to test it).

-- See http://www.python.org for download

  • The python Volity framework, which includes:
  1. The Volity game server source code in Python (currently the file volity-162.tar.gz)
  2. The script that sets up and initializes the Volity game server (currently volityd.py)
  3. The Zymb library, which the Volity Game server uses to communicate with the outside world
-- See http://www.eblong.com/zarf/volity/index.html (which also has sample code for games you can use to test your setup) and  http://www.eblong.com/zarf/zymb/index.html (for zymb) for downloads.

  • A program capable of extracting unix/linux zip (.gz) files.  I downloaded 7-zip for this purpose.  Its interface is a bit odd (it is like a parallel universe version of Windows Explorer), but it'll do the job.  A recent version of WinZip will also do this, but I was too cheap to get and register it.
-- see http://www.7zip.org for downloads

  • The volity testbench software, which will allow you to view and test your SVG User interface files.

-- See http://www.volity.org/projects/gamut/releases/latest/ to download testbench.

  • A recent version of the Java Runtime Environment to run testbench with; if you already have Gamut installed, you should have this already.  Even if you don't have Gamut installed you may be OK here. 

-- See http://java.sun.com if you want to go looking for the latest version of Java

You may want to download the following additional software:

  • Inkscape is a drawing program which has been designed specifically to use the SVG format which Volity uses (and extends).  Inkscape does not implement the interactive extensions Volity uses to create user interfaces, but it does allow you to create graphics, then examine and extract the underlying XML text used to store the graphics you have drawn.  You can then bring this text over into your UI files; your UI will then be able to draw the corresponding SVG graphics using the text you added to the file. 

-- See http://www.inkscape.org for download.

Part 2: Unzip and Install stuff

Install python.  Install your .gz capable unzipper, if needed.  Install Java, if needed.  Unzip the Zymb and Volity .gz files.  Extract the testbench.jar file from thedownloaded file and put it somewhere.

Part 3: Set up the python code to run

Find Python's Library folder.  (It will most likely be something along the lines of C:\python24\Lib (which is what mine was).)  Copy the zymb and volity folders from your unzipping in the previous step into this folder.  I also copied the volityd.py loader file here as well.

Part 3b: Handle some windows-specific issues in the python code

If you're developing under windows, (as I am) a little patching is currently necessary, at least until a couple of changes propagate through the python volity framework.  This step will hopefully become optional very soon.

The first change is required is necessary because sockets, as I understand it, behave in a non-standard way on Windows compared to anywhere else in the known computing universe.  (Bill Gates, if you're reading this, it's all *your* fault!!)  To be specific, there is a Windows-specific error that has to be explicitly ignored.  To patch for this, open the tcp.py file in the Lib\zymb\ folder, and find line 129. 

It should say:

if (errnum == errno.EAGAIN)

You should change this to:

if (errnum == errno.EAGAIN or errnum == errno.WSAEWOULDBLOCK)

(Python is indentation-sensitive, so be sure that you leave the indentation as is when you do this.)

The other thing I had to do when setting up and testing was compensate for my python implementation's lack of an SSL module, and for that the volity code itself has to be patched.  There will be a command line switch in a future version of the framework files - because you wouldn't necessarily want to operate that way consistently, but if you need to boot volity for testing on a PC, you just plain need to get things running.  (Aside: I've gone looking for a canonical SSL module for python 2.4 for Windows and failed to find one - which is part of what delayed this post, BTW.  Successfully finding and setting up an SSL module for python 2.4 would also solve this second issue.)

The edit you'll need is on line 83 of volent.py in the code in the volity folder.  It should say:

        self.conn = jabber.client.JabberAuthResource(self.jid, self.password, host=host)

To unilaterally disable SSL (as was necessary to get the framework running for me), you would change this to:

        self.conn = jabber.client.JabberAuthResource(self.jid, self.password, host=host, secure=jabber.client.SECURE_NONE)
 
Once these issues are corrected in a future release - It was the framework's author that heard me out and suggested these changes, so you know they'll be in an upcoming update - no patching should be necessary.

Part 4: Get the Python stuff running

In this step, you test to ensure you can run the framework - and a sample game server - successfully.

What I did for testing purposes was to create a subfolder in the lib folder called games, and move the python files for whatever games I wanted to test into that folder so that python cound find them easily. Once I did that (and in combination with my previous setup steps), I could open a DOS window, change folders to the Lib folder (where I have everything set up) and type:

python volityd.py --jid [redacted]@volity.net --password [Redacted] --game games.rps.RPS

... where [redacted] replaces things that I don't want you to see.  :-)

You can also create a shortcut to do the same thing, as long as you ensure that python and volityd.py are both findable by Windows when it goes looking.  A well-behaved version of python handles this automatically for python; I set the folder to start in to where volityd.py was, and everything worked fine.

While I'm talking about all of this, I should mention that, until you start registering your referees (more on this later), you have to terminate instances of them running in python by navigating to their window and pressing ctrl-c.  This is very useful knowledge to have.  :-)

Part 5: Get the Java testench stuff working

Set up a shortcut to run testbench with the UI framework you wish to develop.

This is straightforward enough to be done in a shortcut also.  I used this for my shortcut's target:

C:\WINDOWS\system32\java.exe -jar Testbench.jar test.svg

Then I set the folder to start in to the location of the test svg file, and everything worked well.

Next steps:

Currently, referees are generally debug the most easily by getting the UI running first and testing in that way, although Volity's

maintainers have show in interest in getting something released that allows for direct debugging.  Until then, anyway, the next

step is to start iterative development of the UI and get it working.  I've already done a bit of that, and intend to do a bit more

of that soon.  Hopefully you'll be seeing some of the fruits of that labor in a new post next weekend.

Wednesday, June 07, 2006

Basic referee code for Spoons (but no plumbing yet)

I'm still working on being able to test my game referee python code with full Volity plumbing - I'm running into what looks to be a network error or something - but because the mechanics of the game of spoons are so simple, a large portion of the rest of the game referee code is ready and has passed some basic testing. Now all that's needed before I can get it up and running is to be able to hook it up to the Volity RPC stuff. (Next week, I hope...)



# This python code is copyright (c) 2006 by Craig Kasper
# It is provided here for educational use and may be used freely
# for any non-commercial purpose.

# First steps in implementing Spoons for voility:
# passing cards around between hands with python, and grabbing spoons

# At this point, basic card passing and spoon grabbing functionality is present.
# However, much of the error-checking and all of the volity-related plumbing is missing
# because I still don't have all of the packages for developing for voility with python
# downloaded and ready for use in testing.

# Use a list for each hand and a list to hold all of the hands

import random

# Declare our global list of hands.
Hand_list = []

# Have a separate list to hold cards passed from hand to hand

Passed_list = ["","","",""]

# For the spoons, add players to the list as they grab the spoons
Spoons_list = []

# We need an actual player count to handle grabbing of the spoons; we'll set it to zero for now
# and set it at the start of the actual game.
Player_count = 0

def start_game(count_param):
# intialize the card stack
   
# Novice python programmer pitfall: access to global variables within a function definition
# usually requires use of the "global" pitfall

    global Player_count
    global Hand_list
    Player_count = count_param      # Set the global player count
    card_count = count_param * 4    # We need to count off four cards for each player before shuffling
    master_card_list = ["1C", "1D", "1H", "1S", "2C", "2D", "2H", "2S", "3C", "3D", "3H", "3S", "4C", "4D", "4H", "4S", "5C", "5D", "5H", "5S"]
    #Enough cards for 5 players so far, but we can up this later
    card_list=master_card_list[:card_count]
    # Slice off the exact number of cards we need from the start of the card array
    random.shuffle(card_list)
    # Python's random class makes the shuflfing easy

    # start with an empty hand list   
    Hand_list=[]
    while len(card_list) > 0:       # While there are cards left to deal
        Hand_list.append(card_list[:4]) # Add the first four cards to the list of hands as a list
        card_list=card_list[4:]         # remove the first four cards from the card list so they can't get dealt again

def pass_card(player_num, card_num):
# Pass one of the four cards (indexed from 0 to 3) in a player's hand to the next player
# Returns 0 if successful or -1 if the pass could not be completed.

# In theory, we only need full error checking in only the referee code or only the player code;
# In practice, we're going to put it into both sets of code to make it more robust

    if card_num < 0 or card_num > 3:    # Bounds check to ensure that the card number is valid
        return -1
    else:
        if Passed_list[player_num - 1] <> "": # Has this player passed a card that hasn't been picked up yet?
            return -1
            #the player is not allowed to pass more than one card at a time
        else:
            if len(Hand_list[player_num - 1]) <4:
                # The player is not allowed to pass a card twice in a row without picking up in between
                return -1
            else:
                Passed_list[player_num - 1]=Hand_list[player_num - 1][card_num]
                # Copy the passed card into the passed card list
                del Hand_list[player_num - 1][card_num]
                # Erase the passed card from the player's hand
                return 0

def receive_card(player_num):
#Receive the card passed by the previous player
#Returns 0 if successful or -1 if the pick-up could not be completed
   
    if Passed_list[player_num - 2] == "":
        return -1
        #the player cannot receive a card if none is available from a previous player passing it
    else:
        if len(Hand_list[player_num - 1]) >3:
            # The player is not allowed to receive a card with a full hand of 4 cards
            return -1
        else:
            Hand_list[player_num - 1].append(Passed_list[player_num - 2])
            # add the passed card to the player's hand
            Passed_list[player_num - 2]=""
            # erase the passed card from the passed card list
            return 0

#The above functions should successfully pass cards from one hand to another
#with an intermediate stop on the table in between, as with the real game of spoons

#To do: bounds checking on the player number


def end_of_round():
    print "End of game round here"

def end_of_game():
    print "End of game code here"

def grab_spoon(player_num):
    # We're going to grab spoons by adding the grabbing player to a list of spoon-grabbers
    # rather than by removing the spoons from a list because this approach allows us to
    # keep track of who has a spoon (for if and when that player tries to return the spoon)
    if player_num in Spoons_list:
        #One spoon per customer
        return -1
    else:
        if len(Spoons_list) == Player_count - 1:
            return -1
        else:           
            Spoons_list.append(player_num)
            print Spoons_list, Player_count
            if len(Spoons_list) == Player_count - 1:
                end_of_round()

def return_spoon(player_num):
    if len(Spoons_list) == Player_count - 1:
        return -1
    else:
        if player_num in Spoons_list:
            del Spoons_list[Spoons_list.index(player_num)]
            return 0
        else:
            return -1
       

Thursday, June 01, 2006

Current Projects now on the Wiki

Despite the fact that this Wednesday failed to be used for Volity development because a birthday party took a large chunk out of my evening, I have nonetheless accomplished something Volity-related.  The Volity Wiki now has a "current projects" page where developers can actually indicate what they're up to, and indeed I've listed my projects so far there.  I encourage you to list your projects there, as it gives Volity developers a single-page place to see if any other developers are working on their your pet project, and if collaborations are available.

(That's all for now; I will return to our regularly scheduled development discussion format in my next post.)

Craig