<?xml version="1.0" encoding="UTF-8"?>
<rss version="2.0"
	xmlns:content="http://purl.org/rss/1.0/modules/content/"
	xmlns:wfw="http://wellformedweb.org/CommentAPI/"
	xmlns:dc="http://purl.org/dc/elements/1.1/"
	xmlns:atom="http://www.w3.org/2005/Atom"
	xmlns:sy="http://purl.org/rss/1.0/modules/syndication/"
	xmlns:slash="http://purl.org/rss/1.0/modules/slash/"
	>

<channel>
	<title>Sukria.net &#187; Coat::Persistent</title>
	<atom:link href="http://www.sukria.net/fr/archives/tag/coatpersistent/feed/" rel="self" type="application/rss+xml" />
	<link>http://www.sukria.net/fr</link>
	<description>I will press many keys on my keyboard causing an implementation to occur.</description>
	<lastBuildDate>Wed, 18 Jan 2012 17:11:38 +0000</lastBuildDate>
	<language>en</language>
	<sy:updatePeriod>hourly</sy:updatePeriod>
	<sy:updateFrequency>1</sy:updateFrequency>
	<generator>http://wordpress.org/?v=3.3.1</generator>
		<item>
		<title>Migrating homebrew Rails blog-entries to WordPress with Perl</title>
		<link>http://www.sukria.net/fr/archives/2009/12/24/migrating-homebrew-rails-blog-entries-to-wordpress-with-perl/</link>
		<comments>http://www.sukria.net/fr/archives/2009/12/24/migrating-homebrew-rails-blog-entries-to-wordpress-with-perl/#comments</comments>
		<pubDate>Thu, 24 Dec 2009 11:33:19 +0000</pubDate>
		<dc:creator>sukria</dc:creator>
				<category><![CDATA[Programming]]></category>
		<category><![CDATA[Coat::Persistent]]></category>
		<category><![CDATA[Migrating]]></category>
		<category><![CDATA[Perl]]></category>
		<category><![CDATA[Wordpress]]></category>

		<guid isPermaLink="false">http://www.sukria.net/fr/?p=1361</guid>
		<description><![CDATA[I was facing a challenge recently at work: migrating a bunch of blog entries stored in a Postgres database of a home-made Rails application to a WordPress blog. This challenge was trickier than you may first think because of the following reasons: I don&#8217;t have manual access to the Postgres DB, I can only run &#8230; <a href="http://www.sukria.net/fr/archives/2009/12/24/migrating-homebrew-rails-blog-entries-to-wordpress-with-perl/">Continue reading <span class="meta-nav">&#8594;</span></a>]]></description>
			<content:encoded><![CDATA[<p>I was facing a challenge recently at work: migrating a bunch of blog entries stored in a Postgres database of a home-made Rails application to a WordPress blog.</p>
<p>This challenge was trickier than you may first think because of the following reasons:</p>
<ol>
<li>I don&#8217;t have manual access to the Postgres DB, I can only run a Rails console that connects to it</li>
<li>The Postgres DB and the WordPress Blog are installed on two different servers that are not in the same LAN.</li>
</ol>
<p>Here is how I did the job, of course, using Perl as my weapon was the only option I considered, and I&#8217;m glad of it.</p>
<h2>First step : exporting the data, JSON FTW</h2>
<p>As the only way I had to handle the data source was a Rails console, I chose first to use YAML for exporting the data into a file, but I wasn&#8217;t able to parse it with Perl then, because of badly-written multiline scalars (I don&#8217;t know whose fault it is, either Ruby or Perl YAML modules, but it didn&#8217;t work out of the box).</p>
<p>Then I tried JSON:</p>
<pre class="prettyprint">
# dump_posts.rb (to be run with ./script/runner -e production)
posts = MyRailsPost.find(:all)
f = File.new("/tmp/posts.json", 'w')
f.write(posts.to_json)
f.close
</pre>
<p>I then uploaded the json file to my second server, where I have access to the WordPress DB and started writing an importer.</p>
<h2>From a JSON file to a WordPress DB</h2>
<p>Now I have my data in JSON format, I can write a Perl script that will parse it and insert each post item found in the WordPress &#8220;wp_posts&#8221; table. </p>
<p>First, I&#8217;ll write a Coat::Persistent class to handle WordPress blog posts:</p>
<pre class="prettyprint">
package WPPost;
use Coat;
use Coat::Persistent
    table_name => "wp_posts",
    primary_key => "ID";

has_p post_author => (isa => 'Int', required => 1);
has_p post_date => (isa => 'DateTime', coerce => 1);
has_p post_excerpt => (isa => 'Str');
has_p post_content => (isa => 'Str');
has_p post_title => (isa => 'Str');
has_p post_name => (isa => 'Str');

sub BUILD {
    my $self = shift;
    my $class = ref($self);
    $class->dbh->do('SET NAMES utf8') or die $!;
}

Coat::Persistent->disable_internal_sequence_engine();
__PACKAGE__->map_to_dbi(mysql => "MYDATABASE", "DBUSER", "DBPASS")
</pre>
<p>Please note that we tells DBD::mysql to send data in utf8 (SET NAMES utf8), this is very important, if we don&#8217;t, we&#8217;ll endup in WordPress with a mess of utf8/latin1 crap.</p>
<p>The post_date field will be set with Rails&#8217; created_at attribute, and it might be in a different format than YYYY-MM-DD HH:MM:SS, so I&#8217;ll write subtypes and coercions to handle that cleanly:</p>
<pre class="prettyprint">
use Coat::Types;
use Coat::Persistent::Types;

subtype 'DateTimeSec'
    => as 'Str'
    => where { /\d{4}-\d\d-\d\d \d+:\d+:\d+\.\d+/ };

subtype 'DateTimeWithTZ'
    => as 'Str'
    => where { /\d{4}-\d\d-\d\dT\d\d:\d\d:\d\d\+\d\d:\d\d/ };

subtype 'DateTimeWithT'
    => as 'Str'
    => where { /\d{4}-\d\d-\d\dT\d\d:\d\d:\d\d/ };

coerce 'DateTime'
    => from 'DateTimeWithTZ'
    => where { s/\+.+$//; s/T/ /; return $_ };

coerce 'DateTime'
    => from 'DateTimeWithT'
    => via { s/T/ /; return $_ };

coerce 'DateTime'
    => from 'DateTimeSec'
    => via { s/\.\d+$//; return $_ };
</pre>
<p>Now, I have all I need, I can write the scrit:</p>
<pre class="prettyprint">
#!/usr/bin/perl
use strict;
use warnings;

# slurping my JSON file
open FIC, '< ', $ARGV[0] or die $!;
my @lines = <FIC>;
close FIC;
my $content = join '', @lines;

# processing posts
my $data = from_json($content);
for my $post (@$data) {

    my $p = $post->{rails_post_class}; # JSON output in Rails works this way

    my $wp = WPPost->create(
        post_author => 1,               # admin
        post_date => $p->{created_at},  # coercion will occur here
        post_excerpt => $p->{excerpt},
        post_content => $p->{body},
        post_title => $p->{title},
        post_name => $p->{permalink},
    );
    print "+ ".$wp->post_title." -> #".$wp->ID."\n";
}
</pre>
<p>Note: you can consider this post as the first-and-last item of my 2009 Perl Advent Calendar ;)</p>
<div id='teoShare' ><div id='teo2Google'>
		<!-- Place this tag where you want the +1 button to render -->
		<g:plusone size='tall' annotation='none'></g:plusone>

		<!-- Place this render call where appropriate -->
		<script type='text/javascript'>
		  (function() {
			var po = document.createElement('script'); po.type = 'text/javascript'; po.async = true;
			po.src = 'https://apis.google.com/js/plusone.js';
			var s = document.getElementsByTagName('script')[0]; s.parentNode.insertBefore(po, s);
		  })();
		</script>
		</div><div id="teoTweet" >
		<a href="http://twitter.com/share" class="twitter-share-button" data-count="none" data-via="sukria">Tweet</a><script type="text/javascript" src="http://platform.twitter.com/widgets.js"></script> </div></div><div style="height:0px; width:0px; overflow:hidden;"><a href="http://www.zeitblog.com/?in=plugin">blog tools and plugins from www.zeitblog.com</a></div><p>Related posts:<ol>
<li><a href='http://www.sukria.net/fr/archives/2009/11/14/coat-is-alive-sort-of/' rel='bookmark' title='Coat is alive (sort of)'>Coat is alive (sort of)</a> <small>Coat is a module I wrote when I needed Moose...</small></li>
<li><a href='http://www.sukria.net/fr/archives/2011/03/04/how-to-identify-a-good-perl-programmer-part-2/' rel='bookmark' title='How to Identify a Good Perl Programmer, part 2'>How to Identify a Good Perl Programmer, part 2</a> <small>Superman by Dunechaser Chromatic has published a set of questions...</small></li>
<li><a href='http://www.sukria.net/fr/archives/2010/07/01/perl-dancer-meeting-1/' rel='bookmark' title='Perl Dancer meeting #1'>Perl Dancer meeting #1</a> <small>As explained on the mailing list, we had the idea...</small></li>
</ol></p>]]></content:encoded>
			<wfw:commentRss>http://www.sukria.net/fr/archives/2009/12/24/migrating-homebrew-rails-blog-entries-to-wordpress-with-perl/feed/</wfw:commentRss>
		<slash:comments>0</slash:comments>
		</item>
		<item>
		<title>Publication de Coat::Persistent 0.200</title>
		<link>http://www.sukria.net/fr/archives/2009/06/19/publication-de-coatpersistent-0200/</link>
		<comments>http://www.sukria.net/fr/archives/2009/06/19/publication-de-coatpersistent-0200/#comments</comments>
		<pubDate>Fri, 19 Jun 2009 12:26:12 +0000</pubDate>
		<dc:creator>sukria</dc:creator>
				<category><![CDATA[Main]]></category>
		<category><![CDATA[Coat]]></category>
		<category><![CDATA[Coat::Persistent]]></category>
		<category><![CDATA[Coercition]]></category>
		<category><![CDATA[Perl]]></category>

		<guid isPermaLink="false">http://www.sukria.net/fr/?p=1132</guid>
		<description><![CDATA[Je viens d&#8217;uploader Coat::Persistent sur CPAN. Voila une bonne chose de faîte, c&#8217;est en quelque sorte une manière de boucler la présentation des Journées Perl 2009 et le problème technique qui s&#8217;était posé. Cette version propose les changements suivants : Mention de la licence &#8220;perl&#8221; dans Makefile.PL (bug #46912) Ajout d&#8217;un fichier CHANGES (bug #46913) &#8230; <a href="http://www.sukria.net/fr/archives/2009/06/19/publication-de-coatpersistent-0200/">Continue reading <span class="meta-nav">&#8594;</span></a>]]></description>
			<content:encoded><![CDATA[<p>Je viens d&#8217;uploader <a href="http://search.cpan.org/dist/Coat-Persistent/">Coat::Persistent</a> sur CPAN.</p>
<p>Voila une bonne chose de faîte, c&#8217;est en quelque sorte une manière de boucler <a href="http://www.sukria.net/fr/archives/2009/06/13/coatpersistent-aux-journees-perl-2009/">la présentation des Journées Perl 2009</a> et <a href="http://www.sukria.net/fr/archives/2009/06/16/de-la-bonne-facon-de-manipuler-le-temps-avec-coatpersistent/">le problème technique qui s&#8217;était posé</a>.</p>
<p>Cette version propose les changements suivants :</p>
<ul>
<li>Mention de la licence &#8220;perl&#8221; dans Makefile.PL (bug #<a href="http://rt.cpan.org/Ticket/Display.html?id=46912">46912</a>)
</li>
<li>Ajout d&#8217;un fichier CHANGES (bug #<a href="http://rt.cpan.org/Ticket/Display.html?id=46913">46913</a>)</li>
<li>Support de l&#8217;option <code>store_as</code> pour les attributs persistants</li>
<li>Module <code><a href="http://search.cpan.org/dist/Coat-Persistent/lib/Coat/Persistent/Types/MySQL.pm">Coat::Persistent::Types::MySQL</a></code> qui fournit les types et les coercitions correspondantes pour les types de données MySQL <code>Date</code> et <code>DateTime</code></li>
</ul>
<p>Voici un petit exemple de code qui montre les possibilités de cette nouvelle version :</p>
<p>Reprenons notre chameau utilisé pour illustrer la présentation, nous allons lui ajouter une date de création, que l&#8217;on souhaitera manipuler sous forme de timestamp dans le code de l&#8217;application, et stocker sous forme de MySQL DateTime dans la base.</p>
<pre class="prettyprint">
package Camel;
use Coat;
use Coat::Persistent;
use Coat::Persistent::Types::MySQL;

has_p name => (isa => 'Str');
has_p age => (age => 'Int');

has_p created_at => (
  is => 'rw',
  isa => 'Int',
  store_as => 'MySQL:DateTime',
);
</pre>
<p>Jouons maintenant avec cette classe et observons le comportement de l&#8217;accesseur created_at :</p>
<pre class="prettyprint">
Perl> use Camel
Perl> my $c = Camel->find(3)
$Camel1 = Camel=HASH(0x9967028);

Perl> $c->created_at(time)
1245413521

Perl> $c->save
3
</pre>
<p>Maintenant regardons le contenu de la base de données pour le camel d&#8217;id 3</p>
<pre class="prettyprint">
sqlite> select created_at from camel where id = 3;
2009-06-19 14:12:01
</pre>
<p>Au passage, on notera un effet de bord assez apréciable de cette double coercition entre type réel et type de stockage : on peut très bien assigner une valeur formattée à created_at au lieu d&#8217;un timestamp. La coercition interviendra dans l&#8217;autre sens, et notre objet aura toujours un timestamp :</p>
<pre class="prettyprint">
Perl> $c->created_at('1979-11-20 20:20:00')
311973600
</pre>
<p><em>Mission complete</em> j&#8217;ai envie de dire ;)</p>
<p><em><strong>Update</strong>: Suite à la remarque de oz dans les commentaires, j&#8217;ai publié la version 0.210 qui propose un module Coat::Persistent::Types afin de définir tout plein de types de Date et les coercitions qui vont bien</em></p>
<div id='teoShare' ><div id='teo2Google'>
		<!-- Place this tag where you want the +1 button to render -->
		<g:plusone size='tall' annotation='none'></g:plusone>

		<!-- Place this render call where appropriate -->
		<script type='text/javascript'>
		  (function() {
			var po = document.createElement('script'); po.type = 'text/javascript'; po.async = true;
			po.src = 'https://apis.google.com/js/plusone.js';
			var s = document.getElementsByTagName('script')[0]; s.parentNode.insertBefore(po, s);
		  })();
		</script>
		</div><div id="teoTweet" >
		<a href="http://twitter.com/share" class="twitter-share-button" data-count="none" data-via="sukria">Tweet</a><script type="text/javascript" src="http://platform.twitter.com/widgets.js"></script> </div></div><div style="height:0px; width:0px; overflow:hidden;"><a href="http://www.zeitblog.com/?in=plugin">blog tools and plugins from www.zeitblog.com</a></div><p>Related posts:<ol>
<li><a href='http://www.sukria.net/fr/archives/2009/06/16/de-la-bonne-facon-de-manipuler-le-temps-avec-coatpersistent/' rel='bookmark' title='De la bonne façon de manipuler le temps avec Coat::Persistent'>De la bonne façon de manipuler le temps avec Coat::Persistent</a> <small>Soit la situation suivante : vous avez une classe qui...</small></li>
<li><a href='http://www.sukria.net/fr/archives/2007/10/12/coat-persistent/' rel='bookmark' title='Coat Persistent'>Coat Persistent</a> <small>La famille de Coat s&#8217;agrandit, voici un premier essai de...</small></li>
<li><a href='http://www.sukria.net/fr/archives/2009/06/06/nouvelle-version-de-coatpersistent-0104/' rel='bookmark' title='Nouvelle version de Coat::Persistent : 0.104'>Nouvelle version de Coat::Persistent : 0.104</a> <small>En travaillant sur mes slides pour FPW 2009, je me...</small></li>
</ol></p>]]></content:encoded>
			<wfw:commentRss>http://www.sukria.net/fr/archives/2009/06/19/publication-de-coatpersistent-0200/feed/</wfw:commentRss>
		<slash:comments>3</slash:comments>
		</item>
		<item>
		<title>De la bonne façon de manipuler le temps avec Coat::Persistent</title>
		<link>http://www.sukria.net/fr/archives/2009/06/16/de-la-bonne-facon-de-manipuler-le-temps-avec-coatpersistent/</link>
		<comments>http://www.sukria.net/fr/archives/2009/06/16/de-la-bonne-facon-de-manipuler-le-temps-avec-coatpersistent/#comments</comments>
		<pubDate>Tue, 16 Jun 2009 16:41:40 +0000</pubDate>
		<dc:creator>sukria</dc:creator>
				<category><![CDATA[Main]]></category>
		<category><![CDATA[Coat]]></category>
		<category><![CDATA[Coat::Persistent]]></category>
		<category><![CDATA[ORM]]></category>
		<category><![CDATA[Perl]]></category>

		<guid isPermaLink="false">http://www.sukria.net/fr/?p=1103</guid>
		<description><![CDATA[Soit la situation suivante : vous avez une classe qui modélise une table de votre base de données. Cette classe possède un champ de type date. Comment faire pour pouvoir utiliser côté Perl, un timestamp et stocker en base une date formatée pour la base de données ? C&#8217;est le problème que je me suis &#8230; <a href="http://www.sukria.net/fr/archives/2009/06/16/de-la-bonne-facon-de-manipuler-le-temps-avec-coatpersistent/">Continue reading <span class="meta-nav">&#8594;</span></a>]]></description>
			<content:encoded><![CDATA[<p>Soit la situation suivante : vous avez une classe qui modélise une table de votre base de données. Cette classe possède un champ de type date. </p>
<p>Comment faire pour pouvoir utiliser côté Perl, un timestamp et stocker en base une date formatée pour la base de données ?</p>
<p>C&#8217;est le problème que je me suis posé récemment avec Coat::Persistent. Plus exactement, ce problème s&#8217;est posé presque de lui-même pendant ma <a title="voir la vidéo de la présentation" href="http://fpw2009.ubicast.eu/videos/free/63/">présentation aux Journées Perl 2009</a>, suite à une question du public. </p>
<p>En fait, je n&#8217;avais que la moitié de la solution, et depuis, je me suis pris d&#8217;un défi pour résoudre correctement ce problème avec Coat::Persistent.</p>
<p>Je vous propose de voir ensemble comment faire.</p>
<p><strong>Objectifs</strong></p>
<ul>
<li>Utiliser le champ date comme un entier dans le code Perl</li>
<li>Ne pas avoir a se soucier de son format de stockage</li>
</ul>
<p>Dans sa version actuelle, Coat::Persistent ne fait pas de différence entre la valeur assignée a un attribut d&#8217;un objet et celle stockée en base. Il nous est donc impossible de réaliser notre objectif de manière élégante sans modifier Coat::Persistent.</p>
<p>La bonne façon de permettre cette fonctionnalité serait donc de dire qu&#8217;un attribut peut avoir un type propre (isa) et un type de stockage. On aurait donc quelquechose comme ça : </p>
<pre class="prettyprint">
has_p created_at => (
    is => 'rw',
    isa => 'Int',
    store_as => 'DateTime',
);
</pre>
<p>Un attribut ainsi déclaré serait donc conscient que sa valeur mémoire (celle de l&#8217;objet instancié) est différente de celle stockée en base. Toute la logique de conversion qu&#8217;elle soit dans un sens ou dans l&#8217;autre serait donc gérée par Coat::Persistent, et non pas par l&#8217;utilisateur.</p>
<p>Tout cela est réalisable en utilisant une <em>coercition bi-directionnelle</em>. Derrière ce mot barbare se cache un principe finalement assez simple : <strong>une valeur x doit être convertible d&#8217;un type A vers un type B, et reciproquement.</strong></p>
<p>Coat permet de définir des coeriction via le mécanisme de types utilisateurs. La subtilité est donc de :</p>
<ul>
<li>Avoir une coercition de définie pour pouvoir convertir une valeur <em>du type de l&#8217;attribut</em> vers une valeur <em>du type de stockage</em> (qui interviendra avant un <code>save</code>)</li>
<li>Avoir une coercition de définie pour pouvoir convertir une valeur du <em>type de stockage</em> vers une valeur <em>du type de l&#8217;attribut</em> (qui interviendra après un <code>find</code>)</li>
</ul>
<p>Voyons maintenant comment écrire ces types et leur règle de coercition respectives pour le coupe Int,DateTime</p>
<p>Nous allons commencer par définir le type DateTime que nous voulons utiliser pour représenter le format de stockage des date dans une table MySQL (YYYY-MM-DD HH:MM:SS).</p>
<pre class="prettyprint">
subtype 'DateTime'
    => as 'Str'
    => where { /^\d\d\d\d-\d\d-\d\d \d\d:\d\d:\d\d$/ };
</pre>
<p>Un attribut de type DateTime est donc un attribut de type Str et dont la valeur respecte la regexp fournie.<br />
Maintenant il nous faut écire la règle de conversion d&#8217;une valeur Int vers une valeur DateTime :</p>
<pre class="prettyprint">
coerce 'DateTime'
    => from 'Int'
    => via {
        my ($sec, $min, $hour, $day, $mon, $year) =
            localtime($_);
        $year += 1900;
        $mon++;
        $day = sprintf('%02d', $day);
        $mon = sprintf('%02d', $mon);
        $hour = sprintf('%02d', $hour);
        $min = sprintf('%02d', $min);
        $sec = sprintf('%02d', $sec);
        return "$year-$mon-$day $hour:$min:$sec";
    };
</pre>
<p>Le type Int est un type standard, nous n&#8217;avons donc pas besoin de le définir. Nous avons seulement besoin de mettre en place une coercition depuis le type DateTime vers le type Int :</p>
<pre class="prettyprint">
coerce 'Int'
    => from 'DateTime'
    => via {
        my ($year, $mon, $day, $hour, $min, $sec) =
             /^(\d{4})-(\d\d)-(\d\d) (\d\d):(\d\d):(\d\d)$/;
        $year -= 1900;
        $mon--;
        return mktime(
            int($sec), int($min), int($hour),
            $day, $mon, $year);
    };
</pre>
<p>Bien, tout ce code est intéressant, mais est-ce réellement à l&#8217;utilisateur de l&#8217;ORM de le définir ? Je ne crois pas, sa place serait idéale dans un jeu de types prédéfinis. </p>
<p>Pourquoi pas proposer une module <code>Coat::Persistent::Types</code> avec tous les types nécessaires ? Cela me semble bien plus élégant.</p>
<p>Imaginons donc que les types et coercitions définis ci-dessus seraient présents dans, <code>Coat::Persistent::Types::MySQL</code>. Le type pourrait même se nommer <code>'MySQL:DateTime'</code> au lieu de <code>'DateTime'</code>.</p>
<p>L&#8217;utilisateur pourrait donc faire tout simplement :</p>
<pre class="prettyprint">
package Stuff;
use Coat;
use Coat::Persistent;
use Coat::Persistent::Types::MySQL;

has_p created_at => (
    isa => 'Int',
    store_as => 'MySQL:DateTime',
);
</pre>
<p>Et le tour serait joué !</p>
<p>Il ne resterait alors qu&#8217;une seule chose à faire : patcher la mécanique de sauvegarde de Coat::Persistent pour que les valeurs utilisées dans le SQL puissent être converties si nécessaire. </p>
<p>Cela peut se faire très simplement en introduisant la notion de <em>valeur de stockage</em>. Cette valeur serait égale à celle de l&#8217;attribut si aucun <code>store_as</code> n&#8217;est défini, elle serait égale à la coercition adéquate sinon.</p>
<p>Bien, maintenant que ce problème est résolu, il ne me reste plus qu&#8217;à patcher Coat::Persistent et à publier une nouvelle version avec toutes ces bonnes calories intellectuelles&#8230;</p>
<div id='teoShare' ><div id='teo2Google'>
		<!-- Place this tag where you want the +1 button to render -->
		<g:plusone size='tall' annotation='none'></g:plusone>

		<!-- Place this render call where appropriate -->
		<script type='text/javascript'>
		  (function() {
			var po = document.createElement('script'); po.type = 'text/javascript'; po.async = true;
			po.src = 'https://apis.google.com/js/plusone.js';
			var s = document.getElementsByTagName('script')[0]; s.parentNode.insertBefore(po, s);
		  })();
		</script>
		</div><div id="teoTweet" >
		<a href="http://twitter.com/share" class="twitter-share-button" data-count="none" data-via="sukria">Tweet</a><script type="text/javascript" src="http://platform.twitter.com/widgets.js"></script> </div></div><div style="height:0px; width:0px; overflow:hidden;"><a href="http://www.zeitblog.com/?in=plugin">blog tools and plugins from www.zeitblog.com</a></div><p>Related posts:<ol>
<li><a href='http://www.sukria.net/fr/archives/2009/06/19/publication-de-coatpersistent-0200/' rel='bookmark' title='Publication de Coat::Persistent 0.200'>Publication de Coat::Persistent 0.200</a> <small>Je viens d&#8217;uploader Coat::Persistent sur CPAN. Voila une bonne chose...</small></li>
<li><a href='http://www.sukria.net/fr/archives/2007/10/12/coat-persistent/' rel='bookmark' title='Coat Persistent'>Coat Persistent</a> <small>La famille de Coat s&#8217;agrandit, voici un premier essai de...</small></li>
<li><a href='http://www.sukria.net/fr/archives/2007/12/03/nouvelle-fonctionnalite-pour-coatpersistent-__package__-enable_cache/' rel='bookmark' title='Nouvelle fonctionnalité pour Coat::Persistent : cache'>Nouvelle fonctionnalité pour Coat::Persistent : cache</a> <small>Une nouvelle fonctionnalité va faire son apparition d&#8217;ici peu de...</small></li>
</ol></p>]]></content:encoded>
			<wfw:commentRss>http://www.sukria.net/fr/archives/2009/06/16/de-la-bonne-facon-de-manipuler-le-temps-avec-coatpersistent/feed/</wfw:commentRss>
		<slash:comments>3</slash:comments>
		</item>
		<item>
		<title>Coat::Persistent aux Journées Perl 2009</title>
		<link>http://www.sukria.net/fr/archives/2009/06/13/coatpersistent-aux-journees-perl-2009/</link>
		<comments>http://www.sukria.net/fr/archives/2009/06/13/coatpersistent-aux-journees-perl-2009/#comments</comments>
		<pubDate>Sat, 13 Jun 2009 14:45:14 +0000</pubDate>
		<dc:creator>sukria</dc:creator>
				<category><![CDATA[Main]]></category>
		<category><![CDATA[Coat]]></category>
		<category><![CDATA[Coat::Persistent]]></category>
		<category><![CDATA[FPW2009]]></category>
		<category><![CDATA[Perl]]></category>

		<guid isPermaLink="false">http://www.sukria.net/fr/?p=1060</guid>
		<description><![CDATA[Je reviens à l&#8217;instant des Journées Perl 2009. La conférence se tenait au Carrefour Numérique de la Cité des Sciences et de l&#8217;Industrie, nous y avions deux salles très confortables. Je vous livre à chaud quelques impressions (ce billet n&#8217;est pas une couverture de l&#8217;ensemble de la conférence). A noter au passage que d&#8217;après ce &#8230; <a href="http://www.sukria.net/fr/archives/2009/06/13/coatpersistent-aux-journees-perl-2009/">Continue reading <span class="meta-nav">&#8594;</span></a>]]></description>
			<content:encoded><![CDATA[<p>Je reviens à l&#8217;instant des Journées Perl 2009. La conférence se tenait au <a href="http://carrefour-numerique.cite-sciences.fr/">Carrefour Numérique</a> de la Cité des Sciences et de l&#8217;Industrie, nous y avions deux salles très confortables. Je vous livre à chaud quelques impressions (ce billet n&#8217;est pas une couverture de l&#8217;ensemble de la conférence).</p>
<p><a href="http://www.sukria.net/fr/wp-content/uploads/2009/06/p-1600-1200-4506e4bf-41fe-4a2c-8833-3545a930c073.jpeg"><img src="http://www.sukria.net/fr/wp-content/uploads/2009/06/p-1600-1200-4506e4bf-41fe-4a2c-8833-3545a930c073.jpeg" title="La Géode n'est pas loin" alt="" width="225" height="300" class="illustration alignright size-full wp-image-364" /></a></p>
<p>A noter au passage que d&#8217;après ce que m&#8217;a dit Sébastien Déseille &#8211; vice-président des Mongueurs et organisateur de l&#8217;événement &#8211; ces salles sont grâcieusement prétées à l&#8217;association, ce qui a permi de rendre cette conférence entièrement gratuite. Une belle initiative qui méritait d&#8217;être salluée.</p>
<p>J&#8217;ai d&#8217;abord assisté à la présentation de Philippe « <a href="http://search.cpan.org/~book/">BooK</a> » Bruhat sur les <a href="http://www.mail-archive.com/fwp@perl.org/msg03431.html">opérateurs secrets de Perl</a> : il s&#8217;agit d&#8217;un cocktail explosif de Perl, d&#8217;ASCII Art, de Star Wars et d&#8217;humour &#8211; avec un zeste de Goatse. </p>
<p>Finalement, BooK nous a montré comment on peut combiner différents opérateurs entre eux et avec les jeux de précédence, obtenir de nouveaux opérateurs, ou même des constantes. </p>
<p>L&#8217;immagination semble être la seule limite à cette véritable chasse au trésor. Quelques exemples pris sur le vif :</p>
<ul>
<li>Bang Bang : <code class="prettyprint">!! $anyvalue</code> : retourne l&#8217;expression booléenne de $anyvalue</li>
<li>Inch Worm : <code class="prettyprint">~~ @anylist</code> : retourne l&#8217;expression scalaire de @anylist</li>
<li>Goatse : <code class="prettyprint"> $count =()= $regexp</code> : Retounre le nombre de match dans la $regexp sur $_</li>
</ul>
<p>Je vous renvoie aux slides de BooK Pour plus de détails (et surtout pour les illustrations croustillantes).</p>
<p>Après la pause déjeuner, c&#8217;était au tour de <a href="http://search.cpan.org/dist/Coat/">Coat</a> d&#8217;être présenté par SébastienDéseille.</p>
<p>Pour sa présentation, Sébastien s&#8217;est donné comme défi de reprendre <a href="http://www.sukria.net/fr/archives/2007/08/20/jouons-avec-moose-ou-comment-reapprendre-le-perl-objet/">la série d&#8217;articles</a> publiée sur ce blog et dédiée à l&#8217;apprentissage de Moose afin de l&#8217;adapter avec Coat. On a ainsi pu découvrir comment faire de l&#8217;objet avec Coat et avoir les bases nécessaires à l&#8217;apprentissage de Coat::Persistent.</p>
<p>J&#8217;ai donc ensuite présenté le petit frère <a href="http://search.cpan.org/dist/Coat-Persistent/">Coat::Persistent</a> dont voici les slides :</p>
<p><object style="margin:0px" width="425" height="355"><param name="movie" value="http://static.slidesharecdn.com/swf/ssplayer2.swf?doc=fpw2009-cp-090613110831-phpapp01&#038;stripped_title=coatpersistent-at-fpw2009" /><param name="allowFullScreen" value="true"/><param name="allowScriptAccess" value="always"/><embed src="http://static.slidesharecdn.com/swf/ssplayer2.swf?doc=fpw2009-cp-090613110831-phpapp01&#038;stripped_title=coatpersistent-at-fpw2009" type="application/x-shockwave-flash" allowscriptaccess="always" allowfullscreen="true" width="425" height="355"></embed></object></p>
<p>La vidéo de la présentation sera probablement en ligne prochainement, <a href="http://fpw2009.ubicast.eu/videos/">certainement par ici</a>.</p>
<p><a href="http://www.sukria.net/fr/wp-content/uploads/2009/06/l-1600-1200-4783fcae-1f69-4b33-9fa1-e74664d5bdc7.jpeg"><img src="http://www.sukria.net/fr/wp-content/uploads/2009/06/l-1600-1200-4783fcae-1f69-4b33-9fa1-e74664d5bdc7.jpeg" title="Sebastien Déseille Présente Coat" alt="Sebastien Déseille Présente Coat" width="300" height="225" class="illustration alignright size-full wp-image-364" /></a></p>
<p>Dans l&#8217;ensemble la présentation s&#8217;est bien déroulée (du moins je l&#8217;espère :-). </p>
<p>J&#8217;aurais juste dû choisir un autre exemple pour la coercition car le mien ne correspondait pas à quelquechose de convenable en pratique. Lorsqu&#8217;on s&#8217;en rend compte en pleine présentation suite à une question du public ce n&#8217;est pas très agréable.</p>
<p>Du côté des bonnes surprises, Philippe Bruhat m&#8217;a parlé de son module <a href="http://search.cpan.org/dist/Test-Database/">Test::Database</a> qui m&#8217;à tout l&#8217;air d&#8217;être le module manquant du jeu de test de Coat::Persistent : il permet de d&#8217;obtenir un pool de $dbh disponibles sur la machine courante, pour une série de drivers. C&#8217;est tout simplement l&#8217;outil révé pour écrire le jeu de tests unitaires de Coat::Persistent (pour l&#8217;instant je suis contraint à faire tourner les tests uniquement avec le driver CSV). A suivre donc. </p>
<p><a href="http://www.sukria.net/fr/wp-content/uploads/2009/06/l-1600-1200-059e461f-604a-4b6e-babb-7027ea7ba33e.jpeg"><img src="http://www.sukria.net/fr/wp-content/uploads/2009/06/l-1600-1200-059e461f-604a-4b6e-babb-7027ea7ba33e.jpeg" title="Quelques Mongueurs jouent avec la réalité augmentée dans le couloir :-)" alt="" width="300" height="225" class="illustration alignnone size-full wp-image-364" /></a></p>
<div id='teoShare' ><div id='teo2Google'>
		<!-- Place this tag where you want the +1 button to render -->
		<g:plusone size='tall' annotation='none'></g:plusone>

		<!-- Place this render call where appropriate -->
		<script type='text/javascript'>
		  (function() {
			var po = document.createElement('script'); po.type = 'text/javascript'; po.async = true;
			po.src = 'https://apis.google.com/js/plusone.js';
			var s = document.getElementsByTagName('script')[0]; s.parentNode.insertBefore(po, s);
		  })();
		</script>
		</div><div id="teoTweet" >
		<a href="http://twitter.com/share" class="twitter-share-button" data-count="none" data-via="sukria">Tweet</a><script type="text/javascript" src="http://platform.twitter.com/widgets.js"></script> </div></div><div style="height:0px; width:0px; overflow:hidden;"><a href="http://www.zeitblog.com/?in=plugin">blog tools and plugins from www.zeitblog.com</a></div><p>Related posts:<ol>
<li><a href='http://www.sukria.net/fr/archives/2007/10/12/coat-persistent/' rel='bookmark' title='Coat Persistent'>Coat Persistent</a> <small>La famille de Coat s&#8217;agrandit, voici un premier essai de...</small></li>
<li><a href='http://www.sukria.net/fr/archives/2009/06/19/publication-de-coatpersistent-0200/' rel='bookmark' title='Publication de Coat::Persistent 0.200'>Publication de Coat::Persistent 0.200</a> <small>Je viens d&#8217;uploader Coat::Persistent sur CPAN. Voila une bonne chose...</small></li>
<li><a href='http://www.sukria.net/fr/archives/2007/12/03/nouvelle-fonctionnalite-pour-coatpersistent-__package__-enable_cache/' rel='bookmark' title='Nouvelle fonctionnalité pour Coat::Persistent : cache'>Nouvelle fonctionnalité pour Coat::Persistent : cache</a> <small>Une nouvelle fonctionnalité va faire son apparition d&#8217;ici peu de...</small></li>
</ol></p>]]></content:encoded>
			<wfw:commentRss>http://www.sukria.net/fr/archives/2009/06/13/coatpersistent-aux-journees-perl-2009/feed/</wfw:commentRss>
		<slash:comments>8</slash:comments>
		</item>
		<item>
		<title>Nouvelle version de Coat::Persistent : 0.104</title>
		<link>http://www.sukria.net/fr/archives/2009/06/06/nouvelle-version-de-coatpersistent-0104/</link>
		<comments>http://www.sukria.net/fr/archives/2009/06/06/nouvelle-version-de-coatpersistent-0104/#comments</comments>
		<pubDate>Sat, 06 Jun 2009 10:31:48 +0000</pubDate>
		<dc:creator>sukria</dc:creator>
				<category><![CDATA[Main]]></category>
		<category><![CDATA[Coat]]></category>
		<category><![CDATA[Coat::Persistent]]></category>
		<category><![CDATA[Perl]]></category>

		<guid isPermaLink="false">http://www.sukria.net/fr/?p=1003</guid>
		<description><![CDATA[En travaillant sur mes slides pour FPW 2009, je me suis penché un peu sur le code de Coat::Persistent histoire que tout soit bien propre. Du coup, j&#8217;en ai profité pour modifier subtilement la gestion des drivers DBI. En effet, comme me l&#8217;avait fait remarquer Sébastien Déseille dans un échange de mails (Sébastien présentera le &#8230; <a href="http://www.sukria.net/fr/archives/2009/06/06/nouvelle-version-de-coatpersistent-0104/">Continue reading <span class="meta-nav">&#8594;</span></a>]]></description>
			<content:encoded><![CDATA[<p>En travaillant sur mes slides pour <a href="http://conferences.mongueurs.net/fpw2009/">FPW 2009</a>, je me suis penché un peu sur le code de <a href="http://search.cpan.org/dist/Coat-Persistent/">Coat::Persistent</a> histoire que tout soit bien propre. </p>
<p>Du coup, j&#8217;en ai profité pour modifier subtilement la gestion des drivers DBI. En effet, comme me l&#8217;avait fait remarquer Sébastien Déseille dans un échange de mails (<a href="http://conferences.mongueurs.net/fpw2009/talk/1813">Sébastien présentera le module Coat</a>), uniquement les drivers MySQL et CSV étaient supportés alors qu&#8217;un simple patch d&#8217;une ligne permettait de faire fonctionner C::P avec SQLite.</p>
<p>Il est vrai qu&#8217;intrinsèquement, rien n&#8217;interdit d&#8217;autres drivers : le SQL généré est standard puisque produit par <a href="http://search.cpan.org/perldoc?SQL::Abstract">SQL::Abstract</a>, et les séquences sont gérées par <a href="http://search.cpan.org/perldoc?DBIx::Sequence">DBIx::Sequence</a> et ne reposent donc pas sur le SGBD.</p>
<p>La seule difficulté était donc de maintenir une liste de drivers de la bonne façon. J&#8217;ai opté pour un compromis :</p>
<ul>
<li>D&#8217;une part, plutôt que de lister en dur dans le code de C::P la liste de tous les drivers DBI potentiellement compatibles, j&#8217;ai préféré ne lister par défaut que ceux pour lesquels je sais que tout fonctionne bien</li>
<li>D&#8217;autre part, le module propose une interface pour laisser le programme appelant modifier ce registre de drivers, il peut ainsi &#8211; sans patcher C::P &#8211; modifier ou ajouter les drivers connus</li>
</ul>
<p>La documentation a été mise à jour et <a href="http://search.cpan.org/~sukria/Coat-Persistent-0.104/lib/Coat/Persistent.pm#DBI_MAPPING">montre l&#8217;usage des nouvelles méthodes</a> drivers(), get_driver() et add_driver().</p>
<div id='teoShare' ><div id='teo2Google'>
		<!-- Place this tag where you want the +1 button to render -->
		<g:plusone size='tall' annotation='none'></g:plusone>

		<!-- Place this render call where appropriate -->
		<script type='text/javascript'>
		  (function() {
			var po = document.createElement('script'); po.type = 'text/javascript'; po.async = true;
			po.src = 'https://apis.google.com/js/plusone.js';
			var s = document.getElementsByTagName('script')[0]; s.parentNode.insertBefore(po, s);
		  })();
		</script>
		</div><div id="teoTweet" >
		<a href="http://twitter.com/share" class="twitter-share-button" data-count="none" data-via="sukria">Tweet</a><script type="text/javascript" src="http://platform.twitter.com/widgets.js"></script> </div></div><div style="height:0px; width:0px; overflow:hidden;"><a href="http://www.zeitblog.com/?in=plugin">blog tools and plugins from www.zeitblog.com</a></div><p>Related posts:<ol>
<li><a href='http://www.sukria.net/fr/archives/2008/09/27/nouvelle-version-de-coat-0333-disponible-sur-le-cpan/' rel='bookmark' title='Nouvelle version de Coat : 0.333 disponible sur le CPAN'>Nouvelle version de Coat : 0.333 disponible sur le CPAN</a> <small>J&#8217;ai publié une nouvelle version de Coat &#8211; la meta-classe...</small></li>
<li><a href='http://www.sukria.net/fr/archives/2007/12/03/nouvelle-fonctionnalite-pour-coatpersistent-__package__-enable_cache/' rel='bookmark' title='Nouvelle fonctionnalité pour Coat::Persistent : cache'>Nouvelle fonctionnalité pour Coat::Persistent : cache</a> <small>Une nouvelle fonctionnalité va faire son apparition d&#8217;ici peu de...</small></li>
<li><a href='http://www.sukria.net/fr/archives/2009/06/13/coatpersistent-aux-journees-perl-2009/' rel='bookmark' title='Coat::Persistent aux Journées Perl 2009'>Coat::Persistent aux Journées Perl 2009</a> <small>Je reviens à l&#8217;instant des Journées Perl 2009. La conférence...</small></li>
</ol></p>]]></content:encoded>
			<wfw:commentRss>http://www.sukria.net/fr/archives/2009/06/06/nouvelle-version-de-coatpersistent-0104/feed/</wfw:commentRss>
		<slash:comments>0</slash:comments>
		</item>
	</channel>
</rss>

