vendredi, juillet 27, 2012

Petit test Boostrap, Less & nouveau tag HTML5

Pour tester un peu boostrap, less et html5 (les nouveaux tag header, footer, section, article et aside), j'ai d'abord créé une page simple: index-boostrap.html dans laquelle j'ai utilisé boostrap 'normalement'.
C'est à dire quelque chose ça: https://gist.github.com/3189167#file_index_boostrap.html

Puis j'ai essayé de faire disparaitre ces 'class' en transférant les styles sur les tags html5 dans la css. Pour cela j'ai dû partir du code less de boostrap pour ajouter mes nouvelles définitions dans styles.less.
Ce qui donne un html5 plus 'pur'

Voir: https://gist.github.com/3189167#file_styles.less et https://gist.github.com/3189167#file_index.html


Le résultat est assez proche à quelques pixel près ;)
Par contre:
  • la transformation est assez laborieuse et chronophage
  • maintenant ma page subit une transformation less (1 indirection de plus)
  • j'ai laissé tombé la transformation de la navbar (par manque d'envie / temps...)
Donc je conseille plutôt d'utiliser boostrap tel quel si c'est possible et si on a le choix :)

Le projet complet est disponible pendant 30j ici: http://dl.free.fr/qHx7j2rXo

mercredi, avril 28, 2010

Compter le nombres de lignes dans des fichiers

Un petit bout de code pour compter le nombre de lignes par fichier et envoyer le résultat dans un fichier csv pour pouvoir trier et faire de beaux diagrammes excel pour votre chef ;)
require 'fileutils'

File.open("result.csv", File::CREAT|File::RDWR) do |csv_file|
  Dir["JSPs/**/*.jsp"].each do |file|
    lines = File.readlines(file)
    csv_file << "#{file}\t#{lines.length}\n"
  end
end

jeudi, avril 15, 2010

Delicious linkchecker

Je voulais faire un peu de ménage dans mes 2547 Bookmarks delicious et supprimer ou mettre à jour les liens mais aucun outil "officiel" cf le forum.
Voici un petit bout de code qui permet de vérifier tous les bookmarks: http://gist.github.com/366112
Ca m'a permis de faire mon premier "gist", c'est mieux que de créer un projet juste pour ça.
Résultats du nettoyage: Quelque liens 404 (pas trop d'ailleurs) surtout des 301 moved permanently généralement le www à disparu de l'url :)
Technorati tags:

mardi, mars 23, 2010

Query Browser pour MongoDB

Comment chercher dans MongoDB des données (un peu comme mysql en ligne de commande) ?
Très simple utiliser irb et la lib ruby et suivre le tutorial: http://www.mongodb.org/display/DOCS/Ruby+Tutorial#RubyTutorial-AQuickTour
irb(main):001:0> require 'rubygems'
=> true
irb(main):002:0> require 'mongo'
=> true
irb(main):004:0> m = Mongo::Connection.new
=> #
irb(main):005:0> m.database_names.each { |name| puts name }
local
admin
my_db
=> ["local", "admin", "my_db"]
irb(main):008:0> db = Mongo::Connection.new.db("my_db")
=> #
irb(main):009:0> db.collection_names.each { |name| puts name }
people
system.indexes
=> ["people", "system.indexes"]
irb(main):011:0> coll = db["people"]
irb(main):012:0> coll.find.each { |item| puts item.inspect }
(...)
{"_id"=>ObjectID('4ba96c2360d99e0ba1000019'), "birthdate"=>nil, "phone_numbers"=>[],
 "last_name"=>"bar2", "login"=>"foobar2", "password"=>"secret", "first_name"=>"foo2"}
=> nil 
Technorati tags:

lundi, septembre 14, 2009

Zlib::BufError et repository de gems down

J'ai aidé aujourd'hui quelqu'un qui n'arrivait pas à installer watir sous windows, pourtant il faisait tout comme il faut ?! (install ruby one click installer + "gem install watir")
Il avait toujours le message d'erreur suivant: Zlib::BufError buffer error
En essayant sur ma machine, qui avait déjà watir installé, je me suis rendu compte que j'avais le même problème!
C:\>gem update watir --debug
Exception `NameError' at c:/ruby/lib/ruby/site_ruby/1.8/rubygems/command_manager.rb:161 - uninitialized constant Gem::Commands::UpdateCommand
Exception `Gem::LoadError' at c:/ruby/lib/ruby/site_ruby/1.8/rubygems.rb:827 - Could not find RubyGem test-unit (>= 0)

Updating installed gems
Exception `Gem::LoadError' at c:/ruby/lib/ruby/site_ruby/1.8/rubygems.rb:827 - RubyGem version error: sources(0.0.1 not > 0.0.1)

Exception `Zlib::BufError' at c:/ruby/lib/ruby/site_ruby/1.8/rubygems.rb:578 - buffer error
ERROR:  While executing gem ... (Zlib::BufError)
    buffer error
        c:/ruby/lib/ruby/site_ruby/1.8/rubygems.rb:578:in `read'
        c:/ruby/lib/ruby/site_ruby/1.8/rubygems.rb:578:in `gunzip'
        c:/ruby/lib/ruby/site_ruby/1.8/rubygems/remote_fetcher.rb:165:in `fetch_path'
        c:/ruby/lib/ruby/site_ruby/1.8/rubygems/spec_fetcher.rb:219:in `load_specs'
        c:/ruby/lib/ruby/site_ruby/1.8/rubygems/spec_fetcher.rb:192:in `list'
        c:/ruby/lib/ruby/site_ruby/1.8/rubygems/spec_fetcher.rb:188:in `each'
        c:/ruby/lib/ruby/site_ruby/1.8/rubygems/spec_fetcher.rb:188:in `list'
        c:/ruby/lib/ruby/site_ruby/1.8/rubygems/spec_fetcher.rb:123:in `find_matching'
        c:/ruby/lib/ruby/site_ruby/1.8/rubygems/commands/update_command.rb:169:in `which_to_update'
        c:/ruby/lib/ruby/site_ruby/1.8/rubygems/commands/update_command.rb:161:in `each'
        c:/ruby/lib/ruby/site_ruby/1.8/rubygems/commands/update_command.rb:161:in `which_to_update'
        c:/ruby/lib/ruby/site_ruby/1.8/rubygems/commands/update_command.rb:75:in `execute'
        c:/ruby/lib/ruby/site_ruby/1.8/rubygems/command.rb:257:in `invoke'
        c:/ruby/lib/ruby/site_ruby/1.8/rubygems/command_manager.rb:132:in `process_args'
        c:/ruby/lib/ruby/site_ruby/1.8/rubygems/command_manager.rb:102:in `run'
        c:/ruby/lib/ruby/site_ruby/1.8/rubygems/gem_runner.rb:58:in `run'
        c:/ruby/bin/gem:21
Je me suis douté que le serveur de gems était down même si le message d'erreur n'est pas clair du tout!
J'ai trouvé une astuce! On peut transformer sa machine en serveur de gem (mirroir) en lançant la commande:
gem server
On peut voir le résultat aux addresses suivantes:
L'autre personne a juste eu à faire:
gem install watir --source http://xxx.xxx.xxx.xxx:8808/
où xxx.xxx.xxx.xxx est l'ip de la machine où "gem server" a été lancé
Rem: je n'ai pas trouvé l'url que rubygems utilise mais je pense qu'il s'agit de http://gems.rubyforge.org/ et effectivement http://gems.rubyforge.org/gems renvoie une 403 forbidden :(
Technorati tags:

mardi, septembre 08, 2009

Javascript: Orienté Objet

J'ai enfin trouver le temps de regarder comment "bien" programmer en Javascript :)

Je ne suis pas sûr que ça me serve beaucoup, je suis plutôt un "script kiddies" du javascript. Générallement je récupère une lib, je fais quelques lignes de javascript pour faire un petit effet (fade...) et c'est tout!

Mais par curiosité personnelle, je voulais voir comment Javascript gérait l'héritage etc...

En fait c'est assez simple, il suffit de définir une classe et d'utiliser le mot clef "prototype" pour dire que la classe fille hérite de la classe mère.
Je trouve que le concept d'héritage en Javascript ressemble beaucoup à ce qui se passe en ruby avec le mot clef "include".

En tout cas je vous conseille le tutorial que j'ai fait et que je trouve très bien:

Technorati tags:

mercredi, août 26, 2009

Disparition de why et Ma présence online

Je viens de découvrir que why (un fameux codeur ruby) a "disparu".

Il aurait apparemment supprimé ses comptes en ligne (twitter, blogs etc...) http://ejohn.org/blog/eulogy-to-_why/


J'y pense de plus en plus également pour une raison assez simple: je sens que j'ai de moins en moins la maîtrise de ce qui est publié sur moi.

J'ai toujours choisi de parler de sujet "professionnel" mais avec l'avènement de réseaux tels que facebook, google social api, web semantic... font que les "fuites" sur ma vie privée sont de plus en plus nombreuses.

Typiquement je n'ai pas envie de voir ma famille, mes amis ou même mes vacances sur internet...
Globalement ces informations sont noyées dans la masse mais si on veut on peut trouver déjà beaucoup (trop) de choses sur moi...

Tout comme simon, j'essaye de dissocié au maximum vie privée / vie publique

En plus ces informations peuvent être obsolète ou même completement fausse comme par exemple sur 123people, cf mon post sur le sujet.

mardi, juillet 21, 2009

Découverte de Python

Cela faisait un petit moment que je voulais apprendre les bases de python... Quelques remarques que j'ai notées pendant que je déroulais le tutorial officiel
Général
  • Le nom Python vient des 'Monthy Python' pas du serpent > ce qui rend le laguage tout de suite plus sympatique ;)
  • Ca semble s'inspirer assez du C
    Ex: le print() me rappelle le print du C avec des améliorations
  • Utilisation de l'indentation plutôt que les {}...
    pourquoi pas, au moins ça évite les troll interminables sur les { à la fin d'une ligne ou au début de la ligne suivante :p
    Troll qui a dû au moins me prendre 2 journées de ma vie si on cumule toutes les discussions inutiles
  • Les types de base : list (qui sont aussi des stack, queue), set, dictionary (alias map en java) mais je n'ai pas compris la difference avec les tuples
  • Je trouve étrange d'introduire les complexes (i) dans le tutorial ?! Même si cette fonctionnalité est intéressante, elle ne sert pas tout les jours (jamais ?!)
Les plus
  • List/Tableau = même combat
    Contrairement à Java ou les 2 types de classes sont clairement séparées
    En Python ou Ruby la différence à beaucoup moins d'importance et c'est mieux
  • On peut itérer dans les chaines de caractères comme dans un tableau:
b = 'azerty'
for x in b:
print(x)
  • String indice ou slice
    Un peu surpris au début par la manipulation de chaines avec les indices (slices) mais on s'y habitue bien :)
    Et même ça me semble assez puissant ! (a voir si c'est lisible dans un programme plus complexe ou écrit par quelqu'un d'autre...)
>>> word = 'Help' + 'A'
>>> word[4]
'A'
>>> word[0:2]
'He'
>>> word[2:4]
'lp'
>>> word[:2]    # The first two characters
'He'
>>> word[2:]    # Everything except the first two characters
'lpA'
  • Une boucle for peut avoir un else
    Ca semble intéressant mais si je ne vois pas bien l'utilité (cas où il y a un break dans le for)
for x in range(2, n):
if n % x == 0:
  print(n, 'equals', x, '*', n//x)
  break
else:
  # loop fell through without finding a factor
  print(n, 'is a prime number')
Les moins
  • Pas de switch / case mais if ... elif ... elif ...
    Je trouve que les switch/case (même si je ne les utilise pas souvent) expriment mieux ce qu'on veut faire dans ces cas là et la synthaxe est plus concise...
    En java j'ai tendance à considérer pleins de elseif comme un code smell qu'il faudrait autre chose: un switch/case dans le cas les plus simple ou un pattern type builder ou command pour les cas les plus complexe
    Peut-etre juste un habitude à prendre
  • Ruby semble plus orienté object, exemple taille d'une chaine de char
python: len('azerty')
ruby: 'azerty'.size

mercredi, juin 17, 2009

Résistance sociale & langage de script

Justement en lisant cet article: "Why ? Language Archaelogy..." de Bruce Eckel, ça fait écho à ce que disait dans mon post précédent à propos de la résistance sociale lié au langage de script:
Like Smalltalk, everything in Ruby is fungible (this is the very thing that makes dynamic languages so scary to folks who are used to static languages)

lundi, juin 15, 2009

Ruby at ThoughtWorks

Excellent article (comme presque toujours) de Martin Fowler: Ruby at ThoughtWorks
Le sommaire devrait suffire pour vous donner envie de le lire:
Contents
    * The Shape of Our Projects
    * Questions About Ruby
          o Was Ruby the Right Choice?
          o Is Ruby More Productive?
          o Is Ruby Slow?
          o Is a Ruby Code-base Hard to Understand?
          o Is Ruby a Viable Platform
    * Some Development Tips
          o Testing with Active Record
          o Active Record Leaks
          o Long Running Requests
          o Deployment
          o Controlling Gems
          o Schedule Time for Updates
          o Developing on Windows
Les points qui m'ont interessé au vue de mon expérience avec ruby:
  • Développer sous Windows, c'est faisable mais pas évident (c'est d'autant plus frustrant lorsqu'on a fait beaucoup de java et que globalement toutes les lib en java sont indépendantes de l'os).
    la plus grande partie des problèmes que j'ai rencontré vient de l'utilisation dans des scripts Rake de commande système qui n'existent que sous *nix (ssh, etc...).
    Oui il y a cygwin mais bof
  • Le problème de l'adoption de ruby en entreprise est lié à 2 facteurs:
    langage de script perçu comme impossible à maintenir par rapport à un langage compilé (alors que j'ai tant de fois vu des programmes java impossible à maintenir tellement ils sont mal écrit...) et
    résistance sociale au changement (peur de se lancer dans une techno s'il n'y a pas eu déjà 200 articles dans "01 informatique")

lundi, mai 18, 2009

Ruby sous PSPad

J'utilise PsPad comme éditeur de fichier, il y en a pleins d'autres mais j'aime bien celui-là, je trouve qu'il marche bien ;)
Par contre je n'avais pas la coloration syntaxique pour ruby mais en fait tout est déjà fait, il suffit juste de l'activer!
L'explication est disponible sur le forum de PsPad.
En images ça donne :






mercredi, mai 13, 2009

Railsconf 2009 (Trop ?)

En regardant cette vidéo résumant la Railsconf 2009, j'ai plusieurs impressions contradictoires...
Les positives:
  1. il y a beaucoup d'activité dans la communauté rails
  2. il y a plein de sujets très variés qui sont adressés (donc l'argument comme quoi java possède plus de librairies que ruby s'effrite de plus en plus)
  3. La communauté grandit de plus en plus et donc les chances de faire du ruby tous les jours au boulot augmente
Les négatives:
  1. Ca part dans tous les sens
  2. Tout le monde essaye d'inventer son framework pour avoir un peu de gloire
  3. La plupart des frameworks/lib présentés vont tomber dans l'oubli dans l'année qui arrive
  4. Je n'ai pas le temps (et pas l'envie) de suivre tout ce qui se passe
  5. On est plus a l'époque des pionniers où il fallait créer soit même les librairies de base; ce temps est révolue (nostalgie)
En résumé, il y a trop de choses et je n'ai pas le temps / l'envie de suivre tout ça...

Je vais me limité à ce que je fais actuellement c'est à dire suivre les sujets qui m'intéressent au moment où j'en ai besoin pour me libérer du temps pour autre chose que l'informatique ;)

Sinon la video est bien mais un peu trop longue (34 min)

Un sujet qui a attiré mon attention un peu plus est jruby sur google app engine.

mardi, mars 31, 2009

PragDave@github

Je viens juste de découvrir qu'un de mes "guru" favori a un compte sur github ;) Hop "Follow" ! Je sens qu'il va s'y passer des choses intéressantes: "comments as rake desc" :)

lundi, janvier 26, 2009

Différences entre exec et system

Je n'aime pas trop la notation qui utilise les `` pour lancer une commande système parce que:
  • c'est facile de passer à côté et de les confondre avec un ' normal
  • parce que pour un novice c'est difficile à comprendre
  • etc. ...
Jusqu'à présent j'utilisais exec mais en voulant l'utiliser dans une boucle je me suis rendu compte que mon programme ruby s'arrêtait juste après l'exécution (sous windows mais je pense que c'est pareil sur d'autres os).
Dir["folder/foo_*.txt"].each do |path|
    cmd = "echo #{path}"
    exec(cmd)
end
Si on veut que le programme ruby ne soit pas interompu (la plupart du temps) il faut en fait utiliser system.
Dir["folder/foo_*.txt"].each do |path|
    cmd = "echo #{path}"
    system(cmd)
end

mercredi, novembre 26, 2008

Changer l'extension d'un fichier

Un petit bout de script qui m'a fait économiser pas mal de temps ;)
Change l'extension de tous les fichiers d'un répertoire et ses sous-répertoires (ici de .kml en .xml).
Ne fonctionne pas si vos noms de fichiers contiennent des '.'

require 'fileutils.rb'

Dir['**/*.kml'].each do |file|
    puts file
    filename = file.split('.')[0]
    # FileUtils.cp("#{filename}.kml", "#{filename}.xml")
    # FileUtils.mv("#{filename}.kml", "#{filename}.xml")
end

jeudi, juillet 10, 2008

Rubik’s Cube a 25 ans

Vu le nom que j'ai choisi pour mon blog, je me devais de fêter les 25 ans du Rubik's Cube, le vrai!
Une campagne de pub pour l'occasion: Article ViaComIT.

mercredi, juin 25, 2008

LinkedIn la plus grosse application rails au monde ?

Plus précisément "BumperSticker" de LinkedIn.
C'est en tout cas ce que les gens de LinkedIn proclament sur leur blog (relayé par Matt Raible connu pour ses comparaisons de framework web java).
En tout cas si la video dit vrai: "Scaling Rails to 1 Billion Pages a Month"... c'est impressionnant!
Et surtout ça prouve que rails est capable de supporter la charge... (après tout dépend du talent des developpeurs qui font l'application).
Technorati tags:

mercredi, juin 04, 2008

Ruby Riddle: "Trouver le nom de la nouvelle société où je travaille"

J'ai posé ma démission de Anyware Technologies depuis quelques temps maintenant mais j'ai gardé le nom de la nouvelle ssii secrète pour l'instant :p
Pour favoriser les personnes que je connais qui ont installé Ruby sur leur machine et pour pousser ceux qui sont curieux à le faire :D
Voici un petit programme ruby en forme de devinette.
old_job = "anyware"
puts "old job: #{old_job}"
magic = 345756
new_job = old_job[0, 1]
magic.times {|j| new_job = new_job.next}
puts "new job: #{new_job.upcase}"
Il suffit de l'exécuter pour avoir la réponse ;)
Je posterai l'annonce officielle sur mon autre blog demain ;)

lundi, avril 28, 2008

Ruby à l'Université

Comme rien n'échappe à Google et à ses "Google Alerts", je viens de découvrir un cours à l'Université de Toulouse II - Le Mirail qui utilise Ruby comme langage de programmation pour les débutants:
Une bien bonne nouvelle et une excellente initiative :)
Si vous doutez de l'adéquation entre Ruby et Débutants, je vous conseille de jeter un coup d'oeil au livre: "Learn to Program"
Technorati tags:

lundi, mars 31, 2008

Experimentations avec les Mixins / Modules et les méthodes statiques (self)

On m'a posé une question aujourd'hui: "Comment définir une méthode statique dans un module et l'inclure dans une classe ?". Simple non ?!
Justement j'avais eu le même problème quand j'avais fait mon projet "natural_sort"
Spontanément j'ai essayé:

module Carnivore
  def self.manger(oiseau)
    puts "slurp! #{oiseau}"
  end
end

class GrosMinet
  include Carnivore
end

Carnivore.manger "titi" # ok ca foncionnne
GrosMinet.manger "titi" # undefined method `manger' for GrosMinet:Class (NoMethodError) WTF ?!
Malheureusement ça ne fonctionne pas :(

Dans NaturalSort, j'ai contourné le problème:

module NaturalSort
  def self.naturalsort(object)
    ...
  end
  def natural_sort
    NaturalSort::naturalsort(to_a)
  end
end
Je suis tombé sur cet article: "Ruby Mixin Tutorial" et j'ai essayé différentes combinaisons, il n'y en pas une qui soit satifaisante dans tous les cas...
Ici vous pourrez voir les principales étapes jusqu'à la version finale, celle qui me satisfait le plus.
Etape n°1: classe statique dans le module uniquement

module Carnivore
  def self.manger(oiseau)
    puts "slurp! #{oiseau}"
  end
end

Carnivore.manger "titi"
Etape n°2: méthode d'instance de classe (l'utilisation classique du module par excellence):

module Carnivore
  def manger(oiseau)
    puts "slurp! #{oiseau}"
  end
end

class GrosMinet
  include Carnivore
end

gros_minet = GrosMinet.new 
gros_minet.manger "titi"
Version finale. 2 grosses subtilités:
  • il n'y a pas de 'self' dans la définition de la méthode 'manger'
  • On utilise 'extends' au lieu de 'include'

module Carnivore
  def manger(oiseau)
    puts "slurp! #{oiseau}"
  end
end

module Chat
  def miauler
    puts "miaou..."
  end
end

class GrosMinet
  include Chat
  # include Carnivore # necessaire à gros_minet.manger("titi")
  extend Carnivore
end

GrosMinet.manger "titi"

gros_minet = GrosMinet.new 
gros_minet.miauler
# gros_minet.manger "titi" # necessite le include Carnivore pour fonctionner

Conclusion: Ca fonctionne et le code est court mais je trouve la syntax un peut étrange, je ne comprend pas trop pourquoi le 'include' n'inclus pas les méthodes 'self' ? il y a quelquechose qui m'échappe pour l'instant...

Du coup je pense que c'est une bonne pratique de ne pas utiliser de méthodes préfixé par 'self' dans le modules et que faire MonModule.ma_methode n'est peut-être pas très orienté objet... après tout les modules ne sont pas vraiment des objets et ne devraient pas être utilisés seuls ? Qu'est ce que vous en pensez ?


Technorati tags:

mercredi, mars 19, 2008

Rails 2 - Renommer des .rjs en .js.rjs

Suite à ma migration en rails 2.0, j'avais renommé mes .rjs en .js.rjs mais les pages ne fonctionnaient plus :(
C'est parce qu'il faut ajouté le block respond_to dans la méthode qui appelle l'ajax dans le controller:
  # Ajax response for validation
  def validate_new_holiday
    @holiday = self.init_holiday
    respond_to do |format|
      format.js      # validate_new_holiday.js.rjs
    end
  end 
Technorati tags:

mardi, mars 18, 2008

Migration "Pas à pas" d'une application ruby on rails de 1.2.6 vers 2.0.2 (part 2)

Dans le post précédent nous avons vu comment migrer de 1.2.2 vers 1.2.6
Maintenant nous allons passer de 1.2.6 à 2.0.2


Je ne l'ai pas dit dans mon post précédent tellement ça me paraissait évident, mais utiliser systématiquement subversion (ou un autre système: cvs, git...) pour pouvoir revenir en arrière. C'est indispensable !!


> rake rails:unfreeze

> svn delete vendor/rails

> gem list rails

*** LOCAL GEMS ***

rails (1.2.6)

> gem update rails
Successfully installed rails-2.0.2
1 gem installed
Gems updated: rails

> gem list rails

*** LOCAL GEMS ***

rails (2.0.2, 1.2.6)

> edit config/environment.rb

Changer
RAILS_GEM_VERSION = '1.2.6' unless defined? RAILS_GEM_VERSION
En:
RAILS_GEM_VERSION = '2.0.2' unless defined? RAILS_GEM_VERSION

>ruby script\about

      *******************************************************************
      * config.breakpoint_server has been deprecated and has no effect. *
      *******************************************************************

About your application's environment
Ruby version              1.8.6 (i386-mswin32)
RubyGems version          1.0.1
Rails version             2.0.2
Active Record version     2.0.2
Action Pack version       2.0.2
Active Resource version   2.0.2
Action Mailer version     2.0.2
Active Support version    2.0.2
Application root          C:/developement/ruby/appli
Environment               development
Database adapter          mysql
Database schema version   8

Si vous avez le même warning que moi, éditer le fichier development.rb et enlevé les lignes suivantes:

# Enable the breakpoint server that script/breakpointer connects to
config.breakpoint_server = true
puis installer ruby-debug comme expliqué ici: Fixing-config-breakpoint-server

> rake rails:update

modifie 
  config/boot.rb

  public/javascripts/control.js
  public/javascripts/dragdrop.js
  public/javascripts/effects.js
  public/javascripts/prototype.js

> rake log:clear tmp:clear db:test:purge

Editer database.yml J'aime bien ajouter l'encoding pour MySql:

development:
  adapter: mysql
  *encoding: utf8* (sans les étoiles autour ;) )
  database: appli_development
  username: appli
  password: appli
  host: localhost

> rake db:reset
(optionnel ?)

> rake db:migrate:reset
...

m à j de db/schema.rb

> rake test
...
36 tests, 95 assertions, 0 failures, 0 errors
...
33 tests, 63 assertions, 0 failures, 0 errors

> ruby script/server
=> Booting Mongrel (use 'script/server webrick' to force WEBrick)
=> Rails application starting on http://0.0.0.0:3000
=> Call with -d to detach
=> Ctrl-C to shutdown server
** Starting Mongrel listening at 0.0.0.0:3000
** Starting Rails with development environment...
** Rails loaded.
** Loading any Rails specific GemPlugins
** Signals ready.  INT => stop (no restart).
** Mongrel 1.1.4 available at 0.0.0.0:3000
** Use CTRL-C to stop.
/!\ FAILSAFE /!\  Tue Mar 18 10:16:56 +0100 2008
  Status: 500 Internal Server Error
  A secret is required to generate an integrity hash for cookie session data. Use config.action_controller.session = { :session_key => 

"_mya
pp_session", :secret => "some secret phrase of at least 30 characters" } in config/environment.rb
    c:/ruby/lib/ruby/gems/1.8/gems/actionpack-2.0.2/lib/action_controller/session/cookie_store.rb:91:in `ensure_secret_secure'
    c:/ruby/lib/ruby/gems/1.8/gems/actionpack-2.0.2/lib/action_controller/session/cookie_store.rb:60:in `initialize'
...

http://localhost:3000/ retourne une erreur 500
Le plus simple que j'ai trouvé c'est de créer une nouvelle appli rails à côté:
> rails dummy

Remplacer le contenu de appli/config/environment.rb par celui de dummy (en garder une copie qq part)

> rake secret
Remplacer la ligne suivante:

  config.action_controller.session = {
    :session_key => '_dummy_session',
    :secret      => '1234abcde'
  }

par
  config.action_controller.session = {
    :session_key => '_appli_session',
    :secret      => 'secret generé par la commande rake secret'
  }

Ajouter toutes les conf specifiques à votre appli.
Par exemple dans mon cas j'avais ActionMailer, donc j'ai ajouté à la fin:
ActionMailer::Base.smtp_settings = {
  :address  => 'smtp.truc.com',
  :port  => '25',
  :domain => 'www.truc.com'
}

Continuons la migration...
> rake test
...
> svn commit -m "updating to rails 2.0.2"

> rake rails:freeze:gems
...

> svn add vendor/rails

> rake deprecated
...

> ruby script/plugin remove deprecated
...
inutile maintenant

> svn commit -m "freezing rails 2.0.2"

> gem uninstall rails --version 1.2.6

A partir de là vous avez fait l'essentiel!!
Vous pouvez rentrer chez vous après une dure journée de labeur... ou bien fignoler le travail

Créer un fichier migrate_views.rb à la racine de l'appli: Ajouter:

Dir.glob('app/views/**/*.rhtml').each do |file|
  puts `svn mv #{file} #{file.gsub(/\.rhtml$/, '.html.erb')}`
end

> ruby migrate_views.rb
A         app\views\layouts\application.html.erb
D         app\views\layouts\application.rhtml
...

> rake test
...

> ruby script/server
...
Tester l'appli

Ajouter à production.rb

config.action_view.cache_template_loading            = true

Ajouter à test.rb
# Disable request forgery protection in test environment
config.action_controller.allow_forgery_protection    = false
Concernant routes.rb, je n'ai pas de solution miracle :(
Créer un une appli rails dummy à coté et utiliser un outil de conmparaison de fichier pour voir les différences et mettre votre routes.rb à jour.

Par exemple si vous n'utiliser pas le wsdl comme moi vous pouvez supprimer la ligne:
map.connect ':controller/service.wsdl', :action => 'wsdl'

On peut aussi copier/coller le README à la racine de l'appli dummy vers son appli rails

Dans test_helper.rb, on peut ajouter:

  # The only drawback to using transactional fixtures is when you actually 
  # need to test transactions.  Since your test is bracketed by a transaction,
  # any transactions started in your code will be automatically rolled back.
juste avant
self.use_transactional_fixtures = true
et
  # Setup all fixtures in test/fixtures/*.(yml|csv) for all tests in alphabetical order.
  #
  # Note: You'll currently still have to declare fixtures explicitly in integration tests
  # -- they do not yet inherit this setting
  fixtures :all

Si vous avez un script/breakpointer je pense que vous pouvez le supprimer...

Vous pouvez aussi changer tous vos fichiers dans db/migrate pour utiliser la nouvelle syntaxe des migrations (optionnel)

Il y a plein d'autre améliorations possibles dans les controller, dans les fixtures etc... cf: rails-2-0-it-s-done

Et Voilà ! Votre appli est migrée en Rails 2.0.2! Félicitations ;)

Technorati tags:

lundi, mars 17, 2008

Migration "Pas à pas" d'une application ruby on rails de 1.2.2 vers 1.2.6

Comment migrer "Pas à pas" une application ruby on rails de 1.2.2 vers 1.2.6 (Nous verrons de 1.2.6 à 2.0.2 dans un autre post).

Extrait du blog officiel de Ruby on Rails:
So how do I upgrade? If you want to move your application to Rails 2.0, you should first move it to Rails 1.2.6. That’ll include deprecation warnings for most everything we yanked out in 2.0. So if your application runs fine on 1.2.6 with no deprecation warnings, there’s a good chance that it’ll run straight up on 2.0. Of course, if you’re using, say, pagination, you’ll need to install the classic_pagination plugin. If you’re using Oracle, you’ll need to install the activerecord-oracle-adapter gem. And so on and so forth for all the extractions.

> ruby script\about

About your application's environment
Ruby version                 1.8.6 (i386-mswin32)
RubyGems version             1.0.1
Rails version                1.2.2
Active Record version        1.15.2
Action Pack version          1.13.2
Action Web Service version   1.2.2
Action Mailer version        1.3.2
Active Support version       1.4.1
Edge Rails revision          64
Application root             C:/developement/ruby/prophet/trunk/prophet
Environment                  development
Database adapter             mysql
Database schema version      8



> rake rails:unfreeze

> gem install rails --version 1.2.6

> gem list rails

*** LOCAL GEMS ***

rails (2.0.2, 1.2.6)

> edit appli/config/environment.rb
Changer:
RAILS_GEM_VERSION = '1.2.2' unless defined? RAILS_GEM_VERSION
En:
RAILS_GEM_VERSION = '1.2.6' unless defined? RAILS_GEM_VERSION

> rake rails:update:configs

modifie config/boot.rb

> rake rails:update:javascripts

modifie rien ?

> rake rails:update:scripts

modifie rien ?

> rake log:clear tmp:clear db:test:purge

> ruby script\about

About your application's environment
Ruby version                 1.8.6 (i386-mswin32)
RubyGems version             1.0.1
Rails version                1.2.6
Active Record version        1.15.6
Action Pack version          1.13.6
Action Web Service version   1.2.6
Action Mailer version        1.3.6
Active Support version       1.4.4
Application root             C:/developement/ruby/prophet/trunk/prophet
Environment                  development
Database adapter             mysql
Database schema version      8


> rake test
...
36 tests, 95 assertions, 0 failures, 0 errors
...
33 tests, 63 assertions, 0 failures, 0 errors

> ruby script\server
=> Booting Mongrel (use 'script/server webrick' to force WEBrick)
=> Rails application starting on http://0.0.0.0:3000
=> Call with -d to detach
=> Ctrl-C to shutdown server
** Starting Mongrel listening at 0.0.0.0:3000
** Starting Rails with development environment...
** Rails loaded.
** Loading any Rails specific GemPlugins
** Signals ready.  INT => stop (no restart).
** Mongrel 1.1.4 available at 0.0.0.0:3000
** Use CTRL-C to stop.
March 17, 2008 17:39 --

...

Regarder dans les logs s'il y a des "DEPRECATION WARNING"
Jeter un coup d'oeil à http://www.rubyonrails.org/deprecation
Installer le plugin deprecated: http://nubyonrails.com/articles/deprecated-plugin-find-old-rails-code
> ruby script/plugin install http://topfunky.net/svn/plugins/deprecated
+ ./deprecated/CHANGELOG
+ ./deprecated/MIT-LICENSE
+ ./deprecated/README
+ ./deprecated/about.yml
+ ./deprecated/tasks/deprecated.rake


> rake deprecated
(in C:/.../app)
--> component
 Clean! Cheers for you!

--> @session
 Clean! Cheers for you!

--> paginate
 Clean! Cheers for you!
...
S'il n'y a que des "Clean" c'est que tout va bien :)
> svn cleanup .
(optionnel ?)

> svn commit -m "upgrading to 1.2.6"

> rake rails:freeze:gems
(in C:/developement/ruby/prophet/trunk/prophet)
Freezing to the gems for Rails 1.2.6
rake aborted!
uninitialized constant Gem::GemRunner

(See full trace by running task with --trace)


>gem update --system
...

ATTENTION HACK
Supprimer le rep 'rails' dans 'vendor' à la main pour revenir à zéro...
Il y a un probleme de compatibilité entre les dernières version de rubygems et rails 1.2.6
Il faut ajouter 'require 'rubygems/gem_runner' dans C:\ruby\lib\ruby\gems\1.8\gems\rails-1.2.6\lib\tasks\framework.rake
cf: rubygems-095-and-rails-126-uninitialized-constant-gemgemrunner
> gem list rails

*** LOCAL GEMS ***

rails (2.0.2, 1.2.6)

ATTENTION
Bien préciser la version sinon ça va freezer la plus récente (ici on veut 1.2.6 pas la 2.0.2)
> rake rails:freeze:gems VERSION=1.2.6

Maintenant ça devrait marcher...

> svn add vendor/rails

> svn commit -m "freezing to 1.2.6"

Et voilà !
Plus vous avez de tests moins il y aura de régressions ;)
Passez un peu partout dans l'appli pour plus de sureté aussi, on est jamais trop prudent ;)

lundi, mars 10, 2008

Déployer une application ruby on rails dans Tomcat

Je viens de déployer ma première application rails dans un Tomcat avec JRuby.
Contrairement à ce que la longueur de ce post pourrait laisser penser c'est très simple et très rapide (env 2H30 pour quelqu'un d'expérimenté en tomcat, mysql et rails et 30 min si tout est pré-installé).
Le lien indispensable pour déployer une application rail 1.1 ou 2.0 se trouve sur le wiki de JRuby: Jruby on Rails on Tomcat
Je vous conseille la page du wiki qui est beaucoup plus succinte en premier!
Sinon voilà une autre version si vous partez vraiment de zéro avec beaucoup plus de détails et/ou de "bonnes pratiques" ;)
En gros il y a 2 cas:
  1. Vous préparez le ".war" (généralement un développeur ruby/java en charge de la livraison) Nous l'appellerons Mr R.
  2. Vous déployez sur la machine de production (généralement l'admin système ou un gars des opérations). Nous l'appellerons Mr A.
Versions et environnent testé: Windows XP + Tomcat(6.0.16) + JRuby(1.0.3) + MySql(5) + Java(jre1.6.0_03)
Cas numéro 1: Création du "War"
  1. JRuby
    • Télécharger JRuby
    • Dezipper JRuby (ex: c:\jruby)
    • Ajouter JRUBY_HOME=c:\jruby
    • Ajouter %JRUBY_HOME%\bin à la variable d'environnement 'Path'
  2. Tomcat
    • Télécharger l'installer: apache-tomcat-6.0.16.exe
    • Installler (suivre le wizard)
    • Générallement Tomcat est installé ici: C:\Program Files\Apache Software Foundation\Tomcat 6.0
  3. MySQL
    • Installer MySql (encoding utf-8 de préférence)
    • Se Connecter en 'root'
    • Créer un schema appli_[env]
    • Créer un user spécial:
      GRANT ALL ON appli_production.* TO 'appli'@'localhost' IDENTIFIED BY 'appli';
  4. Warbler:
        c:\...\appli> jruby -S gem install -y rails warbler
        c:\...\appli> jruby -S rake db:migrate RAILS_ENV=production
        c:\...\appli> jruby -S rake db:structure:dump RAILS_ENV=production
        c:\...\appli> jruby -S warble war 
        c:\...\appli> jruby -S warble config
    
    Avec la commande "db:structure:dump", on a généré un fichier: production_structure.sql (utilisé par Mr. A)
  5. Aller dans le "appli>tmp/war/WEB-INF/web.xml" et ajouter les lignes suivantes si vous êtes dans le cas décrit dans le wiki de JRuby:
    <context-param>
        <param-name>jruby.session_store</param-name>
        <param-value>db</param-value>
    </context-param>
  6. Copier/Coller le web.xml modifié dans le répertoire config pour ne pas avoir a refaire cette manip à chaque fois que vous généré le war.
  7. Vérifiez les paramètres de connexion à la production dans database.yml
  8. Figer toutes les versions de rails et warbler:
        c:\...\appli> jruby -S rake rails:freeze
        c:\...\appli> jruby -S warble pluginize
    
  9. Regénérer le war avec le web.xml modifié:
        c:\...\appli> jruby -S warble war:clean
        c:\...\appli> jruby -S warble war
    
  10. Le copier dans le répertoire webapp de Tomcat
  11. Démarrer Tomcat
  12. Tester l'application: http://localhost:8080
Cas numéro 2: Production
  1. Installer Tomcat
  2. Installer MySql
  3. Créer le schéma de production
    CREATE SCHEMA appli_production;
  4. Charger les tables dans le schéma (j'utilise MySql Query Browser pour ça; je ne connait pas la commande exacte), On utilisera le fichier généré par Mr. R (cf Cas numéro 1: production_structure.sql)
  5. Se mettre d'accord sur les paramètres de connexion à la base avec Mr. R (schema name/username/password)
  6. Créer un user spécial:
    GRANT ALL ON appli_production.* TO 'appli'@'localhost' IDENTIFIED BY 'appli';
  7. Récupérer le war de Mr. R
  8. Le copier dans le répertoire webapp de Tomcat
  9. Démarrer Tomcat
  10. Tester l'application: http://localhost:8080
Conclusion: Voilà j'ai peut-être oublié des choses ou fait des erreurs de frappe mais l'essentiel est là ;)
Rem: Il y a autant de manières de créer une application que de personnes, certaines pratiques sont une affaire de goût :)
N'hésitez pas à ajouter vos commentaires...
Technorati tags:

jeudi, mars 06, 2008

Améliorer la lisibilité avec Array.collect

Je n'ai pas l'habitude (venant du monde java) d'utiliser les méthodes collect, reject, select de Array.
Voilà un cas d'utilisation où j'ai nettement vu l'amélioration de lisibilité du code.
Imaginez qu'on veut afficher la liste des noms de users séparés par un espace.
Avant:

names = ""
for user in users
    names << "#{user.user_username} "
end
return names.chomp

Après:

names = users.collect { |user| user.user_username}
return names.join(' ')

C'est tout simple mais quand on pas l'habitude; on y pense pas forcément ;)
Je découvre avec un peu de retard l'utilité de "collect()" mais mieux vaut tard que jamais :D
Technorati tags:

mercredi, mars 05, 2008

Assertion en ruby

Je me demandais si le mot clef "assert" existe en ruby... je ne l'ai pas trouvé mais par contre c'est très simple de le réinventer:

def assert(*msg)
  raise "Assertion failed ! #{msg}" unless yield if $DEBUG
end
Un exemple d'utilisation:

$DEBUG = true
i = 0
assert("mon msg") {i == 1}
Résultat:
Exception `RuntimeError' at assert.rb:2 - Assertion failed ! mon msg
assert.rb:2:in `assert': Assertion failed ! mon msg (RuntimeError)
        from assert.rb:7
source: dzone.com

Technorati tags:

mardi, mars 04, 2008

Problème avec Webrick et authenticate_or_request_with_http_basic

Je ne sais pas pourquoi mais il semblerait que le authenticate_or_request_with_http_basic de rails 2 ne fonctionne pas sous webrick (sous windows ?).

Je n'arrive pas à le faire marcher (je passe l'authentification comme si elle n'existait pas) en mode development ou production.
Par contre avec Mongrel ça fonctionne ?!

J'ai trouvé un thread où d'autres personnes ont l'air d'avoir le même genre de problème...
C:\...>ruby --version
ruby 1.8.6 (2007-03-13 patchlevel 0) [i386-mswin32]
C:\...>rails --version
Rails 2.0.2


Technorati tags:

lundi, mars 03, 2008

MD5 en Ruby

Pour mon autentification d'un utilisateur avec rails en utilisant la méthode "authenticate_or_request_with_http_basic" de Basic HTTP authentication, j'avais besoin de vérifer le mot de passe.

Pour celà, j'avais besoin de transformer le mot de passe saisi par l'utilisateur en md5 pour le comparer avec le md5 stocké en base (pour éviter de stocker des mots de passe en clair dans la base de données).
Et bien c'est très simple, il suffit de 2 lignes de code:

require 'digest/md5'
hash = Digest::MD5.hexdigest(password)
Dans le cas de mon application rails pour voir le code complet, ça donne:

class ApplicationController < ActionController::Base
  before_filter :authenticate
  
  protected
  
  def authenticate
    authenticate_or_request_with_http_basic do |username, password|
      auth = User.auth?(username, password)
      if(auth)
        session[:auth], session[:username] = true, username
        return true
      end
      return false
    end
  end
end

require 'digest/md5'

class User < ActiveRecord::Base
  # return true if (username and hashed_password) are correct
  def self.auth?(username, password)
    hashed_password = Digest::MD5.hexdigest(password)
    user = User.find(:first, 
        :conditions => "username='#{username}' and password='#{hashed_password}'" )
    logger.info "#{username}/#{hashed_password} > auth? : #{!user.nil?}"
    return !user.nil?
  end
end
Technorati tags:

jeudi, février 21, 2008

Enlever les lignes en double dans un fichier

Une petie astuce que j'ai posté sur le blog de torsten en réponse à sa question:
Comment afficher les lignes d'un fichier en enlevant les doublons (faire une sorte de 'unique' ou 'distinct' sur les lignes du fichier)

Je me suis dit que ça pourrait servir à d'autres donc je le repost ici:
puts IO.readlines('data.txt').uniq

Par contre sur un gros fichier ça doit consommer beaucoup de mémoire...

Technorati tags:

mardi, février 19, 2008

Convertir de l'utf-16 en iso-8859-1

Je savais que la gestion de l'utf en ruby (1.8) était un gros problème auquel je n'avais pas encore été confronté. Mais dans mon cas le plus gros problème a été de trouver une page web qui m'explique comment faire au lieu de se plaindre :(

Il y a tellement de pages sur des forums etc... qui expliquent qu'il y a un problème sans fournir de solution claire!

Finallement je suis tombé sur cette page de Obie qui m'a mis sur la bonne piste :)

Malheureusement dans son exemple je perdais les accents (c'est embêtant :( )

Voilà l'exemple corrigé pour transfromer de l'UTF-16 vers de l'ISO-8859-1. Pour UTF-8 il suffit de remplacer 16 par 8 étonnant non ? ;)



require 'iconv'

class String
  def utf16_to_iso
    converter = Iconv.new('ISO-8859-1//IGNORE//TRANSLIT', 'UTF-16')
    converter.iconv(self)
  end
end

content = IO.read("utf16.txt").utf16_to_iso
content.split("\n").each do |line|
  do_something(line)
end

IGNORE veut dire que Iconv va ignorer les erreurs de conversion. Si vous voulez être prévenu quand vous avez un caractère exotique, enlevez cette option.


TRANSLIT essaye de trouver le caractère le plus proche de celui de utf dans iso.

Petite astuce UTF-16, je n'ai pas pu utiliser la méthode: IO.foreach que j'aime beaucoup en temps normal parce que cette méthode split sur le caractère "\n" or en UTF-16 comme c'est stocké sur deux octets le retour à la ligne est "\n\000". J'ai donc converti la chaine avant de faire le split.


Technorati tags:

vendredi, février 15, 2008

Comment tester null ou chaine (string) vide

Récemment dans un petit script j'avais le code suivant:
unless(item.nil? or item.emty?)
  ...
end
Où item est soit nil, soit "", soit "foo". Je n'aimais pas faire ça partout dans le code, je voulais être plus "DRY" en faisant quelque chose comme:
unless(item.blank?)
  ...
end
La solution que j'ai trouvé dans un premier temps pour résoudre ce problème "esthétique" était de déclarer la fonction blank? dans String et NilClass; ça conne ça:

class NilClass
  def blank?
    true
  end
end

class String
  def blank?
    to_s.empty?
  end
end
Mais Benjamin C. m'a montrer une manière plus élégante en étendant Object:
class Object
  def blank?
    respond_to?(:empty?) ? empty? : !self
  end
end
Code re-pompé honteusement de rails.
Technorati tags: