Willyfrog http://blog.willyfrog.es/ es Fri, 23 May 2014 18:26:00 GMT http://blog.willyfrog.es/rss/ acrylamid 0.7.6 interrupt.el http://blog.willyfrog.es/2014/interrupt-el/ <p>Pues poco a poco he seguido aprendiendo <a class="reference external" href="http://www.gnu.org/software/emacs/">Emacs</a>, diversos sabores de lisp (ninguno a fondo aun) y entre ellos bastante elisp. La cosa es que cada dia disfruto más con Emacs y cada vez estoy mas enredado con él. Hasta el punto que empiezo a desarrollar pequeñas utilidades para mi día a día.</p> <p>Y de ahí este post, recientemente en mi trabajo tengo la necesidad de mantener un log de las interrupciones sufridas a lo largo de una jornada de trabajo. Es algo que no debería necesitar, pero la freecuencia es elevada y el equipo ha visto la necesidad de hacer algo así. Como cada cual usamos un editor distinto y es algo extraoficial, no hay un metodo común, asi que cada cual se ha buscado las castañas.</p> <p>En mi caso, como queria aprender <a class="reference external" href="http://orgmode.org/">org-mode</a>, empecé con un fichero donde iba anotando las interrupciones. Una interrupcion: un elemento en la lista, añadia la fecha y marcaba mediante tags al <em>interruptor</em>. En unos dias resultaba un tanto tedioso hacer todo esto, por lo que me pareció el momento perfecto para intentar usar un poco <em>elisp-fu</em> y de ahi surgio <a class="reference external" href="https://github.com/Willyfrog/interrupt.el">interrupt.el</a></p> <p>Asi que ahora simplemente pulsando &lt;F12&gt; puedo añadir el nombre de quien me interrumpa, ir a ver qué necesita la persona y cuando vuelva tendré el cursor esperando para añadir el motivo. Como extra, puedo pulsar despues &lt;S-F12&gt; y además apuntará la cantidad de minutos que he estado durante la interrupcion.</p> <p>Desde luego aun está en sus primeras versiones, pero ya hace mucho más de lo que pensaba poner en un principio asi que ya veremos como continua.</p> Fri, 23 May 2014 18:26:00 GMT tag:blog.willyfrog.es,2014-05-23:/2014/interrupt-el Configuracion encriptada en Emacs http://blog.willyfrog.es/2013/configuracion-encriptada-en-emacs/ <p>En mi aprendizaje de Emacs he descubierto el módulo <a class="reference external" href="http://sf.net/p/emacs-jabber">jabber.el</a> que te permite mantener las conversaciones de Google Talk dentro del propio editor. Pero hay que pasarle las contraseñas o guardarlas en modo plano, lo cual es algo incomodo desde el punto de vista de la seguridad. Si bien para la cuenta del trabajo me daria igual ya que no usa la autenticación de dos factores y por lo tanto me se de memoria mi contraseña, en la personal si lo tengo habilitado y la desconozco y aunque la conociera, seria poco afortunado guardarla como texto plano.</p> <p>Por ello, buscando un poco he encontrado como cifrar un fichero <em>.el</em> y la verdad es que es tremendamente sencillo. Para empezar hay que habilitar la extensión <em>.el.gpg</em> por lo que incluimos en nuestro .emacs:</p> <pre class="brush: elisp"> (add-to-list 'load-suffixes ".el.gpg") </pre> <p>A partir de aqui, solo hay que crear un fichero con dicha extensión y Emacs sabrá que hacer ;) Cada vez que guardes, te pedirá la contraseña con la que debe cifrar el contenido.</p> <p>Y ya se puede incluir el fichero con</p> <pre class="brush: elisp"> (require 'fichero-cifrado) </pre> <p>con lo que puedes usar lo que haya dentro en el resto del editor.</p> Mon, 28 Oct 2013 15:52:00 GMT tag:blog.willyfrog.es,2013-10-28:/2013/configuracion-encriptada-en-emacs Mi primera utilidad de elisp http://blog.willyfrog.es/2013/mi-primera-utilidad-de-elisp/ <p>Desde hace algún tiempo me pasé a <a class="reference external" href="http://www.gnu.org/software/emacs/">Emacs</a>, si bien la versatilidad ya la tenía en vim (es un gran editor) con Emacs puedo además aprovechar para aprender un sabor de lisp: <a class="reference external" href="http://en.wikipedia.org/wiki/Emacs_Lisp">emacs lisp</a> (<em>elisp</em> para abreviar). Esta es una tarea que imagino que me llevará años ya que no es un uso intensivo, sino que intentaré ir haciendo pequeñas funciones y cambios que me permitan aprovechar esa caracteristica tan genial de Emacs de ser programable.</p> <p>Lo difícil es encontrar algo que hacer, por lo que lo primero que se me ha ocurrido es algo que ya he tenido que hacer varias veces: cerrar todos los buffers de un proyecto. Estoy seguro que si lo busco, encuentro que alguien ya lo ha hecho y mucho mejor que yo, pero si no voy haciendo pequeñas funciones, no creo que aprenda en la vida.</p> <div class="highlight"><pre><span class="lineno"> 1</span> (defun filter-buffers-by-file-name (partial-path) <span class="lineno"> 2</span> &quot;return list of buffers which contains &#39;partial-path&#39;&quot; <span class="lineno"> 3</span> (remove-if (lambda (x) (or (null (buffer-file-name x)) <span class="lineno"> 4</span> (null (string-match partial-path (buffer-file-name x))))) (buffer-list))) <span class="lineno"> 5</span> <span class="lineno"> 6</span> (defun kill-project-buffers (partial-path) <span class="lineno"> 7</span> &quot;kills all buffers which contains partial-path&quot; <span class="lineno"> 8</span> (interactive &quot;sPartial name:&quot;) <span class="lineno"> 9</span> (let ((kill-list (filter-buffers-by-file-name partial-path))) <span class="lineno">10</span> (if (null kill-list) <span class="lineno">11</span> (message (format &quot;No buffers matched &#39;%s&#39;&quot; partial-path)) <span class="lineno">12</span> (if (y-or-n-p (format &quot;killing %d buffers&quot; (length kill-list))) <span class="lineno">13</span> (kill-some-buffers kill-list) <span class="lineno">14</span> (message (&quot;User cancelled&quot;)) <span class="lineno">15</span> ) <span class="lineno">16</span> ))) </pre></div> <p>Si bien estoy seguro de que se puede hacer mejor, esta es mi segunda aproximación (la original daba mas vueltas a los datos para conseguir lo mismo) que funciona. Lo que hace es buscar entre los buffers que han sido cargados desde un fichero aquellos que contengan en su path el texto indicado, teniendo en cuenta que puede usarse expresiones regulares. De esta manera poniendo el nombre del proyecto (o parte de su path), eliminará los buffers que coincidan.</p> <p>La última versión de esta funcion, podrá encontrarse siempre en <a class="reference external" href="https://github.com/Willyfrog/vimfiles/tree/master/emacs_custom">github</a></p> Wed, 28 Aug 2013 09:49:00 GMT tag:blog.willyfrog.es,2013-08-28:/2013/mi-primera-utilidad-de-elisp Auriculares para correr miomundo http://blog.willyfrog.es/2013/auriculares-para-correr-miomundo/ <p>Recientemente he comprado unos <a class="reference external" href="http://www.amazon.es/gp/product/B00D5SRJDA/ref=oh_details_o00_s00_i00?ie=UTF8&amp;psc=1">auriculares</a> ligeros para correr. Esto no debería ser un motivo de entrada para un blog, ¿no? pero me ha llamado la atención el poco cuidado de los detalles que han tenido hasta el punto de que debe haberles perjudicado gravemente a las ventas.</p> <p>Pero antes de nada decir que me han costado 12 €, por lo que no espero un producto de grandes acabados ni que la calidad de sus graves sea superior. Cuando decidí comprarlos fue porque no queria dejarme mucho dinero en un producto que no se si lo usaré muy a menudo, ya que depende de cuanto continue corriendo. Pero para lo que me ha costado, por ahora estoy contento. Sirve justo para lo que lo quiero, para poner la musica del movil y que no se me caigan los cascos mientras corro, asi que por esa parte me parece genial.</p> <p>El problema que les he visto es que el &quot;manual&quot; de instrucciones (si se le puede llamar así a cuatro parrafos en la cara posterior del envoltorio) es para un producto similar pero sustancialmente diferente. Hasta el punto de que he estado a punto de devolverlo al no ser capaz de enlazar los cascos con el bluetooth del movil. Finalmente, por cabezonería he encontrado la manera: &quot;mantener pulsado boton de play hasta que parpadee de continuo mientras el móvil busca dispositivos&quot;. Pero creo que mucha gente puede haber devuelto el producto por no tener un minimo de detalle y haber puesto la caratula apropiada.</p> Tue, 27 Aug 2013 09:39:00 GMT tag:blog.willyfrog.es,2013-08-27:/2013/auriculares-para-correr-miomundo Traduccion elisp en 15 minutos (aprox.) http://blog.willyfrog.es/2013/traduccion-elisp-en-15-minutos-aprox/ <p>El otro dia vi un tutorial sencillo y rapido para aprender <a class="reference external" href="http://bzg.fr/learn-emacs-lisp-in-15-minutes.html">emacs lisp</a>, me pareció una introducción muy necesaria para la gente que (como yo) está aprendiendo <a class="reference external" href="http://www.gnu.org/s/emacs/‎">Emacs</a>. Más que nada porque aprender Emacs y no aprender como funciona elisp es perder el tiempo. ¡Ojo! esto no quiere decir que sea necesario tener la capacidad de escribir grandes y complicados programas en él, sino simplemente ser capaz de leerlo y entender por donde van los tiros. Y es que su propia configuración está escrita en elisp, por lo que si no lo entiendes, dependes del resto para poder configurar tu editor.</p> <p>En fin, que me gustó lo suficiente como para dedicar un par de horas y hacer un <em>pull request</em> a <a class="reference external" href="https://twitter.com/adambard">adambard</a> y su <a class="reference external" href="http://learnxinyminutes.com/">learnXinYminutes</a> traduciendolo.</p> <p>El resultado lo podeis ver <a class="reference external" href="http://learnxinyminutes.com/docs/es-es/elisp-es/">aqui</a> y si veis un fallo podeis bien <a class="reference external" href="https://github.com/adambard/learnxinyminutes-docs">corregirlo</a> o avisarme y me encargo yo.</p> Tue, 06 Aug 2013 08:17:00 GMT tag:blog.willyfrog.es,2013-08-06:/2013/traduccion-elisp-en-15-minutos-aprox Movimiento estático http://blog.willyfrog.es/2013/movimiento-estatico/ <p>Finalmente he movido el blog de <a class="reference external" href="http://wordpress.com">Wordpress</a> a algo estático, como llevaba queriendo hacer desde hace tiempo. Wordpress no tiene nada malo, me sigue pareciendo una plataforma de blog muy buena, especialmente para gente que no quiera complicarse la vida.</p> <p>Pero (y si no hubiera un pero, seria que me gusta mover por mover) requiere ser activo, cosa que no me pasa con el blog, que durante un tiempo escribo y durante un tiempo dejo de escribir, alternandose ambos a lo largo del tiempo. Y wordpress, siendo dinamico, con php y mysql, require de mimo, cuidado y mantenimiento para evitar que alguien utilize algun exploit, andar con temas de antispam, etc.</p> <p>Resultado, he eliminado los comentarios, he migrado todo de mi <a class="reference external" href="http://blog.willinux.net">antiguo blog</a> y he pasado parte de las vacaciones aprendiendo como funciona acrylamid (toma nombrecito) para ajustarlo a mis necesidades.</p> <p>Además, si alguien me seguia, verá que ahora estreno dominio. Voy a dejar willinux por éste por el que me siento algo más identificado.</p> <p>Por ultimo, si te interesa como está montado el blog, está todo el <a class="reference external" href="https://github.com/Willyfrog/blog">código en github</a>.</p> Sun, 23 Jun 2013 23:22:00 GMT tag:blog.willyfrog.es,2013-06-24:/2013/movimiento-estatico Cargando modulos dinámicamente en la botella http://blog.willyfrog.es/2012/cargando-modulos-dinamicamente-en-la-botella/ <p>Estoy algo liado montando una api para un pequeño proyecto mio, buscando que sea lo más cercano a RESTFUL por motivos laborales y por aprender cosas nuevas (entremedias estoy probando Emacs, que me esta costando un rato el cambio desde Vim).</p> <p>Como framework estoy usando <a class="reference external" href="http://bottlepy.org/">bottle</a>, ya que siempre estoy usando Flask y quería probar otro distinto a ver qué tal. Y más adelante intentaré montarlo de manera asíncrona, pero eso para más adelante.</p> <p>Como digo, es una primera aproximación, imagino que el código final tendrá una pinta muy diferente, pero para alguien que quiera cargar dinámicamente los módulos y tratar de ejecutar las funciones es un buen punto de partida.</p> <div class="highlight"><pre><span class="lineno"> 1</span> <span class="nd">@route</span><span class="p">(</span><span class="s">&#39;/:module/:action&#39;</span><span class="p">)</span> <span class="c"># las rutas dinámicas están compuestas de la forma modulo/función</span> <span class="lineno"> 2</span> <span class="k">def</span> <span class="nf">run_action</span><span class="p">(</span><span class="n">module</span><span class="p">,</span><span class="n">action</span><span class="p">):</span> <span class="lineno"> 3</span> <span class="k">try</span><span class="p">:</span> <span class="lineno"> 4</span> <span class="n">m</span> <span class="o">=</span> <span class="n">import_module</span><span class="p">(</span><span class="n">module</span><span class="p">)</span> <span class="lineno"> 5</span> <span class="k">except</span> <span class="ne">ImportError</span><span class="p">:</span> <span class="lineno"> 6</span> <span class="n">redirect</span><span class="p">(</span><span class="s">&#39;/404&#39;</span><span class="p">)</span> <span class="lineno"> 7</span> <span class="n">action_call</span> <span class="o">=</span> <span class="n">module</span> <span class="o">+</span> <span class="s">&#39;_&#39;</span> <span class="o">+</span> <span class="n">action</span> <span class="lineno"> 8</span> <span class="c"># para evitar que cualquier función del modulo pueda ser llamada, solo aquellos que comienzan por el nombre del modulo pueden ser invocados</span> <span class="lineno"> 9</span> <span class="k">try</span><span class="p">:</span> <span class="lineno">10</span> <span class="n">a</span> <span class="o">=</span> <span class="nb">getattr</span><span class="p">(</span><span class="n">m</span><span class="p">,</span> <span class="n">action_call</span><span class="p">)</span> <span class="lineno">11</span> <span class="k">except</span> <span class="ne">AttributeError</span><span class="p">,</span> <span class="n">e</span><span class="p">:</span> <span class="c"># diferencia entre fallo por modulo o accion</span> <span class="lineno">12</span> <span class="k">print</span> <span class="s">&quot;ERROR: module </span><span class="si">%s</span><span class="s"> doesn&#39;t have a </span><span class="si">%s</span><span class="s"> function&quot;</span> <span class="o">%</span> <span class="p">(</span><span class="n">module</span><span class="p">,</span> <span class="n">action_call</span><span class="p">)</span> <span class="lineno">13</span> <span class="k">print</span> <span class="s">&quot;ERROR: </span><span class="si">%s</span><span class="s">&quot;</span> <span class="o">%</span> <span class="n">e</span> <span class="lineno">14</span> <span class="n">redirect</span><span class="p">(</span><span class="s">&#39;/400&#39;</span><span class="p">)</span> <span class="lineno">15</span> <span class="k">return</span> <span class="n">a</span><span class="p">()</span> </pre></div> <p>A este código habría que añadir el paso de parámetros, así como si hay diferenciación por método (get, post, put o delete)</p> Sun, 04 Nov 2012 09:42:00 GMT tag:blog.willyfrog.es,2012-11-04:/2012/cargando-modulos-dinamicamente-en-la-botella Des/activando virtualenv automaticamente http://blog.willyfrog.es/2012/desactivando-virtualenv-automaticamente/ <p>Algo que no se me habia ocurrido hasta que mi compañero me lo comentó es la activación automática del entorno virtualenv nada más entrar en el propio directorio, de esta manera ahorramos tiempo y problemas de no estar en el virtualenv apropiado.</p> <p>Buscando un poco por la red, encontré este <a class="reference external" href="https://gist.github.com/2198913">gist</a> donde hacía parte de lo que queria, ya que no terminaba el virtualenv al abandonar el directorio, por lo que podria dar problemas al seguir en la misma linea de comandos y ejecutar otras cosas. De ahí que modificara el script para que al abandonar el directorio desactivara el entorno.</p> <p>Para poder utilizarlo se puede poner el entorno en .venv en el directorio o crear un enlace simbolico (llamado igualmente .venv) si queremos tenerlo en otro lado. El script debe incluirse en .bashrc y ya solo falta cerrar e iniciar sesión.</p> <p>El script en cuestión está en un <a class="reference external" href="https://gist.github.com/3226639">fork del enlazado anteriormente</a>.</p> Wed, 01 Aug 2012 13:05:00 GMT tag:blog.willyfrog.es,2012-08-01:/2012/desactivando-virtualenv-automaticamente Receta para introspeccion en python http://blog.willyfrog.es/2012/receta-para-introspeccion-en-python/ <p>Al final no me ha servido ya que hay varias propiedades que no son legibles en el codigo legacy que estoy mirando, pero por si a alguien le sirviera copio una receta vista en <a class="reference external" href="http://stackoverflow.com/questions/1006169/how-do-i-look-inside-a-python-object">SO</a> para ver los contenidos de un objeto:</p> <table class="docutils field-list" frame="void" rules="none"> <col class="field-name"/> <col class="field-body"/> <tbody valign="top"> <tr class="field"><th class="field-name">Original:</th><td class="field-body"><a class="reference external" href="http://diveintopython.net/">diveintopython</a></td> </tr> <tr class="field"><th class="field-name">Author:</th><td class="field-body">Mark Pilgrim (<a class="reference external" href="http://blog.willyfrog.es/2012/receta-para-introspeccion-en-python/mailto:mark@diveintopython.org">mark&#64;diveintopython.org</a>)</td> </tr> </tbody> </table> <div class="highlight"><pre><span class="k">def</span> <span class="nf">info</span><span class="p">(</span><span class="nb">object</span><span class="p">,</span> <span class="n">spacing</span><span class="o">=</span><span class="mi">10</span><span class="p">,</span> <span class="n">collapse</span><span class="o">=</span><span class="mi">1</span><span class="p">):</span> <span class="sd">&quot;&quot;&quot;</span> <span class="sd"> Print methods and doc strings.</span> <span class="sd"> Takes module, class, list, dictionary, or string.</span> <span class="sd"> &quot;&quot;&quot;</span> <span class="n">methodList</span> <span class="o">=</span> <span class="p">[</span><span class="n">e</span> <span class="k">for</span> <span class="n">e</span> <span class="ow">in</span> <span class="nb">dir</span><span class="p">(</span><span class="nb">object</span><span class="p">)</span> <span class="k">if</span> <span class="nb">callable</span><span class="p">(</span><span class="nb">getattr</span><span class="p">(</span><span class="nb">object</span><span class="p">,</span> <span class="n">e</span><span class="p">))]</span> <span class="n">processFunc</span> <span class="o">=</span> <span class="n">collapse</span> <span class="ow">and</span> <span class="p">(</span><span class="k">lambda</span> <span class="n">s</span><span class="p">:</span> <span class="s">&quot; &quot;</span><span class="o">.</span><span class="n">join</span><span class="p">(</span><span class="n">s</span><span class="o">.</span><span class="n">split</span><span class="p">()))</span> <span class="ow">or</span> <span class="p">(</span><span class="n">lambdas</span><span class="p">:</span> <span class="n">s</span><span class="p">)</span> <span class="k">print</span> <span class="s">&quot;</span><span class="se">\\</span><span class="s">n&quot;</span><span class="o">.</span><span class="n">join</span><span class="p">([</span><span class="s">&quot;</span><span class="si">%s</span><span class="s"> </span><span class="si">%s</span><span class="s">&quot;</span> <span class="o">%</span> <span class="p">(</span><span class="n">method</span><span class="o">.</span><span class="n">ljust</span><span class="p">(</span><span class="n">spacing</span><span class="p">),</span> <span class="n">processFunc</span><span class="p">(</span><span class="nb">str</span><span class="p">(</span><span class="nb">getattr</span><span class="p">(</span><span class="nb">object</span><span class="p">,</span> <span class="n">method</span><span class="p">)</span><span class="o">.</span>\<span class="n">_</span>\<span class="n">_doc</span>\<span class="n">_</span>\<span class="n">_</span><span class="p">)))</span> <span class="k">for</span> <span class="n">method</span> <span class="ow">in</span> <span class="n">methodList</span><span class="p">])</span> <span class="k">if</span> <span class="n">name</span> <span class="o">==</span> <span class="s">&quot;main&quot;</span><span class="p">:</span> <span class="k">print</span> <span class="n">help</span><span class="o">.</span><span class="n">doc</span> </pre></div> <p>En parte no lo uso por como está estructurado el codigo este que tenemos, de hecho finalmente una variación más sencilla y peor formateada me sirvió para lo que queria:</p> <div class="highlight"><pre><span class="n">lista_negra</span> <span class="o">=</span> <span class="p">[</span><span class="s">&#39;atributos problematicos aqui&#39;</span><span class="p">]</span> <span class="n">logWARN</span><span class="p">(</span><span class="s">&quot;</span><span class="si">%s</span><span class="s">&quot;</span> <span class="o">%</span> <span class="p">[</span><span class="s">&quot;</span><span class="si">%s</span><span class="s">: </span><span class="si">%s</span><span class="s">&quot;</span> <span class="o">%</span> <span class="p">(</span><span class="n">val</span><span class="p">,</span> <span class="nb">getattr</span><span class="p">(</span><span class="bp">self</span><span class="p">,</span> <span class="n">val</span><span class="p">,</span> <span class="s">&quot;No legible&quot;</span><span class="p">))</span> <span class="k">for</span> <span class="n">val</span> <span class="ow">in</span> <span class="nb">dir</span><span class="p">(</span><span class="bp">self</span><span class="p">)</span> <span class="k">if</span> <span class="n">val</span> <span class="ow">not</span> <span class="ow">in</span> <span class="n">lista_negra</span><span class="p">])</span><span class="sb">``</span> </pre></div> <p>la lista negra es porque en este trozo de codigo concreto hay muchos 'alias' de funcion y al solicitar su valor ejecuta la funcion la cual ni siquiera esta en el contexto apropiado. Supongo que es lo que tiene que el objeto contenga unos 600 valores entre atributos, métodos y alias T_T</p> Fri, 15 Jun 2012 08:43:00 GMT tag:blog.willyfrog.es,2012-06-15:/2012/receta-para-introspeccion-en-python Pyweek 2012 http://blog.willyfrog.es/2012/pyweek/ <p>Despues del &quot;subidon&quot; con ludum dare, me atreví a meterme en pyweek.</p> <p>No ha sido tan excitante en algunos aspectos, ya que tenia mucho más tiempo y eso provocó el &quot;luego lo hago&quot;. La mayor parte del tiempo se me fue en pequeños errores (alguno de los cuales aun siguen existiendo) en el tema de colisiones. En general los consejos y criticas son los que yo mismo pensaba cuando subí la versión final.</p> <p>Quizás me planteo para más adelante recuperar la idea y hacerlo de nuevo con otro engine... pero eso es hablar del futuro.</p> <p><a class="reference external" href="http://pyweek.org/e/bot_uprising/">Mi participación en el pyweek</a></p> Mon, 28 May 2012 10:58:00 GMT tag:blog.willyfrog.es,2012-05-28:/2012/pyweek Ludum dared http://blog.willyfrog.es/2012/ludum-dared/ <p>Este fin de semana me he dado un homenaje tremendamente geek: he participado en un gamejam (<a class="reference external" href="http://www.gamejam.es/">explicacion en castellano</a>) de 48 horas, en concreto el <a class="reference external" href="http://www.ludumdare.com/compo/">ludum dare</a> en su 23ª edición. Las reglas son sencillas: dado un tema tienes 48 horas para hacer un juego completo tú solo (incluye código, animaciones, sonido, etc.)</p> <p>No he podido entregar al final del concurso, aunque ciertamente no esperaba hacerlo. Este fin de semana he tenido varios compromisos que no me iban a permitir sacar el máximo de horas para el evento, pero estoy contento de tener algo jugable</p> <p>Una de las cosas que me ha permitido el ir tan relajado es el probar Scala como lenguaje de programación utilizando Slick2D como framework y la verdad es que ha sido muy cómodo. El mayor problema que he tenido es el no poder generar un .jar ejecutable para poder hacer la entrega. De haber usado Scala antes, eso lo tendría solucionado y es algo que espero hacer en un futuro cercano. Además, como algo insólito en mi, he desarrollado desde windows ya que el stack de Scala es muy cómodo de instalar y quería probarlo, además de facilitarme el hecho de usar otras herramientas de edición de imágenes (pyxeledit) que no tienen versión Linux (no desde que Adobe retiró Air para Linux).</p> <p>La toma de decisiones es una de las cosas que más me ha fascinado. Con la continua falta de tiempo todo es una decisión a tomar en el momento, si pospones la decisión es algo que no harás. Y quizás debería haber llevado una lista más completa de cosas que quería/debería hacer durante el concurso. En ocasiones me daba cuenta de que algo importante lo había desplazado por otra tarea por no haberme acordado de ella. La refactorización y el eliminar código para rehacer algo ha sido contínuo.</p> <p>Por ultimo: el código y assets están disponibles en mi <a class="reference external" href="https://github.com/Willyfrog/tinygod">github</a> Hasta que consiga generar un ejecutable hace falta scala + sbt y ejecutarlo desde ahí.</p> <p>La duda que me queda es: ¿lo continuaré en las próximas semanas?</p> Mon, 23 Apr 2012 09:37:00 GMT tag:blog.willyfrog.es,2012-04-23:/2012/ludum-dared Nuevo mini-proyecto: tuitorrent http://blog.willyfrog.es/2012/nuevo-mini-proyecto-tuitorrent/ <p>En estos días, he intentado sacar algo de tiempo para un mini-proyecto que tenía en mente desde hace mucho y que recordé hace poco gracias a un artículo: <a class="reference external" href="http://inventwithpython.com/blog/2012/03/25/how-to-code-a-twitter-bot-in-python-on-dreamhost/">&quot;Como crear un bot de Twitter en Python&quot;</a> (titulo traducido libremente).</p> <p>La idea es la de un bot de twitter al que mandarle torrents y que este se encargue de gestionarlos. Por ahora solo descarga el torrent y lo coloca en una carpeta, la cual está vigilada por utorrent que al detectar el nuevo archivo lo pone a la cola. Como efecto extra, también descarga archivos que no sean torrent, pero en este caso no se añaden a la lista y se quedan ahí.</p> <p>El <a class="reference external" href="https://github.com/Willyfrog/tuitorrent">proyecto está alojado en github</a> bajo una licencia BSD de 3 clausulas (que podeis trastear con él todo lo que queráis mientras sigáis diciendo de donde salio ;) ). Esto me recuerda que tambien tengo que añadir la licencia al proyecto :P</p> <p>Los requisitos que tiene son pocos por ahora: Python 2.X (solo he probado con la 2.7) y python-twitter</p> <p>Por ahora está en una versión muy primaria (alfa que se suele decir), por lo que tiene varios problemas:</p> <blockquote> <ul class="simple"> <li>la instalación es más complicada de lo que me gustaría. Aunque he</li> </ul> </blockquote> <div class="system-message"> <p class="system-message-title">System Message: WARNING/2 (<tt class="docutils">&lt;string&gt;</tt>, line 25)</p> Block quote ends without a blank line; unexpected unindent.</div> <p>puesto un shell script, hay que asegurarse de al menos tener python y virtualenv, cosa no siempre disponible (para windows ni siquiera he intentado automatizarlo). Además requiere generar un token de oauth en twitter y darle las claves al bot, para que pueda leer y escribir de la cuenta.</p> <blockquote> <ul class="simple"> <li>la ejecución requiere pasos poco necesarios fuera de un entorno de</li> </ul> </blockquote> <div class="system-message"> <p class="system-message-title">System Message: WARNING/2 (<tt class="docutils">&lt;string&gt;</tt>, line 32)</p> Block quote ends without a blank line; unexpected unindent.</div> <p>desarrollo. Cuando desarrollo con python uso virtualenv para abstraer el uso de librerías del resto de la máquina, esto hace que hay que activar el entorno previamente a lanzar el cliente.</p> <blockquote> <ul class="simple"> <li>python-twitter tiene un problema con la codificación de caracteres</li> </ul> </blockquote> <div class="system-message"> <p class="system-message-title">System Message: WARNING/2 (<tt class="docutils">&lt;string&gt;</tt>, line 37)</p> Block quote ends without a blank line; unexpected unindent.</div> <p>que he arrastrado y espero solucionar en algún momento, ya que justo en mac es donde me ha sucedido, aunque espero arreglarlo esta tarde/noche.</p> <p>De cara a un futuro, me gustaría por un lado facilitar el uso e instalación del bot y por otro, montarle un pequeño sistema de plugins de manera que pueda realizar más acciones que solo descarga de archivos y quizás filtrar los tipos de archivos descargables o una lista blanca de quien descargar.</p> Tue, 10 Apr 2012 07:49:00 GMT tag:blog.willyfrog.es,2012-04-10:/2012/nuevo-mini-proyecto-tuitorrent Curva de Hilbert en un 'tuit' http://blog.willyfrog.es/2012/curva-de-hilbert-en-un-tuit/ <p>Actualmente estoy mirando implementaciones de las curvas de <a class="reference external" href="http://es.wikipedia.org/wiki/Curva_de_Peano">Peano</a> y <a class="reference external" href="http://es.wikipedia.org/wiki/Curva_de_Hilbert">Hilbert</a> y me he encontrado en la <a class="reference external" href="http://comments.gmane.org/gmane.org.user-groups.python.argentina/43309">lista de python-argentina</a> con una curiosa implementación de una curva de Hilbert en un tweet (es decir en 140 caracteres o menos):</p> <div class="highlight"><pre><span class="lineno">1</span> <span class="kn">from</span> <span class="nn">turtle</span> <span class="kn">import</span> <span class="o">*</span> <span class="lineno">2</span> <span class="n">l</span><span class="o">=</span><span class="n">left</span> <span class="lineno">3</span> <span class="n">f</span><span class="o">=</span><span class="n">forward</span> <span class="lineno">4</span> <span class="k">def</span> <span class="nf">h</span><span class="p">(</span><span class="n">n</span><span class="p">,</span><span class="n">a</span><span class="p">):</span> <span class="lineno">5</span> <span class="k">if</span> <span class="n">n</span><span class="p">:</span><span class="n">n</span><span class="o">-=</span><span class="mi">1</span><span class="p">;</span><span class="n">l</span><span class="p">(</span><span class="n">a</span><span class="o">*</span><span class="mi">3</span><span class="p">);</span><span class="n">h</span><span class="p">(</span><span class="n">n</span><span class="p">,</span><span class="o">-</span><span class="n">a</span><span class="p">);</span><span class="n">f</span><span class="p">(</span><span class="mi">4</span><span class="p">);</span><span class="n">l</span><span class="p">(</span><span class="n">a</span><span class="p">);</span><span class="n">h</span><span class="p">(</span><span class="n">n</span><span class="p">,</span><span class="n">a</span><span class="p">);</span><span class="n">f</span><span class="p">(</span><span class="mi">4</span><span class="p">);</span><span class="n">h</span><span class="p">(</span><span class="n">n</span><span class="p">,</span><span class="n">a</span><span class="p">);</span><span class="n">l</span><span class="p">(</span><span class="n">a</span><span class="p">);</span><span class="n">f</span><span class="p">(</span><span class="mi">4</span><span class="p">);</span><span class="n">h</span><span class="p">(</span><span class="n">n</span><span class="p">,</span><span class="o">-</span><span class="n">a</span><span class="p">);</span><span class="n">l</span><span class="p">(</span><span class="n">a</span><span class="o">*</span><span class="mi">3</span><span class="p">)</span> <span class="lineno">6</span> <span class="n">h</span><span class="p">(</span><span class="mi">6</span><span class="p">,</span><span class="mi">90</span><span class="p">)</span> </pre></div> <p>Además no conocía el módulo de la tortuga</p> Wed, 14 Mar 2012 15:33:00 GMT tag:blog.willyfrog.es,2012-03-14:/2012/curva-de-hilbert-en-un-tuit Snipet: Generador de etiquetas http://blog.willyfrog.es/2012/snipet-generador-de-etiquetas/ <p>Para poder generar automaticamente etiquetas para nodos (aunque valdria para cualquier otra cosa) he hecho un generador automatico sencillo que va dando strings de texto:</p> <div class="highlight"><pre><span class="k">def</span> <span class="nf">generador_etiquetas</span><span class="p">(</span><span class="bp">self</span><span class="p">):</span> <span class="sd">&#39;&#39;&#39;genera etiquetas,primero minusculas, luego mayusculas y finalmente</span> <span class="sd"> digitos&#39;&#39;&#39;</span> <span class="k">for</span> <span class="n">a</span> <span class="ow">in</span> <span class="n">string</span><span class="o">.</span><span class="n">lowercase</span><span class="p">:</span> <span class="k">yield</span> <span class="n">a</span> <span class="k">for</span> <span class="n">A</span> <span class="ow">in</span> <span class="n">string</span><span class="o">.</span><span class="n">uppercase</span><span class="p">:</span> <span class="c">#cuando termina con las minusculas, pasa a las</span> <span class="c"># mayusculas</span> <span class="k">yield</span> <span class="n">A</span> <span class="n">n</span> <span class="o">=</span> <span class="mi">0</span> <span class="k">while</span> <span class="mi">1</span><span class="p">:</span> <span class="n">n</span> <span class="o">+=</span> <span class="mi">1</span> <span class="k">yield</span> <span class="nb">str</span><span class="p">(</span><span class="n">n</span><span class="p">)</span> </pre></div> <p>El while 1 en vez de while True, es porque al parecer es ligeramente más rapido (imagino que por alguna optimización del compilador del interprete)</p> Thu, 01 Mar 2012 21:49:00 GMT tag:blog.willyfrog.es,2012-03-01:/2012/snipet-generador-de-etiquetas Si un paper fuera un lenguaje de programacion http://blog.willyfrog.es/2012/si-un-paper-fuera-un-lenguaje-de-programacion/ <p>Quizás por verme cercano al momento PFC he encontrado este cómic bastante gracioso: <img alt="if term papers were programming languages" src="http://www.somethingofthatilk.com/comics/135.jpg"/> Una traducción rápida después del salto Fuente: <a class="reference external" href="http://www.somethingofthatilk.com/index.php?id=135">Something of that ilk</a></p> <p><strong>Python</strong>: Esto es plagio. No puedes hacer &quot;import ensayo&quot; y quedarte tan tranquilo.</p> <p><strong>Java</strong>: Llevo leídas dos páginas y aun no tengo ni idea de lo que estás hablando.</p> <p><strong>Ensamblador</strong>: ¿De verdad tenías que redefinir cada palabra del español?</p> <p><strong>C</strong>: Esta muy bien, pero se te olvidó añadir la terminación en null y ahora solo estoy leyendo basura.</p> <p><strong>C++</strong>: Te pedí una copia, no cuatrocientas.</p> <p><strong>Shell</strong>: No tengo permiso para leer esto</p> <p><strong>Latex</strong>: Tu trabajo no tienen ningún sentido, pero es la cosa más bonita que he visto.</p> <p><strong>HTML</strong>: Esto es una maceta.</p> Tue, 14 Feb 2012 08:55:00 GMT tag:blog.willyfrog.es,2012-02-14:/2012/si-un-paper-fuera-un-lenguaje-de-programacion ¿Censura o hactivismo? http://blog.willyfrog.es/2012/censura-o-hactivismo/ <p>Hoy, dia ocupado como pocos en el trabajo (pero de eso igual hago otra entrada), me sorprendian un monton de tuits hablando de censura, cosa que no entendia y me comenzaba a temer una nueva vuelta de tuerca (a mucho peor) de la conocida &quot;Ley de Sinde&quot;. Pero no, era el anuncio por parte de <a class="reference external" href="http://twitter.com">Twitter</a> acerca de su <a class="reference external" href="http://www.genbeta.com/redes-sociales/twitter-censurara-tweets-de-acuerdo-a-la-legislacion-de-cada-pais">nueva politica</a>. En resumidas cuentas viene a decir que en base a legislación y bajo peticion expresa de una entidad competente (entiendo que es alguien designado por el gobierno) no serán visibles los tuits que incumplan algun tipo de ley (y en ocasiones cuestiones morales/culturales).</p> <p>Esto, me ha hecho pensar y mucho en lo que estaba poniendo la gente, sobre como se acababa el mundo, que si a mi no me censura nadie, que menuda mierda de red, etc. Y cada vez lo entendía menos. Quizás lo que mas me ha hecho pensar es la opinion generalizada de que twitter se habia acabado como herramienta de coordinación de movimientos activistas, especialmente explicada en los medios más tradicionales (periodicos y webs de noticias que necesitan tiempo antes de dar una noticia... hum, competencia de twitter, diria)</p> <p>Pero lo cierto es que yo lo estoy viendo de otra manera, como una herramienta más potente de cara al hacktivismo: ahora un chino podría escribir tranquilamente en formato de 140 caracteres sin el temor a que sus vecinos o amigos le denuncien a un regimen, porque su tuit será borrado de su alrededor y sin embargo será accesible al resto del mundo, poniendo de relieve situaciones que se conocen vagamente por el miedo a las consecuencias. Y donde digo un chino, digo un cubano o tantos otros bajo politicas donde la libertad de expresión no es un derecho.</p> <p>Y sigo pensando que seguirá siendo una herramienta magnifica de organización, porque para bloquear un &quot;trending topic&quot; primero habrá tenido que serlo (me acojonaré si empezamos con los precons de minority report, pero mientras tanto...) y una vez ha sido TT las formas de organización ya no pasan por tuits sino por multiples medios (IM, noticias, webs, foros, de palabra, etc.)</p> <p>Mientras no haya un chivatazo por parte de Twitter a las autoridades de turno, creo que aplaudiré este paso y su conversión a una mejor herramienta de hacktivismo.</p> <p>Y de paso mencionar otro <a class="reference external" href="http://gallir.wordpress.com/2012/01/27/la-masa-enfurecida-ahora-contra-twitter/">post interesante</a> de Ricardo Galli (creador de meneame) que ademas me ha puesto en marcha para escribir este.</p> Fri, 27 Jan 2012 17:54:00 GMT tag:blog.willyfrog.es,2012-01-27:/2012/censura-o-hactivismo Acortador de urls terminado http://blog.willyfrog.es/2012/acortador-de-urls-terminado/ <p>He terminado (alcanzado el estado de 1.0) mi pequeño proyecto de <a class="reference external" href="https://bitbucket.org/Willyfrog/misurl/">acortador de urls</a>, no pretende ser útil en el mundo real, pero si quizás sirva como punto de partida para otros en caso de querer ver un ejemplo de flask+redis+sqlalchemy.</p> <p>La motivación principal, es una entrega para la facultad, pero he aprovechado para aprender un par de cosas y puede que acabe usándola para dar una charla introductoria en <a class="reference external" href="http://codemotion.es/">codemotion</a>.</p> <p>Está bajo licencia <a class="reference external" href="http://en.wikipedia.org/wiki/BSD_licenses#3-clause_license_.28.22New_BSD_License.22_or_.22Modified_BSD_License.22.29">BSD (3 clausulas)</a>, por lo que podeis hacer casi cualquier cosa con ella ;)</p> <p>Una última cosa, está pensada para trabajar con postgresql, pero cambiando la configuración de la bb.dd. debería poder adaptarse a otras bases de datos SQL</p> Thu, 26 Jan 2012 14:15:00 GMT tag:blog.willyfrog.es,2012-01-26:/2012/acortador-de-urls-terminado Recomendacion de libro para aprender haskell http://blog.willyfrog.es/2012/recomendacion-de-libro-para-aprender-haskell/ <p>Si estás necesitado de aprender Haskell a marchas forzadas este fin de semana me ayudó mucho el libro: <a class="reference external" href="http://learnyouahaskell.com">learn you a haskell for greater good</a>. Y si lo prefieres en castellano está traducido en <a class="reference external" href="http://aprendehaskell.es/">Aprende Haskell</a> (si pongo los dos es porque al parecer la traducción es muy reciente y esta en proceso de revisión). Bastante conciso y la mayoría de los conceptos (si no eres nuevo completamente en programación funcional) se cogen rápido con la excepción quizás de los functores y las mónadas.</p> <p>Realmente útil para iniciarse en ese complejo mundo que es Haskell. Seguramente acabe cogiendome la versión en papel de amazon</p> Mon, 23 Jan 2012 07:34:00 GMT tag:blog.willyfrog.es,2012-01-23:/2012/recomendacion-de-libro-para-aprender-haskell Calculo de numeros primos http://blog.willyfrog.es/2012/calculo-de-numeros-primos/ <p>En un problema que resolvi recientemente en el proyecto euler, descubrí un algoritmo sencillo para calcular numeros primos: la <a class="reference external" href="http://es.wikipedia.org/wiki/Criba_de_Erat%C3%B3stenes">criba de eratóstenes</a> El cual inicialmente traduje a python como:</p> <div class="highlight"><pre><span class="lineno">1</span> <span class="k">def</span> <span class="nf">eratostenes</span><span class="p">(</span><span class="n">m</span><span class="p">):</span> <span class="lineno">2</span> <span class="n">primos</span> <span class="o">=</span> <span class="nb">set</span><span class="p">(</span><span class="nb">range</span><span class="p">(</span><span class="mi">2</span><span class="p">,</span><span class="n">m</span><span class="o">+</span><span class="mi">1</span><span class="p">))</span> <span class="lineno">3</span> <span class="k">for</span> <span class="n">i</span> <span class="ow">in</span> <span class="nb">xrange</span><span class="p">(</span><span class="mi">2</span><span class="p">,</span><span class="nb">int</span><span class="p">(</span><span class="n">sqrt</span><span class="p">(</span><span class="n">m</span><span class="p">))</span><span class="o">+</span><span class="mi">1</span><span class="p">):</span> <span class="lineno">4</span> <span class="k">if</span> <span class="n">i</span> <span class="ow">in</span> <span class="n">primos</span><span class="p">:</span> <span class="lineno">5</span> <span class="k">for</span> <span class="n">j</span> <span class="ow">in</span> <span class="nb">xrange</span><span class="p">(</span><span class="mi">2</span><span class="p">,</span><span class="n">m</span><span class="o">/</span><span class="n">i</span><span class="o">+</span><span class="mi">1</span><span class="p">):</span> <span class="lineno">6</span> <span class="n">primos</span><span class="o">.</span><span class="n">discard</span><span class="p">(</span><span class="n">i</span><span class="o">*</span><span class="n">j</span><span class="p">)</span> <span class="lineno">7</span> <span class="lineno">8</span> <span class="k">return</span> <span class="n">primos</span> </pre></div> <p>No comento el código por ser una transcripción directa del algoritmo enlazado, asi que si no entiendes como funciona, mejor leer el articulo de la <a class="reference external" href="http://es.wikipedia.org/wiki/Criba_de_Erat%C3%B3stenes">wikipedia</a>.</p> <p>El problema de esta solución es que para números grandes, el conjunto inicial es muy grande y se crea nada mas empezar, por lo que en caso de no tener memoria suficiente pasará lo que tiene que pasar cuando el 99% de las aplicaciones están en la <a class="reference external" href="http://es.wikipedia.org/wiki/Espacio_de_intercambio">partición de swap</a>.</p> <p>Por ello, cree una modificación del algoritmo de eratóstenes, para continuar un set de primos dado:</p> <div class="highlight"><pre><span class="k">def</span> <span class="nf">erato_follow</span><span class="p">(</span><span class="n">m</span><span class="p">,</span><span class="n">primes</span><span class="p">):</span> <span class="sd">&#39;&#39;&#39;</span> <span class="sd"> m - nuevo numero tope</span> <span class="sd"> primes - antiguo set de numeros primos</span> <span class="sd"> &#39;&#39;&#39;</span> <span class="k">if</span> <span class="ow">not</span> <span class="n">primes</span><span class="p">:</span> <span class="c">#si se ha llamado sin pasar un set antiguo</span> <span class="c"># se trabaja como siempre</span> <span class="n">pl</span> <span class="o">=</span> <span class="n">eratostenes</span><span class="p">(</span><span class="n">m</span><span class="p">)</span> <span class="k">else</span><span class="p">:</span> <span class="n">pl</span> <span class="o">=</span> <span class="n">primes</span> <span class="n">np</span> <span class="o">=</span> <span class="nb">set</span><span class="p">(</span><span class="nb">range</span><span class="p">(</span><span class="nb">max</span><span class="p">(</span><span class="n">pl</span><span class="p">),</span><span class="n">m</span><span class="p">))</span> <span class="c">#nuevos primos</span> <span class="k">for</span> <span class="n">i</span> <span class="ow">in</span> <span class="nb">xrange</span><span class="p">(</span><span class="mi">2</span><span class="p">,</span><span class="nb">int</span><span class="p">(</span><span class="n">sqrt</span><span class="p">(</span><span class="n">m</span><span class="p">))</span><span class="o">+</span><span class="mi">1</span><span class="p">):</span> <span class="k">if</span> <span class="n">i</span> <span class="o">&lt;</span> <span class="nb">max</span><span class="p">(</span><span class="n">pl</span><span class="p">):</span> <span class="c">#aprovechamos lo aprendido</span> <span class="k">for</span> <span class="n">j</span> <span class="ow">in</span> <span class="nb">xrange</span><span class="p">(</span><span class="mi">2</span><span class="p">,</span><span class="n">m</span><span class="o">/</span><span class="n">i</span><span class="o">+</span><span class="mi">1</span><span class="p">):</span> <span class="n">np</span><span class="o">.</span><span class="n">discard</span><span class="p">(</span><span class="n">i</span><span class="o">*</span><span class="n">j</span><span class="p">)</span> <span class="k">else</span><span class="p">:</span> <span class="c">#sacamos los que quedan</span> <span class="k">for</span> <span class="n">j</span> <span class="ow">in</span> <span class="nb">xrange</span><span class="p">(</span><span class="mi">2</span><span class="p">,</span><span class="n">m</span><span class="o">/</span><span class="n">i</span><span class="o">+</span><span class="mi">1</span><span class="p">):</span> <span class="n">np</span><span class="o">.</span><span class="n">discard</span><span class="p">(</span><span class="n">i</span><span class="o">*</span><span class="n">j</span><span class="p">)</span> <span class="k">return</span> <span class="n">pl</span> <span class="o">|</span> <span class="n">np</span> <span class="c"># union de ambos</span> </pre></div> <p>mediante esta funcion, podemos seguir aumentando el numero de primos en incrementos manejables, si bien habrá que tener cuidado con el set completo ya que al final acabará creciendo por encima de la memoria disponible.</p> Tue, 10 Jan 2012 07:55:00 GMT tag:blog.willyfrog.es,2012-01-10:/2012/calculo-de-numeros-primos El arbol de navidad de QBert http://blog.willyfrog.es/2011/el-arbol-de-navidad-de-qbert/ <p>En el <a class="reference external" href="http://www.pythonweekly.com/">weekly python newsletter</a>, enlazaban al problema de sacar la suma máxima, dado un árbol y utilizando el movimiento del clásico videojuego <a class="reference external" href="http://en.wikipedia.org/wiki/Qbert">qBert</a> (no del todo cierto porque qBert sí podía subir, pero eso son tecnicismos). El árbol en concreto sería así:</p> <div class="highlight"><pre><span class="n">chrismasTree</span> <span class="o">=</span> <span class="p">[</span> <span class="p">[</span><span class="mi">75</span><span class="p">],</span> <span class="p">[</span><span class="mi">95</span><span class="p">,</span><span class="mi">64</span><span class="p">],</span> <span class="p">[</span><span class="mi">17</span><span class="p">,</span><span class="mi">47</span><span class="p">,</span><span class="mi">82</span><span class="p">],</span> <span class="p">[</span><span class="mi">18</span><span class="p">,</span><span class="mi">35</span><span class="p">,</span><span class="mi">87</span><span class="p">,</span><span class="mi">10</span><span class="p">],</span> <span class="p">[</span><span class="mi">20</span><span class="p">,</span> <span class="mi">4</span><span class="p">,</span><span class="mi">82</span><span class="p">,</span><span class="mi">47</span><span class="p">,</span><span class="mi">65</span><span class="p">],</span> <span class="p">[</span><span class="mi">19</span><span class="p">,</span> <span class="mi">1</span><span class="p">,</span><span class="mi">23</span><span class="p">,</span><span class="mi">75</span><span class="p">,</span> <span class="mi">3</span><span class="p">,</span><span class="mi">34</span><span class="p">]</span> <span class="p">]</span> </pre></div> <p>Y las reglas son que qBert siempre debe bajar por el árbol, bien hacia un lado o el otro.</p> <p>Dado el pequeño número de datos, se puede hacer una aproximación por fuerza bruta (recorriendo todos los posibles caminos y comparándolos) pero esta solución sería malísima si cambiamos el set de datos al proporcionado por el <a class="reference external" href="http://projecteuler.net">proyecto euler</a> en su <a class="reference external" href="http://projecteuler.net/problem=67">problema 67</a>, que aseguran que si pudieras comprobar 10⁷ rutas por segundo, tadarías 20 billones americanos (millardos por aquí) de años en resolverlo.</p> <p>El código para generar este segundo árbol, sería:</p> <div class="highlight"><pre><span class="n">tri</span> <span class="o">=</span> <span class="p">[]</span> <span class="n">peUrl</span> <span class="o">=</span> <span class="s">&quot;http://projecteuler.net/project/triangle.txt&quot;</span> <span class="k">for</span> <span class="n">line</span> <span class="ow">in</span> <span class="n">urllib</span><span class="o">.</span><span class="n">urlopen</span><span class="p">(</span><span class="n">peUrl</span><span class="p">)</span><span class="o">.</span><span class="n">readlines</span><span class="p">():</span> <span class="n">tri</span><span class="o">.</span><span class="n">append</span><span class="p">([</span><span class="nb">int</span><span class="p">(</span><span class="n">x</span><span class="p">)</span> <span class="k">for</span> <span class="n">x</span> <span class="ow">in</span> <span class="n">line</span><span class="o">.</span><span class="n">strip</span><span class="p">()</span><span class="o">.</span><span class="n">split</span><span class="p">(</span><span class="s">&quot; &quot;</span><span class="p">)])</span> <span class="k">print</span> <span class="n">qbertRun</span><span class="p">(</span><span class="n">tri</span><span class="p">)</span> </pre></div> <p>Y por si quieres intentar resolverlo, no pondré mi solución hasta después de &quot;Leer mas&quot;.</p> <p><strong>Solucion:</strong> La parte crítica de la solución está en la siguiente función:</p> <div class="highlight"><pre><span class="k">def</span> <span class="nf">solve_line</span><span class="p">(</span><span class="n">curr</span><span class="p">,</span> <span class="n">prev</span><span class="p">):</span> <span class="sd">&quot;&quot;&quot;</span> <span class="sd"> dadas 2 lineas, hacemos una nueva con cada elemento</span> <span class="sd"> max(curr[i]+prev[i],curr[i]+prev[i+1])</span> <span class="sd"> &quot;&quot;&quot;</span> <span class="k">if</span> <span class="ow">not</span> <span class="n">prev</span><span class="p">:</span> <span class="n">res</span><span class="o">=</span><span class="n">curr</span> <span class="k">else</span><span class="p">:</span> <span class="n">res</span><span class="o">=</span><span class="p">[</span><span class="nb">max</span><span class="p">(</span><span class="n">prev</span><span class="p">[</span><span class="n">i</span><span class="p">],</span><span class="n">prev</span><span class="p">[</span><span class="n">i</span><span class="o">+</span><span class="mi">1</span><span class="p">])</span><span class="o">+</span><span class="n">curr</span><span class="p">[</span><span class="n">i</span><span class="p">]</span> <span class="k">for</span> <span class="n">i</span> <span class="ow">in</span> <span class="nb">range</span><span class="p">(</span><span class="nb">len</span><span class="p">(</span><span class="n">curr</span><span class="p">))]</span> <span class="k">return</span> <span class="n">res</span> </pre></div> <p>Es decir, tomamos el árbol de navidad como un <a class="reference external" href="http://es.wikipedia.org/wiki/%C3%81rbol_binario">árbol binario</a> y sumamos a cada línea el máximo de los dos caminos posibles.</p> <p>Esto en realidad, nos daría un resultado incorrecto si lo aplicaramos de arriba a abajo, ya que se iria por el primer resultado gordo que hubiera, pero si invertimos el arbol y empezamos por el lado ancho, lo que hace es recortar los caminos, dejando cada vez el camino mayor. Por ello, la funcion que tratará los datos, lo primero que hace es invertir la lista:</p> <div class="highlight"><pre><span class="k">def</span> <span class="nf">qbert</span><span class="p">(</span><span class="n">arbol</span><span class="p">):</span> <span class="sd">&quot;&quot;&quot;</span> <span class="sd"> dada una lista en forma de arbol (ver variable tree) resolver</span> <span class="sd"> la suma maxima pudiendo solo bajar bien a izda. o dcha.</span> <span class="sd"> &quot;&quot;&quot;</span> <span class="n">arbol</span><span class="o">.</span><span class="n">reverse</span><span class="p">()</span> <span class="c">#invertimos la lista</span> <span class="n">res</span> <span class="o">=</span> <span class="p">[]</span> <span class="k">for</span> <span class="n">l</span> <span class="ow">in</span> <span class="n">arbol</span><span class="p">:</span> <span class="n">res</span> <span class="o">=</span> <span class="n">solve_line</span><span class="p">(</span><span class="n">l</span><span class="p">,</span><span class="n">res</span><span class="p">)</span> <span class="k">return</span> <span class="n">res</span> </pre></div> Fri, 23 Dec 2011 09:43:00 GMT tag:blog.willyfrog.es,2011-12-23:/2011/el-arbol-de-navidad-de-qbert Salpimentar al gusto la contraseña http://blog.willyfrog.es/2011/salpimentar-al-gusto-la-contrasena/ <p>Actualmente estoy trabajando en un prototipo de aplicación bastante sencillo, pero que requiere que aprenda bastantes cosas (y por lo tanto voy algo lento). Uno de los pequeños problemas con los que me enfrenté fue la creación de un sistema de usuarios y no quería dejar el <em>password</em> en abierto en la base de datos aun cuando solo fuera a utilizarlo yo.</p> <p>Echando un ojo al módulo <a class="reference external" href="http://docs.python.org/library/crypt.html">crypt</a> de <em>Python</em>, requería un poco de <a class="reference external" href="http://en.wikipedia.org/wiki/Salt_%28cryptography%29">sal</a>, por lo que me vi de repente con que no quería meter la sal ni en un fichero de configuración ni obviamente en el código fuente. Por lo que finalmente decidí que la sal iba a ser parte del login y de ahí sacar todo. Así para comprobar la clave únicamente hago:</p> <div class="highlight"><pre><span class="kn">from</span> <span class="nn">crypt</span> <span class="kn">import</span> <span class="n">crypt</span> <span class="k">def</span> <span class="nf">auth</span><span class="p">(</span><span class="bp">self</span><span class="p">,</span> <span class="n">clave</span><span class="p">):</span> <span class="sd">&#39;&#39;&#39;Comprueba si la clave suministrada es la del usuario&#39;&#39;&#39;</span> <span class="k">return</span> <span class="bp">self</span><span class="o">.</span><span class="n">password</span> <span class="o">==</span> <span class="n">crypt</span><span class="p">(</span><span class="n">clave</span><span class="p">,</span><span class="bp">self</span><span class="o">.</span><span class="n">username</span><span class="p">[:</span><span class="mi">2</span><span class="p">])</span> </pre></div> <p>El motivo de usar 2 caracteres es únicamente debido a los ejemplos que vi de uso de <em>crypt</em>, imagino que se podrían más caracteres, pero no me pareció necesario.</p> <p>Si acaso, mi mayor problema con esta implementación es que la sal se vuelve conocida con solo leer el código y no se si a partir de ahí y con la clave encriptada se podría sacar la clave en plano. Por ahora y para la demo, esto me sirve y más adelante estudiaré mejores sistemas de encriptación y almacenado.</p> <p>Mientras tanto, lo publico por si a alguien le viene bien.</p> <p>Ah, decir además que esto lo uso mezclado con el decorator que sugieren en un <a class="reference external" href="http://flask.pocoo.org/snippets/8/">snippet</a> de <a class="reference external" href="http://flask.pocoo.org">Flask</a>, pero obviamente cambiando la manera de acceso.</p> Fri, 09 Dec 2011 11:14:00 GMT tag:blog.willyfrog.es,2011-12-09:/2011/salpimentar-al-gusto-la-contrasena Peores momentos de programar http://blog.willyfrog.es/2011/peores-momentos-de-programar/ <p>Ultimamente he sufrido las dos cosas que mas me molestan al programar (provocadas por mi, claro está)</p> <ul class="simple"> <li>una es la búsqueda de un error en el sitio incorrecto. El log de error</li> </ul> <div class="system-message"> <p class="system-message-title">System Message: WARNING/2 (<tt class="docutils">&lt;string&gt;</tt>, line 5)</p> Bullet list ends without a blank line; unexpected unindent.</div> <p>no deja claro o puede llevar a error y te tiras un par de días buscando un error en el sitio equivocado.</p> <ul class="simple"> <li>en el proceso de aprendizaje de una librería, descubrir que algo lo</li> </ul> <div class="system-message"> <p class="system-message-title">System Message: WARNING/2 (<tt class="docutils">&lt;string&gt;</tt>, line 9)</p> Bullet list ends without a blank line; unexpected unindent.</div> <p>entendiste mal</p> <p>Estoy seguro de que hay más cosas que me molesten, pero esas dos suelen aparecer antes o después. Si me acuerdo ya pondré mas, porque por ahora parece una excusa promocional para stackoverflow :)</p> Mon, 28 Nov 2011 11:19:00 GMT tag:blog.willyfrog.es,2011-11-28:/2011/peores-momentos-de-programar Más recursos para leer de python http://blog.willyfrog.es/2011/mas-recursos-para-leer-de-python/ <p>Esta mañana he encontrado un <a class="reference external" href="http://jessenoller.com/good-to-great-python-reads/">enlace recopilatorio</a> de lecturas sobre python, cubriendo un buen montón de temas. Algunos de los mencionados ya los <a class="reference external" href="http://blog.willyfrog.es/2011/recursos-de-python">nombré hace poco</a>.</p> <p>Intentaré actualizar este post segun vaya leyendo algunos de esos artículos si merecen algún tipo de reseña especial.</p> Wed, 16 Nov 2011 07:23:00 GMT tag:blog.willyfrog.es,2011-11-16:/2011/mas-recursos-para-leer-de-python Música en una línea de codigo, versión Python http://blog.willyfrog.es/2011/musica-en-una-linea-de-codigo-version-python/ <p>El otro día vi en reddit <a class="reference external" href="http://www.reddit.com/r/programming/comments/mbakl/experimental_oneline_algorithmic_music/">algunos</a> <a class="reference external" href="http://xiatek.org/?p=296">temas</a> en los cuales se creaba música a partir de una línea de código (en realidad algo mas, pero la parte importante estaba solo en una) y como curiosidad personal, decidí hacer una versión serpentina de dicho programa, una vez conseguido (fácil, rápido y feo) quise poner un poco mas interés y utilizar funciones lambda y generadores. El resultado es el siguiente:</p> <div class="highlight"><pre><span class="c">#redirigir el output a /dev/audio o instalar sox y redirigir a</span> <span class="c"># &quot;play -u -b 8 -t raw -r 8000 -&quot; sin las comillas</span> <span class="c">#modificar la parte interna de la funcion int() para variar la &quot;melodia&quot;</span> <span class="n">func</span> <span class="o">=</span> <span class="k">lambda</span> <span class="n">t</span><span class="p">:</span> <span class="nb">int</span><span class="p">((</span><span class="n">t</span><span class="o">&lt;&lt;</span><span class="mi">6</span><span class="o">|</span><span class="n">t</span><span class="o">|</span><span class="p">(</span><span class="n">t</span><span class="o">&lt;&lt;</span><span class="mi">16</span><span class="p">))</span> <span class="o">*</span> <span class="mi">10</span> <span class="o">+</span> <span class="p">((</span><span class="n">t</span><span class="o">&lt;&lt;</span><span class="mi">11</span><span class="p">)</span> <span class="o">&amp;</span> <span class="mi">7</span><span class="p">))</span> <span class="c">#generador de secuencia</span> <span class="k">def</span> <span class="nf">inc</span><span class="p">(</span><span class="n">t</span><span class="p">):</span> <span class="sd">&#39;&#39;&#39;generador para incrementar t&#39;&#39;&#39;</span> <span class="n">c</span><span class="o">=</span><span class="n">t</span> <span class="k">while</span> <span class="bp">True</span><span class="p">:</span> <span class="k">yield</span> <span class="n">c</span> <span class="n">c</span><span class="o">=</span><span class="n">c</span><span class="o">+</span><span class="mi">1</span> <span class="k">if</span> <span class="n">__name__</span><span class="o">==</span><span class="s">&#39;__main__&#39;</span><span class="p">:</span> <span class="k">for</span> <span class="n">t</span> <span class="ow">in</span> <span class="n">inc</span><span class="p">(</span><span class="mi">0</span><span class="p">):</span> <span class="k">print</span><span class="p">(</span><span class="s">&quot;</span><span class="si">%X</span><span class="s">&quot;</span> <span class="o">%</span> <span class="n">func</span><span class="p">(</span><span class="n">t</span><span class="p">))</span> </pre></div> <p>Espero que alguien lo encuentre interesante y genere nuevas &quot;melodías&quot;. Quizás decir que el generador es lo que menos me ha gustado ya que queda poco &quot;pitónico&quot;, si bien el programa final queda bastante legible ya que añadí algo de codigo por variar la funcionalidad inicial y dar mas opciones con el fichero.</p> <p>Para ver el resultado final recomiendo <a class="reference external" href="http://www.youtube.com/watch?feature=player_detailpage&amp;v=qlrs2Vorw2Y#t=16s">el video</a> de los enlaces</p> Tue, 15 Nov 2011 15:25:00 GMT tag:blog.willyfrog.es,2011-11-15:/2011/musica-en-una-linea-de-codigo-version-python time, datetime, epoch y el lio de todos los tiempos http://blog.willyfrog.es/2011/time-datetime-epoch-y-el-lio-de-todos-los-tiempos/ <p>Estoy intentando hacer la transición del código de Python de una aplicación desde mx.DateTime a la librería estándar de Python datetime. Las razones son 2, por un lado quitarnos una dependencia que no hace falta realmente y por otra facilitar la transición a psycopg2 (estamos usando el 1 ahora mismo). La mayor parte de las cosas son fácilmente traducibles ya que en ocasiones usan la misma sintaxis o parecida.</p> <p>Si bien nadie debería jugar demasiado con <a class="reference external" href="http://en.wikipedia.org/wiki/Unix_epoch">epoch</a> dado que Python ofrece múltiples opciones para jugar con la fecha tanto en mx.DateTime como en datetime, parece que los tiempos de C no se olvidan y hay quien lo prefiere. Esto, me ha supuesto un problema ya que para traer una fecha de un timestamp a datetime es sencillo, pero la inversa no. En cualquier otro caso estaría estudiando un poco el código para adaptarlo y hacerlo sencillo, pero dado que tengo que cambiar algo más de 100 ficheros, me corre cierta prisa cambiar todo y que de primeras funcione igual, es decir que no puedo cambiar apenas el código mas que lo estrictamente necesario.</p> <p>Por si alguien más lo necesita, el código para pasar de datetime al número de segundos desde 'epoch' sería el siguiente:</p> <div class="highlight"><pre><span class="n">time</span><span class="o">.</span><span class="n">mktime</span><span class="p">(</span><span class="n">datetime</span><span class="o">.</span><span class="n">datetime</span><span class="o">.</span><span class="n">today</span><span class="p">()</span><span class="o">.</span><span class="n">timetuple</span><span class="p">())</span> </pre></div> <p>datetime.datetime.today() puede sustituirse por cualquier otra fecha en formato datetime, además se pierde la precisión del float debido al uso de timetuple, por lo que si los microsegundos son importantes, esta no es una forma válida, aunque posiblemente sería parcheable.</p> <p><a class="reference external" href="http://bytes.com/topic/python/answers/522572-datetime-timestamp">Fuente</a></p> Mon, 31 Oct 2011 11:17:00 GMT tag:blog.willyfrog.es,2011-10-31:/2011/time-datetime-epoch-y-el-lio-de-todos-los-tiempos