<?xml version="1.0" encoding="UTF-8"?>
<rss version="2.0" xmlns:atom="http://www.w3.org/2005/Atom">
  <channel>
    <title>paradoxxx zero</title>
    <description>Random interesting things</description>
    <link>http://paradoxxxzero.github.io</link>
    <atom:link href="http://paradoxxxzero.github.io/feed.xml" rel="self" type="application/rss+xml" />
    
      <item>
        <title>wdb 3.0 material style</title>
        <description>&lt;p&gt;The future is here! wdb 3.0 is out get it while it’s hot (and certainly bugged)&lt;/p&gt;

&lt;p&gt;&lt;img src=&quot;https://camo.githubusercontent.com/4cd1a4503682803966e706dec1de84b0d178da29/68747470733a2f2f7261772e6769746875622e636f6d2f4b6f7a65612f7764622f6d61737465722f7764622e706e67&quot; alt=&quot;Material wdb&quot; /&gt;&lt;/p&gt;

&lt;h2 id=&quot;changes&quot;&gt;Changes&lt;/h2&gt;

&lt;ul&gt;
  &lt;li&gt;A whole new material design lite interface (a bit responsive)&lt;/li&gt;
  &lt;li&gt;Clickable icons instead of hotkeys only&lt;/li&gt;
  &lt;li&gt;Visual distinction between stepping, post_mortem and shell&lt;/li&gt;
  &lt;li&gt;An actually readable help (you can get by clicking on the help icons)&lt;/li&gt;
  &lt;li&gt;A code mirror prompt with syntaxic coloration and more classic completion&lt;/li&gt;
  &lt;li&gt;New commands (open current file in external editor, respawn current process…)&lt;/li&gt;
  &lt;li&gt;A far better post mortem interaction when using ext and wdb is disabled.&lt;/li&gt;
  &lt;li&gt;&lt;code class=&quot;highlighter-rouge&quot;&gt;importmagic&lt;/code&gt; suggestions on NameError&lt;/li&gt;
  &lt;li&gt;Use of &lt;code class=&quot;highlighter-rouge&quot;&gt;uncompyle6&lt;/code&gt; to get python source when only byte-code is available&lt;/li&gt;
&lt;/ul&gt;

&lt;p&gt;For more information about wdb, see &lt;a href=&quot;/2014/05/12/wdb-2.html&quot;&gt;previous post&lt;/a&gt;&lt;/p&gt;
</description>
        <pubDate>Wed, 08 Jun 2016 00:00:00 +0000</pubDate>
        <link>http://paradoxxxzero.github.io/2016/06/08/wdb-3.html</link>
        <guid isPermaLink="true">http://paradoxxxzero.github.io/2016/06/08/wdb-3.html</guid>
      </item>
    
      <item>
        <title>Announcing butterfly 2.0</title>
        <description>&lt;p&gt;After 1 year and a half of waiting butterfly 2.0 is finally here!&lt;/p&gt;

&lt;p&gt;&lt;img src=&quot;/assets/butterfly_2.0_1.gif&quot; alt=&quot;butterfly screen cast&quot; /&gt;&lt;/p&gt;

&lt;p&gt;For those who don’t know butterfly is an open source in-browser nix terminal written in coffeescript (yeah I know, maybe I will port it to ES7 but you know… braces) and backed up by a python tornado server.&lt;/p&gt;

&lt;h2 id=&quot;features&quot;&gt;Features&lt;/h2&gt;

&lt;p&gt;Its feature list keep on growing:&lt;/p&gt;

&lt;ul&gt;
  &lt;li&gt;xterm compatible (support a lot of unused features!)&lt;/li&gt;
  &lt;li&gt;Native browser scroll and search&lt;/li&gt;
  &lt;li&gt;Theming in css / sass &lt;a href=&quot;https://github.com/paradoxxxzero/butterfly-themes&quot;&gt;(18 preset themes)&lt;/a&gt; endless possibilities!&lt;/li&gt;
  &lt;li&gt;HTML in your terminal! cat images and use &amp;lt;table&amp;gt;&lt;/li&gt;
  &lt;li&gt;Multiple sessions support (à la screen -x) to simultaneously access a terminal from several places on the planet!&lt;/li&gt;
  &lt;li&gt;Secure authentication with X509 certificates!&lt;/li&gt;
  &lt;li&gt;16,777,216 colors support!&lt;/li&gt;
  &lt;li&gt;Keyboard text selection!&lt;/li&gt;
  &lt;li&gt;Desktop notifications on terminal output!&lt;/li&gt;
  &lt;li&gt;Geolocation from browser!&lt;/li&gt;
  &lt;li&gt;May work on firefox too!&lt;/li&gt;
&lt;/ul&gt;

&lt;h2 id=&quot;give-it-a-shot&quot;&gt;Give it a shot&lt;/h2&gt;

&lt;p&gt;It’s even easier to try it now, just:&lt;/p&gt;

&lt;div class=&quot;language-bash highlighter-rouge&quot;&gt;&lt;div class=&quot;highlight&quot;&gt;&lt;pre class=&quot;highlight&quot;&gt;&lt;code&gt;    &lt;span class=&quot;nv&quot;&gt;$ &lt;/span&gt;pip install butterfly
    &lt;span class=&quot;nv&quot;&gt;$ &lt;/span&gt;pip install libsass  &lt;span class=&quot;c&quot;&gt;# If you want to use themes&lt;/span&gt;
    &lt;span class=&quot;nv&quot;&gt;$ &lt;/span&gt;butterfly
&lt;/code&gt;&lt;/pre&gt;&lt;/div&gt;&lt;/div&gt;

&lt;p&gt;A new tab should appear in your browser. Then type&lt;/p&gt;

&lt;div class=&quot;language-bash highlighter-rouge&quot;&gt;&lt;div class=&quot;highlight&quot;&gt;&lt;pre class=&quot;highlight&quot;&gt;&lt;code&gt;    &lt;span class=&quot;nv&quot;&gt;$ &lt;/span&gt;butterfly &lt;span class=&quot;nb&quot;&gt;help&lt;/span&gt;
&lt;/code&gt;&lt;/pre&gt;&lt;/div&gt;&lt;/div&gt;

&lt;p&gt;To get an overview of butterfly features.&lt;/p&gt;

&lt;p&gt;For everyday use, it’s better to use a server see below.&lt;/p&gt;

&lt;h2 id=&quot;moar&quot;&gt;Moar&lt;/h2&gt;

&lt;p&gt;Go to the &lt;a href=&quot;https://github.com/paradoxxxzero/butterfly&quot;&gt;github page&lt;/a&gt; for more informations, like setting up a server using systemd (I know I know) or why not contributing!&lt;/p&gt;

&lt;p&gt;Take a look at existing themes at &lt;a href=&quot;https://github.com/paradoxxxzero/butterfly-themes&quot;&gt;butterfly-themes&lt;/a&gt; and current demos showing the butterfly potential at &lt;a href=&quot;https://github.com/paradoxxxzero/butterfly-demos&quot;&gt;butterfly-demos&lt;/a&gt;.&lt;/p&gt;

&lt;h3 id=&quot;end&quot;&gt;End&lt;/h3&gt;

&lt;p&gt;We are in 2015, we can finally display images and html in a terminal and use 16M colors, what a world we are living in.&lt;/p&gt;
</description>
        <pubDate>Fri, 30 Oct 2015 00:00:00 +0000</pubDate>
        <link>http://paradoxxxzero.github.io/2015/10/30/butterfly_2.0.html</link>
        <guid isPermaLink="true">http://paradoxxxzero.github.io/2015/10/30/butterfly_2.0.html</guid>
      </item>
    
      <item>
        <title>Cutter</title>
        <description>&lt;p&gt;Today I finally wrote a readme for my little python tool &lt;a href=&quot;https://github.com/paradoxxxzero/cutter/&quot;&gt;cutter&lt;/a&gt;.&lt;/p&gt;

&lt;p&gt;This is something that I use daily in conjuction of &lt;a href=&quot;https://github.com/Kozea/wdb&quot;&gt;wdb&lt;/a&gt; for debugging python code.&lt;/p&gt;

&lt;p&gt;Here’s the readme, I hope you will find it as useful as I am:&lt;/p&gt;

&lt;h1 id=&quot;readme&quot;&gt;Readme&lt;/h1&gt;

&lt;p&gt;Cutter is a small python module that add some sugar on top of list like objects to ease list traversal.&lt;/p&gt;

&lt;h2 id=&quot;usage&quot;&gt;Usage&lt;/h2&gt;

&lt;div class=&quot;language-python highlighter-rouge&quot;&gt;&lt;div class=&quot;highlight&quot;&gt;&lt;pre class=&quot;highlight&quot;&gt;&lt;code&gt;&lt;span class=&quot;kn&quot;&gt;from&lt;/span&gt; &lt;span class=&quot;nn&quot;&gt;cutter&lt;/span&gt; &lt;span class=&quot;kn&quot;&gt;import&lt;/span&gt; &lt;span class=&quot;n&quot;&gt;cut&lt;/span&gt;

&lt;span class=&quot;n&quot;&gt;numbers&lt;/span&gt; &lt;span class=&quot;o&quot;&gt;=&lt;/span&gt; &lt;span class=&quot;p&quot;&gt;[&lt;/span&gt;
  &lt;span class=&quot;p&quot;&gt;[&lt;/span&gt;&lt;span class=&quot;s&quot;&gt;'one'&lt;/span&gt;&lt;span class=&quot;p&quot;&gt;,&lt;/span&gt; &lt;span class=&quot;s&quot;&gt;'two'&lt;/span&gt;&lt;span class=&quot;p&quot;&gt;,&lt;/span&gt; &lt;span class=&quot;s&quot;&gt;'three'&lt;/span&gt;&lt;span class=&quot;p&quot;&gt;],&lt;/span&gt;
  &lt;span class=&quot;p&quot;&gt;[&lt;/span&gt;&lt;span class=&quot;s&quot;&gt;'un'&lt;/span&gt;&lt;span class=&quot;p&quot;&gt;,&lt;/span&gt; &lt;span class=&quot;s&quot;&gt;'deux'&lt;/span&gt;&lt;span class=&quot;p&quot;&gt;,&lt;/span&gt; &lt;span class=&quot;s&quot;&gt;'trois'&lt;/span&gt;&lt;span class=&quot;p&quot;&gt;],&lt;/span&gt;
  &lt;span class=&quot;p&quot;&gt;[&lt;/span&gt;&lt;span class=&quot;s&quot;&gt;'uno'&lt;/span&gt;&lt;span class=&quot;p&quot;&gt;,&lt;/span&gt; &lt;span class=&quot;s&quot;&gt;'due'&lt;/span&gt;&lt;span class=&quot;p&quot;&gt;,&lt;/span&gt; &lt;span class=&quot;s&quot;&gt;'tre'&lt;/span&gt;&lt;span class=&quot;p&quot;&gt;]&lt;/span&gt;
&lt;span class=&quot;p&quot;&gt;]&lt;/span&gt;

&lt;span class=&quot;o&quot;&gt;&amp;gt;&amp;gt;&amp;gt;&lt;/span&gt; &lt;span class=&quot;n&quot;&gt;cut&lt;/span&gt;&lt;span class=&quot;p&quot;&gt;(&lt;/span&gt;&lt;span class=&quot;n&quot;&gt;numbers&lt;/span&gt;&lt;span class=&quot;p&quot;&gt;)[&lt;/span&gt;&lt;span class=&quot;mi&quot;&gt;1&lt;/span&gt;&lt;span class=&quot;p&quot;&gt;]&lt;/span&gt;  &lt;span class=&quot;c&quot;&gt;# Take the second element of every sublist&lt;/span&gt;
&lt;span class=&quot;p&quot;&gt;[&lt;/span&gt;&lt;span class=&quot;s&quot;&gt;'two'&lt;/span&gt;&lt;span class=&quot;p&quot;&gt;,&lt;/span&gt; &lt;span class=&quot;s&quot;&gt;'deux'&lt;/span&gt;&lt;span class=&quot;p&quot;&gt;,&lt;/span&gt; &lt;span class=&quot;s&quot;&gt;'due'&lt;/span&gt;&lt;span class=&quot;p&quot;&gt;]&lt;/span&gt;&lt;span class=&quot;o&quot;&gt;*&lt;/span&gt;

&lt;span class=&quot;o&quot;&gt;&amp;gt;&amp;gt;&amp;gt;&lt;/span&gt; &lt;span class=&quot;n&quot;&gt;cut&lt;/span&gt;&lt;span class=&quot;p&quot;&gt;(&lt;/span&gt;&lt;span class=&quot;n&quot;&gt;numbers&lt;/span&gt;&lt;span class=&quot;p&quot;&gt;)[&lt;/span&gt;&lt;span class=&quot;mi&quot;&gt;1&lt;/span&gt;&lt;span class=&quot;p&quot;&gt;,&lt;/span&gt; &lt;span class=&quot;mi&quot;&gt;2&lt;/span&gt;&lt;span class=&quot;p&quot;&gt;]&lt;/span&gt;  &lt;span class=&quot;c&quot;&gt;# Take the third letter of every second element of every sublist&lt;/span&gt;
&lt;span class=&quot;p&quot;&gt;[&lt;/span&gt;&lt;span class=&quot;s&quot;&gt;'o'&lt;/span&gt;&lt;span class=&quot;p&quot;&gt;,&lt;/span&gt; &lt;span class=&quot;s&quot;&gt;'u'&lt;/span&gt;&lt;span class=&quot;p&quot;&gt;,&lt;/span&gt; &lt;span class=&quot;s&quot;&gt;'e'&lt;/span&gt;&lt;span class=&quot;p&quot;&gt;]&lt;/span&gt;&lt;span class=&quot;o&quot;&gt;.&lt;/span&gt;

&lt;span class=&quot;o&quot;&gt;&amp;gt;&amp;gt;&amp;gt;&lt;/span&gt; &lt;span class=&quot;n&quot;&gt;cut&lt;/span&gt;&lt;span class=&quot;p&quot;&gt;(&lt;/span&gt;&lt;span class=&quot;n&quot;&gt;numbers&lt;/span&gt;&lt;span class=&quot;p&quot;&gt;)[&lt;/span&gt;&lt;span class=&quot;mi&quot;&gt;1&lt;/span&gt;&lt;span class=&quot;p&quot;&gt;][&lt;/span&gt;&lt;span class=&quot;mi&quot;&gt;2&lt;/span&gt;&lt;span class=&quot;p&quot;&gt;]&lt;/span&gt;  &lt;span class=&quot;c&quot;&gt;# Idem&lt;/span&gt;
&lt;span class=&quot;p&quot;&gt;[&lt;/span&gt;&lt;span class=&quot;s&quot;&gt;'o'&lt;/span&gt;&lt;span class=&quot;p&quot;&gt;,&lt;/span&gt; &lt;span class=&quot;s&quot;&gt;'u'&lt;/span&gt;&lt;span class=&quot;p&quot;&gt;,&lt;/span&gt; &lt;span class=&quot;s&quot;&gt;'e'&lt;/span&gt;&lt;span class=&quot;p&quot;&gt;]&lt;/span&gt;&lt;span class=&quot;o&quot;&gt;.&lt;/span&gt;

&lt;span class=&quot;o&quot;&gt;&amp;gt;&amp;gt;&amp;gt;&lt;/span&gt; &lt;span class=&quot;n&quot;&gt;cut&lt;/span&gt;&lt;span class=&quot;p&quot;&gt;(&lt;/span&gt;&lt;span class=&quot;n&quot;&gt;numbers&lt;/span&gt;&lt;span class=&quot;p&quot;&gt;)[&lt;/span&gt;&lt;span class=&quot;mi&quot;&gt;1&lt;/span&gt;&lt;span class=&quot;p&quot;&gt;,&lt;/span&gt; &lt;span class=&quot;mi&quot;&gt;2&lt;/span&gt;&lt;span class=&quot;p&quot;&gt;]&lt;/span&gt;&lt;span class=&quot;o&quot;&gt;.&lt;/span&gt;&lt;span class=&quot;n&quot;&gt;upper&lt;/span&gt;&lt;span class=&quot;p&quot;&gt;()&lt;/span&gt;  &lt;span class=&quot;c&quot;&gt;# Upper case the third letter of every second element of every sublist&lt;/span&gt;
&lt;span class=&quot;p&quot;&gt;[&lt;/span&gt;&lt;span class=&quot;s&quot;&gt;'O'&lt;/span&gt;&lt;span class=&quot;p&quot;&gt;,&lt;/span&gt; &lt;span class=&quot;s&quot;&gt;'U'&lt;/span&gt;&lt;span class=&quot;p&quot;&gt;,&lt;/span&gt; &lt;span class=&quot;s&quot;&gt;'E'&lt;/span&gt;&lt;span class=&quot;p&quot;&gt;]&lt;/span&gt;

&lt;span class=&quot;o&quot;&gt;&amp;gt;&amp;gt;&amp;gt;&lt;/span&gt; &lt;span class=&quot;n&quot;&gt;cut&lt;/span&gt;&lt;span class=&quot;p&quot;&gt;(&lt;/span&gt;&lt;span class=&quot;n&quot;&gt;numbers&lt;/span&gt;&lt;span class=&quot;p&quot;&gt;)[&lt;/span&gt;&lt;span class=&quot;o&quot;&gt;...&lt;/span&gt;&lt;span class=&quot;p&quot;&gt;]&lt;/span&gt;  &lt;span class=&quot;c&quot;&gt;# Flatten the list for one level&lt;/span&gt;
&lt;span class=&quot;p&quot;&gt;[&lt;/span&gt;&lt;span class=&quot;s&quot;&gt;'one'&lt;/span&gt;&lt;span class=&quot;p&quot;&gt;,&lt;/span&gt; &lt;span class=&quot;s&quot;&gt;'two'&lt;/span&gt;&lt;span class=&quot;p&quot;&gt;,&lt;/span&gt; &lt;span class=&quot;s&quot;&gt;'three'&lt;/span&gt;&lt;span class=&quot;p&quot;&gt;,&lt;/span&gt; &lt;span class=&quot;s&quot;&gt;'un'&lt;/span&gt;&lt;span class=&quot;p&quot;&gt;,&lt;/span&gt; &lt;span class=&quot;s&quot;&gt;'deux'&lt;/span&gt;&lt;span class=&quot;p&quot;&gt;,&lt;/span&gt; &lt;span class=&quot;s&quot;&gt;'trois'&lt;/span&gt;&lt;span class=&quot;p&quot;&gt;,&lt;/span&gt; &lt;span class=&quot;s&quot;&gt;'uno'&lt;/span&gt;&lt;span class=&quot;p&quot;&gt;,&lt;/span&gt; &lt;span class=&quot;s&quot;&gt;'due'&lt;/span&gt;&lt;span class=&quot;p&quot;&gt;,&lt;/span&gt; &lt;span class=&quot;s&quot;&gt;'tre'&lt;/span&gt;&lt;span class=&quot;p&quot;&gt;]&lt;/span&gt;

&lt;span class=&quot;o&quot;&gt;&amp;gt;&amp;gt;&amp;gt;&lt;/span&gt; &lt;span class=&quot;n&quot;&gt;cut&lt;/span&gt;&lt;span class=&quot;p&quot;&gt;(&lt;/span&gt;&lt;span class=&quot;n&quot;&gt;numbers&lt;/span&gt;&lt;span class=&quot;p&quot;&gt;)[&lt;/span&gt;&lt;span class=&quot;o&quot;&gt;...&lt;/span&gt;&lt;span class=&quot;p&quot;&gt;,&lt;/span&gt; &lt;span class=&quot;mi&quot;&gt;0&lt;/span&gt;&lt;span class=&quot;p&quot;&gt;]&lt;/span&gt;  &lt;span class=&quot;c&quot;&gt;# First letter of every elements&lt;/span&gt;
&lt;span class=&quot;p&quot;&gt;[&lt;/span&gt;&lt;span class=&quot;s&quot;&gt;'o'&lt;/span&gt;&lt;span class=&quot;p&quot;&gt;,&lt;/span&gt; &lt;span class=&quot;s&quot;&gt;'t'&lt;/span&gt;&lt;span class=&quot;p&quot;&gt;,&lt;/span&gt; &lt;span class=&quot;s&quot;&gt;'t'&lt;/span&gt;&lt;span class=&quot;p&quot;&gt;,&lt;/span&gt; &lt;span class=&quot;s&quot;&gt;'u'&lt;/span&gt;&lt;span class=&quot;p&quot;&gt;,&lt;/span&gt; &lt;span class=&quot;s&quot;&gt;'d'&lt;/span&gt;&lt;span class=&quot;p&quot;&gt;,&lt;/span&gt; &lt;span class=&quot;s&quot;&gt;'t'&lt;/span&gt;&lt;span class=&quot;p&quot;&gt;,&lt;/span&gt; &lt;span class=&quot;s&quot;&gt;'u'&lt;/span&gt;&lt;span class=&quot;p&quot;&gt;,&lt;/span&gt; &lt;span class=&quot;s&quot;&gt;'d'&lt;/span&gt;&lt;span class=&quot;p&quot;&gt;,&lt;/span&gt; &lt;span class=&quot;s&quot;&gt;'t'&lt;/span&gt;&lt;span class=&quot;p&quot;&gt;]&lt;/span&gt;&lt;span class=&quot;o&quot;&gt;.&lt;/span&gt;
&lt;/code&gt;&lt;/pre&gt;&lt;/div&gt;&lt;/div&gt;

&lt;h2 id=&quot;syntaxic-sugars&quot;&gt;Syntaxic sugars&lt;/h2&gt;

&lt;p&gt;There are two syntaxic sugars for an even easier use:&lt;/p&gt;

&lt;h3 id=&quot;the-_-syntax&quot;&gt;The |_ syntax&lt;/h3&gt;

&lt;div class=&quot;language-python highlighter-rouge&quot;&gt;&lt;div class=&quot;highlight&quot;&gt;&lt;pre class=&quot;highlight&quot;&gt;&lt;code&gt;&lt;span class=&quot;kn&quot;&gt;from&lt;/span&gt; &lt;span class=&quot;nn&quot;&gt;cutter&lt;/span&gt; &lt;span class=&quot;kn&quot;&gt;import&lt;/span&gt; &lt;span class=&quot;n&quot;&gt;_&lt;/span&gt;

&lt;span class=&quot;o&quot;&gt;&amp;gt;&amp;gt;&amp;gt;&lt;/span&gt; &lt;span class=&quot;n&quot;&gt;numbers&lt;/span&gt; &lt;span class=&quot;o&quot;&gt;|&lt;/span&gt;&lt;span class=&quot;n&quot;&gt;_&lt;/span&gt; &lt;span class=&quot;p&quot;&gt;[&lt;/span&gt;&lt;span class=&quot;mi&quot;&gt;1&lt;/span&gt;&lt;span class=&quot;p&quot;&gt;]&lt;/span&gt;
&lt;span class=&quot;p&quot;&gt;[&lt;/span&gt;&lt;span class=&quot;s&quot;&gt;'two'&lt;/span&gt;&lt;span class=&quot;p&quot;&gt;,&lt;/span&gt; &lt;span class=&quot;s&quot;&gt;'deux'&lt;/span&gt;&lt;span class=&quot;p&quot;&gt;,&lt;/span&gt; &lt;span class=&quot;s&quot;&gt;'due'&lt;/span&gt;&lt;span class=&quot;p&quot;&gt;]&lt;/span&gt;&lt;span class=&quot;o&quot;&gt;*&lt;/span&gt;

&lt;span class=&quot;o&quot;&gt;&amp;gt;&amp;gt;&amp;gt;&lt;/span&gt; &lt;span class=&quot;p&quot;&gt;(&lt;/span&gt;&lt;span class=&quot;n&quot;&gt;numbers&lt;/span&gt; &lt;span class=&quot;o&quot;&gt;|&lt;/span&gt;&lt;span class=&quot;n&quot;&gt;_&lt;/span&gt; &lt;span class=&quot;p&quot;&gt;[&lt;/span&gt;&lt;span class=&quot;mi&quot;&gt;1&lt;/span&gt;&lt;span class=&quot;p&quot;&gt;]&lt;/span&gt; &lt;span class=&quot;o&quot;&gt;|&lt;/span&gt;&lt;span class=&quot;n&quot;&gt;_&lt;/span&gt; &lt;span class=&quot;p&quot;&gt;[&lt;/span&gt;&lt;span class=&quot;mi&quot;&gt;2&lt;/span&gt;&lt;span class=&quot;p&quot;&gt;]&lt;/span&gt; &lt;span class=&quot;o&quot;&gt;|&lt;/span&gt;&lt;span class=&quot;n&quot;&gt;_&lt;/span&gt; &lt;span class=&quot;o&quot;&gt;.&lt;/span&gt;&lt;span class=&quot;n&quot;&gt;upper&lt;/span&gt;&lt;span class=&quot;p&quot;&gt;)()&lt;/span&gt;
&lt;span class=&quot;p&quot;&gt;[&lt;/span&gt;&lt;span class=&quot;s&quot;&gt;'O'&lt;/span&gt;&lt;span class=&quot;p&quot;&gt;,&lt;/span&gt; &lt;span class=&quot;s&quot;&gt;'U'&lt;/span&gt;&lt;span class=&quot;p&quot;&gt;,&lt;/span&gt; &lt;span class=&quot;s&quot;&gt;'E'&lt;/span&gt;&lt;span class=&quot;p&quot;&gt;]&lt;/span&gt;

&lt;span class=&quot;o&quot;&gt;&amp;gt;&amp;gt;&amp;gt;&lt;/span&gt; &lt;span class=&quot;n&quot;&gt;numbers&lt;/span&gt; &lt;span class=&quot;o&quot;&gt;|&lt;/span&gt;&lt;span class=&quot;n&quot;&gt;_&lt;/span&gt; &lt;span class=&quot;p&quot;&gt;[&lt;/span&gt;&lt;span class=&quot;o&quot;&gt;...&lt;/span&gt;&lt;span class=&quot;p&quot;&gt;]&lt;/span&gt;
&lt;span class=&quot;p&quot;&gt;[&lt;/span&gt;&lt;span class=&quot;s&quot;&gt;'one'&lt;/span&gt;&lt;span class=&quot;p&quot;&gt;,&lt;/span&gt; &lt;span class=&quot;s&quot;&gt;'two'&lt;/span&gt;&lt;span class=&quot;p&quot;&gt;,&lt;/span&gt; &lt;span class=&quot;s&quot;&gt;'three'&lt;/span&gt;&lt;span class=&quot;p&quot;&gt;,&lt;/span&gt; &lt;span class=&quot;s&quot;&gt;'un'&lt;/span&gt;&lt;span class=&quot;p&quot;&gt;,&lt;/span&gt; &lt;span class=&quot;s&quot;&gt;'deux'&lt;/span&gt;&lt;span class=&quot;p&quot;&gt;,&lt;/span&gt; &lt;span class=&quot;s&quot;&gt;'trois'&lt;/span&gt;&lt;span class=&quot;p&quot;&gt;,&lt;/span&gt; &lt;span class=&quot;s&quot;&gt;'uno'&lt;/span&gt;&lt;span class=&quot;p&quot;&gt;,&lt;/span&gt; &lt;span class=&quot;s&quot;&gt;'due'&lt;/span&gt;&lt;span class=&quot;p&quot;&gt;,&lt;/span&gt; &lt;span class=&quot;s&quot;&gt;'tre'&lt;/span&gt;&lt;span class=&quot;p&quot;&gt;]&lt;/span&gt;

&lt;span class=&quot;o&quot;&gt;&amp;gt;&amp;gt;&amp;gt;&lt;/span&gt; &lt;span class=&quot;n&quot;&gt;numbers&lt;/span&gt; &lt;span class=&quot;o&quot;&gt;|&lt;/span&gt;&lt;span class=&quot;n&quot;&gt;_&lt;/span&gt; &lt;span class=&quot;p&quot;&gt;[&lt;/span&gt;&lt;span class=&quot;o&quot;&gt;...&lt;/span&gt;&lt;span class=&quot;p&quot;&gt;,&lt;/span&gt; &lt;span class=&quot;mi&quot;&gt;0&lt;/span&gt;&lt;span class=&quot;p&quot;&gt;]&lt;/span&gt;
&lt;span class=&quot;p&quot;&gt;[&lt;/span&gt;&lt;span class=&quot;s&quot;&gt;'o'&lt;/span&gt;&lt;span class=&quot;p&quot;&gt;,&lt;/span&gt; &lt;span class=&quot;s&quot;&gt;'t'&lt;/span&gt;&lt;span class=&quot;p&quot;&gt;,&lt;/span&gt; &lt;span class=&quot;s&quot;&gt;'t'&lt;/span&gt;&lt;span class=&quot;p&quot;&gt;,&lt;/span&gt; &lt;span class=&quot;s&quot;&gt;'u'&lt;/span&gt;&lt;span class=&quot;p&quot;&gt;,&lt;/span&gt; &lt;span class=&quot;s&quot;&gt;'d'&lt;/span&gt;&lt;span class=&quot;p&quot;&gt;,&lt;/span&gt; &lt;span class=&quot;s&quot;&gt;'t'&lt;/span&gt;&lt;span class=&quot;p&quot;&gt;,&lt;/span&gt; &lt;span class=&quot;s&quot;&gt;'u'&lt;/span&gt;&lt;span class=&quot;p&quot;&gt;,&lt;/span&gt; &lt;span class=&quot;s&quot;&gt;'d'&lt;/span&gt;&lt;span class=&quot;p&quot;&gt;,&lt;/span&gt; &lt;span class=&quot;s&quot;&gt;'t'&lt;/span&gt;&lt;span class=&quot;p&quot;&gt;]&lt;/span&gt;&lt;span class=&quot;o&quot;&gt;.&lt;/span&gt;

&lt;/code&gt;&lt;/pre&gt;&lt;/div&gt;&lt;/div&gt;
&lt;h3 id=&quot;the--syntax&quot;&gt;The ! syntax&lt;/h3&gt;
&lt;p&gt;This syntax is meant for use in shells.&lt;/p&gt;

&lt;p&gt;It is for example used in &lt;a href=&quot;https://github.com/Kozea/wdb&quot;&gt;wdb&lt;/a&gt; and the bundled cut.py interpreter.&lt;/p&gt;

&lt;p&gt;Cutter provide a &lt;code class=&quot;highlighter-rouge&quot;&gt;bang_compile&lt;/code&gt; function which is a wrapper of the python builtin &lt;code class=&quot;highlighter-rouge&quot;&gt;compile&lt;/code&gt; function.&lt;/p&gt;

&lt;div class=&quot;language-python highlighter-rouge&quot;&gt;&lt;div class=&quot;highlight&quot;&gt;&lt;pre class=&quot;highlight&quot;&gt;&lt;code&gt;&lt;span class=&quot;c&quot;&gt;# This code muste be compiled with cutter.utils.bang_compile&lt;/span&gt;

&lt;span class=&quot;o&quot;&gt;&amp;gt;&amp;gt;&amp;gt;&lt;/span&gt; &lt;span class=&quot;n&quot;&gt;numbers&lt;/span&gt;&lt;span class=&quot;err&quot;&gt;!&lt;/span&gt;&lt;span class=&quot;mi&quot;&gt;1&lt;/span&gt;
&lt;span class=&quot;p&quot;&gt;[&lt;/span&gt;&lt;span class=&quot;s&quot;&gt;'two'&lt;/span&gt;&lt;span class=&quot;p&quot;&gt;,&lt;/span&gt; &lt;span class=&quot;s&quot;&gt;'deux'&lt;/span&gt;&lt;span class=&quot;p&quot;&gt;,&lt;/span&gt; &lt;span class=&quot;s&quot;&gt;'due'&lt;/span&gt;&lt;span class=&quot;p&quot;&gt;]&lt;/span&gt;&lt;span class=&quot;o&quot;&gt;*&lt;/span&gt;

&lt;span class=&quot;o&quot;&gt;&amp;gt;&amp;gt;&amp;gt;&lt;/span&gt; &lt;span class=&quot;n&quot;&gt;numbers&lt;/span&gt;&lt;span class=&quot;err&quot;&gt;!&lt;/span&gt;&lt;span class=&quot;mi&quot;&gt;1&lt;/span&gt;&lt;span class=&quot;err&quot;&gt;!&lt;/span&gt;&lt;span class=&quot;mi&quot;&gt;2&lt;/span&gt;&lt;span class=&quot;err&quot;&gt;!&lt;/span&gt;&lt;span class=&quot;n&quot;&gt;upper&lt;/span&gt;&lt;span class=&quot;p&quot;&gt;()&lt;/span&gt;
&lt;span class=&quot;p&quot;&gt;[&lt;/span&gt;&lt;span class=&quot;s&quot;&gt;'O'&lt;/span&gt;&lt;span class=&quot;p&quot;&gt;,&lt;/span&gt; &lt;span class=&quot;s&quot;&gt;'U'&lt;/span&gt;&lt;span class=&quot;p&quot;&gt;,&lt;/span&gt; &lt;span class=&quot;s&quot;&gt;'E'&lt;/span&gt;&lt;span class=&quot;p&quot;&gt;]&lt;/span&gt;

&lt;span class=&quot;o&quot;&gt;&amp;gt;&amp;gt;&amp;gt;&lt;/span&gt; &lt;span class=&quot;n&quot;&gt;numbers&lt;/span&gt;&lt;span class=&quot;err&quot;&gt;!&lt;/span&gt;&lt;span class=&quot;o&quot;&gt;*&lt;/span&gt;
&lt;span class=&quot;p&quot;&gt;[&lt;/span&gt;&lt;span class=&quot;s&quot;&gt;'one'&lt;/span&gt;&lt;span class=&quot;p&quot;&gt;,&lt;/span&gt; &lt;span class=&quot;s&quot;&gt;'two'&lt;/span&gt;&lt;span class=&quot;p&quot;&gt;,&lt;/span&gt; &lt;span class=&quot;s&quot;&gt;'three'&lt;/span&gt;&lt;span class=&quot;p&quot;&gt;,&lt;/span&gt; &lt;span class=&quot;s&quot;&gt;'un'&lt;/span&gt;&lt;span class=&quot;p&quot;&gt;,&lt;/span&gt; &lt;span class=&quot;s&quot;&gt;'deux'&lt;/span&gt;&lt;span class=&quot;p&quot;&gt;,&lt;/span&gt; &lt;span class=&quot;s&quot;&gt;'trois'&lt;/span&gt;&lt;span class=&quot;p&quot;&gt;,&lt;/span&gt; &lt;span class=&quot;s&quot;&gt;'uno'&lt;/span&gt;&lt;span class=&quot;p&quot;&gt;,&lt;/span&gt; &lt;span class=&quot;s&quot;&gt;'due'&lt;/span&gt;&lt;span class=&quot;p&quot;&gt;,&lt;/span&gt; &lt;span class=&quot;s&quot;&gt;'tre'&lt;/span&gt;&lt;span class=&quot;p&quot;&gt;]&lt;/span&gt;

&lt;span class=&quot;o&quot;&gt;&amp;gt;&amp;gt;&amp;gt;&lt;/span&gt; &lt;span class=&quot;n&quot;&gt;numbers&lt;/span&gt;&lt;span class=&quot;err&quot;&gt;!&lt;/span&gt;&lt;span class=&quot;o&quot;&gt;*&lt;/span&gt;&lt;span class=&quot;err&quot;&gt;!&lt;/span&gt;&lt;span class=&quot;mi&quot;&gt;0&lt;/span&gt;
&lt;span class=&quot;p&quot;&gt;[&lt;/span&gt;&lt;span class=&quot;s&quot;&gt;'o'&lt;/span&gt;&lt;span class=&quot;p&quot;&gt;,&lt;/span&gt; &lt;span class=&quot;s&quot;&gt;'t'&lt;/span&gt;&lt;span class=&quot;p&quot;&gt;,&lt;/span&gt; &lt;span class=&quot;s&quot;&gt;'t'&lt;/span&gt;&lt;span class=&quot;p&quot;&gt;,&lt;/span&gt; &lt;span class=&quot;s&quot;&gt;'u'&lt;/span&gt;&lt;span class=&quot;p&quot;&gt;,&lt;/span&gt; &lt;span class=&quot;s&quot;&gt;'d'&lt;/span&gt;&lt;span class=&quot;p&quot;&gt;,&lt;/span&gt; &lt;span class=&quot;s&quot;&gt;'t'&lt;/span&gt;&lt;span class=&quot;p&quot;&gt;,&lt;/span&gt; &lt;span class=&quot;s&quot;&gt;'u'&lt;/span&gt;&lt;span class=&quot;p&quot;&gt;,&lt;/span&gt; &lt;span class=&quot;s&quot;&gt;'d'&lt;/span&gt;&lt;span class=&quot;p&quot;&gt;,&lt;/span&gt; &lt;span class=&quot;s&quot;&gt;'t'&lt;/span&gt;&lt;span class=&quot;p&quot;&gt;]&lt;/span&gt;&lt;span class=&quot;o&quot;&gt;.&lt;/span&gt;
&lt;/code&gt;&lt;/pre&gt;&lt;/div&gt;&lt;/div&gt;

&lt;p&gt;This syntax use the python tokenizer and ast to make it work. This is really useful when debugging to inspect list content.&lt;/p&gt;

&lt;h3 id=&quot;more&quot;&gt;More&lt;/h3&gt;

&lt;p&gt;Cutter works with dictionaries too:&lt;/p&gt;
&lt;div class=&quot;language-python highlighter-rouge&quot;&gt;&lt;div class=&quot;highlight&quot;&gt;&lt;pre class=&quot;highlight&quot;&gt;&lt;code&gt;&lt;span class=&quot;n&quot;&gt;cut&lt;/span&gt;&lt;span class=&quot;p&quot;&gt;(&lt;/span&gt;&lt;span class=&quot;nb&quot;&gt;dict&lt;/span&gt;&lt;span class=&quot;p&quot;&gt;)[&lt;/span&gt;&lt;span class=&quot;s&quot;&gt;'key'&lt;/span&gt;&lt;span class=&quot;p&quot;&gt;]&lt;/span&gt;
&lt;/code&gt;&lt;/pre&gt;&lt;/div&gt;&lt;/div&gt;

&lt;p&gt;slices:&lt;/p&gt;
&lt;div class=&quot;language-python highlighter-rouge&quot;&gt;&lt;div class=&quot;highlight&quot;&gt;&lt;pre class=&quot;highlight&quot;&gt;&lt;code&gt;&lt;span class=&quot;n&quot;&gt;cut&lt;/span&gt;&lt;span class=&quot;p&quot;&gt;(&lt;/span&gt;&lt;span class=&quot;nb&quot;&gt;list&lt;/span&gt;&lt;span class=&quot;p&quot;&gt;)[:&lt;/span&gt;&lt;span class=&quot;mi&quot;&gt;5&lt;/span&gt;&lt;span class=&quot;p&quot;&gt;]&lt;/span&gt;
&lt;/code&gt;&lt;/pre&gt;&lt;/div&gt;&lt;/div&gt;

&lt;p&gt;For more examples see the test files : &lt;a href=&quot;https://github.com/paradoxxxzero/cutter/tree/master/test&quot;&gt;test&lt;/a&gt;&lt;/p&gt;

&lt;p&gt;Cutter is compatible with at least: python 2.6, 2.7, 3.2, 3.3, 3.4 and pypy and is licensed under &lt;a href=&quot;http://www.gnu.org/licenses/lgpl.html&quot;&gt;lgpl v3&lt;/a&gt;&lt;/p&gt;
</description>
        <pubDate>Wed, 13 May 2015 15:04:42 +0000</pubDate>
        <link>http://paradoxxxzero.github.io/2015/05/13/cutter.html</link>
        <guid isPermaLink="true">http://paradoxxxzero.github.io/2015/05/13/cutter.html</guid>
      </item>
    
      <item>
        <title>wdb 2.1 new features and a new clean theme</title>
        <description>&lt;p&gt;Hi everybody!&lt;/p&gt;

&lt;p&gt;wdb 2.1 is here with a new more serious light theme:&lt;/p&gt;

&lt;p&gt;&lt;img src=&quot;/assets/wdb21_light.png&quot; alt=&quot;wdb light theme&quot; /&gt;&lt;/p&gt;

&lt;p&gt;and some new features:&lt;/p&gt;

&lt;ul&gt;
  &lt;li&gt;New completion mechanism, should complete a lot better&lt;/li&gt;
  &lt;li&gt;Experimental object tree search (&lt;code class=&quot;highlighter-rouge&quot;&gt;.f&lt;/code&gt; command) to look for a key or a value condition in an object&lt;/li&gt;
  &lt;li&gt;Media display in debugger (&lt;code class=&quot;highlighter-rouge&quot;&gt;.i&lt;/code&gt; command)&lt;/li&gt;
  &lt;li&gt;New diff function to print differences between string / obj / files (&lt;code class=&quot;highlighter-rouge&quot;&gt;.x&lt;/code&gt; command)&lt;/li&gt;
  &lt;li&gt;Timing for expression evaluation (remote and local)&lt;/li&gt;
  &lt;li&gt;Prompt exception inspection (click on it)&lt;/li&gt;
  &lt;li&gt;A new shell mode and an executable &lt;code class=&quot;highlighter-rouge&quot;&gt;wdb&lt;/code&gt;&lt;/li&gt;
  &lt;li&gt;A new multiline prompt mode triggered by &lt;code class=&quot;highlighter-rouge&quot;&gt;[ctrl]&lt;/code&gt; + &lt;code class=&quot;highlighter-rouge&quot;&gt;[enter]&lt;/code&gt;&lt;/li&gt;
  &lt;li&gt;And a lot of bug fixes&lt;/li&gt;
&lt;/ul&gt;

&lt;p&gt;For more features see the &lt;a href=&quot;/2014/05/12/wdb-2.html&quot;&gt;last wdb 2.0 post&lt;/a&gt; and the &lt;a href=&quot;https://github.com/Kozea/wdb&quot;&gt;github readme page&lt;/a&gt;.&lt;/p&gt;
</description>
        <pubDate>Thu, 07 May 2015 00:00:00 +0000</pubDate>
        <link>http://paradoxxxzero.github.io/2015/05/07/wdb-2.1.html</link>
        <guid isPermaLink="true">http://paradoxxxzero.github.io/2015/05/07/wdb-2.1.html</guid>
      </item>
    
      <item>
        <title>wdb 2.0 the python debugger in your web browser</title>
        <description>&lt;p&gt;I released wdb 2.0 this morning with a lot of fixes and features added.&lt;/p&gt;

&lt;p&gt;&lt;img src=&quot;/assets/wdb.gif&quot; alt=&quot;wdb screen cast&quot; /&gt;&lt;/p&gt;
&lt;h4 id=&quot;wdb-debugging-itself-colors-are-awful-due-to-gif-compression&quot;&gt;&lt;strong&gt;wdb&lt;/strong&gt; debugging itself &lt;small&gt;(colors are awful due to gif compression)&lt;/small&gt;&lt;/h4&gt;

&lt;h2 id=&quot;introduction&quot;&gt;Introduction&lt;/h2&gt;

&lt;p&gt;For those who don’t know the project, &lt;strong&gt;wdb&lt;/strong&gt; is a python debugger like &lt;a href=&quot;https://docs.python.org/3/library/pdb.html&quot;&gt;pdb&lt;/a&gt;, but with a slick web front-end and a lot of additional features, such as:&lt;/p&gt;

&lt;ul&gt;
  &lt;li&gt;Source syntax highlighting&lt;/li&gt;
  &lt;li&gt;Visual breakpoints&lt;/li&gt;
  &lt;li&gt;Interactive code completion using &lt;a href=&quot;http://jedi.jedidjah.ch/&quot;&gt;jedi&lt;/a&gt;&lt;/li&gt;
  &lt;li&gt;Persistent breakpoints&lt;/li&gt;
  &lt;li&gt;Deep objects inspection using mouse&lt;/li&gt;
  &lt;li&gt;Multithreading / Multiprocessing support&lt;/li&gt;
  &lt;li&gt;Remote debugging&lt;/li&gt;
  &lt;li&gt;Watch expressions&lt;/li&gt;
  &lt;li&gt;In debugger code edition&lt;/li&gt;
  &lt;li&gt;Popular web servers integration to break on error&lt;/li&gt;
  &lt;li&gt;In exception breaking during trace (not post-mortem) in contrary to the werkzeug debugger for instance&lt;/li&gt;
  &lt;li&gt;Breaking in currently running programs through code injection (on supported systems)&lt;/li&gt;
&lt;/ul&gt;

&lt;p&gt;This project started because I wanted to improve the werkzeug debugger and it ended up as a whole project.&lt;/p&gt;

&lt;p&gt;&lt;strong&gt;wdb&lt;/strong&gt; supports python 2.6, 2.7, 3.2, 3.3, 3.4 and pypy.&lt;/p&gt;

&lt;h2 id=&quot;try-it&quot;&gt;Try it&lt;/h2&gt;

&lt;p&gt;Enough with the introduction, let’s just try it.&lt;/p&gt;

&lt;div class=&quot;language-bash highlighter-rouge&quot;&gt;&lt;div class=&quot;highlight&quot;&gt;&lt;pre class=&quot;highlight&quot;&gt;&lt;code&gt;    &lt;span class=&quot;nv&quot;&gt;$ &lt;/span&gt;pip install wdb.server      &lt;span class=&quot;c&quot;&gt;# Install both debugging client and server&lt;/span&gt;
    &lt;span class=&quot;nv&quot;&gt;$ &lt;/span&gt;wdb.server.py &amp;amp;             &lt;span class=&quot;c&quot;&gt;# Start the server as a background process&lt;/span&gt;
    &lt;span class=&quot;nv&quot;&gt;$ &lt;/span&gt;python &lt;span class=&quot;nt&quot;&gt;-m&lt;/span&gt; wdb your_file.py  &lt;span class=&quot;c&quot;&gt;# Start debugging your_file.py&lt;/span&gt;
&lt;/code&gt;&lt;/pre&gt;&lt;/div&gt;&lt;/div&gt;

&lt;p&gt;With the last command, your default web browser should open with the debugger paused at the first executable line of your file. You can get some help by typing &lt;code class=&quot;highlighter-rouge&quot;&gt;.h&lt;/code&gt; in the prompt. But for the basic commands:&lt;/p&gt;

&lt;ul&gt;
  &lt;li&gt;&lt;code class=&quot;highlighter-rouge&quot;&gt;.s or [Ctrl] + [↓] or [F11]&lt;/code&gt; : Step into&lt;/li&gt;
  &lt;li&gt;&lt;code class=&quot;highlighter-rouge&quot;&gt;.n or [Ctrl] + [→] or [F10]&lt;/code&gt; : Step over (Next)&lt;/li&gt;
  &lt;li&gt;&lt;code class=&quot;highlighter-rouge&quot;&gt;.r or [Ctrl] + [↑] or [F9] &lt;/code&gt; : Step out (Return)&lt;/li&gt;
  &lt;li&gt;&lt;code class=&quot;highlighter-rouge&quot;&gt;.c or [Ctrl] + [←] or [F8] &lt;/code&gt; : Continue&lt;/li&gt;
  &lt;li&gt;&lt;code class=&quot;highlighter-rouge&quot;&gt;.u or [F7]                 &lt;/code&gt; : Until (Next over loops)&lt;/li&gt;
  &lt;li&gt;&lt;code class=&quot;highlighter-rouge&quot;&gt;.h                         &lt;/code&gt; : Get some help&lt;/li&gt;
  &lt;li&gt;&lt;code class=&quot;highlighter-rouge&quot;&gt;.q                         &lt;/code&gt; : Quit&lt;/li&gt;
&lt;/ul&gt;

&lt;p&gt;While the server is running, you can access the following url &lt;a href=&quot;http://localhost:1984/&quot;&gt;localhost:1984/&lt;/a&gt; to have an overview of currently opened debugging sessions, global breakpoints and running python process (which you may pause see &lt;a href=&quot;https://github.com/Kozea/wdb#wdb-server&quot;&gt;here&lt;/a&gt; for more information).&lt;/p&gt;

&lt;p&gt;Wdb can also be used just like pdb, by putting:&lt;/p&gt;

&lt;div class=&quot;language-python highlighter-rouge&quot;&gt;&lt;div class=&quot;highlight&quot;&gt;&lt;pre class=&quot;highlight&quot;&gt;&lt;code&gt;    &lt;span class=&quot;kn&quot;&gt;import&lt;/span&gt; &lt;span class=&quot;nn&quot;&gt;wdb&lt;/span&gt;
    &lt;span class=&quot;n&quot;&gt;wdb&lt;/span&gt;&lt;span class=&quot;o&quot;&gt;.&lt;/span&gt;&lt;span class=&quot;n&quot;&gt;set_trace&lt;/span&gt;&lt;span class=&quot;p&quot;&gt;()&lt;/span&gt;
&lt;/code&gt;&lt;/pre&gt;&lt;/div&gt;&lt;/div&gt;

&lt;p&gt;anywhere in your code, where you want it to pause.&lt;/p&gt;

&lt;h2 id=&quot;upgrade-from-v1x&quot;&gt;Upgrade from v1.x&lt;/h2&gt;

&lt;p&gt;To upgrade to wdb 2.0 you need to run:&lt;/p&gt;

&lt;div class=&quot;language-bash highlighter-rouge&quot;&gt;&lt;div class=&quot;highlight&quot;&gt;&lt;pre class=&quot;highlight&quot;&gt;&lt;code&gt;    &lt;span class=&quot;nv&quot;&gt;$ &lt;/span&gt;pip install wdb.server
    &lt;span class=&quot;nv&quot;&gt;$ &lt;/span&gt;pip install &lt;span class=&quot;nt&quot;&gt;--upgrade&lt;/span&gt; wdb
&lt;/code&gt;&lt;/pre&gt;&lt;/div&gt;&lt;/div&gt;

&lt;p&gt;This is because now the package is split between the server and the debugging client to avoid unneeded dependencies installed in virtualenvs or other python versions (once the server is installed and running you just have to install &lt;code class=&quot;highlighter-rouge&quot;&gt;wdb&lt;/code&gt; in your virtualenvs).&lt;/p&gt;

&lt;h2 id=&quot;more&quot;&gt;More&lt;/h2&gt;

&lt;p&gt;For more detailed instructions and other features please read the &lt;code class=&quot;highlighter-rouge&quot;&gt;README.md&lt;/code&gt; on the github page: &lt;a href=&quot;https://github.com/Kozea/wdb&quot;&gt;github.com/Kozea/wdb&lt;/a&gt;&lt;/p&gt;

&lt;p&gt;As always, you are more than welcome to contribute to the project (GPLv3), by adding new features, creating themes or supporting me financially see &lt;a href=&quot;/about/&quot;&gt;about page&lt;/a&gt; &lt;em&gt;(Encouraged!)&lt;/em&gt;&lt;/p&gt;
</description>
        <pubDate>Mon, 12 May 2014 00:00:00 +0000</pubDate>
        <link>http://paradoxxxzero.github.io/2014/05/12/wdb-2.html</link>
        <guid isPermaLink="true">http://paradoxxxzero.github.io/2014/05/12/wdb-2.html</guid>
      </item>
    
      <item>
        <title>butterfly now with ssl authentication</title>
        <description>&lt;p&gt;First of all thanks for all your feedback on butterfly, I never expected this to have such success (wow).
Google Analytics says about 50k visits in 2 days scattered among 164 countries. That’s a lot and this led to a fair amount of constructive criticism (and obviously a huge amount of destructive criticism too).&lt;/p&gt;

&lt;h2 id=&quot;security&quot;&gt;Security&lt;/h2&gt;

&lt;p&gt;Security was a major topic among all discussions and therefore several security breaches were fixed. That’s why starting with the new 1.4.0 release, the SSL secure mode will now be enabled by default. This works by creating a certificate and a client authentication file (&lt;a href=&quot;http://en.wikipedia.org/wiki/Transport_Layer_Security&quot;&gt;More info&lt;/a&gt;).&lt;/p&gt;

&lt;p&gt;This is a boring process so I added some options in butterfly to make your life easier.&lt;/p&gt;

&lt;p&gt;NB: Before going into the details, be aware that the previous behavior is still available using the &lt;code class=&quot;highlighter-rouge&quot;&gt;--unsecure&lt;/code&gt; option.&lt;/p&gt;

&lt;h3 id=&quot;use-butterfly--140&quot;&gt;Use butterfly &amp;gt; 1.4.0&lt;/h3&gt;

&lt;p&gt;As you launch &lt;code class=&quot;highlighter-rouge&quot;&gt;butterfly.server.py&lt;/code&gt; you will see that it can’t start without generating certificates. So let’s make them!&lt;/p&gt;

&lt;h4 id=&quot;generating-certificates-files&quot;&gt;Generating certificates files&lt;/h4&gt;
&lt;p&gt;In this example I suppose you run the server as root, you want to access it as user &lt;code class=&quot;highlighter-rouge&quot;&gt;foo&lt;/code&gt; on a local network and your ip address on this network is &lt;code class=&quot;highlighter-rouge&quot;&gt;192.168.0.1&lt;/code&gt;:&lt;/p&gt;

&lt;figure class=&quot;highlight&quot;&gt;&lt;pre&gt;&lt;code class=&quot;language-bash&quot; data-lang=&quot;bash&quot;&gt;&lt;span class=&quot;nv&quot;&gt;$ &lt;/span&gt;&lt;span class=&quot;nb&quot;&gt;sudo &lt;/span&gt;butterfly.server.py &lt;span class=&quot;nt&quot;&gt;--generate-certs&lt;/span&gt; &lt;span class=&quot;nt&quot;&gt;--host&lt;/span&gt;&lt;span class=&quot;o&quot;&gt;=&lt;/span&gt;&lt;span class=&quot;s2&quot;&gt;&quot;192.168.0.1&quot;&lt;/span&gt; &lt;span class=&quot;c&quot;&gt;# Generate the root certificate for running on local network&lt;/span&gt;&lt;br data-jekyll-commonmark-ghpages=&quot;&quot; /&gt;&lt;span class=&quot;nv&quot;&gt;$ &lt;/span&gt;&lt;span class=&quot;nb&quot;&gt;sudo &lt;/span&gt;butterfly.server.py &lt;span class=&quot;nt&quot;&gt;--generate-user-pkcs&lt;/span&gt;&lt;span class=&quot;o&quot;&gt;=&lt;/span&gt;foo              &lt;span class=&quot;c&quot;&gt;# Generate PKCS#12 auth file for user foo&lt;/span&gt;&lt;/code&gt;&lt;/pre&gt;&lt;/figure&gt;

&lt;p&gt;If you run as root the certificates will be placed in &lt;code class=&quot;highlighter-rouge&quot;&gt;/etc/buttefly/ssl/&lt;/code&gt;, as user in &lt;code class=&quot;highlighter-rouge&quot;&gt;~/.butterfly/ssl&lt;/code&gt;.&lt;/p&gt;

&lt;p&gt;Once generated, you will have to configure your browser to accept your self signed certificate and the PKCS#12 file (which contains a private key for your user to log in).&lt;/p&gt;

&lt;h4 id=&quot;importing-the-root-certificate-in-google-chrome&quot;&gt;Importing the root certificate in Google Chrome&lt;/h4&gt;

&lt;p&gt;Under Google Chrome go to &lt;em&gt;Settings &amp;gt; Show advanced settings… &amp;gt; HTTPS/SSL &amp;gt; Manage certificates… &amp;gt; Authorities &amp;gt; Import…&lt;/em&gt; and specify the &lt;code class=&quot;highlighter-rouge&quot;&gt;butterfly_ca.crt&lt;/code&gt; file.&lt;/p&gt;

&lt;h4 id=&quot;importing-the-root-certificate-in-firefox&quot;&gt;Importing the root certificate in Firefox&lt;/h4&gt;

&lt;p&gt;Under Firefox go to &lt;em&gt;Edit &amp;gt; Preferences &amp;gt; Advanced &amp;gt; View Certificates &amp;gt; Authorities &amp;gt; Import…&lt;/em&gt; and specify the &lt;code class=&quot;highlighter-rouge&quot;&gt;butterfly_ca.crt&lt;/code&gt; file.&lt;/p&gt;

&lt;h4 id=&quot;importing-the-user-client-authentication-file&quot;&gt;Importing the user client authentication file&lt;/h4&gt;
&lt;p&gt;For the PKCS#12 you will have to change the file ownership before importing it since this file is a secured file:&lt;/p&gt;

&lt;figure class=&quot;highlight&quot;&gt;&lt;pre&gt;&lt;code class=&quot;language-bash&quot; data-lang=&quot;bash&quot;&gt;&lt;span class=&quot;nv&quot;&gt;$ &lt;/span&gt;&lt;span class=&quot;nb&quot;&gt;sudo &lt;/span&gt;cp /etc/butterfly/ssl/foo.p12 /tmp &lt;span class=&quot;c&quot;&gt;# Copy it in /tmp&lt;/span&gt;&lt;br data-jekyll-commonmark-ghpages=&quot;&quot; /&gt;&lt;span class=&quot;nv&quot;&gt;$ &lt;/span&gt;&lt;span class=&quot;nb&quot;&gt;sudo &lt;/span&gt;chown foo /tmp/foo.p12             &lt;span class=&quot;c&quot;&gt;# Make it readable to the browser&lt;/span&gt;&lt;/code&gt;&lt;/pre&gt;&lt;/figure&gt;

&lt;p&gt;Then go to the &lt;em&gt;Your Certificates&lt;/em&gt; tab (for both Google Chrome and Firefox) and import it from &lt;code class=&quot;highlighter-rouge&quot;&gt;/tmp&lt;/code&gt;. If you set a password during the creation you will have to type it otherwise leave it blank. Some browsers don’t seem to appreciate blank passwords, if this doesn’t work retry &lt;code class=&quot;highlighter-rouge&quot;&gt;--generate-user-pkcs&lt;/code&gt; with a password.&lt;/p&gt;

&lt;figure class=&quot;highlight&quot;&gt;&lt;pre&gt;&lt;code class=&quot;language-bash&quot; data-lang=&quot;bash&quot;&gt;&lt;span class=&quot;nv&quot;&gt;$ &lt;/span&gt;&lt;span class=&quot;nb&quot;&gt;sudo &lt;/span&gt;butterfly.server.py &lt;span class=&quot;nt&quot;&gt;--host&lt;/span&gt;&lt;span class=&quot;o&quot;&gt;=&lt;/span&gt;&lt;span class=&quot;s2&quot;&gt;&quot;192.168.0.1&quot;&lt;/span&gt; &lt;span class=&quot;c&quot;&gt;# Run the server&lt;/span&gt;&lt;/code&gt;&lt;/pre&gt;&lt;/figure&gt;

&lt;p&gt;Now point your browser at &lt;a href=&quot;https://192.168.0.1:57575/&quot;&gt;&lt;strong&gt;https://&lt;/strong&gt;192.168.0.1:57575/&lt;/a&gt; and you are done without typing any password (The authentication informations are contained in the PKCS#12 file).&lt;/p&gt;

&lt;p&gt;You will have to run &lt;code class=&quot;highlighter-rouge&quot;&gt;--generate-certs&lt;/code&gt; every time you want to use a different &lt;code class=&quot;highlighter-rouge&quot;&gt;--host&lt;/code&gt;, the PKCS#12 will still work though since both host and user files comes from the same CA.&lt;/p&gt;

&lt;h4 id=&quot;other-browsers&quot;&gt;Other browsers&lt;/h4&gt;

&lt;p&gt;For Safari under Mac OS X, you will have to go to the Keychain Access application to import the certificate and the user PKCS#12 files. Sadly Safari seems to have some issues with WebSockets and self-signed certificates and I didn’t manage to make it work but I’m far from being an Apple tech and if anyone has succeeded in making it work I’m really interested in your procedure, otherwise file a bug to Apple. It works with the previous browsers in Mac OS though.&lt;/p&gt;

&lt;p&gt;For other platforms, search for the browser / OS specific way to import certificates and pray for X509 client auth support. If it’s unavailable, use the &lt;code class=&quot;highlighter-rouge&quot;&gt;--unsecure&lt;/code&gt; option and use ssh tunnels.&lt;/p&gt;

&lt;h2 id=&quot;changes&quot;&gt;Changes&lt;/h2&gt;

&lt;p&gt;Since the previous post, a lot of effort has been put into making butterfly safer by removing the HTML escapes for now and by checking websocket origin but also into making it run on more operating systems (FreeBSD, Mac OS, Cygwin, …).&lt;/p&gt;

&lt;p&gt;Incoming changes are listed &lt;a href=&quot;https://github.com/paradoxxxzero/butterfly/issues?labels=enhancement&amp;amp;page=1&amp;amp;state=open&quot;&gt;on the github page&lt;/a&gt;. Please feel free to discuss them.&lt;/p&gt;

&lt;p&gt;Have fun and if you want to support my free work go to the &lt;a href=&quot;/about/&quot;&gt;about page&lt;/a&gt; (Encouraged!)&lt;/p&gt;
</description>
        <pubDate>Fri, 21 Mar 2014 00:00:00 +0000</pubDate>
        <link>http://paradoxxxzero.github.io/2014/03/21/butterfly-with-ssl-auth.html</link>
        <guid isPermaLink="true">http://paradoxxxzero.github.io/2014/03/21/butterfly-with-ssl-auth.html</guid>
      </item>
    
      <item>
        <title>butterfly your everyday terminal in your web browser</title>
        <description>&lt;p&gt;There’s something I’ve been working on recently, at start just for fun but it turned out I use it everyday, so here it is:&lt;/p&gt;

&lt;p&gt;&lt;img src=&quot;/assets/butterfly_1.gif&quot; alt=&quot;butterfly screen cast&quot; /&gt;&lt;/p&gt;
&lt;h4 id=&quot;butterfly-your-everyday-terminal-right-in-your-browser&quot;&gt;&lt;strong&gt;butterfly&lt;/strong&gt; your everyday terminal right in your browser&lt;/h4&gt;

&lt;h1 id=&quot;version-20-has-been-released-please-see-here&quot;&gt;&lt;a href=&quot;/2015/10/30/butterfly_2.0.html&quot;&gt;Version 2.0 has been released please see here&lt;/a&gt;&lt;/h1&gt;

&lt;h2 id=&quot;try-it&quot;&gt;Try it&lt;/h2&gt;

&lt;p&gt;To try it, it’s as simple as:&lt;/p&gt;

&lt;figure class=&quot;highlight&quot;&gt;&lt;pre&gt;&lt;code class=&quot;language-bash&quot; data-lang=&quot;bash&quot;&gt;&lt;span class=&quot;nv&quot;&gt;$ &lt;/span&gt;&lt;span class=&quot;nb&quot;&gt;sudo &lt;/span&gt;pip install butterfly      &lt;span class=&quot;c&quot;&gt;# Install butterfly&lt;/span&gt;&lt;br data-jekyll-commonmark-ghpages=&quot;&quot; /&gt;&lt;span class=&quot;nv&quot;&gt;$ &lt;/span&gt;butterfly.server.py &lt;span class=&quot;nt&quot;&gt;--unsecure&lt;/span&gt;  &lt;span class=&quot;c&quot;&gt;# Launch the web server unsecure&lt;/span&gt;&lt;/code&gt;&lt;/pre&gt;&lt;/figure&gt;

&lt;p&gt;Then all you have to do is go to : &lt;a href=&quot;http://localhost:57575&quot;&gt;localhost:57575&lt;/a&gt;&lt;/p&gt;

&lt;h2 id=&quot;use-it&quot;&gt;Use it&lt;/h2&gt;

&lt;h3 id=&quot;server-side&quot;&gt;Server side&lt;/h3&gt;

&lt;h4 id=&quot;run-at-start-up-with-systemd&quot;&gt;Run at start-up with systemd&lt;/h4&gt;

&lt;p&gt;As it relies on a server running in background, you can download the systemd file from here &lt;a href=&quot;https://github.com/paradoxxxzero/butterfly/raw/master/butterfly.service&quot;&gt;butterfly.service&lt;/a&gt; put it in &lt;code class=&quot;highlighter-rouge&quot;&gt;/etc/systemd/system/&lt;/code&gt; or equivalent and:&lt;/p&gt;

&lt;figure class=&quot;highlight&quot;&gt;&lt;pre&gt;&lt;code class=&quot;language-bash&quot; data-lang=&quot;bash&quot;&gt;&lt;span class=&quot;nv&quot;&gt;$ &lt;/span&gt;&lt;span class=&quot;nb&quot;&gt;sudo &lt;/span&gt;systemctl &lt;span class=&quot;nb&quot;&gt;enable &lt;/span&gt;butterfly&lt;br data-jekyll-commonmark-ghpages=&quot;&quot; /&gt;&lt;span class=&quot;nv&quot;&gt;$ &lt;/span&gt;&lt;span class=&quot;nb&quot;&gt;sudo &lt;/span&gt;systemctl start butterfly&lt;/code&gt;&lt;/pre&gt;&lt;/figure&gt;

&lt;p&gt;Now the server will always be running.&lt;/p&gt;

&lt;h4 id=&quot;running-as-root-and-multiple-users&quot;&gt;Running as root and multiple users&lt;/h4&gt;

&lt;p&gt;butterfly is made to work as the terminal user and as root. When running as root the privileges are dropped to the user that initiated the connection, i.e. the user that opened the web page. You can change the terminal user by going to  &lt;a href=&quot;http://localhost:57575/user/user&quot;&gt;localhost:57575/user/&amp;lt;user&amp;gt;&lt;/a&gt;.&lt;/p&gt;

&lt;h4 id=&quot;access-from-a-remote-machine&quot;&gt;Access from a remote machine&lt;/h4&gt;

&lt;p&gt;You can set the bind host with &lt;code class=&quot;highlighter-rouge&quot;&gt;butterfly.server.py --host=&quot;0.0.0.0&quot;&lt;/code&gt; which will allow other users to connect to your terminal.
A password will be asked but IT IS NOT SECURE! So it’s recommended as of now to run this only on local network for testing purposes.&lt;/p&gt;

&lt;h4 id=&quot;run-a-specific-shell&quot;&gt;Run a specific shell&lt;/h4&gt;

&lt;p&gt;Use the &lt;strong&gt;shell&lt;/strong&gt; command line option: &lt;code class=&quot;highlighter-rouge&quot;&gt;butterfly.server.py --shell=/bin/fish&lt;/code&gt;&lt;/p&gt;

&lt;h3 id=&quot;client-side&quot;&gt;Client side&lt;/h3&gt;

&lt;p&gt;butterfly works like any other terminal but it has some specific features.&lt;/p&gt;

&lt;h4 id=&quot;enter-browser-shortcuts&quot;&gt;Enter browser shortcuts&lt;/h4&gt;

&lt;p&gt;When the terminal is focused all the hot-keys are caught (except from those the browser prevent to catch).
If you want to access the address bar with &lt;strong&gt;CTRL + L&lt;/strong&gt; it won’t work. That why the &lt;strong&gt;ALT + Z&lt;/strong&gt; combination is here to tell butterfly that the next key combination must not be caught. In other word accessing the address bar is done by doing &lt;strong&gt;ALT + Z&lt;/strong&gt; then &lt;strong&gt;CTRL + L&lt;/strong&gt;. (If your browser support css3 filters the screen will turn sepia).&lt;/p&gt;

&lt;h4 id=&quot;quick-selection-of-history-text&quot;&gt;Quick selection of history text&lt;/h4&gt;

&lt;p&gt;This is a feature I’ve always wanted to find in a terminal, quick history selection. I’ve started to implement some ideas you can try by hitting &lt;strong&gt;CTRL + SHIFT + up&lt;/strong&gt; and then navigating the selection with &lt;strong&gt;CTRL + SHIFT + up/down&lt;/strong&gt; and the arrow keys. When you got it right, hit &lt;strong&gt;ENTER&lt;/strong&gt; and butterfly will enter the selection right into the prompt:&lt;/p&gt;

&lt;p&gt;&lt;img src=&quot;/assets/butterfly_2.gif&quot; alt=&quot;butterfly selection&quot; /&gt;&lt;/p&gt;

&lt;h2 id=&quot;other-features&quot;&gt;Other features&lt;/h2&gt;

&lt;ul&gt;
  &lt;li&gt;The visual style is done using css so it’s entirely editable.&lt;/li&gt;
  &lt;li&gt;You can easily extend the in-browser terminal behavior by writing javascript (like it’s done for the quick selection).&lt;/li&gt;
  &lt;li&gt;As apparently it’s trendy to use a “modern” text editor based on a browser, so…&lt;/li&gt;
&lt;/ul&gt;

&lt;p&gt;&lt;img src=&quot;/assets/butterfly_3.gif&quot; alt=&quot;butterfly modern editors&quot; /&gt;&lt;/p&gt;

&lt;h2 id=&quot;missing-features&quot;&gt;Missing features&lt;/h2&gt;

&lt;ul&gt;
  &lt;li&gt;
    &lt;p&gt;The scroll-back is actually limited to 100 000 lines as of now in order to limit the memory footprint. It would be a good idea to store the scroll-back on the disk using the server.&lt;/p&gt;
  &lt;/li&gt;
  &lt;li&gt;
    &lt;p&gt;Embedding html directly into the terminal is possible and &lt;strong&gt;edit: not supported anymore&lt;/strong&gt; with the &lt;code class=&quot;highlighter-rouge&quot;&gt;ESC]99;&lt;span class=&quot;nt&quot;&gt;&amp;lt;html&amp;gt;&lt;/span&gt;BELL&lt;/code&gt; command for security reasons.&lt;/p&gt;
  &lt;/li&gt;
  &lt;li&gt;16.7 millions of color is not yet supported but it’s one hack away using the &lt;code class=&quot;highlighter-rouge&quot;&gt;ESC[38;2;r;g;bm&lt;/code&gt; escape.&lt;/li&gt;
  &lt;li&gt;An in-terminal css/js editing options with why not a package repository for theme and extensions.&lt;/li&gt;
&lt;/ul&gt;

&lt;h2 id=&quot;thats-all&quot;&gt;That’s all&lt;/h2&gt;

&lt;p&gt;&lt;a href=&quot;https://github.com/paradoxxxzero/butterfly&quot;&gt;The repository is hosted on github&lt;/a&gt;.&lt;/p&gt;

&lt;p&gt;So let me know what do you think of it, I had a lot of fun writing it anyway.&lt;/p&gt;
</description>
        <pubDate>Fri, 28 Feb 2014 00:00:00 +0000</pubDate>
        <link>http://paradoxxxzero.github.io/2014/02/28/butterfly.html</link>
        <guid isPermaLink="true">http://paradoxxxzero.github.io/2014/02/28/butterfly.html</guid>
      </item>
    
      <item>
        <title>wsreload 2.0 technical details</title>
        <description>&lt;p&gt;Following to the last post about &lt;a href=&quot;/2013/11/29/wsreload.html&quot;&gt;wsreload&lt;/a&gt;, I will now explain how I implemented the reload mechanism.&lt;/p&gt;

&lt;p&gt;Most existing browser reloaders use dirty X hacks like &lt;code class=&quot;highlighter-rouge&quot;&gt;xdotool&lt;/code&gt; or &lt;code class=&quot;highlighter-rouge&quot;&gt;xvkbd&lt;/code&gt; to send the &lt;code class=&quot;highlighter-rouge&quot;&gt;[F5]&lt;/code&gt; key to the browser window. This doesn’t work reliably. No really. There’s also other solutions, like launching your browser with a debug backdoor (&lt;code class=&quot;highlighter-rouge&quot;&gt;--remote-shell-port&lt;/code&gt;) but it kinda mess up everything else.&lt;/p&gt;

&lt;p&gt;So to avoid these dirty hacks wsreload ships with a browser extension which role is to tell the browser to reload the page directly from the extension API with the added bonus of bypassing the cache.&lt;/p&gt;

&lt;p&gt;But this does not solve the shell &amp;lt;-&amp;gt; browser communication problem, for that I used a python websocket server: the well known &lt;a href=&quot;http://www.tornadoweb.org/&quot;&gt;tornado server&lt;/a&gt;.
The server is used to relay reload requests from the command line client to the listening extension in the browser all through websockets.&lt;/p&gt;

&lt;p&gt;It looks like this:&lt;/p&gt;

&lt;p&gt;&lt;a href=&quot;http://kozea.github.io/umlaut/#N4IgdghgtgpiBcIBiAbA9gdwMIAsICcAXEAGhEIEtCU5EMBnfGdCAEwAI0A3GfLimBlIgUFMAGt6hAJ40EILAFc+cMgC80aKAlD0AxhDnwAjGUL4IYeigiFaAbQC0x4wAYSADgAsAXQC+ZMwwsGCE9Aj2oJCw8lhooRBivMIAHggAbABMAKxk0ghu7iAYFKyEOAiZrgDsZDgwFADmOMTwmW5k+GiEthTxCEV2Ka3FjMxobAB09Lw8+JMADvlkAGYUKTCsCCuGM2S25uHwwH4BUdC0CvE9SfipGTl5CNmPxaXlCF5eAJx1Dc2tTJVTrdXr9eCDGDDeQAIS6DGSq3Wm22uxg+0Ihx0pxI5xiiDiCVu9zar3yJmyRRKZQqJmqRXqTRaBVcRS6PUo4Mh0LoYxYrEWyxAaw2W3gOxQexAB3wRxOZ3AF3kAGVFAAjAAKXT0MHo4TIaXgWVyIHJlKp71pYEUKBQfyZrWttpBHL6YAGZihIzhmBm+HYXpgVjdwhFKPFaIxWOOOLxlwAImg9IoQsQDQ8TeT0qyyNSPvAnXaQIyAQhCy6we6IZ6eSAfQj/T01aHkWKJVKZXLY4r8SBVZrtbr9SBDcanvBKbnLWWbUWS8yC7OK5yq9yRmJuhQVkKw23I9LMbLsQropctUmhySgZmCtUTXmrUvi/8F+WQOzKx7yF75AwmPz2D0UQgzTYVW1RSV0QPaN5VxHtLgACQIVh4woehxBJMdTQQABmV4HxnZ1nwdQiiw/FcCkmbJ6Wqb50jw9JaK8Dw6PSGsRgAdRgNV6CTcQYFA3cII7Q8uxPJVEAASQAeUwslb3vadFyI+dHSfciQ2rb9awAFTQfBIFYNB2Awbi/TmFtRWEqDO2PODT3kJD8BQtCMPTeAvHkulFJpUj7VLZSyNBCitKGTizL4gTLPDdsbNEuy43keNbAgNUIBmElPJvCcLV8wL/NfdTgs0td5CQCgaHoaQpGCaK90gqMjxjcTe0TZNU0yu9x2zXL8zfVS/PfYquXY2F4T9dgmzq6zGrE+yJJANqUxAklqi67CjRzN48v6l81KIjSRu071xt4SbUumiMGugpr5R8MhRAkI5Ingsr0CEMheOUHUMjMAhGii+BfhAL78B1AB9Sw9BwfSvx6fAAcISGwGh2H4BwyZjC8Yxsm+TJ0myHDshY2icNG44QbQb7LmEeHEfkEAAhuubEsQVBMGEUGfvRv6EcBlxPqpsGYGR1G7hMKiaLohimJY9I2PIf6BNFmHxa8SZqmMTIcI8FiPFcHWvHSYGwp0SnqYZ3n6cQRnZoS162fezmhe5snFb51ovEF6mVbR4xJZqaX8dl1ireVqHVc+DWtZ1vWDe8Y3yd0F2abDkZbeZ+2HMdjnveFhBand623a5kWI7RwYlaR8u1ej7Xde+fXDcT46zdLy2i8BkAM9simCHhG4UHkAA5fSoEMRmWsudmPvN/Oec71otbziGa8o6jA/o4PvmY0PF992vNfruPm5Nn8Kfbm204Zpne9g1nkCdlfaBNquEA8Z+D7hquv/y1uL5Th3OmXce7xWavNXsM9nYW3gIXYBrQP5z1XijSO6NMbY1xvjQmxNvikzTr/f2G9aJb0YjvOWCtTYAJgd3a+Ntb5gNAP3TAg8R5jwnvsfA8J6hsFYfgceQ9uzZ0frnJBtATTwPfp/NeEsiFB1IbveW+DpFvkocnahtM350LtuAh+UDn7PFoR0URv8MZYxxnjAmRMSbfDdvA3+pUqHzxoYvG+2i7qrH0tzWKfggAA&quot;&gt;&lt;img src=&quot;data:image/svg+xml;base64,<svg xmlns='http://www.w3.org/2000/svg'  xmlns:xlink='http://www.w3.org/1999/xlink' width='829.5' height='790'><!--Generated with umlaut (http://kozea.github.io/umlaut/) (c) Mounier Florian Kozea 2013 on Tue Dec 03 2013 18:15:34 GMT+0100 (CET)--><defs><pattern id="grid" viewBox="0 0 10 10" x="0" y="0" width="25" height="25" patternUnits="userSpaceOnUse" patternTransform="translate(-110,84)scale(1)"><path d="M 10 0 L 0 0 L 0 10"></path></pattern><marker id="None" class="marker fill-fg stroke-fg" viewBox="-23 -13 26 26" markerUnits="userSpaceOnUse" markerWidth="20" markerHeight="20" orient="auto"><path d="M 0 0"></path></marker><marker id="NoneOpen" class="marker fill-bg stroke-fg" viewBox="-23 -13 26 26" markerUnits="userSpaceOnUse" markerWidth="20" markerHeight="20" orient="auto"><path d="M 0 0"></path></marker><marker id="NoneStart" class="marker fill-fg stroke-fg" viewBox="0 -13 26 26" markerUnits="userSpaceOnUse" markerWidth="20" markerHeight="20" orient="auto"><path d="M 0 0"></path></marker><marker id="NoneOpenStart" class="marker fill-bg stroke-fg" viewBox="0 -13 26 26" markerUnits="userSpaceOnUse" markerWidth="20" markerHeight="20" orient="auto"><path d="M 0 0"></path></marker><marker id="Vee" class="marker fill-fg stroke-fg" viewBox="-23 -13 26 26" markerUnits="userSpaceOnUse" markerWidth="20" markerHeight="20" orient="auto"><path d="M 0 0 L -20 -10 L -6.666666666666667 0 L -20 10 z"></path></marker><marker id="VeeOpen" class="marker fill-bg stroke-fg" viewBox="-23 -13 26 26" markerUnits="userSpaceOnUse" markerWidth="20" markerHeight="20" orient="auto"><path d="M 0 0 L -20 -10 L -6.666666666666667 0 L -20 10 z"></path></marker><marker id="VeeStart" class="marker fill-fg stroke-fg" viewBox="0 -13 26 26" markerUnits="userSpaceOnUse" markerWidth="20" markerHeight="20" orient="auto"><path d="M 0 0 L 20 -10 L 6.666666666666667 0 L 20 10 z"></path></marker><marker id="VeeOpenStart" class="marker fill-bg stroke-fg" viewBox="0 -13 26 26" markerUnits="userSpaceOnUse" markerWidth="20" markerHeight="20" orient="auto"><path d="M 0 0 L 20 -10 L 6.666666666666667 0 L 20 10 z"></path></marker><marker id="Crow" class="marker fill-fg stroke-fg" viewBox="-23 -13 26 26" markerUnits="userSpaceOnUse" markerWidth="20" markerHeight="20" orient="auto"><path d="M 0 -10 L -20 0 L 0 10 L -13.333333333333334 0 z"></path></marker><marker id="CrowOpen" class="marker fill-bg stroke-fg" viewBox="-23 -13 26 26" markerUnits="userSpaceOnUse" markerWidth="20" markerHeight="20" orient="auto"><path d="M 0 -10 L -20 0 L 0 10 L -13.333333333333334 0 z"></path></marker><marker id="CrowStart" class="marker fill-fg stroke-fg" viewBox="0 -13 26 26" markerUnits="userSpaceOnUse" markerWidth="20" markerHeight="20" orient="auto"><path d="M 0 -10 L 20 0 L 0 10 L 13.333333333333334 0 z"></path></marker><marker id="CrowOpenStart" class="marker fill-bg stroke-fg" viewBox="0 -13 26 26" markerUnits="userSpaceOnUse" markerWidth="20" markerHeight="20" orient="auto"><path d="M 0 -10 L 20 0 L 0 10 L 13.333333333333334 0 z"></path></marker><marker id="Normal" class="marker fill-fg stroke-fg" viewBox="-23 -13 26 26" markerUnits="userSpaceOnUse" markerWidth="20" markerHeight="20" orient="auto"><path d="M 0 0 L -20 -10 L -20 10 z"></path></marker><marker id="NormalOpen" class="marker fill-bg stroke-fg" viewBox="-23 -13 26 26" markerUnits="userSpaceOnUse" markerWidth="20" markerHeight="20" orient="auto"><path d="M 0 0 L -20 -10 L -20 10 z"></path></marker><marker id="NormalStart" class="marker fill-fg stroke-fg" viewBox="0 -13 26 26" markerUnits="userSpaceOnUse" markerWidth="20" markerHeight="20" orient="auto"><path d="M 0 0 L 20 -10 L 20 10 z"></path></marker><marker id="NormalOpenStart" class="marker fill-bg stroke-fg" viewBox="0 -13 26 26" markerUnits="userSpaceOnUse" markerWidth="20" markerHeight="20" orient="auto"><path d="M 0 0 L 20 -10 L 20 10 z"></path></marker><marker id="Inv" class="marker fill-fg stroke-fg" viewBox="-23 -13 26 26" markerUnits="userSpaceOnUse" markerWidth="20" markerHeight="20" orient="auto"><path d="M 0 -10 L -20 0 L 0 10 z"></path></marker><marker id="InvOpen" class="marker fill-bg stroke-fg" viewBox="-23 -13 26 26" markerUnits="userSpaceOnUse" markerWidth="20" markerHeight="20" orient="auto"><path d="M 0 -10 L -20 0 L 0 10 z"></path></marker><marker id="InvStart" class="marker fill-fg stroke-fg" viewBox="0 -13 26 26" markerUnits="userSpaceOnUse" markerWidth="20" markerHeight="20" orient="auto"><path d="M 0 -10 L 20 0 L 0 10 z"></path></marker><marker id="InvOpenStart" class="marker fill-bg stroke-fg" viewBox="0 -13 26 26" markerUnits="userSpaceOnUse" markerWidth="20" markerHeight="20" orient="auto"><path d="M 0 -10 L 20 0 L 0 10 z"></path></marker><marker id="Diamond" class="marker fill-fg stroke-fg" viewBox="-43 -13 46 26" markerUnits="userSpaceOnUse" markerWidth="40" markerHeight="20" orient="auto"><path d="M 0 0 L -20 -10 L -40 0 L -20 10 z"></path></marker><marker id="DiamondOpen" class="marker fill-bg stroke-fg" viewBox="-43 -13 46 26" markerUnits="userSpaceOnUse" markerWidth="40" markerHeight="20" orient="auto"><path d="M 0 0 L -20 -10 L -40 0 L -20 10 z"></path></marker><marker id="DiamondStart" class="marker fill-fg stroke-fg" viewBox="0 -13 46 26" markerUnits="userSpaceOnUse" markerWidth="40" markerHeight="20" orient="auto"><path d="M 0 0 L 20 -10 L 40 0 L 20 10 z"></path></marker><marker id="DiamondOpenStart" class="marker fill-bg stroke-fg" viewBox="0 -13 46 26" markerUnits="userSpaceOnUse" markerWidth="40" markerHeight="20" orient="auto"><path d="M 0 0 L 20 -10 L 40 0 L 20 10 z"></path></marker><marker id="Dot" class="marker fill-fg stroke-fg" viewBox="-23 -13 26 26" markerUnits="userSpaceOnUse" markerWidth="20" markerHeight="20" orient="auto"><path d="M 0 0         A 10 10 0 1 1 -20 0         A 10 10 0 1 1 0 0"></path></marker><marker id="DotOpen" class="marker fill-bg stroke-fg" viewBox="-23 -13 26 26" markerUnits="userSpaceOnUse" markerWidth="20" markerHeight="20" orient="auto"><path d="M 0 0         A 10 10 0 1 1 -20 0         A 10 10 0 1 1 0 0"></path></marker><marker id="DotStart" class="marker fill-fg stroke-fg" viewBox="0 -13 26 26" markerUnits="userSpaceOnUse" markerWidth="20" markerHeight="20" orient="auto"><path d="M 0 0         A -10 10 0 1 1 20 0         A -10 10 0 1 1 0 0"></path></marker><marker id="DotOpenStart" class="marker fill-bg stroke-fg" viewBox="0 -13 26 26" markerUnits="userSpaceOnUse" markerWidth="20" markerHeight="20" orient="auto"><path d="M 0 0         A -10 10 0 1 1 20 0         A -10 10 0 1 1 0 0"></path></marker><marker id="Box" class="marker fill-fg stroke-fg" viewBox="-23 -13 26 26" markerUnits="userSpaceOnUse" markerWidth="20" markerHeight="20" orient="auto"><path d="M 0 -10         L 0 10         L -20 10         L -20 -10        z"></path></marker><marker id="BoxOpen" class="marker fill-bg stroke-fg" viewBox="-23 -13 26 26" markerUnits="userSpaceOnUse" markerWidth="20" markerHeight="20" orient="auto"><path d="M 0 -10         L 0 10         L -20 10         L -20 -10        z"></path></marker><marker id="BoxStart" class="marker fill-fg stroke-fg" viewBox="0 -13 26 26" markerUnits="userSpaceOnUse" markerWidth="20" markerHeight="20" orient="auto"><path d="M 0 -10         L 0 10         L 20 10         L 20 -10        z"></path></marker><marker id="BoxOpenStart" class="marker fill-bg stroke-fg" viewBox="0 -13 26 26" markerUnits="userSpaceOnUse" markerWidth="20" markerHeight="20" orient="auto"><path d="M 0 -10         L 0 10         L 20 10         L 20 -10        z"></path></marker><marker id="Tee" class="marker fill-fg stroke-fg" viewBox="-10.5 -13 13.5 26" markerUnits="userSpaceOnUse" markerWidth="7.5" markerHeight="20" orient="auto"><path d="M 0 -10         L 0 10         L -7.5 10         L -7.5 -10        z"></path></marker><marker id="TeeOpen" class="marker fill-bg stroke-fg" viewBox="-10.5 -13 13.5 26" markerUnits="userSpaceOnUse" markerWidth="7.5" markerHeight="20" orient="auto"><path d="M 0 -10         L 0 10         L -7.5 10         L -7.5 -10        z"></path></marker><marker id="TeeStart" class="marker fill-fg stroke-fg" viewBox="0 -13 13.5 26" markerUnits="userSpaceOnUse" markerWidth="7.5" markerHeight="20" orient="auto"><path d="M 0 -10         L 0 10         L 7.5 10         L 7.5 -10        z"></path></marker><marker id="TeeOpenStart" class="marker fill-bg stroke-fg" viewBox="0 -13 13.5 26" markerUnits="userSpaceOnUse" markerWidth="7.5" markerHeight="20" orient="auto"><path d="M 0 -10         L 0 10         L 7.5 10         L 7.5 -10        z"></path></marker><style>.fill-bg { fill: #ffffff; }.fill-fg { fill: #000000; }.fill-none { fill: rgba(0, 0, 0, 0); }.stroke-bg { stroke: #ffffff; }.stroke-fg { stroke: #000000; }.stroke-none { stroke: rgba(0, 0, 0, 0); }#title { text-anchor: middle; font-size: 28px; }#grid path { fill: rgba(0, 0, 0, 0); stroke: rgba(0, 0, 255, 0.11764705882352941); }.background.move { cursor: move; }text { text-anchor: middle; font-family: sans; font-size: 14px; }.link .dashed { stroke-dasharray: 5px, 5px; }.link .ghost { fill: none; stroke-width: 30px; stroke: rgba(0, 0, 255, 0); }.group .ghost { cursor: crosshair; }.shape { stroke: #000000; stroke-width: 1.3px; }.selected .shape { stroke: #0000ff; }.selected.moving .shape { stroke: #ff00ff; }.moving .shape { stroke: #ff0000; }.link .shape { fill: none; }.link text { fill: #000000; stroke: none; }.element, .group { cursor: move; }.element .ghost, .group .ghost { fill: rgba(0, 0, 0, 0); stroke: none; }.element .handle.NW, .group .handle.NW { cursor: nw-resize; }.element .handle.NE, .group .handle.NE { cursor: ne-resize; }.element .handle.SW, .group .handle.SW { cursor: sw-resize; }.element .handle.SE, .group .handle.SE { cursor: se-resize; }.element .handle.O, .group .handle.O { cursor: crosshair; }.element .anchor, .group .anchor { cursor: pointer; }.element.selected .handle, .group.selected .handle { fill: rgba(0, 0, 0, 0.023529411764705882); stroke: rgba(0, 0, 0, 0.14901960784313725); }.element.selected .anchor, .group.selected .anchor { fill: rgba(255, 0, 0, 0.023529411764705882); stroke: rgba(255, 0, 0, 0.24705882352941178); }.selection { fill: rgba(0, 0, 255, 0.047058823529411764); stroke: rgba(0, 0, 255, 0.2980392156862745); stroke-width: 2px; }</style></defs><g id="bg"><g class="root" transform="translate(-94,81)scale(1)"><g class="underlay"></g><g class="links"><g class="link"><path class="shape full" marker-start="url(#NoneStart)" marker-end="url(#Normal)" d="M 286.5 175 C 326.25 175 326.25 175 366 175"></path></g><g class="link"><path class="shape full" marker-start="url(#NoneStart)" marker-end="url(#Normal)" d="M 625 513 C 625 595.924005119145 475 500.8259948808549 475 583.75"></path></g><g class="link"><path class="shape full" marker-start="url(#NoneStart)" marker-end="url(#Normal)" d="M 625 513 C 625 548.375 625 548.375 625 583.75"></path></g><g class="link"><path class="shape full" marker-start="url(#NormalStart)" marker-end="url(#Normal)" d="M 625 384 C 625 435.5 625 435.5 625 487"></path></g><g class="link"><path class="shape full" marker-start="url(#NoneStart)" marker-end="url(#Normal)" d="M 625 513 C 625 595.924005119145 775 500.8259948808549 775 583.75"></path></g><g class="link"><path class="shape full" marker-start="url(#NoneStart)" marker-end="url(#Normal)" d="M 484 175 C 512.75 175 512.75 175 541.5 175"></path></g><g class="link"><path class="shape full" marker-start="url(#NormalStart)" marker-end="url(#Normal)" d="M 625 266 C 625 227 625 227 625 188"></path></g><g class="link"><path class="shape full" marker-start="url(#NoneStart)" marker-end="url(#Normal)" d="M 625 63 C 625 112.5 625 112.5 625 162"></path></g><g class="link"><path class="shape full" marker-start="url(#NoneStart)" marker-end="url(#Normal)" d="M 589 50 C 530.25 50 530.25 50 471.5 50"></path></g></g><g class="elements"><g class="element" transform="translate(625,100)rotate(0)"><path class="shape fill-bg stroke-fg" d="M -103.5 -105         L 103.5 -105         L 103.5 -74         L -103.5 -74         z         M 103.5 -74         L 103.5 105         M -103.5 -74         L -103.5 105         M -103.5 105         L 103.5 105        "></path><text x="0" y="-84"><tspan x="0">wsreload.server.py</tspan></text></g><g class="element" transform="translate(625,525)rotate(0)"><path class="shape fill-bg stroke-fg" d="M -224.5 -110         L 224.5 -110         L 224.5 -79         L -224.5 -79         z         M 224.5 -79         L 224.5 110         M -224.5 -79         L -224.5 110         M -224.5 110         L 224.5 110        "></path><text x="0" y="-89"><tspan x="0">Browser</tspan></text></g><g class="element" transform="translate(225,150)rotate(0)"><path class="shape fill-bg stroke-fg" d="M -85 -50         L 85 -50         L 85 -19         L -85 -19         z         M 85 -19         L 85 50         M -85 -19         L -85 50         M -85 50         L 85 50        "></path><text x="0" y="-29"><tspan x="0">wsreload.py</tspan></text></g><g class="element" transform="translate(625,500)rotate(0)"><path class="shape fill-bg stroke-fg" d="M -89.39999999999999 -13         L 89.39999999999999 -13         L 89.39999999999999 13         L -89.39999999999999 13         z         M -74.5 -13         L -74.5 13         M 74.5 -13         L 74.5 13        "></path><text x="0" y="5"><tspan x="0">Browser extension</tspan></text></g><g class="element" transform="translate(625,600)rotate(0)"><path class="shape fill-bg stroke-fg" d="M -51.5 -16.25         L 51.5 -16.25         L 51.5 16.25         Q 25.75 8.125 0 16.25         T -51.5 16.25         z"></path><text x="0" y="2.96875"><tspan x="0">Browser tab</tspan></text></g><g class="element" transform="translate(625,50)rotate(0)"><path class="shape fill-bg stroke-fg" d="M -36 -13         L 36 -13         L 36 13         L -36 13         z         M -30 -13         L -30 13         M 30 -13         L 30 13        "></path><text x="0" y="5"><tspan x="0">inotify</tspan></text></g><g class="element" transform="translate(225,175)rotate(0)"><path class="shape fill-bg stroke-fg" d="M -61.5 -13         L 61.5 -13         L 61.5 13         L -61.5 13         z"></path><text x="0" y="5"><tspan x="0">wsreload client</tspan></text></g><g class="element" transform="translate(625,325)rotate(90)"><path class="shape fill-bg stroke-fg" d="M 54 13         A 5 13 0 1 1 54 -13         A 5 13 0 1 1 54 13         L -54 13         A 5 13 0 1 1 -54 -13         L 54 -13        "></path><text x="-2.5" y="5"><tspan x="-2.5">Websocket</tspan></text></g><g class="element" transform="translate(625,175)rotate(0)"><path class="shape fill-bg stroke-fg" d="M -90 -13         L 77 -13         L 90 13         L -77 13         z"></path><text x="0" y="5"><tspan x="0">Tornado webserver</tspan></text></g><g class="element" transform="translate(425,175)rotate(0)"><path class="shape fill-bg stroke-fg" d="M 54 13         A 5 13 0 1 1 54 -13         A 5 13 0 1 1 54 13         L -54 13         A 5 13 0 1 1 -54 -13         L 54 -13        "></path><text x="-2.5" y="5"><tspan x="-2.5">Websocket</tspan></text></g><g class="element" transform="translate(425,50)rotate(0)"><path class="shape fill-bg stroke-fg" d="M -46.5 -18         A 46.5 5 0 1 1 46.5 -18         A 46.5 5 0 1 1 -46.5 -18         M 46.5 -18         L 46.5 18         A 46.5 5 0 1 1 -46.5 18         L -46.5 -18"></path><text x="0" y="7.5"><tspan x="0">Filesystem</tspan></text></g><g class="element" transform="translate(475,600)rotate(0)"><path class="shape fill-bg stroke-fg" d="M -51.5 -16.25         L 51.5 -16.25         L 51.5 16.25         Q 25.75 8.125 0 16.25         T -51.5 16.25         z"></path><text x="0" y="2.96875"><tspan x="0">Browser tab</tspan></text></g><g class="element" transform="translate(775,600)rotate(0)"><path class="shape fill-bg stroke-fg" d="M -51.5 -16.25         L 51.5 -16.25         L 51.5 16.25         Q 25.75 8.125 0 16.25         T -51.5 16.25         z"></path><text x="0" y="2.96875"><tspan x="0">Browser tab</tspan></text></g></g><g class="overlay"></g></g></g><text id="title" x="414.75" y="50">wsreload overview</text><image xlink:href="data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAAGQAAAAnCAYAAAD5Lu2WAAAABHNCSVQICAgIfAhkiAAAAAlwSFlzAAAEKQAABCkBfcZRfgAAABl0RVh0U29mdHdhcmUAd3d3Lmlua3NjYXBlLm9yZ5vuPBoAAAfPSURBVGiB7Zp/bFPXFce/5z7b+QUJPxMYhYSqDQuidKloIAnaaiBpkqKsWcV7L6VjsK1j7To6pm2q2kko60Q3dWpVtBVYW7VFBOIHBRFo4piiFTFIwo/BYIVN3fhRGHSUEaAhjsHvnv0RbJzEv5KYxs38kSz73nvOuee9r9+75z6bmBkJ4gcx2Akk6E5CkDgjIUickRAkzkgIEmdYojEqamoqgEQZEfVbQGYcuav96nZjwQKzvzH+H6BIZW9xo8vFhJKYTAacvSm9BfsrKj6NRTwflZWVqfX19R2xjDlYOYT9xs9yfjC7pxjFY8fi6dx7oWZPQobV2jsgEeaNH4cfTclF1cSJSFYU/xgDE63C8v2BJh2I3W63pKamXNQ07euxjNufHHRdnzXQWGFvWQqb5Uzkb79w3zQ8MuEr/va3756MH7TsxwW3G0CXGC8/kI/CsWP8NmrOJDzZ0or2m14AAAPlAH490MR95OTkWNzujjQAo2IVs785CCFHDDRW2CtECuH/ej8walQ3MQBgdFISnsq919+eMy6rmxgAkJ2WhkWTJ9/uIChIEJKoF+kp6cOD92ekB9ikB7XJzQjeH88UNDSMC2wTQMUuV+adnjdqQc51BF+vzl3v6JONj5qamm5z2+12C1HA/bEHS5cu7b1g9ZFIcwAA1dSIQufOSouwXChy7lzg6y90up5jiU9nO51TKCB3VVUVj8eTAQCmSWlVVVUjBpJrVGUvAOz97BKOtl3B9JG3b5M3pcRb//yXv+08fx5adjZyhqX5+9q9Xqw/dbpXvBMnjtdqmvaxENzKTCuysjKna5p6Q9fV/VLiZ4ZhHCEi0jTtKYCfAZCr69o1gPcqiny2tnbzyWjyVlU1hYhWEKEsKyszT9NUj66rxwCsdzg2reGAMrPY6Xq6qKDwUVKwWHr5NcUqWv2BWDiJzDHy84xPCmcWtRQ6XX9oLit9VwhRK6WpAQARNicl2XD1alsrgH4t8FELIpmx7MBBLMiehOkjR+CSx4P3zpzFyfZ2v43HlHiypRXVOdnITU/HebcbG0+dxn86O3vFI8J4AOXMNBzAGoB+BWAYwEuEoP26ri/UNFUDeD5Arwohd0spMgE8Y5rKUVVVZxuGcSRczqqqThWCtgDIAHg1EQ5JSWlEVAjgVVVVq6qqqtStW7deueWSKQmZzSUl5wH8JDDWvvJ5hwEcBoAi585MAsYDgM12Y5nHY32dCLsBWg6ghUj8O9rz2pOoBQGAG1Ki9tRp1J4KbXPd68WbAVdNBDIAUuvq6jYF9NXpuv4mwA4ALCXPMQzHbt+gqqq1RLRDCHqHiPIXL14cNLCqqilC0GaAz0iJasMwLgcMG6qqviEEvZ+UZFsNoDrahHuybt2Wi0uWLLnmdndACPn3DRuMlv7GAgb90Qm39hADAKAoys8BMIB6wzB2B44ZhmES0U8B3P/444/dEyqyEOIFAKOEsCzsIYYvznEisRDAAl3X5w/4UGLEoApCRB8F66+trW0DcAbgw8HG8/Ly/gHAK6WYFjq6nMeMdzZs2HAplMXGjRv3AbwPwLw+JX4HGVRBmHEzzPCNW69erFixQgIwmUVSsPGuSoqmE1HYNeaW7UdE/LWoEv4CGJJPe29VTpcBOSayLY1kxohbflcIyIhihuCbshgwJAXpgg8z0zfCWdjtdguAB/0exEcA5MzctWt0KJ9ZjY13ocdjGpvNZgJgZqQMLOchLAgzrSbCY7qul4ayGTcu85cA3+1rdwhxAMAFi9d8MZSPIMvKnn1r1669CeACM1UMNO8hK4jD4WgAaA3AhqZp3w0cq6ioSNI07WVmPA/AX6P/tbT0OjOWMuOHxY1Nv7V/+GGyb2zGjh2pxU7XKoCfAODtPSPtAfAtXdfDFBqR6dM+JJ4hwjJd12c6HI7nfbtvKeWPiegsEVbruvYiwH8BKC09fXg+gOtScgkRlROhzBenubx0e7Fz5yImrPJ03vhekbPpIDEJm8U2g0HXwLCD8G5XVX4bKeUvhKA9AB/Tde0EQL+pq6tb19fjCCtIxF+v+gGz70j4j8x0MYzl74mUvWHGVyqKt/ntt+s6dV1/BcAEZu5WdRmGYQJ4Sdf195h5DhHdB1AbgFfcbvfubdu2fV5dXd3JzN22unvLStYXNDR8oCjW+YIxnQmdDLkmOTnZ9aeHHmovdLpeY8hdPeb6pLKyMi8lJWURgDwhpHXGoUNW22eX5zPJrFBHoZji4J8rSg762mHPeZFzZxXAW0KflL7DwKrmstJnYxkznpi2aZNt+LCMuc3lpY1FTtdzAF6K4MKKqdyz55G5J4EIa0hSsnU7gHMxyhUAJLO5Nobx4o6MtBFTifB+UZNrK4imRuFCpsWb629EuivNrq8fblqTn6Cun3Lz+58qXybQCQbaI9t+qRkNwjeJIZnQCMajET2Iy/c9/LCz62Piv70xpdjlmsYSR5ixkAjlAL4T0SlAkCFb9g4We0tL/2Zalazm8lJHf/wTgtwBWufO/W9/fROCfAHkjxqJ5XlfxYTUVKzMvx/DLBa8XvBgUNshszGMZw5fbsPxq1fhMSVqjh6Dx5RYfuhQUNvEFXIHYcbHvs8eUwZ9B8BSCP/jm8QVcgcZCfN3V2A5CQ69U2eIAy0lJX7hEmVvnJG4ZcUZCUHijIQgccb/AGU94E0OVgKPAAAAAElFTkSuQmCC" x="10" width="100" y="740" height="39"></image></svg>&quot; alt=&quot;wsreload overview&quot; /&gt;&lt;/a&gt;&lt;/p&gt;

&lt;p&gt;As you can see the server is also used to watch over file modification using &lt;a href=&quot;https://github.com/seb-m/pyinotify&quot;&gt;pyinotify&lt;/a&gt;.&lt;/p&gt;

&lt;p&gt;For instance if you have told wsreload to watch over your &lt;code class=&quot;highlighter-rouge&quot;&gt;/tmp&lt;/code&gt; directory and to reload all the &lt;code class=&quot;highlighter-rouge&quot;&gt;http://localhost:1234/*&lt;/code&gt; pages with the following command:&lt;/p&gt;

&lt;figure class=&quot;highlight&quot;&gt;&lt;pre&gt;&lt;code class=&quot;language-bash&quot; data-lang=&quot;bash&quot;&gt;&lt;span class=&quot;nv&quot;&gt;$ &lt;/span&gt;wsreload &lt;span class=&quot;nt&quot;&gt;--watch&lt;/span&gt; /tmp &lt;span class=&quot;nt&quot;&gt;--url&lt;/span&gt; &lt;span class=&quot;s2&quot;&gt;&quot;http://localhost:1234/*&quot;&lt;/span&gt;&lt;/code&gt;&lt;/pre&gt;&lt;/figure&gt;

&lt;p&gt;a inotify watcher will be added to the &lt;code class=&quot;highlighter-rouge&quot;&gt;/tmp&lt;/code&gt; directory with the url query attach to it.&lt;/p&gt;

&lt;p&gt;When a file change in &lt;code class=&quot;highlighter-rouge&quot;&gt;/tmp&lt;/code&gt;, inotify will tell the server and the server will send the corresponding query through all the connected websockets (opened by the browser extensions).&lt;/p&gt;

&lt;p&gt;All the extensions will receive the query and try to match any open tab with it, reloading the one matching.&lt;/p&gt;

&lt;p&gt;NB: The extension is also able to tell the server to watch for files when a url match the &lt;code class=&quot;highlighter-rouge&quot;&gt;file://&lt;/code&gt; prefix.&lt;/p&gt;

&lt;p&gt;That’s all for now.&lt;/p&gt;

&lt;p&gt;Ask me if you have any question :)&lt;/p&gt;

</description>
        <pubDate>Mon, 02 Dec 2013 00:00:00 +0000</pubDate>
        <link>http://paradoxxxzero.github.io/2013/12/02/wsreload-technical.html</link>
        <guid isPermaLink="true">http://paradoxxxzero.github.io/2013/12/02/wsreload-technical.html</guid>
      </item>
    
      <item>
        <title>wsreload 2.0</title>
        <description>&lt;p&gt;Today I released wsreload 2.0 which is IMHO a neat tool to increase productivity.&lt;/p&gt;

&lt;p&gt;&lt;em&gt;NB: As it depends on inotify, it’s only available on linux (if you are interested in making it work on other OS feel free to make a &lt;a href=&quot;http://github.com/paradoxxxzero/wsreload&quot;&gt;pull request&lt;/a&gt; :D)&lt;/em&gt;&lt;/p&gt;

&lt;p&gt;&lt;em&gt;NB: As of now it is only compatible with google chrome / chromium. Nothing prevents it to work under other browsers though, except from the fact that someone has to port the &lt;a href=&quot;http://github.com/paradoxxxzero/wsreload/tree/master/chrome-wsreload&quot;&gt;chrome extension code&lt;/a&gt; to his preferred browser again : &lt;a href=&quot;http://github.com/paradoxxxzero/wsreload&quot;&gt;pull request&lt;/a&gt; :)&lt;/em&gt;&lt;/p&gt;

&lt;h3 id=&quot;reloading-your-tabs&quot;&gt;Reloading your tabs&lt;/h3&gt;

&lt;p&gt;wsreload has one aim: making you forget about the &lt;code class=&quot;highlighter-rouge&quot;&gt;[F5]&lt;/code&gt; key.&lt;/p&gt;

&lt;h4 id=&quot;file-reloading&quot;&gt;file:// reloading&lt;/h4&gt;

&lt;p&gt;Every local file opened in your browser with a &lt;code class=&quot;highlighter-rouge&quot;&gt;file://&lt;/code&gt; scheme will be automatically synchronized with the file.&lt;/p&gt;

&lt;h4 id=&quot;selective-tabs-reloading&quot;&gt;Selective tabs reloading&lt;/h4&gt;

&lt;p&gt;But that’s not all of it, wsreload comes with a handy executable: &lt;code class=&quot;highlighter-rouge&quot;&gt;wsreload&lt;/code&gt; which can reload any tab matching a criteria when a file is modified.&lt;/p&gt;

&lt;p&gt;For example, I’m currently using it while writing the draft of this blog post.&lt;/p&gt;

&lt;p&gt;The following command:&lt;/p&gt;

&lt;figure class=&quot;highlight&quot;&gt;&lt;pre&gt;&lt;code class=&quot;language-bash&quot; data-lang=&quot;bash&quot;&gt;&lt;span class=&quot;nv&quot;&gt;$ &lt;/span&gt;wsreload &lt;span class=&quot;nt&quot;&gt;--watch&lt;/span&gt; _site/ &lt;span class=&quot;nt&quot;&gt;--url&lt;/span&gt; &lt;span class=&quot;s2&quot;&gt;&quot;http://localhost:4000/*&quot;&lt;/span&gt;&lt;/code&gt;&lt;/pre&gt;&lt;/figure&gt;

&lt;p&gt;tells the browser to reload all tabs that match the following url pattern : &lt;code class=&quot;highlighter-rouge&quot;&gt;http://localhost:4000/*&lt;/code&gt; whenever a file in the &lt;code class=&quot;highlighter-rouge&quot;&gt;_site/&lt;/code&gt; directory is modified.
(This is the directory in which jekyll generates this site.)
With it my draft preview is automatically reloaded every time I save the markdown file
(in conjunction with &lt;code class=&quot;highlighter-rouge&quot;&gt;jekill serve -w&lt;/code&gt; which regenerates the site when a file change).&lt;/p&gt;

&lt;p&gt;The &lt;code class=&quot;highlighter-rouge&quot;&gt;wsreload&lt;/code&gt; command comes with several matching criteria:&lt;/p&gt;

&lt;figure class=&quot;highlight&quot;&gt;&lt;pre&gt;&lt;code class=&quot;language-bash&quot; data-lang=&quot;bash&quot;&gt;&lt;span class=&quot;nv&quot;&gt;$ &lt;/span&gt; wsreload &lt;span class=&quot;nt&quot;&gt;--help&lt;/span&gt;&lt;br data-jekyll-commonmark-ghpages=&quot;&quot; /&gt;&lt;br data-jekyll-commonmark-ghpages=&quot;&quot; /&gt;Reload all tabs matching query through websocket&lt;br data-jekyll-commonmark-ghpages=&quot;&quot; /&gt;&lt;br data-jekyll-commonmark-ghpages=&quot;&quot; /&gt;optional arguments:&lt;br data-jekyll-commonmark-ghpages=&quot;&quot; /&gt;  &lt;span class=&quot;nt&quot;&gt;-h&lt;/span&gt;, &lt;span class=&quot;nt&quot;&gt;--help&lt;/span&gt;            show this &lt;span class=&quot;nb&quot;&gt;help &lt;/span&gt;message and &lt;span class=&quot;nb&quot;&gt;exit&lt;/span&gt;&lt;br data-jekyll-commonmark-ghpages=&quot;&quot; /&gt;  &lt;span class=&quot;nt&quot;&gt;--active&lt;/span&gt; ACTIVE       Whether the tabs are active &lt;span class=&quot;k&quot;&gt;in &lt;/span&gt;their windows.&lt;br data-jekyll-commonmark-ghpages=&quot;&quot; /&gt;  &lt;span class=&quot;nt&quot;&gt;--pinned&lt;/span&gt; PINNED       Whether the tabs are pinned.&lt;br data-jekyll-commonmark-ghpages=&quot;&quot; /&gt;  &lt;span class=&quot;nt&quot;&gt;--highlighted&lt;/span&gt; HIGHLIGHTED&lt;br data-jekyll-commonmark-ghpages=&quot;&quot; /&gt;                        Whether the tabs are highlighted.&lt;br data-jekyll-commonmark-ghpages=&quot;&quot; /&gt;  &lt;span class=&quot;nt&quot;&gt;--status&lt;/span&gt; &lt;span class=&quot;o&quot;&gt;{&lt;/span&gt;loading,complete&lt;span class=&quot;o&quot;&gt;}&lt;/span&gt;&lt;br data-jekyll-commonmark-ghpages=&quot;&quot; /&gt;                        Whether the tabs have completed loading.&lt;br data-jekyll-commonmark-ghpages=&quot;&quot; /&gt;  &lt;span class=&quot;nt&quot;&gt;--title&lt;/span&gt; TITLE         Match page titles against a pattern.&lt;br data-jekyll-commonmark-ghpages=&quot;&quot; /&gt;  &lt;span class=&quot;nt&quot;&gt;--url&lt;/span&gt; URL             Match tabs against a URL pattern.&lt;br data-jekyll-commonmark-ghpages=&quot;&quot; /&gt;  &lt;span class=&quot;nt&quot;&gt;--windowId&lt;/span&gt; WINDOWID   The ID of the parent window, or&lt;br data-jekyll-commonmark-ghpages=&quot;&quot; /&gt;                        chrome.windows.WINDOW_ID_CURRENT &lt;span class=&quot;k&quot;&gt;for &lt;/span&gt;the current&lt;br data-jekyll-commonmark-ghpages=&quot;&quot; /&gt;                        window.&lt;br data-jekyll-commonmark-ghpages=&quot;&quot; /&gt;  &lt;span class=&quot;nt&quot;&gt;--windowType&lt;/span&gt; &lt;span class=&quot;o&quot;&gt;{&lt;/span&gt;normal,popup,panel,app&lt;span class=&quot;o&quot;&gt;}&lt;/span&gt;&lt;br data-jekyll-commonmark-ghpages=&quot;&quot; /&gt;                        The &lt;span class=&quot;nb&quot;&gt;type &lt;/span&gt;of window the tabs are &lt;span class=&quot;k&quot;&gt;in&lt;/span&gt;&lt;span class=&quot;nb&quot;&gt;.&lt;/span&gt;&lt;br data-jekyll-commonmark-ghpages=&quot;&quot; /&gt;  &lt;span class=&quot;nt&quot;&gt;--index&lt;/span&gt; INDEX         The position of the tabs within their windows.&lt;br data-jekyll-commonmark-ghpages=&quot;&quot; /&gt;  &lt;span class=&quot;nt&quot;&gt;--delay&lt;/span&gt; DELAY         Delay to &lt;span class=&quot;nb&quot;&gt;wait &lt;/span&gt;before refreshing.&lt;br data-jekyll-commonmark-ghpages=&quot;&quot; /&gt;  &lt;span class=&quot;nt&quot;&gt;-H&lt;/span&gt; HOST, &lt;span class=&quot;nt&quot;&gt;--host&lt;/span&gt; HOST&lt;br data-jekyll-commonmark-ghpages=&quot;&quot; /&gt;  &lt;span class=&quot;nt&quot;&gt;-P&lt;/span&gt; PORT, &lt;span class=&quot;nt&quot;&gt;--port&lt;/span&gt; PORT&lt;br data-jekyll-commonmark-ghpages=&quot;&quot; /&gt;  &lt;span class=&quot;nt&quot;&gt;-E&lt;/span&gt; ENDPOINT, &lt;span class=&quot;nt&quot;&gt;--endpoint&lt;/span&gt; ENDPOINT&lt;br data-jekyll-commonmark-ghpages=&quot;&quot; /&gt;  &lt;span class=&quot;nt&quot;&gt;--watch&lt;/span&gt; WATCH &lt;span class=&quot;o&quot;&gt;[&lt;/span&gt;WATCH ...]&lt;br data-jekyll-commonmark-ghpages=&quot;&quot; /&gt;                        Tell the server to watch files and reload according to&lt;br data-jekyll-commonmark-ghpages=&quot;&quot; /&gt;                        the other options&lt;br data-jekyll-commonmark-ghpages=&quot;&quot; /&gt;  &lt;span class=&quot;nt&quot;&gt;--unwatch&lt;/span&gt; UNWATCH &lt;span class=&quot;o&quot;&gt;[&lt;/span&gt;UNWATCH ...]&lt;br data-jekyll-commonmark-ghpages=&quot;&quot; /&gt;                        Tell the server to unwatch the following files&lt;/code&gt;&lt;/pre&gt;&lt;/figure&gt;

&lt;p&gt;These rules follow the &lt;a href=&quot;http://developer.chrome.com/extensions/tabs.html#method-query&quot;&gt;chrome extension api tabs query criteria&lt;/a&gt;&lt;/p&gt;

&lt;h3 id=&quot;installing-it&quot;&gt;Installing it&lt;/h3&gt;

&lt;p&gt;To forget about the &lt;code class=&quot;highlighter-rouge&quot;&gt;F5&lt;/code&gt; key you have to install the wsreload library like any other python package:&lt;/p&gt;

&lt;figure class=&quot;highlight&quot;&gt;&lt;pre&gt;&lt;code class=&quot;language-bash&quot; data-lang=&quot;bash&quot;&gt;&lt;span class=&quot;nv&quot;&gt;$ &lt;/span&gt;pip install wsreload&lt;/code&gt;&lt;/pre&gt;&lt;/figure&gt;

&lt;p&gt;Then start the server using:&lt;/p&gt;

&lt;figure class=&quot;highlight&quot;&gt;&lt;pre&gt;&lt;code class=&quot;language-bash&quot; data-lang=&quot;bash&quot;&gt;&lt;span class=&quot;nv&quot;&gt;$ &lt;/span&gt;wsreload-server&amp;amp;&lt;/code&gt;&lt;/pre&gt;&lt;/figure&gt;

&lt;p&gt;or if you are using systemd:&lt;/p&gt;

&lt;figure class=&quot;highlight&quot;&gt;&lt;pre&gt;&lt;code class=&quot;language-bash&quot; data-lang=&quot;bash&quot;&gt;&lt;span class=&quot;nv&quot;&gt;$ &lt;/span&gt;wget https://github.com/paradoxxxzero/wsreload/raw/master/wsreload.service&lt;br data-jekyll-commonmark-ghpages=&quot;&quot; /&gt;&lt;span class=&quot;nv&quot;&gt;$ &lt;/span&gt;&lt;span class=&quot;nb&quot;&gt;sudo &lt;/span&gt;mv wsreload.service /etc/systemd/system&lt;br data-jekyll-commonmark-ghpages=&quot;&quot; /&gt;&lt;span class=&quot;nv&quot;&gt;$ &lt;/span&gt;&lt;span class=&quot;nb&quot;&gt;sudo &lt;/span&gt;systemctl &lt;span class=&quot;nb&quot;&gt;enable &lt;/span&gt;wsreload&lt;br data-jekyll-commonmark-ghpages=&quot;&quot; /&gt;&lt;span class=&quot;nv&quot;&gt;$ &lt;/span&gt;&lt;span class=&quot;nb&quot;&gt;sudo &lt;/span&gt;systemctl start wsreload&lt;/code&gt;&lt;/pre&gt;&lt;/figure&gt;

&lt;p&gt;And finally just install the &lt;a href=&quot;http://chrome.google.com/webstore/detail/wsreload/knefplbckfcppebehbomeankfgjalmak&quot;&gt;wsreload chrome extension&lt;/a&gt; like any other chrome extension.
That’s it !&lt;/p&gt;

&lt;p&gt;In a following post I will write about the technical details behind all of this.&lt;/p&gt;

</description>
        <pubDate>Fri, 29 Nov 2013 00:00:00 +0000</pubDate>
        <link>http://paradoxxxzero.github.io/2013/11/29/wsreload.html</link>
        <guid isPermaLink="true">http://paradoxxxzero.github.io/2013/11/29/wsreload.html</guid>
      </item>
    
      <item>
        <title>Prelude</title>
        <description>&lt;p&gt;Hello there and welcome to my brand new site &lt;a href=&quot;http://paradoxxxzero.github.io/&quot;&gt;paradoxxxzero.github.io&lt;/a&gt;.&lt;/p&gt;

&lt;p&gt;I’ve decided to build up this site in order to write about my personal and professional projects and some less serious stuffs.&lt;/p&gt;

&lt;p&gt;I hope that you will enjoy reading it.&lt;/p&gt;

</description>
        <pubDate>Sun, 24 Nov 2013 15:24:42 +0000</pubDate>
        <link>http://paradoxxxzero.github.io/2013/11/24/prelude.html</link>
        <guid isPermaLink="true">http://paradoxxxzero.github.io/2013/11/24/prelude.html</guid>
      </item>
    
  </channel>
</rss>
