Programmation

Dancer 0.9903 released

I’ve been working a lot last days nights on Dancer. I’m quite pleased by the resulting package, even if it’s not complete yet. The provided features work well I must say, and there are an honnest amount of ideas in my head for the next moves.

This new release – 0.9903fixes a bug that was blocking Dancer from being installed with a Perl interpreter lower than 5.10.0. This bug was found by Enric Joffrion, and I thank him a lot for taking the time to report it.

New features are also provided in this release, like support for YAML configuration files and environments.
The test suite counts 208 tests and covers 78.1% of the source code.

If you feel in the appropriate mood, enjoy a dance!

I’ll be on vacation for the next three weeks, so don’t expect any new release until then.
See you in september ;)

Posted in Programmation 1 Comment »

The easiest way to build web applications with Perl

Choose the original!

Tags: ,
Posted in Internet, Programmation 2 Comments »

Suis-je maudit ?

On dirait qu’une sorte de malédiction pèse sur mes petites épaules de développeur Perl.

Lorsque j’ai parlé de l’idée de faire un Moose “light”, à savoir écrit en Perl 5 natif, on m’a rit au nez (surtout Matt Trout). J’ai alors écrit Coat dans une ambiance un peu bizarre où j’avais l’impression de faire cavalier seul.

Plus tard, les gens de Moose ont commencé le projet Mouse, qui n’était ni plus ni moins qu’un Moose sans dépendances, et cette fois-ci, mst leur donnait tout son soutien.

Cet épisode m’a un peu dégouté de l’équipe Moose, et j’ai plus ou moins coupé les ponts à ce moment là.

Je me suis dit que c’était une histoire de personnalités fortes, et j’ai laissé tomber. Mais aujourd’hui, je remarque qu’à peu de choses près, la même histoire vient de se produire avec mon petit dernier : Dancer, une réécriture de Sinatra en Perl :

Les gens de Mojolicious ont vu Dancer, puis l’ont réécrit en le renommant Mojolicious::Lite.

Je ne comprend pas très bien, suis-je maudit par le Dieu Perl ? Si c’est le cas j’aimerai qu’on me le dise…

Du coup je ne sais plus quoi faire avec Dancer, est-il condamné à disparaître ?

Tags:
Posted in Programmation 7 Comments »

Dancer 0.99 publié sur CPAN

La toute première version de Dancer vient d’être publiée. C’est une version destinée aux développeurs. Si des bonnes âmes ont envie de tester et de chercher des bugs, c’est maintenant :-)

http://search.cpan.org/~sukria/Dancer-0_0.99/lib/Dancer.pm

Tags: ,
Posted in Programmation 1 Comment »

Teaser…

Je suis tombé récemment sur Sinatra, sûrement grâce au blog de oz ou à celui de sa société, j’ai trouvé l’idée amusante, et même plutôt brillante.

Bon on ne se refait pas, je dois avoir chopé une sorte de maladie qui fait que j’aime réécrire des trucs en Perl, ça commence à devenir inquiétant diront certains, probablement, mais tant que ça m’amuse je ne vois pas pourquoi arrêter.

Bon venons-en au fait, j’ai un script bin/mawebapp.pl :

#!/usr/bin/perl

use Dancer;

get '/' => sub {
    "Hello There!"
};

get '/hello/:name' => sub {
    my $params = shift;

    "Hey ".$params->{name}.", how are you?";
};

Dancer->dance;

Je lance cet objet numérique étrange …

$ perl bin/mawebapp.pl
== Entering the dance floor ...
>> Listening on 0.0.0.0:8080

… Puis je joue avec

$ curl http://localhost:8080/
Hello There!

$ curl http://localhost:8080/hello/YouFreak
Hey YouFreak, how are you?

Merci oz !

Bientôt disponible sur github pour ceux qui veulent jouer avec moi c’est ici.

UPDATE 23 Juillet

Quelques corrections dans ce billet sur les exemple de code, pour un détail plus complet sur ce framework, je vous invite à lire le README ou à regarder l’application d’exemple fournie avec le source.

Tags: ,
Posted in Programmation 9 Comments »

Publication de Coat::Persistent 0.200

Je viens d’uploader Coat::Persistent sur CPAN.

Voila une bonne chose de faîte, c’est en quelque sorte une manière de boucler la présentation des Journées Perl 2009 et le problème technique qui s’était posé.

Cette version propose les changements suivants :

  • Mention de la licence “perl” dans Makefile.PL (bug #46912)
  • Ajout d’un fichier CHANGES (bug #46913)
  • Support de l’option store_as pour les attributs persistants
  • Module Coat::Persistent::Types::MySQL qui fournit les types et les coercitions correspondantes pour les types de données MySQL Date et DateTime

Voici un petit exemple de code qui montre les possibilités de cette nouvelle version :

Reprenons notre chameau utilisé pour illustrer la présentation, nous allons lui ajouter une date de création, que l’on souhaitera manipuler sous forme de timestamp dans le code de l’application, et stocker sous forme de MySQL DateTime dans la base.

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',
);

Jouons maintenant avec cette classe et observons le comportement de l’accesseur created_at :

Perl> use Camel
Perl> my $c = Camel->find(3)
$Camel1 = Camel=HASH(0x9967028);

Perl> $c->created_at(time)
1245413521

Perl> $c->save
3

Maintenant regardons le contenu de la base de données pour le camel d’id 3

sqlite> select created_at from camel where id = 3;
2009-06-19 14:12:01

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’un timestamp. La coercition interviendra dans l’autre sens, et notre objet aura toujours un timestamp :

Perl> $c->created_at('1979-11-20 20:20:00')
311973600

Mission complete j’ai envie de dire ;)

Update: Suite à la remarque de oz dans les commentaires, j’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

Tags: , , ,
Posted in Programmation 3 Comments »

De la bonne façon de manipuler le temps avec Coat::Persistent

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’est le problème que je me suis posé récemment avec Coat::Persistent. Plus exactement, ce problème s’est posé presque de lui-même pendant ma présentation aux Journées Perl 2009, suite à une question du public.

En fait, je n’avais que la moitié de la solution, et depuis, je me suis pris d’un défi pour résoudre correctement ce problème avec Coat::Persistent.

Je vous propose de voir ensemble comment faire.

Objectifs

  • Utiliser le champ date comme un entier dans le code Perl
  • Ne pas avoir a se soucier de son format de stockage

Dans sa version actuelle, Coat::Persistent ne fait pas de différence entre la valeur assignée a un attribut d’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.

La bonne façon de permettre cette fonctionnalité serait donc de dire qu’un attribut peut avoir un type propre (isa) et un type de stockage. On aurait donc quelquechose comme ça :

has_p created_at => (
    is => 'rw',
    isa => 'Int',
    store_as => 'DateTime',
);

Un attribut ainsi déclaré serait donc conscient que sa valeur mémoire (celle de l’objet instancié) est différente de celle stockée en base. Toute la logique de conversion qu’elle soit dans un sens ou dans l’autre serait donc gérée par Coat::Persistent, et non pas par l’utilisateur.

Tout cela est réalisable en utilisant une coercition bi-directionnelle. Derrière ce mot barbare se cache un principe finalement assez simple : une valeur x doit être convertible d’un type A vers un type B, et reciproquement.

Coat permet de définir des coeriction via le mécanisme de types utilisateurs. La subtilité est donc de :

  • Avoir une coercition de définie pour pouvoir convertir une valeur du type de l’attribut vers une valeur du type de stockage (qui interviendra avant un save)
  • Avoir une coercition de définie pour pouvoir convertir une valeur du type de stockage vers une valeur du type de l’attribut (qui interviendra après un find)

Voyons maintenant comment écrire ces types et leur règle de coercition respectives pour le coupe Int,DateTime

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).

subtype 'DateTime'
    => as 'Str'
    => where { /^\d\d\d\d-\d\d-\d\d \d\d:\d\d:\d\d$/ };

Un attribut de type DateTime est donc un attribut de type Str et dont la valeur respecte la regexp fournie.
Maintenant il nous faut écire la règle de conversion d’une valeur Int vers une valeur DateTime :

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";
    };

Le type Int est un type standard, nous n’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 :

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);
    };

Bien, tout ce code est intéressant, mais est-ce réellement à l’utilisateur de l’ORM de le définir ? Je ne crois pas, sa place serait idéale dans un jeu de types prédéfinis.

Pourquoi pas proposer une module Coat::Persistent::Types avec tous les types nécessaires ? Cela me semble bien plus élégant.

Imaginons donc que les types et coercitions définis ci-dessus seraient présents dans, Coat::Persistent::Types::MySQL. Le type pourrait même se nommer 'MySQL:DateTime' au lieu de 'DateTime'.

L’utilisateur pourrait donc faire tout simplement :

package Stuff;
use Coat;
use Coat::Persistent;
use Coat::Persistent::Types::MySQL;

has_p created_at => (
    isa => 'Int',
    store_as => 'MySQL:DateTime',
);

Et le tour serait joué !

Il ne resterait alors qu’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.

Cela peut se faire très simplement en introduisant la notion de valeur de stockage. Cette valeur serait égale à celle de l’attribut si aucun store_as n’est défini, elle serait égale à la coercition adéquate sinon.

Bien, maintenant que ce problème est résolu, il ne me reste plus qu’à patcher Coat::Persistent et à publier une nouvelle version avec toutes ces bonnes calories intellectuelles…

Tags: , , ,
Posted in Programmation 3 Comments »

Coat::Persistent aux Journées Perl 2009

Je reviens à l’instant des Journées Perl 2009. La conférence se tenait au Carrefour Numérique de la Cité des Sciences et de l’Industrie, nous y avions deux salles très confortables. Je vous livre à chaud quelques impressions (ce billet n’est pas une couverture de l’ensemble de la conférence).

A noter au passage que d’après ce que m’a dit Sébastien Déseille – vice-président des Mongueurs et organisateur de l’événement – ces salles sont grâcieusement prétées à l’association, ce qui a permi de rendre cette conférence entièrement gratuite. Une belle initiative qui méritait d’être salluée.

J’ai d’abord assisté à la présentation de Philippe « BooK » Bruhat sur les opérateurs secrets de Perl : il s’agit d’un cocktail explosif de Perl, d’ASCII Art, de Star Wars et d’humour – avec un zeste de Goatse.

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.

L’immagination semble être la seule limite à cette véritable chasse au trésor. Quelques exemples pris sur le vif :

  • Bang Bang : !! $anyvalue : retourne l’expression booléenne de $anyvalue
  • Inch Worm : ~~ @anylist : retourne l’expression scalaire de @anylist
  • Goatse : $count =()= $regexp : Retounre le nombre de match dans la $regexp sur $_

Je vous renvoie aux slides de BooK Pour plus de détails (et surtout pour les illustrations croustillantes).

Après la pause déjeuner, c’était au tour de Coat d’être présenté par SébastienDéseille.

Pour sa présentation, Sébastien s’est donné comme défi de reprendre la série d’articles publiée sur ce blog et dédiée à l’apprentissage de Moose afin de l’adapter avec Coat. On a ainsi pu découvrir comment faire de l’objet avec Coat et avoir les bases nécessaires à l’apprentissage de Coat::Persistent.

J’ai donc ensuite présenté le petit frère Coat::Persistent dont voici les slides :

La vidéo de la présentation sera probablement en ligne prochainement, certainement par ici.

Sebastien Déseille Présente Coat

Dans l’ensemble la présentation s’est bien déroulée (du moins je l’espère :-).

J’aurais juste dû choisir un autre exemple pour la coercition car le mien ne correspondait pas à quelquechose de convenable en pratique. Lorsqu’on s’en rend compte en pleine présentation suite à une question du public ce n’est pas très agréable.

Du côté des bonnes surprises, Philippe Bruhat m’a parlé de son module Test::Database qui m’à tout l’air d’être le module manquant du jeu de test de Coat::Persistent : il permet de d’obtenir un pool de $dbh disponibles sur la machine courante, pour une série de drivers. C’est tout simplement l’outil révé pour écrire le jeu de tests unitaires de Coat::Persistent (pour l’instant je suis contraint à faire tourner les tests uniquement avec le driver CSV). A suivre donc.

Tags: , , ,
Posted in Programmation 8 Comments »

Nouvelle version de Coat::Persistent : 0.104

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’en ai profité pour modifier subtilement la gestion des drivers DBI. En effet, comme me l’avait fait remarquer Sébastien Déseille dans un échange de mails (Sébastien présentera le module Coat), uniquement les drivers MySQL et CSV étaient supportés alors qu’un simple patch d’une ligne permettait de faire fonctionner C::P avec SQLite.

Il est vrai qu’intrinsèquement, rien n’interdit d’autres drivers : le SQL généré est standard puisque produit par SQL::Abstract, et les séquences sont gérées par DBIx::Sequence et ne reposent donc pas sur le SGBD.

La seule difficulté était donc de maintenir une liste de drivers de la bonne façon. J’ai opté pour un compromis :

  • D’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’ai préféré ne lister par défaut que ceux pour lesquels je sais que tout fonctionne bien
  • D’autre part, le module propose une interface pour laisser le programme appelant modifier ce registre de drivers, il peut ainsi – sans patcher C::P – modifier ou ajouter les drivers connus

La documentation a été mise à jour et montre l’usage des nouvelles méthodes drivers(), get_driver() et add_driver().

Tags: , ,
Posted in Programmation Comments Off

OpenURI de Ruby n’est pas sortable

Je réalise avec stupeur que la lib open-uri.rb, installée sur mon système crée systématiquement des fichiers temporaires et ne les efface jamais.

Comme j’ai chopé la crêve et que je suis, du coup, facilement irritable je vais donc troller :

La communauté Ruby est composée d’ados pré-puberts ou quoi ?

Enfin bref, une lib avec un nom pareil, installée dans /usr/lib, ça se doit au minimu de se comporter proprement, non ?

Le pire, c’est que personne n’en parle sur Google ; à croire que je suis encore un vieux con réac qui se choque pour pas grand chose…

Tags: , ,
Posted in Programmation 4 Comments »

Get Adobe Flash playerPlugin by wpburn.com wordpress themes