Händler-Anwendung (Ruby-on-Rails-Beispiel)

aus GlossarWiki, der Glossar-Datenbank der Fachhochschule Augsburg

Dieser Artikel wird derzeit von einem Autor gründlich bearbeitet. Die Inhalte sind daher evtl. noch inkonsistent.

Zweck

Anhand einer einfachen Web-Anwendung sollten die Prinzipien von Ruby on Rails erläutert werden.

Diesem Beispiel liegt die Händler-Datenbank zugrunde, die Wolfgang Kowarschick in der Vorlesung Datenmanagement II (Multimedia-Datenbanksysteme)[1] als Beispiel einsetzt:

Medium:Haendler.png

Voraussetzungen

Folgende Softwarepakete wurden installiert:

Ruby on Rails wurde zusammen mit einigen Erweiterungs-Modulen mit Hilfe des Ruby-Paketmanagers gem installiert:

RUBY_DIR=...; export RUBY_DIR
POSTGRESQL_DIR=...; export  POSTGRESQL_DIR

gem install rails --include-dependencies
gem install postgres -r -- --with-pgsql-dir=${POSTGRESQL_DIR}
gem install passenger
${RUBY_DIR}/bin/passenger-install-apache2-module

Auf dem Test-Server steht in der Datei /etc/hosts ein Eintrag für den gewünschten virtuellen Server, auf dem die Anwendung laufen soll (die IP-Adresse und der Rechnernamen können natürlich angepasst werden):

192.168.0.161   haendler.kowa haendler

Diese IP-Adresse ist einem echten oder virtuellen Rechner zugeordnet. Unter SuSE 11 kann man einem Rechner ganz einfach mehrere IP-Adressen zuordnen. Man trägt die neue IP-Adresse in die Datei /etc/sysconfig/network/ifcfg-eth0 ein:

IPADDR_haendler='192.168.0.161'
NETMASK_haendler='255.255.255.0'

Danach muss der Netzwerkzugriff neu gestartet werden:

/etc/init.d/network restart

Anlegen der Händlerdatenbanken

Für jede Rails-Anwendung müssen stets drei Datenbanken existieren, eine für die Entwicklung, eine für die automatischen Tests und eine für den produktiven Betrieb. Außerdem sollte für jede Rails-Anwendung ein spezieller Benutzer definiert werden, dem diese drei Datenbanken gehören. Der Benutzer und die Datenbanken können über eine Web-Schnittstelle wie phpPgAdmin oder auch mit Hilfe von einfachen PostgreSQL-Befehlen angelegt werden:

createuser haendler -d -e -E -P -S -R

createdb -Uhaendler -Ohaendler haendler_development
createdb -Uhaendler -Ohaendler haendler_test
createdb -Uhaendler -Ohaendler haendler_production

Anlegen der Rails-Anwendung haendler

Zunächst werden im Rails-Webverzeichnis die Ordner und Dateien einer neuen Rails-Anwendung namens haendler erzeugt:

cd /web/rails
rails --database postgresql haendler

In der Konfigurations-Datei database.yml, die den Zugriff auf das DB-System regelt, wird das Passwort für den Benutzer haendler an drei Stellen eingetragen:

vi /web/rails/haendler/config/database.yml
development:
  adapter: postgresql
  encoding: unicode
  database: haendler_development
  pool: 5
  username: haendler
  password: geHEIM!?

test:
  adapter: postgresql
  encoding: unicode
  database: haendler_test
  pool: 5
  username: haendler
  password: geHEIM!?

production:
  adapter: postgresql
  encoding: unicode
  database: haendler_production
  pool: 5
  username: haendler
  password: geHEIM!?

Mit Hilfe eines rake-Kommandos kann überprüft werden, ob die Verbindung zur Datenbank hergestellt werden kann:

cd /web/rails/haendler
rake db:migrate

Wenn das rake-Kommando keinen Fehler ausgibt, ist alles in Ordnung.

Zugang zur Web-Anwendung haendler mit Hilfe von Apache

Unter der Annahme, dass Rails-Anwendungen im Verzeichnis /web/rails angelegt werden, kann der virtuelle Server in der Apache-Konfigurationsdatei httpd.conf eingetragen werden:

<VirtualHost 192.168.0.161:80>
   ServerName haendler.kowa
   DocumentRoot /web/rails/haendler/public
   <Directory /web/rails/haendler/public>
    AllowOverride All
    Order allow,deny
    Allow from all
  </Directory>

  CustomLog    logs/haendler-access combined
  ErrorLog     logs/haendler-error

</VirtualHost>

Nach einem Neustart von Apache mit /etc/rc.d/apache restart oder, schneller, kill -1 <Prozessnummer des Apache-Root-Prozesses> sollte man die Applikation im Browser ansehen können:

http://haendler.kowa/

Anmerkung

Wann immer an der Rails-Anwendung eine Änderung vorgenommen wurde, sollte man im tmp-Verzeichnis der Anwendung eine Datei namens restart.txt (mit beliebigem Inhalt) angelegt werden. Dadurch wird Apache informiert, dass der die Anwendung neu einlesen soll. Und somit werden alle Änderungen direkt im Browser sichtbar.

 touch /web/rails/haendler/tmp/restart.txt

Konvention: Klassennamen stehen im Singular und Datenbanktabellen-Namen im Plural

Für Ruby on Rails wurde die Konvention eingeführt, dass Klassennamen stets in Singular stehen und mit einem Großbuchstaben beginnen: Haendler, Liefert, Ware. Die zugehörigen Tabellennamen sollen dagegen im Plural stehen und kleingeschrieben werden. Die zweite Regel führt allerdings bei deutschen Tabellennamen sowie bei englischnamigen Beziehungstabellen zu unschönen Ergebnissen, wenn die Tabellen automatisch gemäß Konvention z.B. mittels ruby script/generate erzeugt werden: haendlers, lieferts, wares bzw. traders, supplies, items (hier wäre supply besser).

Wenn man die Ruby-Konvention beachten will, sollte man die falsche automatische Plural-Bildung verhindern, indem man Ausnahmen in die Datei inflections.rb einfügt:

vi /web/rails/haendler/config/initializers/inflections.rb
ActiveSupport::Inflector.inflections do |inflect|
  inflect.uncountable 'haendler'
  inflect.irregular   'liefert', 'liefern'
  inflect.irregular   'ware',    'waren'
end

Anmerkung 1

Klassennamen für die Singular und Plural übereinstimmen, bereiten beim automatischen Erzeugen von Zugriffsmethoden („Scaffolding“) Probleme, da in den URLs nicht zwischen der Menge aller zugehörigen Objekten und einem einzelen Objekt unterschieden werden kann.

Diese Probleme kommen hier nicht zum Tragen, da Scaffolding sowieso nicht verwendet wird.

Anmerkung 2

In allen meinen Programmen und Vorlesungsbeispielen halte ich mich normalerweise an die Konvention, dass sowohl Klassen-, als auch Tabellennamen im Singular geschrieben werden sollten. Dies könnte man in Ruby on Rails erreichen, indem man eine Initialisierungs-Datei pluralize_false.rb mit folgenden Inhalt anlegt:

# vi /web/rails/haendler/config/initializers/pluralize_false.rb
ActiveRecord::Base.pluralize_table_names = false

Das es nun einmal Ruby-on-Rails-Konvention ist, Tabellennamen im Plural zu schreiben, greife ich in diesem Beispiel nicht auf diese Möglichkeit zurück.

Anlegen des Modells

Anlegen der Modell-Dateien

Mit Hilfe des Ruby-Skriptes ruby script/generate model kann nun die Dateien des Modells, d.h. die Dateien für die Klassen- und Tabellen-Definitionen des Modells angelegt werden.

cd /web/rails/haendler/
ruby script/generate model haendler
ruby script/generate model ware
ruby script/generate model liefert

Ergebnis:

exists  app/models/
exists  test/unit/
exists  test/fixtures/
create  app/models/haendler.rb
create  test/unit/haendler_test.rb
create  test/fixtures/haendlers.yml
create  db/migrate
create  db/migrate/20081208135556_create_haendlers.rb

...

Primär- und Fremdschlüssel

Laut Ruby-on-Rails-Philosophie sollte jede Tabelle ein Primärschlüssel-Attribut namens id enthalten. Dieses wird, wenn man in den zugehörigen Migrate-Dateien nichts anderes angibt, mittels

rake db:migrate

automatisch erzeugt.

Nun benötigen Beziehungstabellen kein künstliches Schlüssel-Attribut. Eine übliche Methode ist es daher, in Rails-Anwendungen bei Beziehungstabellen ganz auf Primärschlüssel zu verzichten und anstatt dessen einen Index für die Beziehungsattribute zu definieren (siehe z.B. den Abschnitt "Migration" in der Rails-Kurz-Referenz von InVisible[2]). Sollte eine Beziehungstabelle weitere Attribute haben, so müsste für die Beziehungsattribute zusätzlich noch eine gemeinsame Unique-Bedingung angegeben werden. Dies wird für mehr als ein Attribut von Rails leider auch nicht direkt unterstüzt (vgl. z.B. die Dokumentation zur Klasse ActiveRecord::Migration[3], Abschnitt „More examples“).

In einem sehr schönen Blog-Beitrag[4] beschreibt Jeff Smith, warum im Falle von Beziehungstabellen „Composite Primary Keys“, wie Multi-Attribut-Primärschlüssel im Englischen genannt werden, so wichtig sind. Er widerlegt dabei auch in einer Antwort auf eine Leserzuschrift die Argumente von Lee Richardson[5], der gegen „Composite Primary Keys“ argumentiert.

Ruby on Rails unterstützt nicht nur die Angabe von Multi-Attribut-Primärschlüsseln, auch die Angabe von Fremdschlüsseln wird nicht direkt unterstützt.

Glücklicherweise gibt es zwei Erweiterungspakete, die diese Probleme beheben.

Das Multi-Primär-Schlüssel-Problem lässt sich mit dem Rails-Paket composite_primary_keys beheben. Und mit Hilfe des Pakets foreign_key_migrations können auch Foreign Keys innerhalb von Migration-Dateien definiert werde:

gem install composite_primary_keys
gem install foreign_key_migrations

Um diese Pakete verwenden zu können, müssen sie in der Händler-Anwendung noch aktiviert werden:

vi /web/rails/haendler/config/initializers/db_extensions.rb
require 'composite_primary_keys'
require 'composite_primary_keys/migration'
require 'foreign_key_migrations'

ActiveRecord-PostgreSQL-Patch

Wenn man mit Fremdschlüssel arbeitet, stolpert man leicht über einen Rail-Bug[6]. Diesen kann man mit folgendem Patch beheben:

vi /wkcms/web/rails/haendler/config/initializers/active_record_postgresql.rb
module ActiveRecord
  module ConnectionAdapters
    class PostgreSQLAdapter < AbstractAdapter
      def disable_referential_integrity(&block)
        transaction {
           begin
             execute "SET CONSTRAINTS ALL DEFERRED"
             yield
           ensure
             execute "SET CONSTRAINTS ALL IMMEDIATE"
           end
        }
      end
    end
  end
end

Bearbeiten der Migrate-Dateien

Nun werden die Attribute, Schlüssel, Fremdschlüssel und Indexe der Einzelenen Klassen des Modells in den zugehörigen migrate-Dateien eingetragen.

vi /web/rails/haendler/db/migrate/*_create_haendler.rb
class CreateHaendler < ActiveRecord::Migration
  def self.up
    create_table :haendler do |t|
      t.string :name,   :null => false
      t.string :adresse

#     t.timestamps
    end
  end

  def self.down
    remove_index :liefern, [:ware_id]
    drop_table :haendler
  end
end


vi /web/rails/haendler/db/migrate/*_create_waren.rb
class CreateWaren < ActiveRecord::Migration
  def self.up
    create_table :waren do |t|
      t.string :typ,         :null => false
      t.string :bezeichnung, :null => false

#     t.timestamps
    end

    execute "ALTER TABLE waren ADD CONSTRAINT unique_typ_bezeichnung UNIQUE (typ, bezeichnung)"
  end

  def self.down
    drop_table :waren
  end
end


vi /web/rails/haendler/db/migrate/*_create_liefern.rb
class CreateLiefern < ActiveRecord::Migration
  def self.up
    create_table :liefern, :primary_key => [:haendler_id, :ware_id, :preis] do |t|
      t.integer :haendler_id, :null => false, :references => :haendler
      t.integer :ware_id,     :null => false, :references => :waren
      t.decimal :preis,       :null => false, :precision => 6, :scale => 2
      t.integer :lieferzeit

#     t.timestamps
    end

    add_index :liefern, [:ware_id]
  end

  def self.down
    drop_table :liefern
  end
end

Anschließend werden die Tabellen mit dem Ruby-Make-Befehl rake erzeugt:

cd /web/rails/haendler
rake db:migrate
rake db:migrate RAILS_ENV="production"

Anmerkung

Mit

rake db:migrate                        VERSION="0"
rake db:migrate RAILS_ENV="production" VERSION="0"

können alle Tabellen wieder gelöscht werden.

Bearbeitung der Klassendateien des Modells

vi /web/rails/haendler/app/models/haendler.rb
class Haendler < ActiveRecord::Base
  has_many :waren, :through => :liefern
  has_many :liefern
end


vi /web/rails/haendler/app/models/ware.rb
class Ware < ActiveRecord::Base
  has_many :haendler, :through => :liefern
  has_many :liefern
end


vi /web/rails/haendler/app/models/liefert.rb
class Liefert < ActiveRecord::Base
  set_primary_keys [:haendler_id, :ware_id]
  belongs_to :haendler, :foreign_key => 'haendler_id'
  belongs_to :waren,    :foreign_key => 'ware_id'
end

Anschließend sollte mit einem Browser Ihrer Wahl überprüft werden, ob irgendwelche Fehlermeldungen ausgegeben werden:

touch /web/rails/haendler/tmp/restart.txt
lynx http://haendler.kowa/test

Wenn der Server mit der Fehlermeldung 404 anzeigt, dass es diese Seite nicht gibt, ist alles OK. Views wurden ja noch nicht definiert. Wenn der Server allerdings mit der Fehlermeldung 500 anzeigt, dass ein interner Server-Fehler vorliegt, wurden die Modell-Dateien nicht korrekt definiert. Der Server gibt dann auch eine genauere Fehlermeldung aus.

Anlegen der Testdaten

In Rails sollten für jede Anwendung Tests generiert werden, um die Integrität der Anwendung jederzeit auotmatisch testen zu können. Insbesondere müssen Test-Daten definiert werden, mit denen bei Start eines Test die Test-Datenbank initialisiert wird.

vi /web/rails/haendler/test/fixtures/haendler.yml
maier_koenigsbrunn:
  id:      1
  name:    Maier
  adresse: Königsbrunn

mueller_koenigsbrunn:
  id:      2
  name:    Müller
  adresse: Königsbrunn

maier_augsburg:
  id:      3
  name:    Maier
  adresse: Augsburg

huber:
  id:      4
  name:    Huber


vi /web/rails/haendler/test/fixtures/waren.yml
cup1:
  id:          1
  typ:         CPU
  bezeichnung: Pentium IV 3,8

cpu2:
  id:      2
  typ:         CPU
  bezeichnung: Celeron 2,6'

cpu3:
  id:      3
  typ:         CPU
  bezeichnung: Athlon XP 3000+

eieruhr:
  id:      4
  typ:         Sonstiges
  bezeichnung: Eieruhr


vi /web/rails/haendler/test/fixtures/liefern.yml
mak_c1:
  haendler_id:      1
  ware_id:          1
  preis:       200.00
  lieferzeit:       1

mak_c2:
  haendler_id:      1
  ware_id:          2
  preis:       100.00
  lieferzeit:       1

mak_c3:
  haendler_id:      1
  ware_id:          3
  preis:       150.00

mak_3:
  haendler_id:      1
  ware_id:          4
  preis:        10.00
  lieferzeit:       1

muk_cp1:
  haendler_id:      2
  ware_id:          1
  preis:       160.00
  lieferzeit:       1

muk_cp2:
  haendler_id:      2
  ware_id:          2
  preis:       180.00

muk_cp3:
  haendler_id:      2
  ware_id:          3
  preis:       150.00
  lieferzeit:       4

maa_cp1:
  haendler_id:      3
  ware_id:          1
  preis:       160.00
  lieferzeit:       4

maa_cp2:
  haendler_id:      3
  ware_id:          2
  preis:       190.00
  lieferzeit:       1

hu_cp1:
  haendler_id:      4
  ware_id:          1
  preis:       150.00
  lieferzeit:       4

hu_cp3:
  haendler_id:      4
  ware_id:          3
  preis:       180.00
  lieferzeit:       5

hu_cp3_2:
  haendler_id:      4
  ware_id:          3
  preis:       199.00
  lieferzeit:       1

Ob alle Daten korrekt eingetragen wurden, kann man überprüfen, indem man eine Rails-Testlauf startet. Dies geschieht einfach durch Aufruf des Befehls rake. Daran, dass das Testen die Default-Aktion von rake ist, erkennt man die Bedeutung, die die Rails-Gemeinde dem Testen beimisst.

cd /web/rails/haendler
rake

Testen des Modells

Probleme mit Primär- und Fremdschlüsseln sowie Unique-Attributen

Wenn man die Test-Tabellen mit db:migrate erzeugt, werden alle Primär- und Fremdschlüssel sowie die Integritätsbedingung unique_typ_bezeichnung wie gewünscht in der Test-Datenbank angelegt:

rake db:migrate RAILS_ENV="test" VERSION=0
rake db:migrate RAILS_ENV="test"

Erzeugt man die Test-Datenbank hingegen mit

rake

oder

rake db:test:prepare

fehlen die mit Hilfe der gem-Pakete composite_primary_keys und foreign_key_migrations angelegten Primär- und Fremdschlüssel sowie der Index unique_typ_bezeichnung.

Dies ist ziemlich nachteilig, da sich die Datenbank-Struktur der Test-Datenbank nicht von der Datenbankstruktur der beiden anderen Datenbanken unterscheiden sollte.

Der Grund für dieses Verhalten ist, dass anstelle der Migrations-Dateien die Datei /web/rails/haendler/db/schema.db zum Erzeugen der Test-Datenbank verwendet wird. Diese Datei wird bei jedem Aufruf von rake db:migrate automatisch erzeugt. Leider werden dabei die Primär- und Fremdschlüssel nicht korrekt übernommen.

Dieses Problem kann derzeit nur behoben werden, indem man eine Datei schema_sound.db von Hand erzeugt, die die korrekten Schema-Definitionen enthält. Jedesmal, wenn die Datei schema.db neu generiert wird, sollte man diese Datei mit der Datei schema_sound.db überschreiben. Eventuell muss zuvor die Datei schema_sound.db gemäß den an den Migrations-Dateien vorgenommenen Änderungen ebenfalls modifiziert werden.

vi /web/rails/haendler/db/schema_sound.db
# Die Versionsnummer sollte aus der aktuellen schema.db übernommen werden.
ActiveRecord::Schema.define(:version => 20081216130913) do

  create_table "haendler", :force => true do |t|
    t.string "name",    :null => false
    t.string "adresse"
  end

  create_table "waren", :force => true do |t|
    t.string "typ",         :null => false
    t.string "bezeichnung", :null => false
  end

  execute "ALTER TABLE waren ADD CONSTRAINT unique_typ_bezeichnung UNIQUE (typ, bezeichnung)"

  # Man beachte, dass "liefern" nach "haendler" und "waren" definiert werden muss,
  # da es sonst Probleme beim Erzeugen der Fremdschlüssel gibt.
  create_table "liefern", :primary_key => [:haendler_id, :ware_id, :preis], :force => true  do |t|
    t.integer "haendler_id",                               :null => false, :references => :haendler
    t.integer "ware_id",                                   :null => false, :references => :waren
    t.decimal "preis",       :precision => 6, :scale => 2, :null => false
    t.integer "lieferzeit"
  end

  add_index "liefern", ["ware_id"], :name => "index_liefern_on_ware_id"

end

Anlegen der Tests

Beim Testen der Anwendung werden die Test-Methoden aller Dateien, die in einem der Ordner

  • test/unit (Modell-Tests)
  • test/functional (Controller- und View-Tests)
  • test/integration (Test zur Interaktion zwischen Benutzer und Anwendung)
  • test/performance (Performanz-Tests)

gefunden werden und deren Namen mit _test enden, der Reihe nach durchgeführt. Jede dieser Test-Methoden soll mit Hilfe der Methoden assert, assert_generates etc. sicherstellen, dass bestimmte Integritätsbedingungen von der Anwendung nicht verletzt werden.

Zum Beispiel kann man überprüfen, ob der Primär- und die Fremdschlüssel der Tabelle liefern korrekt beachtet werden:

vi /wkcms/web/rails/haendler/test/unit/liefert_test.rb
require 'test_helper'

class LiefertTest < ActiveSupport::TestCase
  fixtures :haendler, :waren, :liefern

  test "foreign key haendler" do
    liefert1 = Liefert.new(:haendler_id => 6,
                           :ware_id     => 1,
                           :preis       => 199.00
                          )
    result1 = false 
    begin
      liefert1.save
    rescue
      result1 = true # Händler 6 existiert nicht
    end
    assert result1
  end

  test "foreign key waren" do
    liefert2 = Liefert.new(:haendler_id => 1,
                           :ware_id     => 7,
                           :preis       => 399.00
                          )
    result2 = false
    begin
      liefert2.save
    rescue
      result2 = true # Ware 7 existiert nicht
    end
    assert result2
  end

  test "primary key" do
    liefert = Liefert.new(:haendler_id => 1,
                          :ware_id     => 1,
                          :preis       => 200.00,
                          :lieferzeit  => 5
                         )
    result = false
    begin
      liefert.save
    rescue
      result = true # Händler 1 liefert Ware 1 zum Preis von 200.00
                    # schon innerhalb eines Tages.
    end
    assert result
  end

end

Durchführen der Tests

Wenn man nun die Tests mit der originalen schema.db durchführt, erhält man drei Assert-Fehler:

cd /web/rails/haendler
rake

> 5 tests, 5 assertions, 3 failures, 0 errors

Der Grund ist, dass alle drei Objekte, die in der Datei liefert_test.rb definiert werden, problemlos in die Test-Datenbank eingefügt werden können. Das heißt, der Primär- und die Fremdschlüssel werden nicht beachtet.

Um dies zu erreichen, muss man die falsche Datei schema.db durch die richtige ersetzen.

cp /web/rails/haendler/db/schema_sound.rb /web/rails/haendler/db/schema.rb

Allerdings funktioniert der Test jetzt immer noch nicht, sondern endet nun sogar mit fünf Laufzeit-Fehlern. Der Grund ist, dass die Datenbank-Tabellen in alphabetischer Reihenfolge gefüllt werden. Aufgrund der Fremdschlüssel muss aber die Tabelle liefern als letzte gefüllt werden. Die erreicht man, in dem man die Reihenfolge der drei Tabellen explizit in der Datei test_helper.rb angibt:

vi /web/rails/haendler/test/test_helper.rb
 #fixtures :all  ## :all bedeutet "Initialisierung in alphabetischer Reihenfolge"
  fixtures :haendler, :waren, :liefern

Nun sollten alle Tests fehlerfrei funktionieren:

cd /web/rails/haendler
rake

> 5 tests, 5 assertions, 0 failures, 0 errors

Anlegen der Controller-Dateien

Scaffolding

In diesem Beispiel werde ich auf Scaffolding verzichten.

Mit Hilfe des Scaffold-Skriptes könnten ein paar Controller- und View-Dateien angelegt werden, um Waren erfassen, bearbeiten und löschen zu können.

cd /web/rails/haendler
ruby script/generate scaffold ware typ:string bezeichnung:string

touch /web/rails/haendler/tmp/restart.txt
lynx http://haendler.kowa/waren/

Leider funktioniert diese Methode nicht problemlos für die Klasse Haendler, da hier „Singular gleich Plural“ gilt.

Ein Controller für die Klasse Haendler

Quellen

  1. Kowarschick, W.: Content-Management
  2. Agiles Web Development with Rails
  3. Learn all about Ruby ob Rails
  4. http://wiki.rubyonrails.com/rails/pages/ActiveRecord


Dieser Artikel ist GlossarWiki-konform.