Учебник по Ruby / DBI

В этой главе вы узнаете, как получить доступ к базе данных с помощью Ruby. Модуль Ruby DBI предоставляет независимый от базы данных интерфейс для сценариев Ruby, аналогичный интерфейсу модуля Perl DBI.

DBI расшифровывается как Database Independent Interface for Ruby, что означает, что DBI обеспечивает уровень абстракции между кодом Ruby и базовой базой данных, что позволяет очень легко переключать реализации базы данных. Он определяет набор методов, переменных и соглашений, которые обеспечивают согласованный интерфейс базы данных, независимо от фактического использования базы данных.

DBI может взаимодействовать со следующим:

  • ADO (объекты данных ActiveX)
  • DB2
  • Frontbase
  • mSQL
  • MySQL
  • ODBC
  • Oracle
  • OCI8 (Oracle)
  • PostgreSQL
  • Proxy/Server
  • SQLite
  • SQLRelay

Архитектура приложения DBI

DBI не зависит от какой-либо базы данных, доступной в серверной части. DBI можно использовать независимо от того, работаете ли вы с Oracle, MySQL или Informix и т. Д. Это ясно из следующей диаграммы архитектуры.

Общая архитектура Ruby DBI использует два уровня:

  • Уровень интерфейса базы данных (DBI). Этот уровень не зависит от базы данных и предоставляет набор общих методов доступа, которые используются одинаково независимо от типа сервера базы данных, с которым вы общаетесь.

  • Уровень драйвера базы данных (DBD). Этот уровень зависит от базы данных; разные драйверы предоставляют доступ к разным ядрам баз данных. Есть один драйвер для MySQL, другой для PostgreSQL, третий для InterBase, третий для Oracle и так далее. Каждый драйвер интерпретирует запросы от уровня DBI и сопоставляет их с запросами, подходящими для данного типа сервера базы данных.

Предпосылки

Если вы хотите писать сценарии Ruby для доступа к базам данных MySQL, вам потребуется установить модуль Ruby MySQL.

Этот модуль действует как DBD, как описано выше, и может быть загружен с https://www.tmtm.org/en/mysql/ruby/

Получение и установка Ruby / DBI

Вы можете загрузить и установить модуль Ruby DBI из следующего места -

https://imgur.com/NFEuWe4/embed

Перед началом установки убедитесь, что у вас есть права root. Теперь выполните шаги, указанные ниже -

Шаг 1

$ tar zxf dbi-0.2.0.tar.gz

Шаг 2

Перейдите в каталог распространения dbi-0.2.0 и настройте его с помощью сценария setup.rb в этом каталоге. Самая общая команда настройки выглядит так, без аргументов после аргумента config. Эта команда настраивает дистрибутив на установку всех драйверов по умолчанию.

$ ruby setup.rb config

Чтобы быть более конкретным, укажите параметр --with, в котором перечислены определенные части дистрибутива, которые вы хотите использовать. Например, чтобы настроить только основной модуль DBI и драйвер уровня MySQL DBD, введите следующую команду:

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

Шаг 3

Последний шаг - собрать драйвер и установить его с помощью следующих команд:

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

Подключение к базе данных

Предполагая, что мы собираемся работать с базой данных MySQL, перед подключением к базе данных убедитесь в следующем:

  • Вы создали базу данных TESTDB.

  • Вы создали EMPLOYEE в TESTDB.

  • В этой таблице есть поля FIRST_NAME, LAST_NAME, AGE, SEX и INCOME.

  • Идентификатор пользователя testuser и пароль test123 установлены для доступа к TESTDB.

  • Ruby Module DBI правильно установлен на вашем компьютере.

  • Вы прошли обучение MySQL, чтобы понять основы MySQL.

Ниже приведен пример подключения к базе данных 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

При запуске этого сценария на нашей машине с Linux он дает следующий результат.

Server version: 5.0.45

Если соединение установлено с источником данных, то дескриптор базы данных возвращается и сохраняется в dbh для дальнейшего использования в противном случае dbhустанавливается равным нулю, и e.err и e :: errstr возвращают код ошибки и строку ошибки соответственно.

И, наконец, перед выходом убедитесь, что соединение с базой данных закрыто и ресурсы освобождены.

ВСТАВИТЬ операцию

Операция INSERT требуется, когда вы хотите создать свои записи в таблице базы данных.

Как только соединение с базой данных установлено, мы готовы создавать таблицы или записи в таблицах базы данных, используя do метод или prepare и execute метод.

Использование инструкции do

Операторы, которые не возвращают строки, могут быть выполнены с помощью вызова doметод дескриптора базы данных. Этот метод принимает строковый аргумент оператора и возвращает счетчик количества строк, затронутых оператором.

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 )" );

Точно так же вы можете выполнить инструкцию SQL INSERT, чтобы создать запись в таблице 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

Использование подготовить и выполнить

Вы можете использовать методы подготовки и выполнения класса DBI для выполнения оператора SQL через код Ruby.

Создание записи требует следующих шагов -

  • Подготовка оператора SQL с помощью оператора INSERT. Это будет сделано с помощьюprepare метод.

  • Выполнение SQL-запроса для выбора всех результатов из базы данных. Это будет сделано с помощьюexecute метод.

  • Деблокировать дескриптор заявления. Это будет сделано с помощьюfinish API

  • Если все будет хорошо, то commit эта операция в противном случае вы можете rollback полная транзакция.

Ниже приведен синтаксис использования этих двух методов:

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

Эти два метода можно использовать для передачи bindзначения для операторов SQL. Возможен случай, когда значения для ввода не указаны заранее. В таком случае используются значения привязки. Знак вопроса (?) используется вместо фактических значений, а затем фактические значения передаются через API-интерфейс execute ().

Ниже приведен пример создания двух записей в таблице 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

Если одновременно выполняется несколько операций INSERT, то сначала подготовка оператора, а затем его выполнение несколько раз в цикле более эффективна, чем вызов do каждый раз в цикле.

ПРОЧИТАТЬ операцию

READ Операция с любой базой данных означает получение некоторой полезной информации из базы данных.

Как только наша база данных будет установлена, мы готовы сделать запрос в эту базу данных. Мы можем использовать либоdo метод или prepare и execute методы для получения значений из таблицы базы данных.

Получение записи требует следующих шагов -

  • Подготовка SQL-запроса на основе необходимых условий. Это будет сделано с помощьюprepare метод.

  • Выполнение SQL-запроса для выбора всех результатов из базы данных. Это будет сделано с помощьюexecute метод.

  • Получение всех результатов один за другим и их печать. Это будет сделано с помощьюfetch метод.

  • Деблокировка заявления. Это будет сделано с помощьюfinish метод.

Ниже приводится процедура запроса всех записей из таблицы EMPLOYEE с зарплатой более 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

Это даст следующий результат -

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

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

Есть более короткие методы для получения записей из базы данных. Если вам интересно, пройдите через Получение результата, в противном случае переходите к следующему разделу.

Операция обновления

ОБНОВЛЕНИЕ Операция с любой базой данных означает обновление одной или нескольких записей, которые уже доступны в базе данных. Ниже приведена процедура обновления всех записей, в которых SEX обозначается буквой «M». Здесь мы увеличим ВОЗРАСТ всех мужчин на один год. Это займет три шага -

  • Подготовка SQL-запроса на основе необходимых условий. Это будет сделано с помощьюprepare метод.

  • Выполнение SQL-запроса для выбора всех результатов из базы данных. Это будет сделано с помощьюexecute метод.

  • Деблокировка заявления. Это будет сделано с помощьюfinish метод.

  • Если все пойдет хорошо, тогда commit эта операция в противном случае вы можете rollback полная транзакция.

#!/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

УДАЛИТЬ операцию

Операция DELETE требуется, если вы хотите удалить некоторые записи из своей базы данных. Ниже приведена процедура удаления всех записей из СОТРУДНИКА, у которого ВОЗРАСТ больше 20. Эта операция потребует следующих шагов.

  • Подготовка SQL-запроса на основе необходимых условий. Это будет сделано с помощьюprepare метод.

  • Выполнение SQL-запроса для удаления необходимых записей из базы данных. Это будет сделано с помощьюexecute метод.

  • Деблокировка заявления. Это будет сделано с помощьюfinish метод.

  • Если все пойдет хорошо, тогда commit эта операция в противном случае вы можете rollback полная транзакция.

#!/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

Выполнение транзакций

Транзакции - это механизм, обеспечивающий согласованность данных. Транзакции должны иметь следующие четыре свойства -

  • Atomicity - Либо транзакция завершается, либо вообще ничего не происходит.

  • Consistency - Транзакция должна начинаться в согласованном состоянии и выходить из системы в согласованном состоянии.

  • Isolation - Промежуточные результаты транзакции не видны вне текущей транзакции.

  • Durability - После фиксации транзакции последствия сохраняются даже после сбоя системы.

DBI предоставляет два метода для фиксации или отката транзакции. Есть еще один метод, называемый транзакцией, который можно использовать для выполнения транзакций. Есть два простых подхода к реализации транзакций:

Подход I

Первый подход использует методы фиксации и отката DBI для явной фиксации или отмены транзакции -

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

Подход II

Второй подход использует метод транзакции . Это проще, поскольку для этого требуется блок кода, содержащий операторы, составляющие транзакцию. Метод транзакции выполняет блок, а затем автоматически вызывает фиксацию или откат , в зависимости от того, успешен блок или нет -

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

COMMIT Operation

Фиксация - это операция, которая дает зеленый сигнал базе данных о завершении изменений, и после этой операции никакие изменения не могут быть отменены.

Вот простой пример вызова commit метод.

dbh.commit

ROLLBACK Операция

Если вас не устраивает одно или несколько изменений и вы хотите полностью отменить эти изменения, используйте rollback метод.

Вот простой пример вызова rollback метод.

dbh.rollback

Отключение базы данных

Чтобы отключить соединение с базой данных, используйте API отключения.

dbh.disconnect

Если соединение с базой данных закрывается пользователем с помощью метода отключения, все невыполненные транзакции откатываются DBI. Однако вместо того, чтобы зависеть от каких-либо деталей реализации DBI, вашему приложению было бы лучше явно вызывать фиксацию или откат.

Обработка ошибок

Есть много источников ошибок. Несколько примеров - это синтаксическая ошибка в выполненном операторе SQL, сбой соединения или вызов метода выборки для уже отмененного или завершенного дескриптора оператора.

Если метод DBI не работает, DBI вызывает исключение. Методы DBI могут вызывать любой из нескольких типов исключений, но двумя наиболее важными классами исключений являются DBI :: InterfaceError и DBI :: DatabaseError .

Объекты исключений этих классов имеют три атрибута с именами err , errstr и state , которые представляют номер ошибки, строку описания ошибки и стандартный код ошибки. Атрибуты объяснены ниже -

  • err- Возвращает целочисленное представление произошедшей ошибки или ноль, если это не поддерживается DBD. Oracle DBD, например, возвращает числовую часть сообщения об ошибке ORA-XXXX .

  • errstr - Возвращает строковое представление произошедшей ошибки.

  • state- Возвращает код SQLSTATE возникшей ошибки. SQLSTATE представляет собой строку из пяти символов. Большинство DBD не поддерживают это и вместо этого возвращают nil.

Вы видели следующий код выше в большинстве примеров -

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

Чтобы получить отладочную информацию о том, что делает ваш скрипт во время выполнения, вы можете включить трассировку. Для этого вы должны сначала загрузить модуль dbi / trace, а затем вызвать метод трассировки, который управляет режимом трассировки и местом назначения вывода -

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

trace(mode, destination)

Значение режима может быть 0 (выключено), 1, 2 или 3, а местом назначения должен быть объект ввода-вывода. Значения по умолчанию - 2 и STDERR соответственно.

Блоки кода с методами

Есть несколько методов создания дескрипторов. Эти методы можно вызывать с помощью блока кода. Преимущество использования блока кода вместе с методами заключается в том, что они предоставляют дескриптор блока кода в качестве его параметра и автоматически очищают дескриптор при завершении блока. Есть несколько примеров для понимания концепции.

  • DBI.connect- Этот метод генерирует дескриптор базы данных, и рекомендуется вызвать отключение в конце блока, чтобы отключить базу данных.

  • dbh.prepare- Этот метод генерирует дескриптор оператора, и его рекомендуется завершить в конце блока. В блоке, вы должны вызвать выполнить метод выполнения оператора.

  • dbh.execute- Этот метод аналогичен, за исключением того, что нам не нужно вызывать выполнение внутри блока. Дескриптор оператора выполняется автоматически.

Пример 1

DBI.connect может взять блок кода, передать ему дескриптор базы данных и автоматически отсоединить дескриптор в конце блока следующим образом.

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

Пример 2

dbh.prepare может взять блок кода, передать ему дескриптор оператора и автоматически вызвать finish в конце блока, как показано ниже.

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

Пример 3

dbh.execute может взять блок кода, передать ему дескриптор оператора и автоматически вызвать finish в конце блока следующим образом:

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

Метод транзакции DBI также принимает блок кода, описанный выше.

Функции и атрибуты, специфичные для драйвера

DBI позволяет драйверам базы данных предоставлять дополнительные специфичные для базы данных функции, которые могут быть вызваны пользователем с помощью метода func любого объекта Handle.

Атрибуты, специфичные для драйвера, поддерживаются и могут быть установлены или получены с помощью []= или же [] методы.

Sr.No. Функции и описание
1

dbh.func(:createdb, db_name)

Создает новую базу данных.

2

dbh.func(:dropdb, db_name)

Удаляет базу данных.

3

dbh.func(:reload)

Выполняет операцию перезагрузки.

4

dbh.func(:shutdown)

Выключает сервер.

5

dbh.func(:insert_id) => Fixnum

Возвращает самое последнее значение AUTO_INCREMENT для соединения.

6

dbh.func(:client_info) => String

Возвращает информацию о клиенте MySQL с указанием версии.

7

dbh.func(:client_version) => Fixnum

Возвращает информацию о клиенте с указанием версии. Он похож на: client_info, но возвращает fixnum вместо sting.

8

dbh.func(:host_info) => String

Возвращает информацию о хосте.

9

dbh.func(:proto_info) => Fixnum

Возвращает протокол, используемый для связи.

10

dbh.func(:server_info) => String

Возвращает информацию о сервере MySQL с указанием версии.

11

dbh.func(:stat) => String

Возвращает текущее состояние базы данных.

12

dbh.func(:thread_id) => Fixnum

Возвращает идентификатор текущего потока.

пример

#!/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

Это даст следующий результат -

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