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