lunes, enero 31, 2011
voy a tener que cambiar de maquina :(
se le rompio la tecla esc.
no parece mucho?
cabe recordar que fuera de firefox, vim es lo que mas uso en la maquina
:P
viernes, enero 28, 2011
como empece "con esto de las computadoras"
Después de reírnos un rato de las anécdotas una de mis compañeras de trabajo decidió recopilarlas para publicarlas en un blog interno de IBM, yo mande la miá y como casi nunca la conté decidí traducirla y postearla acá.
Tengo computadoras desde que tenia 8 años, pero solo las usaba para jugar algunos juegos (principalmente sim city 2000 y indiana jones) hasta que tuve 14 años.
Ese año salio una noticia en CNN sobre una chica de mi edad que tenia una pagina web sobre HTML, yo pensé "aparece en CNN porque tiene 14 años y sabe como hacer paginas web? no debe ser tan difícil!".
Ese verano fui algunas mañanas a un lugar que tenia internet gratis (si lavabas el auto ahí ;), en esas épocas internet era muy cara (10 dolares la hora aproximadamente), así que bajaba toda la información sobre HTML que podía obtener en una hora, los guardaba en diskettes y me iba a mi casa a leerlos e intentar hacer paginas en notepad e internet explorer 4.0.
Cuando supe suficiente cree algunos sitios que hostee en freeservers y me interese por otras cosas (mi guitarra, streetboard y basket si mal no recuerdo).
A los 16 años estaba en mi casa, mi hermano volvía de una de sus primeras clases en la universidad, me tiro un libro y dijo "a vos te gustan las computadoras, lee esto y explicame".
Era un libro llamado "Programación en C para ingenieros", lo leí todo y lo ayude a corregir algunos problemas pero nunca escribí un programa yo debido a que no podía entender la enorme IDE que usaba (alguna versión de Borland C++ para windows).
Un año después nos mudamos a una casa nueva y lo único que tenia era una computadora vieja y una radio. Por las tardes empezaba a escuchar la radio a las 8PM hasta las 2AM (cucuruchos en la frente de peña y zero a la izquierda en la rock and pop y después dolina). Mientras escuchaba la radio no tenia nada que hacer así que fui a un cyber y me baje TurboC++ 1.01, una IDE muy simple para DOS para programar en C y C++, la idea vino debido a que un tiempo atrás estaba jugando el juego de la viborita en un celular de mi madrina:
Y pensé, como se hacen estos juegos? puedo hacer uno de estos?
empece a programar todas las noches mientras escuchaba la radio y por prueba y error (No conocía a ninguna persona que supiera programar y no tenia internet) aprendí a programar e hice la primera versión del juego.
después aprendí assembly para hacer juegos gráficos en increíbles 320x200 pixels a impresionantes 256 colores, luego aprendí C++ porque escuche que la programación orientada a objetos era lo mejor desde el pan rebanado.
un año después estaba por empezar a estudiar Administración de empresas y mi viejo me dijo "hay mucha gente estudiando economía, tendrías que estudiar algo relacionado con computadoras, sos bueno en eso" y acá estoy :)
La ultima cosa graciosa fue que cuando le pregunte a un vecino a donde debería estudiar me dijo "en famaf (facultad de astronomía, matemática y física) tienen una buena carrera de Ciencias de la Computación", fui a la ciudad universitaria a averiguar y como no sabia donde quedaba famaf fui a la UTN y termine estudiando Ingeniería en Sistemas :D
viernes, enero 07, 2011
busco
tengo guitarra eléctrica con ampli y criolla.
se aceptan canciones similares o zapadas del estilo.
jueves, enero 06, 2011
del correo y otros menesteres
en primer lugar nos ponemos anarcos y tratamos de subvertir el sistema.
salvando las estampillas con tracking, que evita reutilizar un sobre o estampilla si se suele enviar y recibir información de y a los mismos lugares?
usted en este momento pensara*, "que rata es este ser humano", lo que usted no tuvo en cuenta es el costo de la estampilla.
para enviar hasta 100g a buenos aires sale ~ $25, si señor, escucho bien, la módica suma de $25 para la cartera de la dama y el bolsillo del caballero.
esto significa que si quisiera enviar mi humilde existencia** a buenos aires me saldría algo así como $17000, el calculo de mi peso (una operación solo realizable por la computación cuántica y hábiles mercaderes) queda como tarea para el lector.
para darle un toque místico a la cuestión permitanme advertirlos de una situación que la ciencia todavía no ha podido explicar:
por las noches los (escasos) buzones de la empresa OCA desaparecen de la faz de la tierra.
abundan teorías sobre el destino de los mismos***, algunas de ellas involucran seres mágicos como hadas, magos o políticos honestos.
lo único que les puedo asegurar es que si algo de bueno tenia el correo era su carácter asincronico, ahora dicha asincronicidad queda circunscripta**** al horario laboral, momento en el cual su dichosa asincronicidad me importa un bledo ya que estoy trabajando (o debería).
ultimo pero no menos importante (?) es la ausencia en esta etapa madura de dicho servicio de interoperabilidad entre servicios competidores. El hecho se aprecia en este mundo post moderno en el cual la cantidad de lugares donde uno puede dejar su carta disminuye a la misma velocidad que mi capacidad de inventar frases graciosas a esta altura del post, en donde la cantidad de buzones de OCA en el centro de la república separatista de córdoba es cercano a 1, el correo argentino no me permite depositar el sobre en sus buzones para luego de buena manera intercambiar los mismos con su empresa competidora de alguna manera*****.
A esta epifanía llegue de manera abrupta al preguntar a un empleado de OCA si tal facilidad para el usuario del servicio existía, recibiendo como respuesta una risa y la simple explicación "como va a existir eso?".
con este simple post pretendo advertirlos sobre algunos de los oscuros rincones de un servicio con el que me he visto obligado a interactuar por cuestiones laborales.
* si, no solo que leo la mente sino que si nota el tiempo verbal le induzco que pensar.
** una forma cheta de decir cuerpo.
*** los mismos buzones
**** mira mama, uso palabras complicadas!
***** la cual poco me importa, para eso me cobran $25
compilar ruby en fedora (RHEL o Centos) e instalar gems y rails
disclaimer: no programo en ruby ni en rails (tampoco uso fedora) :D
estableciendo el entorno
su yum install gcc make binutils wget openssl-devel zlib zlib-devel automake readline-devel sqlite-devel exit
compilando e instalando
cd mkdir ruby cd ruby wget ftp://ftp.ruby-lang.org/pub/ruby/1.8/ruby-1.8.7-p330.tar.gz tar -xzf ruby-1.8.7-p330.tar.gz cd ruby-1.8.7-p330 ./configure --prefix=/opt/ruby-1.8.7-p330 --enable-pthread --enable-shared --enable-openssl --enable-readline --enable-zlib make su make install exit
facilitandonos la vida y probando que todo ande
echo "export PATH=/opt/ruby-1.8.7-p330/bin/:$PATH" >> ~/.bashrc bash # aca deberia andar irb
instalando gems y rails
cd .. wget http://rubyforge.org/frs/download.php/73779/rubygems-1.4.1.tgz tar -xzf rubygems-1.4.1.tgz cd rubygems-1.4.1 su ruby setup.rb gem install rails --include-dependencies exit
probando que anda
rails new testapp cd testapp # por alguna razon tuve que hacer esto aca su bundle install exit rails server
y visitando http://0.0.0.0:3000/ en el browser me salio la pagina de bienvenida.
miércoles, enero 05, 2011
Indice Punk de Precios (IPP)
alguna vez dijo ricky espinoza de la mitica banda punk argentina flema:
hoy pegamos un 50, tomaremos 1000 cervezas
considerando que con mucha suerte hoy "pegando un 50" conseguimos máximo 10 cervezas podemos concluir que el poder adquisitivo punk (PAP) disminuyo en 100 veces considerando la canasta basica punk (CBP) que consiste de los siguientes elementos
* cerveza
espero que les haya gustado, esto fue el Indice Punk de Precios para los fichines, ahora seguimos con mas nivel X
action!
viernes, diciembre 31, 2010
la naranja mecanica, el punk argentino y no me gusta stanley kubrick
nunca le presto atención a las letras así que no fue muy obvio para mi, pero el estribillo tenia muchas referencias al libro.
Al terminar el libro leí la letra y definitivamente esta lleno de términos del libro :D
acá va la letra:
Uno-Dos Ultraviolento
Uno-Dos Ultraviolento
Uno-Dos Ultraviolento
Uno-Dos Ultraviolento
Varias debotchas
caminan por ahi
Mueven sus scahrros
con frenesi
Los mal chicos de cuero
nos queremos divertir
Con mis drugos
al ataque vamos a ir
Y ahora que pasa, eh?
Y ahora que pasa, eh?
Y ahora que pasa, pasa:
Uno-Dos Ultraviolento.
Sin militscos en la esquina
Es mas fácil para mi
El dremcom en la goloba
Me hace decidir
La de grudos mas bolches
La quiero para mi
Crobo rojo entre sus capas
Les haremos salir
Y ahora que pasa, eh?
Nos quieren transformar
No lo lograran
No lo lograran
No, no lo lograran
No... no, no, no
Y ahora que pasa, eh?
Uno-Dos-Ultravio-len-to.
después de terminar el libro vi la película del mismo titulo de stanley kubrick.
y acá es donde le doy de comer a los amantes del cine arte para que me bardeen de lo lindo.
me parece muy mala la película.
No soy muy critico de las películas, son películas y listo, o me gustan o no. Normalmente cuando leo el libro antes suelo terminar pensando "el libro esta mejor" pero supongo que es una cosa normal ya que tratar de meter un libro en dos horas es difícil, pero en este caso me pareció definitivamente mala.
ya me había pasado con el mismo director después de leer 2001 odisea del espacio, no pude terminar la película porque me aburrí a los 30 minutos. En ese caso se justificaba porque el libro era puro relato que no se podía llevar a la pantalla fácilmente, pero en el caso de la naranja mecánica esta relatada por el personaje principal así que no tenia una excusa similar.
bue no voy a seguir porque no soy critico de cine (tampoco soy camionero) pero para cerrar, las omisiones, las modificaciones y el hecho que la película termine unos cuantos capítulos antes que el libro justifica al menos mi opinión.
para los amantes del cine arte con ganas de bardear a verdaderos trolls del cine los invito a pasar por el cine apesta :P
jueves, diciembre 30, 2010
an idea
one step further:
document the design and make it generic for any IM client so people can store all their chat logs in the same database.
sábado, noviembre 13, 2010
code coverage en python
convirtiéndome en un unittester necesito motivación para seguir testeando, esa motivación me sirve para mejorar, en el caso de python compito contra el puntaje que me da pylint, en el caso de testing estaba buscando algo para competir y encontré coverage, que es una herramienta que mide cuando código cubren tus tests.
si bien el code coverage no es indicativo que no hay bugs, es una buena forma de intentar superarse y agregar mas pruebas.
acá va una mini receta de como instalarlo y usarlo, en mi caso ubuntu 10.10 pero debería adaptarse fácilmente a otros SO.
instalando lo necesario
sudo apt-get install build-essential python-dev sudo easy_install pip
instalando coverage
sudo pip install coverage
probandolo
yo tengo todos los tests en un directorio llamado test (que original) y en el archivo __main__.py importo todas las clases que heredan de unittest.TestCase y llamo a unittest.main()
para correr coverage ejecuto
coverage run test/__main__.py coverage html --omit="/usr/*"
con el primero se mide la cobertura de código, con el segundo se genera un reporte html, ahora miramos el reporte
firefox htmlcov/index.html
para tener algo mas simple y fácil de correr para medir el avance podemos usar
coverage report --omit="/usr/*"
que nos muestra el reporte en consola, podemos hacerle un watch o un grep para seguir mas de cerca algún modulo
bueno, esta fue la forma de hacer el testing algo mas parecido a una competencia por quien consigue el puntaje mas alto, al menos a mi eso me sirve
martes, noviembre 09, 2010
si mis hermanos fueran canciones
si, leiste bien, este post iba a ser sobre la banda kyuss, pero en el medio estaba escuchando un tema de la banda y pense
si war fuera una cancion seria esta
(war es mi hermano mas chico) y automaticamete pense, que idea chiflada, que cancion seria pablo? (mi otro hermano, el mas grande), sin mucho pensar se me vino un tema a la mente
aqui estan, estos son... ante ustedes les presento a mis hermanos
pablo, tambien llamado Immigrant Song de Led Zeppelin
war, Whitewater de Kyuss
espero que mis canciones, ehem, hermanos les caigan bien
domingo, noviembre 07, 2010
apoyo moral a Martín Gaitán en su carrera
espero que este link le permita subir un poco mas en el ranking de google :P
ya hablando en serio, no solo me parecio interesante su carrera sino que el articulo esta muy bien escrito.
PD: me voy a mirar contra quienes compito ;)
martes, noviembre 02, 2010
charla sobre efene en la utn cordoba
del anuncio del grupo:
los esperamos este Viernes 5 de Noviembre, a las 17:30 Hs en el 1º Piso del Edificio Central de la U.T.N.
PS: se emocionaron un poco con el cartel, yo no tuve nada que ver con la concepción del mismo :P
CouchApp V: filtrar documentos por tag ordenados por fecha
ahora, como filtro documentos por tag?
si puedo hacer eso, como filtro por tag y ordeno por fecha?
ya que estamos, no seria lindo poder filtrar por tag *y* por fecha?
vamos a resolver todos estos requerimientos con una simple vista
primero creamos la vista
couchapp generate view by-tag
esto crea una vista en el directorio views llamado by-tag, este es un directorio que contiene dos archivos, map.js y reduce.js
en este caso vamos a usar solo map.js asi que borra reduce.js
cd views/by-tag rm reduce.js
ahora edita map.js para que se vea como esto
function(doc) {
if (doc.tags) {
doc.tags.forEach(function (element, index) {
emit([element, doc.created], doc);
});
}
}
por cada documento en la base de datos, vemos si tiene el atributo tags, si lo tiene, por cada tag en tags emitimos un documento cuya llave es un array con el tag y la fecha y cuyo valor es el documento en si.
ahora empujamos los cambios a couchdb
couchapp push
para probar que funciona, crea algunos documentos con el un campo llamado tags que contenga un array de strings y otro campo llamado created que contenga el timestamp en el que el documento fue creado
si pongo esta URL http://localhost:5984/datos/_design/datos/_view/by-tag/ en mi navegador, obtengo algo así:
{"total_rows":8,"offset":0,"rows":[
{"id":"d6de7e9a63039dc1af500a40af0014d7","key":["bar",1288644825761],"value":{"_id":"d6de7e9a63039dc1af500a40af0014d7","_rev":"1-eab86fbc2b4c24f31e1d60dfdd762793","author":"wariano", "created":1288644825761, "tags":["test","foo","bar"], ...}},
...
]}
esto significa que la vista funciono, ahora para filtrar por tag la URL se va a poner un poco rara
vamos a usar filtros en la vista para filtrar solo los documentos con un tag especifico
http://localhost:5984/datos/_design/datos/_view/by-tag?descending=false&startkey=["test", 0]&endkey=["test", 9999999999999]
con este request decimos que queremos los resultados de la vista llamada by-tag, filtrando los documentos empezando con la llave ["test", 0] y terminando con la llave ["test", 9999999999999]. Esto significa que solo queremos los documentos con la llave "test" y que queremos todos los timestamps (por eso el numero enorme en endkey)
si queremos ordenar los tags en orden descendente deberíamos cambiar el orden de starkey y endkey: http://localhost:5984/datos/_design/datos/_view/by-tag?descending=true&startkey=["test", 9999999999999]&endkey=["test", 0]
podemos jugar con startkey y endkey para obtener rangos de tags o un tag en un periodo de tiempo especifico, por ejemplo: "cosas taggeadas con fun en los últimos dos días"
el código para hacer el request a couchdb desde javascript es el siguiente
datos.getByTag = function (tag, descending, okCb, errorCb, startStamp, endStamp) {
var tmp;
startStamp = startStamp || 0;
endStamp = endStamp || 9999999999999;
if (descending) {
tmp = endStamp;
endStamp = startStamp;
startStamp = tmp;
}
$.couch.db(datos.db).view("datos/by-tag",
{"descending": descending, "startkey": [tag, startStamp], "endkey": [tag, endStamp],
"success": okCb, "error": errorCb});
};
con esto tenes una forma de listar documentos por uno o mas campos, podes modificar este ejemplo un poco para listar por usuario o por otras cosas
[EN] CouchApp V: filter documents by tag ordered by timestamp
you now have documents and to be in this century you decide to add tags to some elements so you can filter and categorize (say folksonomy and you will sound really hip!)
now, how do I filter the documents by tag?
if I can do that, how do I filter by tag and order by date?
now that we are at it, wouldn't be nice to have a filter by tag *and* date?
we will solve all this requirements with a simple view
first we create the view
couchapp generate view by-tag
this creates a view in the views called by-tags, this is a directory that contains two files, map.js and reduce.js
in this case we will only use the map.js file, so remove the reduce.js file
cd views/by-tag rm reduce.js
now edit the map.js file to look like this
function(doc) {
if (doc.tags) {
doc.tags.forEach(function (element, index) {
emit([element, doc.created], doc);
});
}
}
here for each document in the database we check if the document has the tags attribute and if it has the attribute, for each tag we emit a document that contains as key an array with the tag and timestamp when the document was created and as value the document itself
now we push our changes to couchdb
couchapp push
to test that this works, create some documents with a field called tags that contains a list of strings and a field called created that contain the timestamp when the item was created
if I put this URL http://localhost:5984/datos/_design/datos/_view/by-tag/ in my browser I get something like this
{"total_rows":8,"offset":0,"rows":[
{"id":"d6de7e9a63039dc1af500a40af0014d7","key":["bar",1288644825761],"value":{"_id":"d6de7e9a63039dc1af500a40af0014d7","_rev":"1-eab86fbc2b4c24f31e1d60dfdd762793","author":"wariano", "created":1288644825761, "tags":["test","foo","bar"], ...}},
...
]}
this means the view worked, now to filter by tag the URL will get weird
we will use filters in the view to filter only for a specific tag
http://localhost:5984/datos/_design/datos/_view/by-tag?descending=false&startkey=["test", 0]&endkey=["test", 9999999999999]
with this request we say that we want to get the result of the view called by-tag, filtering starting with the key ["test", 0] and ending with the key ["test", 9999999999999]. This means that we only want the documents with the key "test" and we want all the timestamps (that's why the huge number in the endkey
if we want to sort the tags in descending order we should switch the start and endkey like this: http://localhost:5984/datos/_design/datos/_view/by-tag?descending=true&startkey=["test", 9999999999999]&endkey=["test", 0]
we can play with startkey and endkey to get a range of tags or one tag in a specific period, for example, "things tagged fun in the last 2 days"
the code to do the request to couchdb from javascript is the following
datos.getByTag = function (tag, descending, okCb, errorCb, startStamp, endStamp) {
var tmp;
startStamp = startStamp || 0;
endStamp = endStamp || 9999999999999;
if (descending) {
tmp = endStamp;
endStamp = startStamp;
startStamp = tmp;
}
$.couch.db(datos.db).view("datos/by-tag",
{"descending": descending, "startkey": [tag, startStamp], "endkey": [tag, endStamp],
"success": okCb, "error": errorCb});
};
with this you have a way to list documents by one or more fields, you can modify this a little to list by users, or by some other thing
viernes, octubre 29, 2010
Como generar archivos .exe e instaladores para una aplicación python (y pygtk)
Como generar archivos .exe e instaladores para una aplicación python
Este documento describe los pasos necesarios para crear un archivo ejecutable
de una aplicación python y como generar un instalador y una versión portable
para dicha instalación.
Este documento asume que la aplicación se basa en GTK pero debería funcionar
con menores cambios en otros toolkits.
porque un instalador
- se requiere instalar muchos componentes a mano por el usuario final para una sola aplicación
- muchos instaladores pequeños
- difíciles de encontrar
- difícil encontrar las versiones exactas que funcionan en conjunto
- requiere instalarlos en un orden definido
- rezar
- algunas veces incluso haciendo todo bien puede no funcionar
- fácil de automatizar y documentar para replicar con cada nueva versión
- liberar al usuario final de los problemas para poder usar la aplicación
componentes requeridos
- python
- todas las librerías utilizadas por la aplicación
- py2exe
- nsis
- tiempo y suerte
instaladores
aquí se listan los links a los instaladores de todos los componentes usados en el ejemplo.
- http://python.org/ftp/python/2.6.6/python-2.6.6.msi
- http://sourceforge.net/projects/py2exe/files/py2exe/0.6.9/py2exe-0.6.9.win32-py2.6.exe/download
- http://ftp.gnome.org/pub/GNOME/binaries/win32/pycairo/1.8/pycairo-1.8.6.win32-py2.6.exe
- http://ftp.gnome.org/pub/GNOME/binaries/win32/pygobject/2.20/pygobject-2.20.0.win32-py2.6.exe
- http://ftp.gnome.org/pub/GNOME/binaries/win32/pygtk/2.16/pygtk-2.16.0+glade.win32-py2.6.exe
- http://sourceforge.net/projects/pywin32/files/pywin32/Build%20214/pywin32-214.win32-py2.6.exe/download
- http://sourceforge.net/projects/gtk-win/files/GTK%2B%20Runtime%20Environment/GTK%2B%202.22/gtk2-runtime-2.22.0-2010-10-01-ash.exe/download
- http://sourceforge.net/projects/gtk-win/files/GTK%2B%20Themes%20Package/2009-09-07/gtk2-themes-2009-09-07-ash.exe/download
- http://prdownloads.sourceforge.net/nsis/nsis-2.46-setup.exe?download
orden de instalación
algunos instaladores son independientes de otros, pero para evitar posibles problemas recomiendo la instalación en el siguiente orden.
- python
- gtk-runtime
- gtk2-themes
- nsis
- pygobject
- pycairo
- pygtk
- pywin32
- py2exe
tareas extra
- setear la variable de entorno PATH para agregar el path a la instalación de python
- probar la instalación con una pequeña aplicación gtk
>>> import gtk
>>> w = gtk.Window()
>>> l = gtk.Label("asd")
>>> w.add(l)
>>> w.show_all()
>>> gtk.main()
prueba con una aplicación de ejemplo
Cree un repositorio con una aplicación de ejemplo para probar los pasos, la aplicación esta disponible en github acá:
http://github.com/marianoguerra/PyGtkOnWindows
pasos
- descargarla
- descomprimirla
- ejecutar python setup.py py2exe
- copiar los directorios lib y share de la instalación del runtime de gtk (no de la instalación de pygtk) al directorio dist
- copiar todos los archivos del directorio dll al directorio dist
- borrar los locales y temas no usados de los directorios copiados a dist (yo solo dejo el theme MS-Windows)
- crear la siguiente estructura de directorios dentro de dist: etc/gtk-2.0
- dentro de ese directorio crear un archivo llamado gtkrc con una linea como la siguiente dentro:
- gtk-theme-name = "MS-Windows"
- podes cambiar el tema usado manteniendo otro theme dentro de share/themes y cambiando el nombre del theme en gtkrc
- right click en ejemplo.nsi y seleccionar "Compile NSIS Script"
- right click en ejemplo-portable.nsi y seleccionar "Compile NSIS Script"
- deberías tener el instalador y la versión portable disponibles
- para probar que funciona correctamente, correr el instalador y la versión portable en una instalación de windows sin los paquetes que instalaste anteriormente
probar con una aplicación real
ahora para sentirlo mas real, creemos un instalador y una versión portable de
un programa real, en este caso, un proyecto personal llamado emesene 2
(http://www.emesene.org/).
pasos
- descargarlo de http://github.com/emesene/emesene
- descomprimirlo
- copiar setup.py and ez_setup.py al directorio emesene
- cd emesene
- correr python setup.py py2exe
- cd ..
- copiar los directorios lib y share de la instalación del runtime de gtk (no de la instalación de pygtk) al directorio dist
- copiar todos los archivos del directorio dll al directorio dist
- borrar los locales y temas no usados de los directorios copiados a dist (yo solo dejo el theme MS-Windows)
- crear la siguiente estructura de directorios dentro de dist: etc/gtk-2.0
- dentro de ese directorio crear un archivo llamado gtkrc con una linea como la siguiente dentro:
- gtk-theme-name = "MS-Windows"
- podes cambiar el tema usado manteniendo otro theme dentro de share/themes y cambiando el nombre del theme en gtkrc
- right click en emesene.nsi y seleccionar "Compile NSIS Script"
- right click en emesene-portable.nsi y seleccionar "Compile NSIS Script"
- deberías tener el instalador y la versión portable disponibles
- para probar que funciona correctamente, correr el instalador y la versión portable en una instalación de windows sin los paquetes que instalaste anteriormente
notas
- obtengo algunos de los dlls requeridos de portable python (http://www.portablepython.com/) e inkscape (http://inkscape.org/)
[EN] How to generate .exe files and installers for a python (and pygtk) applications
How to generate .exe files and installers for a python applications
This document describes the steps required to create an executable file from a
python program and how to build an installer and portable file from that
application.
The document assumes that the application is based on GTK but it should work
with minor changes for other toolkits.
why an installer
- many components are required to install by hand by the end user for a simple application
- a lot of small installers
- hard to find
- hard to match the exact versions that work together
- install them in the required order
- pray
- sometimes even doing everything right it may not work
- easy to automate and document to replicate with each new version
- free the end user from problems to use the app
required components
- python
- all the libraries used by the application
- py2exe
- nsis
- time and luck ;)
installers
here are listed the links to the installers of all the components used in the example.
- http://python.org/ftp/python/2.6.6/python-2.6.6.msi
- http://sourceforge.net/projects/py2exe/files/py2exe/0.6.9/py2exe-0.6.9.win32-py2.6.exe/download
- http://ftp.gnome.org/pub/GNOME/binaries/win32/pycairo/1.8/pycairo-1.8.6.win32-py2.6.exe
- http://ftp.gnome.org/pub/GNOME/binaries/win32/pygobject/2.20/pygobject-2.20.0.win32-py2.6.exe
- http://ftp.gnome.org/pub/GNOME/binaries/win32/pygtk/2.16/pygtk-2.16.0+glade.win32-py2.6.exe
- http://sourceforge.net/projects/pywin32/files/pywin32/Build%20214/pywin32-214.win32-py2.6.exe/download
- http://sourceforge.net/projects/gtk-win/files/GTK%2B%20Runtime%20Environment/GTK%2B%202.22/gtk2-runtime-2.22.0-2010-10-01-ash.exe/download
- http://sourceforge.net/projects/gtk-win/files/GTK%2B%20Themes%20Package/2009-09-07/gtk2-themes-2009-09-07-ash.exe/download
- http://prdownloads.sourceforge.net/nsis/nsis-2.46-setup.exe?download
installation order
some installers are independent from the others, but to avoid posible problems I recommend the installation in this order.
- python
- gtk-runtime
- gtk2-themes
- nsis
- pygobject
- pycairo
- pygtk
- pywin32
- py2exe
extra tasks
- set the PATH environment variable to add the path to the python installation
- test that the installation works with a simple gtk application
>>> import gtk
>>> w = gtk.Window()
>>> l = gtk.Label("asd")
>>> w.add(l)
>>> w.show_all()
>>> gtk.main()
test with a sample application
I created a repository with a sample application to test the steps, the application is available in github here:
http://github.com/marianoguerra/PyGtkOnWindows
steps
- download it
- unpack it
- run python setup.py py2exe
- copy the lib and share directory from the gtk runtime installation (not the pygtk installation) to the dist directory
- copy all the files from the dll directory to the dist directory
- remove unused locales and unused themes (I keep only ms theme)
- create the following dirs inside dist: etc/gtk-2.0
- inside that create a file called gtkrc with a line like this inside:
- gtk-theme-name = "MS-Windows"
- you can change the theme by keeping that theme inside share/themes and changing the name in gtkrc
- right click on ejemplo.nsi and select "Compile NSIS Script"
- right click on ejemplo-portable.nsi and select "Compile NSIS Script"
- you should have the installer and portable versions available
- to test that it works correctly, run the installer and portable versions in a windows installation without the packages you installed previously
test with a real application
now to make it feel more real let's create an installer and portable versions
for a real world program, in this case, a project of mine called emesene 2
(http://www.emesene.org/).
steps
- download it from http://github.com/emesene/emesene
- unpack it
- copy setup.py and ez_setup.py to the emesene directory
- cd to emesene
- run python setup.py py2exe
- cd ..
- copy the lib and share directory from the gtk runtime installation (not the pygtk installation) to the dist directory
- copy all the files from the dll directory to the dist directory
- remove unused locales and unused themes (I keep only ms theme)
- create the following dirs inside dist: etc/gtk-2.0
- inside that create a file called gtkrc with a line like this inside:
- gtk-theme-name = "MS-Windows"
- you can change the theme by keeping that theme inside share/themes and changing the name in gtkrc
- right click on emesene.nsi and select "Compile NSIS Script"
- right click on emesene-portable.nsi and select "Compile NSIS Script"
- you should have the installer and portable versions available
- to test that it works correctly, run the installer and portable versions in a windows installation without the packages you installed previously
notes
- I get some needed dlls from portable python (http://www.portablepython.com/) and inkscape (http://inkscape.org/)
viernes, octubre 22, 2010
CouchApp IV: creando funciones show con templates HTML
necesitamos mostrar una entidad en su propia pagina, para eso vamos a necesitar un template html con los valores del documento almacenado en la base de datos, un template es una buena herramienta para eso, veamos como hacerlo en nuestra couchapp
primero creamos la funcion show, esta funcion es llamada para mostrar un documento en algun formato, en nuestro caso html
# generamos la funcion show couchapp generate show dato # la editamos con un editor de texto vim shows/dato.js
en un principio vamos a ver este contenido
function(doc, req) {
}
reemplazamos por algo parecido a esto
function(doc, req) {
if (doc !== null && doc.name) {
var ddoc = this,
Mustache = require("vendor/couchapp/lib/mustache"),
path = require("vendor/couchapp/lib/path").init(req),
assetPath = path.asset();
provides("html", function() {
return Mustache.to_html(ddoc.templates.dato, {
assetPath: assetPath,
doc: doc
});
});
}
else {
return "not found";
}
}
en el codigo controlamos que tenemos un documento, sino mostramos "not found",
tambien creamos algunos objetos que nos ayudan y finalmente renderizamos el template pasandole algunos valores que seran usados dentro del template.
mi template esta en templates/dato.html (por eso ddoc.templates.dato) y contiene un template mustache, mira la documentacion de mustache para mas informacion sobre el formato
la url par acceder a esta funcion es [database]/_design/[app]/_show/[showname]/[docid] un ejemplo podria ser http://localhost:5984/datos/_design/datos/_show/dato/6bd97648d74961996c8f0d42b2005761
[EN] CouchApp IV: creating show functions with html templates
we need to display some entity in its own web page, for that we need to fill an html template with the values of the document stores in the database, a template is a nice fit for this, let's see how we do this in our couchapp
first, create the show function, this is a function that when called displays a document in some format, in our case in HTML
# generate the show function couchapp generate show dato # open it with a text editor vim shows/dato.js
we will see this content
function(doc, req) {
}
we replace the content with something like this
function(doc, req) {
if (doc !== null && doc.name) {
var ddoc = this,
Mustache = require("vendor/couchapp/lib/mustache"),
path = require("vendor/couchapp/lib/path").init(req),
assetPath = path.asset();
provides("html", function() {
return Mustache.to_html(ddoc.templates.dato, {
assetPath: assetPath,
doc: doc
});
});
}
else {
return "not found";
}
}
here we check that we have a document, if not return "not found", also we create some objects that will help us and finally we render the template passing some values that will be used in the template.
my template is located at templates/dato.html (that's why ddoc.templates.dato) and contains a mustache template, see mustache documentation for information about the format
the url to access this function is [database]/_design/[app]/_show/[showname]/[docid] an example could be http://localhost:5984/datos/_design/datos/_show/dato/6bd97648d74961996c8f0d42b2005761
jquery.couch snippets
algunos snippets para hacer tareas comunes en jquery.couch
var datos = {};
datos.db = "datos";
datos.login = function (user, password, okCb, errorCb) {
$.couch.login({"name": user, "password": password, "success": okCb, "error": errorCb});
};
datos.logout = function (okCb, errorCb) {
$.couch.logout({"success": okCb, "error": errorCb});
};
datos.checkSession = function (okCb, errorCb) {
$.couch.session({"success": okCb, "error": errorCb});
};
datos.create = function (data, okCb, errorCb) {
$.couch.db(datos.db).saveDoc(data, {"success": okCb, "error": errorCb});
};
datos.get = function (id, okCb, errorCb, rev) {
$.couch.db(datos.db).openDoc(id, {"success": okCb, "error": errorCb, "rev": rev});
};
datos.remove = function (id, rev, okCb, errorCb) {
$.couch.db(datos.db).removeDoc({"_id": id, "_rev": rev}, {"success": okCb, "error": errorCb});
};
datos.getRecent = function (limit, okCb, errorCb) {
$.couch.db(datos.db).view("datos/recent-items",
{"limit": limit, "descending": true, "success": okCb, "error": errorCb});
};
CouchApp III: subiendo los cambios automaticamente
ahora que tenemos todo lo que necesitamos para crear nuestra couchapp, podemos ir al directorio _attachments y empezar a modificar el código para que haga lo que queramos
después de un rato vas a notar que necesitas hacer push de los cambios a couchdb cada vez que queres probarlos, si sos como yo esto se va a poner molesto bastante rápido, hagamos que las herramientas nos ayuden
#install inotify-tools sudo apt-get install inotify-tools
con inotify tools vamos a correr un script que va a monitorear cualquier cambio de archivos y va a hacer un push cuando eso suceda, voy a excluir los cambios en *.swp ya que vim crea esos archivos y cambian bastante seguido
corre este comando en algún shell y dejalo corriendo
inotifywait -q -e modify -m -r . | while read line; do if echo $line | grep -v .*.swp; then couchapp push; fi; done
ahora edita algún archivo y guardalo, anda al shell donde tenes el script corriendo, vas a ver algo como esto:
./_attachments/ MODIFY index.html 2010-10-22 11:00:48 [INFO] Visit your CouchApp here: http://wariano:secret@localhost:5984/datos/_design/datos/index.html
ahora podes editar tus archivos y los cambios van a ser automáticamente subidos a couchdb
Seguidores
Archivo del Blog
-
►
2011
(74)
- ► septiembre (4)
-
►
2010
(111)
- ► septiembre (8)
-
►
2009
(96)
- ► septiembre (3)
-
►
2008
(60)
- ► septiembre (8)
-
►
2007
(64)
- ► septiembre (1)
-
►
2006
(81)
- ► septiembre (1)


