Tutoriel Ruby / DBI

Ce chapitre vous apprend à accéder à une base de données à l'aide de Ruby. Le module Ruby DBI fournit une interface indépendante de la base de données pour les scripts Ruby similaire à celle du module Perl DBI.

DBI signifie Database Independent Interface pour Ruby, ce qui signifie que DBI fournit une couche d'abstraction entre le code Ruby et la base de données sous-jacente, vous permettant de changer d'implémentation de base de données très facilement. Il définit un ensemble de méthodes, de variables et de conventions qui fournissent une interface de base de données cohérente, indépendante de la base de données réelle utilisée.

DBI peut s'interfacer avec les éléments suivants -

  • ADO (objets de données ActiveX)
  • DB2
  • Frontbase
  • mSQL
  • MySQL
  • ODBC
  • Oracle
  • OCI8 (Oracle)
  • PostgreSQL
  • Proxy/Server
  • SQLite
  • SQLRelay

Architecture d'une application DBI

DBI est indépendant de toute base de données disponible dans le backend. Vous pouvez utiliser DBI si vous travaillez avec Oracle, MySQL ou Informix, etc. Cela ressort clairement du diagramme d'architecture suivant.

L'architecture générale de Ruby DBI utilise deux couches -

  • La couche d'interface de base de données (DBI). Cette couche est indépendante de la base de données et fournit un ensemble de méthodes d'accès communes qui sont utilisées de la même manière quel que soit le type de serveur de base de données avec lequel vous communiquez.

  • La couche du pilote de base de données (DBD). Cette couche dépend de la base de données; différents pilotes donnent accès à différents moteurs de base de données. Il existe un pilote pour MySQL, un autre pour PostgreSQL, un autre pour InterBase, un autre pour Oracle, et ainsi de suite. Chaque pilote interprète les requêtes de la couche DBI et les mappe sur les requêtes appropriées pour un type donné de serveur de base de données.

Conditions préalables

Si vous souhaitez écrire des scripts Ruby pour accéder aux bases de données MySQL, vous devrez installer le module Ruby MySQL.

Ce module agit comme un DBD comme expliqué ci-dessus et peut être téléchargé depuis https://www.tmtm.org/en/mysql/ruby/

Obtention et installation de Ruby / DBI

Vous pouvez télécharger et installer le module Ruby DBI à partir de l'emplacement suivant -

https://imgur.com/NFEuWe4/embed

Avant de commencer cette installation, assurez-vous que vous disposez du privilège root. Maintenant, suivez les étapes ci-dessous -

Étape 1

$ tar zxf dbi-0.2.0.tar.gz

Étape 2

Allez dans le répertoire de distribution dbi-0.2.0 et configurez-le en utilisant le script setup.rb dans ce répertoire. La commande de configuration la plus générale ressemble à ceci, sans argument après l'argument config. Cette commande configure la distribution pour installer tous les pilotes par défaut.

$ ruby setup.rb config

Pour être plus précis, fournissez une option --with qui répertorie les parties particulières de la distribution que vous souhaitez utiliser. Par exemple, pour configurer uniquement le module DBI principal et le pilote de niveau DBD MySQL, exécutez la commande suivante -

$ ruby setup.rb config --with = dbi,dbd_mysql

Étape 3

La dernière étape consiste à créer le pilote et à l'installer à l'aide des commandes suivantes -

$ ruby setup.rb setup
$ ruby setup.rb install

Connexion à la base de données

En supposant que nous allons travailler avec la base de données MySQL, avant de vous connecter à une base de données, assurez-vous de ce qui suit -

  • Vous avez créé une base de données TESTDB.

  • Vous avez créé EMPLOYEE dans TESTDB.

  • Cette table contient les champs FIRST_NAME, LAST_NAME, AGE, SEX et INCOME.

  • L'ID utilisateur "testuser" et le mot de passe "test123" sont définis pour accéder à TESTDB.

  • Ruby Module DBI est correctement installé sur votre machine.

  • Vous avez parcouru le didacticiel MySQL pour comprendre les bases de MySQL.

Voici l'exemple de connexion avec la base de données MySQL "TESTDB"

#!/usr/bin/ruby -w

require "dbi"

begin
   # connect to the MySQL server
   dbh = DBI.connect("DBI:Mysql:TESTDB:localhost", "testuser", "test123")
   # get server version string and display it
   row = dbh.select_one("SELECT VERSION()")
   puts "Server version: " + row[0]
rescue DBI::DatabaseError => e
   puts "An error occurred"
   puts "Error code:    #{e.err}"
   puts "Error message: #{e.errstr}"
ensure
   # disconnect from server
   dbh.disconnect if dbh
end

Lors de l'exécution de ce script, il produit le résultat suivant sur notre machine Linux.

Server version: 5.0.45

Si une connexion est établie avec la source de données, un handle de base de données est renvoyé et enregistré dans dbh pour une utilisation ultérieure autrement dbha la valeur nil et e.err et e :: errstr renvoient respectivement un code d'erreur et une chaîne d'erreur.

Enfin, avant de le sortir, assurez-vous que la connexion à la base de données est fermée et que les ressources sont libérées.

Opération INSERT

L'opération INSERT est requise lorsque vous souhaitez créer vos enregistrements dans une table de base de données.

Une fois la connexion à la base de données établie, nous sommes prêts à créer des tables ou des enregistrements dans les tables de la base de données en utilisant do méthode ou prepare et execute méthode.

Utilisation de l'instruction do

Les instructions qui ne renvoient pas de lignes peuvent être émises en appelant le dométhode de poignée de base de données. Cette méthode prend un argument de chaîne d'instruction et renvoie un décompte du nombre de lignes affectées par l'instruction.

dbh.do("DROP TABLE IF EXISTS EMPLOYEE")
dbh.do("CREATE TABLE EMPLOYEE (
   FIRST_NAME  CHAR(20) NOT NULL,
   LAST_NAME  CHAR(20),
   AGE INT,  
   SEX CHAR(1),
   INCOME FLOAT )" );

De même, vous pouvez exécuter l'instruction SQL INSERT pour créer un enregistrement dans la table EMPLOYEE.

#!/usr/bin/ruby -w

require "dbi"

begin
   # connect to the MySQL server
   dbh = DBI.connect("DBI:Mysql:TESTDB:localhost", "testuser", "test123")
   dbh.do( "INSERT INTO EMPLOYEE(FIRST_NAME, LAST_NAME, AGE, SEX, INCOME)
      VALUES ('Mac', 'Mohan', 20, 'M', 2000)" )
   puts "Record has been created"
   dbh.commit
rescue DBI::DatabaseError => e
   puts "An error occurred"
   puts "Error code:    #{e.err}"
   puts "Error message: #{e.errstr}"
   dbh.rollback
ensure
   # disconnect from server
   dbh.disconnect if dbh
end

Utilisation de la préparation et de l'exécution

Vous pouvez utiliser les méthodes de préparation et d' exécution de la classe DBI pour exécuter l'instruction SQL via du code Ruby.

La création d'un enregistrement nécessite les étapes suivantes -

  • Préparation de l'instruction SQL avec l'instruction INSERT. Cela se fera en utilisant leprepare méthode.

  • Exécution d'une requête SQL pour sélectionner tous les résultats de la base de données. Cela se fera en utilisant leexecute méthode.

  • Poignée de déclaration de libération. Cela se fera en utilisantfinish API

  • Si tout va bien, alors commit cette opération sinon vous pouvez rollback la transaction complète.

Voici la syntaxe pour utiliser ces deux méthodes -

sth = dbh.prepare(statement)
sth.execute
   ... zero or more SQL operations ...
sth.finish

Ces deux méthodes peuvent être utilisées pour réussir bindvaleurs en instructions SQL. Il peut y avoir un cas où les valeurs à saisir ne sont pas données à l'avance. Dans un tel cas, des valeurs de liaison sont utilisées. Un point d'interrogation (?) est utilisé à la place des valeurs réelles, puis les valeurs réelles sont transmises via l'API execute ().

Voici l'exemple pour créer deux enregistrements dans la table EMPLOYEE -

#!/usr/bin/ruby -w

require "dbi"

begin
   # connect to the MySQL server
   dbh = DBI.connect("DBI:Mysql:TESTDB:localhost", "testuser", "test123")
   sth = dbh.prepare( "INSERT INTO EMPLOYEE(FIRST_NAME, LAST_NAME, AGE, SEX, INCOME)
      VALUES (?, ?, ?, ?, ?)" )
   sth.execute('John', 'Poul', 25, 'M', 2300)
   sth.execute('Zara', 'Ali', 17, 'F', 1000)
   sth.finish
   dbh.commit
   puts "Record has been created"
rescue DBI::DatabaseError => e
   puts "An error occurred"
   puts "Error code:    #{e.err}"
   puts "Error message: #{e.errstr}"
   dbh.rollback
ensure
   # disconnect from server
   dbh.disconnect if dbh
end

S'il y a plusieurs INSERT à la fois, il est plus efficace de préparer d'abord une instruction, puis de l'exécuter plusieurs fois dans une boucle que d'appeler do à chaque fois dans la boucle.

LIRE l'opération

LIRE L'opération sur n'importe quelle base de données signifie récupérer des informations utiles dans la base de données.

Une fois notre connexion à la base de données établie, nous sommes prêts à effectuer une requête dans cette base de données. Nous pouvons utiliser soitdo méthode ou prepare et execute méthodes pour récupérer les valeurs d'une table de base de données.

La récupération des enregistrements s'effectue comme suit -

  • Préparation de la requête SQL en fonction des conditions requises. Cela se fera en utilisant leprepare méthode.

  • Exécution d'une requête SQL pour sélectionner tous les résultats de la base de données. Cela se fera en utilisant leexecute méthode.

  • Récupérer tous les résultats un par un et imprimer ces résultats. Cela se fera en utilisant lefetch méthode.

  • Poignée de déclaration de libération. Cela se fera en utilisant lefinish méthode.

Voici la procédure pour interroger tous les enregistrements de la table EMPLOYEE ayant un salaire supérieur à 1000.

#!/usr/bin/ruby -w

require "dbi"

begin
   # connect to the MySQL server
   dbh = DBI.connect("DBI:Mysql:TESTDB:localhost", "testuser", "test123")
   sth = dbh.prepare("SELECT * FROM EMPLOYEE WHERE INCOME > ?")
   sth.execute(1000)

   sth.fetch do |row|
   printf "First Name: %s, Last Name : %s\n", row[0], row[1]
   printf "Age: %d, Sex : %s\n", row[2], row[3]
   printf "Salary :%d \n\n", row[4]
end
   sth.finish
rescue DBI::DatabaseError => e
   puts "An error occurred"
   puts "Error code:    #{e.err}"
   puts "Error message: #{e.errstr}"
ensure
   # disconnect from server
   dbh.disconnect if dbh
end

Cela produira le résultat suivant -

First Name: Mac, Last Name : Mohan
Age: 20, Sex : M
Salary :2000

First Name: John, Last Name : Poul
Age: 25, Sex : M
Salary :2300

Il existe d'autres méthodes de raccourci pour récupérer les enregistrements de la base de données. Si vous êtes intéressé, passez par la récupération du résultat, sinon passez à la section suivante.

Opération de mise à jour

UPDATE L'opération sur n'importe quelle base de données signifie mettre à jour un ou plusieurs enregistrements, qui sont déjà disponibles dans la base de données. Voici la procédure pour mettre à jour tous les enregistrements ayant SEX comme «M». Ici, nous augmenterons l'âge de tous les mâles d'un an. Cela prendra trois étapes -

  • Préparation de la requête SQL en fonction des conditions requises. Cela se fera en utilisant leprepare méthode.

  • Exécution d'une requête SQL pour sélectionner tous les résultats de la base de données. Cela se fera en utilisant leexecute méthode.

  • Poignée de déclaration de libération. Cela se fera en utilisant lefinish méthode.

  • Si tout va bien alors commit cette opération sinon vous pouvez rollback la transaction complète.

#!/usr/bin/ruby -w

require "dbi"

begin
   # connect to the MySQL server
   dbh = DBI.connect("DBI:Mysql:TESTDB:localhost", "testuser", "test123")
   sth = dbh.prepare("UPDATE EMPLOYEE SET AGE = AGE + 1 WHERE SEX = ?")
   sth.execute('M')
   sth.finish
   dbh.commit
rescue DBI::DatabaseError => e
   puts "An error occurred"
   puts "Error code:    #{e.err}"
   puts "Error message: #{e.errstr}"
   dbh.rollback
ensure
   # disconnect from server
   dbh.disconnect if dbh
end

Opération DELETE

L'opération DELETE est requise lorsque vous souhaitez supprimer certains enregistrements de votre base de données. Voici la procédure pour supprimer tous les enregistrements d'EMPLOYÉ où AGE est supérieur à 20. Cette opération comprendra les étapes suivantes.

  • Préparation de la requête SQL en fonction des conditions requises. Cela se fera en utilisant leprepare méthode.

  • Exécution d'une requête SQL pour supprimer les enregistrements requis de la base de données. Cela se fera en utilisant leexecute méthode.

  • Poignée de déclaration de libération. Cela se fera en utilisant lefinish méthode.

  • Si tout va bien alors commit cette opération sinon vous pouvez rollback la transaction complète.

#!/usr/bin/ruby -w

require "dbi"

begin
   # connect to the MySQL server
   dbh = DBI.connect("DBI:Mysql:TESTDB:localhost", "testuser", "test123")
   sth = dbh.prepare("DELETE FROM EMPLOYEE WHERE AGE > ?")
   sth.execute(20)
   sth.finish
   dbh.commit
rescue DBI::DatabaseError => e
   puts "An error occurred"
   puts "Error code:    #{e.err}"
   puts "Error message: #{e.errstr}"
   dbh.rollback
ensure
   # disconnect from server
   dbh.disconnect if dbh
end

Effectuer des transactions

Les transactions sont un mécanisme qui garantit la cohérence des données. Les transactions doivent avoir les quatre propriétés suivantes -

  • Atomicity - Soit une transaction se termine, soit rien ne se passe du tout.

  • Consistency - Une transaction doit démarrer dans un état cohérent et quitter le système dans un état cohérent.

  • Isolation - Les résultats intermédiaires d'une transaction ne sont pas visibles en dehors de la transaction en cours.

  • Durability - Une fois qu'une transaction a été validée, les effets sont persistants, même après une panne du système.

Le DBI fournit deux méthodes pour valider ou annuler une transaction. Il existe une autre méthode appelée transaction qui peut être utilisée pour implémenter des transactions. Il existe deux approches simples pour mettre en œuvre des transactions -

Approche I

La première approche utilise les méthodes de validation et de restauration de DBI pour valider ou annuler explicitement la transaction -

dbh['AutoCommit'] = false # Set auto commit to false.
begin
   dbh.do("UPDATE EMPLOYEE SET AGE = AGE+1 WHERE FIRST_NAME = 'John'")
   dbh.do("UPDATE EMPLOYEE SET AGE = AGE+1 WHERE FIRST_NAME = 'Zara'")
   dbh.commit
rescue
   puts "transaction failed"
   dbh.rollback
end
dbh['AutoCommit'] = true

Approche II

La seconde approche utilise la méthode transactionnelle . C'est plus simple, car il faut un bloc de code contenant les instructions qui composent la transaction. La méthode de transaction exécute le bloc, puis appelle automatiquement la validation ou la restauration , selon que le bloc réussit ou échoue -

dbh['AutoCommit'] = false # Set auto commit to false.
dbh.transaction do |dbh|
   dbh.do("UPDATE EMPLOYEE SET AGE = AGE+1 WHERE FIRST_NAME = 'John'")
   dbh.do("UPDATE EMPLOYEE SET AGE = AGE+1 WHERE FIRST_NAME = 'Zara'")
end
dbh['AutoCommit'] = true

Opération COMMIT

Valider est l'opération qui donne un signal vert à la base de données pour finaliser les modifications, et après cette opération, aucune modification ne peut être annulée.

Voici un exemple simple pour appeler le commit méthode.

dbh.commit

Fonctionnement ROLLBACK

Si vous n'êtes pas satisfait d'une ou plusieurs des modifications et que vous souhaitez annuler complètement ces modifications, utilisez le rollback méthode.

Voici un exemple simple pour appeler le rollback méthode.

dbh.rollback

Déconnexion de la base de données

Pour déconnecter la connexion à la base de données, utilisez l'API de déconnexion.

dbh.disconnect

Si la connexion à une base de données est fermée par l'utilisateur avec la méthode de déconnexion, toutes les transactions en attente sont annulées par le DBI. Cependant, au lieu de dépendre des détails d'implémentation de DBI, votre application ferait mieux d'appeler explicitement la validation ou la restauration.

Traitement des erreurs

Il existe de nombreuses sources d'erreurs. Quelques exemples sont une erreur de syntaxe dans une instruction SQL exécutée, un échec de connexion ou l'appel de la méthode fetch pour un descripteur d'instruction déjà annulé ou terminé.

Si une méthode DBI échoue, DBI lève une exception. Les méthodes DBI peuvent déclencher l'un de plusieurs types d'exceptions, mais les deux classes d'exception les plus importantes sont DBI :: InterfaceError et DBI :: DatabaseError .

Les objets d'exception de ces classes ont trois attributs nommés err , errstr et state , qui représentent le numéro d'erreur, une chaîne d'erreur descriptive et un code d'erreur standard. Les attributs sont expliqués ci-dessous -

  • err- Renvoie une représentation entière de l'erreur survenue ou nil si elle n'est pas prise en charge par le DBD. Oracle DBD par exemple renvoie la partie numérique d'un message d'erreur ORA-XXXX .

  • errstr - Renvoie une représentation sous forme de chaîne de l'erreur survenue.

  • state- Renvoie le code SQLSTATE de l'erreur survenue. SQLSTATE est une chaîne de cinq caractères. La plupart des DBD ne prennent pas en charge cela et renvoient nil à la place.

Vous avez vu le code suivant ci-dessus dans la plupart des exemples -

rescue DBI::DatabaseError => e
   puts "An error occurred"
   puts "Error code:    #{e.err}"
   puts "Error message: #{e.errstr}"
   dbh.rollback
ensure
   # disconnect from server
   dbh.disconnect if dbh
end

Pour obtenir des informations de débogage sur ce que fait votre script lors de son exécution, vous pouvez activer le traçage. Pour ce faire, vous devez d'abord charger le module dbi / trace, puis appeler la méthode trace qui contrôle le mode trace et la destination de sortie -

require "dbi/trace"
..............

trace(mode, destination)

La valeur de mode peut être 0 (désactivé), 1, 2 ou 3, et la destination doit être un objet IO. Les valeurs par défaut sont respectivement 2 et STDERR.

Blocs de code avec méthodes

Il existe certaines méthodes qui créent des poignées. Ces méthodes peuvent être appelées avec un bloc de code. L'avantage d'utiliser le bloc de code avec les méthodes est qu'ils fournissent le handle du bloc de code comme paramètre et nettoie automatiquement le handle lorsque le bloc se termine. Il existe peu d'exemples pour comprendre le concept.

  • DBI.connect- Cette méthode génère un handle de base de données et il est recommandé d'appeler disconnect à la fin du bloc pour déconnecter la base de données.

  • dbh.prepare- Cette méthode génère un descripteur d'instruction et il est recommandé de terminer à la fin du bloc. Au sein du bloc, vous devez appeler exécuter la méthode pour exécuter l'instruction.

  • dbh.execute- Cette méthode est similaire sauf que nous n'avons pas besoin d'appeler execute dans le bloc. Le descripteur de l'instruction est automatiquement exécuté.

Exemple 1

DBI.connect peut prendre un bloc de code, lui passe le handle de base de données et déconnecte automatiquement le handle à la fin du bloc comme suit.

dbh = DBI.connect("DBI:Mysql:TESTDB:localhost", "testuser", "test123") do |dbh|

Exemple 2

dbh.prepare peut prendre un bloc de code, lui transmet le descripteur d'instruction et appelle automatiquement finish à la fin du bloc comme suit.

dbh.prepare("SHOW DATABASES") do |sth|
   sth.execute
   puts "Databases: " + sth.fetch_all.join(", ")
end

Exemple 3

dbh.execute peut prendre un bloc de code, lui passe le descripteur de l'instruction et appelle automatiquement finish à la fin du bloc comme suit -

dbh.execute("SHOW DATABASES") do |sth|
   puts "Databases: " + sth.fetch_all.join(", ")
end

La méthode de transaction DBI prend également un bloc de code qui a été décrit ci-dessus.

Fonctions et attributs spécifiques au pilote

Le DBI permet aux pilotes de base de données de fournir des fonctions supplémentaires spécifiques à la base de données, qui peuvent être appelées par l'utilisateur via la méthode func de n'importe quel objet Handle.

Les attributs spécifiques au pilote sont pris en charge et peuvent être définis ou obtenus à l'aide du []= ou [] méthodes.

N ° Sr. Fonctions et description
1

dbh.func(:createdb, db_name)

Crée une nouvelle base de données.

2

dbh.func(:dropdb, db_name)

Supprime une base de données.

3

dbh.func(:reload)

Effectue une opération de rechargement.

4

dbh.func(:shutdown)

Arrête le serveur.

5

dbh.func(:insert_id) => Fixnum

Renvoie la valeur AUTO_INCREMENT la plus récente pour une connexion.

6

dbh.func(:client_info) => String

Renvoie les informations du client MySQL en termes de version.

sept

dbh.func(:client_version) => Fixnum

Renvoie les informations client en termes de version. C'est similaire à: client_info mais il renvoie un fixnum au lieu de sting.

8

dbh.func(:host_info) => String

Renvoie les informations sur l'hôte.

9

dbh.func(:proto_info) => Fixnum

Renvoie le protocole utilisé pour la communication.

dix

dbh.func(:server_info) => String

Renvoie les informations du serveur MySQL en termes de version.

11

dbh.func(:stat) => String

Renvoie l'état actuel de la base de données.

12

dbh.func(:thread_id) => Fixnum

Renvoie l'ID de thread actuel.

Exemple

#!/usr/bin/ruby

require "dbi"
begin
   # connect to the MySQL server
   dbh = DBI.connect("DBI:Mysql:TESTDB:localhost", "testuser", "test123") 
   puts dbh.func(:client_info)
   puts dbh.func(:client_version)
   puts dbh.func(:host_info)
   puts dbh.func(:proto_info)
   puts dbh.func(:server_info)
   puts dbh.func(:thread_id)
   puts dbh.func(:stat)
rescue DBI::DatabaseError => e
   puts "An error occurred"
   puts "Error code:    #{e.err}"
   puts "Error message: #{e.errstr}"
ensure
   dbh.disconnect if dbh
end

Cela produira le résultat suivant -

5.0.45
50045
Localhost via UNIX socket
10
5.0.45
150621
Uptime: 384981  Threads: 1  Questions: 1101078  Slow queries: 4 \
Opens: 324  Flush tables: 1  Open tables: 64  \
Queries per second avg: 2.860