[three]Bean
In which I avoid the inverse unicode sandwich
Jun 22, 2012 | categories: python, toscawidgets, testing, turbogears View CommentsProblem #1 - I need to test tons of HTML output for correctness (because I maintain toscawidgets2). That output varies slightly because tw2 supports five different templating languages (mako, genshi, jinja2, kajiki, and chameleon). Using double-equals (==) just won't do it.
Solution #1 - We used strainer. It works!
Problem #2 - Imagine porting this to Python 3. Yes, that's right. The encoding is sniffed by hand and then used to encode regular expressions; these are in turn applied to parse XML. Think "inverse unicode sandwich with a side of Cthulhu."
Solution #2 - I wrote sieve: a baby module child of one corner of FormEncode and another corner of strainer. It works on pythons 2.6, 2.7, and 3.2. If you like, you may use it:
>>> from sieve.operators import eq_xml, in_xml >>> a = "<foo><bar>Value</bar></foo>" >>> b = """ ... <foo> ... <bar> ... Value ... </bar> ... </foo> ... """ >>> eq_xml(a, b) True >>> c = "<html><body><foo><bar>Value</bar></foo></body></html" >>> in_xml(a, c) # 'needle' in a 'haystack' True
p.s. -- I looked into xmldiff. Awesome!
Ship It!
Jun 15, 2012 | categories: python, fedora View CommentsI just got into this whole Fedora packaging thing and found myself annoyed by all the waiting-on-koji-to-build. It's always the same kind of thing:
$ git commit -a -m 'Initial commit (#12345)' $ fedpkg push $ fedpkg build <wait wait wait> $ fedpkg switch-branch f17 $ git merge master $ fedpkg push $ fedpkg build <wait wait wait> $ fedpkg update <type type type> $ bodhi --buildroot-override BLAH $ fedpkg switch-branch el6 $ git merge master $ fedpkg push $ fedpkg build <wait wait wait> $ fedpkg update <type type type> $ bodhi --buildroot-override BLAH $ echo "le sigh"
I had to do this for the whole python-tw2 stack!
In answer, I wrote a baby scriptlet called ship-it to smooth it over. It uses the unbelievably re__dict__ulous pbs python module to do the dirty work (coming soon to a Beefy Miracle near you as python-pbs-subprocess):
#!/usr/bin/env python """ Script for merging, pushing, building, updating, and buildroot-overriding changes to a package on multiple branches. Run this after you have made a change, committed and built the master branch for your new package. """ import argparse import textwrap # The pbs module is amazing. from pbs import git, fedpkg, bodhi, rpmspec, glob branches = [ { 'short': 'f17', 'long': 'fc17', }, { 'short': 'el6', 'long': 'el6', }, ] def config(): parser = argparse.ArgumentParser( description=textwrap.dedent(__doc__), formatter_class=argparse.RawDescriptionHelpFormatter, ) parser.add_argument( '--user', dest='user', help='Your FAS username.') parser.add_argument( '--type', dest='type', help="Type. (bugfix, enhancement, security)") parser.add_argument( '--bugs', dest='bugs', help="Specify any number of Bugzilla IDs (--bugs=1234,5678)") parser.add_argument( '--notes', dest='notes', help='Update notes') parser.add_argument( '--duration', dest='duration', type=int, default=7, help="Duration of the buildroot override in days.") parser.add_argument( '--forgive-build', dest='forgive', action='store_true', default=False, help="Don't stop the script if a build fails.") args = parser.parse_args() required_args = ['user', 'type', 'notes'] for required in required_args: if not getattr(args, required): parser.print_usage() raise ValueError("%r is required." % required) return args def main(): args = config() spec = glob("*.spec")[0] nevr = rpmspec(q=spec).split()[0].rsplit('.', 2)[0] print "Processing %r" % nevr for branch in branches: nevra = nevr + '.' + branch['long'] print "Working on %r, %r" % (branch['short'], nevra) print git.checkout(branch['short']) # Merge, push, build git.merge("master", _fg=True) fedpkg.push(_fg=True) if args.forgive: try: fedpkg.build(_fg=True) except Exception, e: print str(e) else: fedpkg.build(_fg=True) # Submit a new update. kwargs = { '_fg': True, 'new': True, 'user': args.user, 'type': args.type, 'notes': args.notes, } bodhi(nevra, **kwargs) # Buildroot override kwargs = { '_fg': True, 'user': args.user, 'buildroot-override': nevra, 'duration': args.duration, 'notes': args.notes, } bodhi(**kwargs) if __name__ == '__main__': main()
Are my Fedora packages handled by Upstream Release Monitoring?
May 02, 2012 | categories: python, fedora View CommentsI just recently started packaging up a bunch of stuff for Fedora 17 and have had to learn the ins-and-outs of package review. The final step in the how-to-get-your-package-into-Fedora story is to enable upstream release monitoring for your package.
I'd forgotten to do this a few times. Oop! So I wrote the following little script to ask pkgdb for all my packages, compare them against the list of monitored packages, and print the result. Maybe you'll find it useful:
#!/usr/bin/env python """ List the Fedora packages I own by their status in Upstream Release Monitoring. Requires python-fedora:: $ sudo yum install python-fedora """ import fedora.client import requests from kitchen.text.converters import to_unicode import os URL = "http://fedoraproject.org/wiki/Upstream_release_monitoring" symbols = { False: ' - ', True: ' + ', } if __name__ == '__main__': pkgdb = fedora.client.PackageDB() username = os.environ.get("BODHI_USER") print "* Packages for FAS user %r" % username pkgs = pkgdb.user_packages(username).pkgs # Bypass varnish cache -> http://bit.ly/XF4YBy headers = {'Cookie': 'this-is-not-a-cookie'} page = requests.get(URL, headers=headers).text for pkg in pkgs: print symbols[pkg.name in page], pkg.name
Teaching Open Source, Web-Based, Game Programming at RIT
Mar 06, 2012 | categories: python, teachingopensource, openshift, turbogears View CommentsThis last quarter at RIT I taught (for the first time) a course on open source, web-based, game programming. The quarter just came to an end and I really want to brag on my students. But before I do that, here's a glimmer of what we covered:
- How to use bash and vim
- How to use git and github
- How to submit patches to projects you know nothing about
- What makes a good, casual game
- How to make games with HTML5
- How to program in JavaScript, CoffeeScript, and Python
- Serverside programming with Turbogears2
- How to get your code running on Red Hat's Openshift cloud.
- How to give a lightning talk(!)
Really cool stuff was built into the class; it was about making open source software, so we open sourced the syllabus! Really cool stuff happened along the way; we hit ugly problems with openshift, so we patched the quickstarter script!
You can endure my self-indulgent drivel yes, but you get the best picture from reading the students' blog posts themselves. Enough of this! Les projets de cours!
#1 - Lazorz teamed up with the Boston Museum of Science to make "[An] educational game about the physics of light. In it's current incarnation it helps demonstrate the concepts of reflection and color filtering. The development team has plans to include other concepts such as refraction and prisms in future releases."
They got a nasty front-end built, by hand, with javascript and HTML5 that works on every mobile device we could test it on. It is intuitive and fun. They have almost-working Facebook auth and they have an almost-working Turbogears2 backed JSON store. (really, this close to completion!)
#2 - Gold Rush is visually and game-ly amazing. If I had to pick which of the three project was going to make one million dollars, it'd be this one.
The front-end is built in Unity (closed source, but compiles to a ton of platforms) and the back-end JSON store is built in TurboGears2. These guys completely flew through development, mastering skills they'd never heard of in a day. Its 3D with a moving camera and an incredibly fun game. The original game idea belongs to team member Eric Heaney. It's fun to play as a normal old card game (I've played it at a number of parties since he pitched it to the class).
#3 -- WebBotWar (the python web robot fighting game) wins the prize for master hack. They:
- Forked pybotwar, ripped it's UI off and made it export JSON.
- Reimplemented the front-end with jCanvas (and cooked up all their own art assets!).
- Wrote a TurboGears2 app that spins up serverside instances of their pybotwar fork.
- Those instances dump their state into mongodb (or memcached, long story).
- Their javascript client polls the TG2 app for the game state and voilà.
They have (awesome) plans to:
- Allow you to upload your own scripts (they have it working, just not secured).
- Make a built-in script editor.
- Tighter facebook integration (challenge your friends!)
BTW, it works on android and iPad. It even works on Epiphany (wat?) They also want you to know that python box2d is a pain in the ass.
ToscaWidgets2 Bugsprint Report
Mar 04, 2012 | categories: python, toscawidgets, pyramid, turbogears View CommentsHere's the rundown of the pre-PyCon tw2 bugsprint. The unstoppable Greg Jurman and I coordinated in google+ and IRC (with lots of ad-hoc visitors in both) and did all the damage we could do to the bug list. Our goal was to close enough that we could justify a solid 2.0 non-beta release but we didn't quite get there. We settled on 2.0 release candidate 1.
tl;dr - Big progress. We'll seal the deal on 2.0 at PyCon US next week.
Infrastructure:
- tw2 sourcecode is spread out in repos all over the place. We centralized everything at http://github.com/toscawidgets.
- We're using git-flow to manage team development and releases.
- We setup a http://cia.vc bot in #toscawidgets/IRC, which is awesome.
- We're in the continuous integration queue with Shining Panda.
Tickets closed:
- Duplicated TW Encoders
- Graceful degradation when JS disabled for tw2.dynforms
- HTML5 Prompt Text
- Configurable 'location' for resources
- Disable resource injection on a per-resource basis
- Document middleware configuration values
- Handling SCRIPT_NAME
- Internationalization
- Inline templates
- Refactor resources for better compatibility with tw0.9
- Automatic 'modname' for resources
- Shortcuts for controller registration / Tidy up tutorial
Tickets worked on, but not complete:
Tickets created:
« Previous Page -- Next Page »