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
mercredi, avril 28, 2010
Compter le nombres de lignes dans des fichiers
jeudi, avril 15, 2010
Delicious linkchecker
Voici un petit bout de code qui permet de vérifier tous les bookmarks: http://gist.github.com/366112
mardi, mars 23, 2010
Query Browser pour MongoDB
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
lundi, septembre 14, 2009
Zlib::BufError et repository de gems down
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
gem server
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émardi, septembre 08, 2009
Javascript: Orienté Objet
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...
Je trouve que le concept d'héritage en Javascript ressemble beaucoup à ce qui se passe en ruby avec le mot clef "include".
mercredi, août 26, 2009
Disparition de why et Ma présence online
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
- 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 ?!)
- 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
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
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
- 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
mercredi, mai 13, 2009
Railsconf 2009 (Trop ?)
Les positives:
- il y a beaucoup d'activité dans la communauté rails
- 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)
- La communauté grandit de plus en plus et donc les chances de faire du ruby tous les jours au boulot augmente
- Ca part dans tous les sens
- Tout le monde essaye d'inventer son framework pour avoir un peu de gloire
- La plupart des frameworks/lib présentés vont tomber dans l'oubli dans l'année qui arrive
- Je n'ai pas le temps (et pas l'envie) de suivre tout ce qui se passe
- 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)
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
lundi, janvier 26, 2009
Différences entre exec et system
- c'est facile de passer à côté et de les confondre avec un ' normal
- parce que pour un novice c'est difficile à comprendre
- etc. ...
Dir["folder/foo_*.txt"].each do |path|
cmd = "echo #{path}"
exec(cmd)
end
Dir["folder/foo_*.txt"].each do |path|
cmd = "echo #{path}"
system(cmd)
end
mercredi, novembre 26, 2008
Changer l'extension d'un fichier
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
mercredi, juin 25, 2008
LinkedIn la plus grosse application rails au monde ?
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).
mercredi, juin 04, 2008
Ruby Riddle: "Trouver le nom de la nouvelle société où je travaille"
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}"
lundi, avril 28, 2008
Ruby à l'Université
lundi, mars 31, 2008
Experimentations avec les Mixins / Modules et les méthodes statiques (self)
Justement j'avais eu le même problème quand j'avais fait mon projet "natural_sort"
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 ?!
Dans NaturalSort, j'ai contourné le problème:
module NaturalSort
def self.naturalsort(object)
...
end
def natural_sort
NaturalSort::naturalsort(to_a)
end
end
Ici vous pourrez voir les principales étapes jusqu'à la version finale, celle qui me satisfait le plus.
module Carnivore
def self.manger(oiseau)
puts "slurp! #{oiseau}"
end
end
Carnivore.manger "titi"
module Carnivore
def manger(oiseau)
puts "slurp! #{oiseau}"
end
end
class GrosMinet
include Carnivore
end
gros_minet = GrosMinet.new
gros_minet.manger "titi"
- 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 ?
mercredi, mars 19, 2008
Rails 2 - Renommer des .rjs en .js.rjs
# 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
mardi, mars 18, 2008
Migration "Pas à pas" d'une application ruby on rails de 1.2.6 vers 2.0.2 (part 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 = truepuis 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 = falseConcernant 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 ;)
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"
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
- Vous préparez le ".war" (généralement un développeur ruby/java en charge de la livraison) Nous l'appellerons Mr R.
- 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.
- JRuby
- Télécharger JRuby
- Dezipper JRuby (ex: c:\jruby)
- Ajouter JRUBY_HOME=c:\jruby
- Ajouter %JRUBY_HOME%\bin à la variable d'environnement 'Path'
- 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
- 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';
- 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 configAvec la commande "db:structure:dump", on a généré un fichier: production_structure.sql (utilisé par Mr. A) - 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> - 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.
- Vérifiez les paramètres de connexion à la production dans database.yml
- Figer toutes les versions de rails et warbler:
c:\...\appli> jruby -S rake rails:freeze c:\...\appli> jruby -S warble pluginize - Regénérer le war avec le web.xml modifié:
c:\...\appli> jruby -S warble war:clean c:\...\appli> jruby -S warble war - Le copier dans le répertoire webapp de Tomcat
- Démarrer Tomcat
- Tester l'application: http://localhost:8080
- Installer Tomcat
- Installer MySql
- Créer le schéma de production
CREATE SCHEMA appli_production;
- 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)
- Se mettre d'accord sur les paramètres de connexion à la base avec Mr. R (schema name/username/password)
- Créer un user spécial:
GRANT ALL ON appli_production.* TO 'appli'@'localhost' IDENTIFIED BY 'appli';
- Récupérer le war de Mr. R
- Le copier dans le répertoire webapp de Tomcat
- Démarrer Tomcat
- Tester l'application: http://localhost:8080
N'hésitez pas à ajouter vos commentaires...
jeudi, mars 06, 2008
Améliorer la lisibilité avec Array.collect
Imaginez qu'on veut afficher la liste des noms de users séparés par un espace.
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
mercredi, mars 05, 2008
Assertion en ruby
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.commardi, mars 04, 2008
Problème avec Webrick et authenticate_or_request_with_http_basic
Par contre avec Mongrel ça fonctionne ?!
C:\...>ruby --version ruby 1.8.6 (2007-03-13 patchlevel 0) [i386-mswin32] C:\...>rails --version Rails 2.0.2
lundi, mars 03, 2008
MD5 en Ruby
require 'digest/md5'
hash = Digest::MD5.hexdigest(password)
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
jeudi, février 21, 2008
Enlever les lignes en double dans un fichier
Comment afficher les lignes d'un fichier en enlevant les doublons (faire une sorte de 'unique' ou 'distinct' sur les lignes du fichier)
puts IO.readlines('data.txt').uniqmardi, février 19, 2008
Convertir de l'utf-16 en iso-8859-1
Il y a tellement de pages sur des forums etc... qui expliquent qu'il y a un problème sans fournir de solution claire!
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)
endTRANSLIT essaye de trouver le caractère le plus proche de celui de utf dans iso.
vendredi, février 15, 2008
Comment tester null ou chaine (string) vide
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
endclass Object
def blank?
respond_to?(:empty?) ? empty? : !self
end
end
Code re-pompé honteusement de rails.