Ecosyste.ms: Awesome
An open API service indexing awesome lists of open source software.
https://github.com/suketa/ruby-duckdb
Ruby binding for DuckDB
https://github.com/suketa/ruby-duckdb
database hacktoberfest ruby
Last synced: about 2 months ago
JSON representation
Ruby binding for DuckDB
- Host: GitHub
- URL: https://github.com/suketa/ruby-duckdb
- Owner: suketa
- License: mit
- Created: 2019-10-28T10:53:24.000Z (about 5 years ago)
- Default Branch: main
- Last Pushed: 2024-05-22T23:28:14.000Z (8 months ago)
- Last Synced: 2024-05-23T00:30:25.887Z (8 months ago)
- Topics: database, hacktoberfest, ruby
- Language: Ruby
- Homepage: https://github.com/suketa/ruby-duckdb
- Size: 744 KB
- Stars: 97
- Watchers: 7
- Forks: 24
- Open Issues: 16
-
Metadata Files:
- Readme: README.md
- Changelog: CHANGELOG.md
- Funding: .github/FUNDING.yml
- License: LICENSE
Awesome Lists containing this project
- awesome-duckdb - Ruby
README
# ruby-duckdb
[![Ubuntu](https://github.com/suketa/ruby-duckdb/workflows/Ubuntu/badge.svg)](https://github.com/suketa/ruby-duckdb/actions?query=workflow%3AUbuntu)
[![MacOS](https://github.com/suketa/ruby-duckdb/workflows/MacOS/badge.svg)](https://github.com/suketa/ruby-duckdb/actions?query=workflow%3AMacOS)
[![Windows](https://github.com/suketa/ruby-duckdb/workflows/Windows/badge.svg)](https://github.com/suketa/ruby-duckdb/actions?query=workflow%3AWindows)
[![Gem Version](https://badge.fury.io/rb/duckdb.svg)](https://badge.fury.io/rb/duckdb)## Description
This gem `duckdb` is Ruby client for the [DuckDB](https://www.duckdb.org) database engine.
## Requirement
You must have [DuckDB](https://www.duckdb.org) engine installed in order to use this gem.
## Pre-requisite setup (Linux):
1. Head over to the [DuckDB](https://duckdb.org/) webpage.2. Download the latest C++ package release for DuckDB.
3. Move the files to their respective location:
- Extract the `duckdb.h` and `duckdb.hpp` file to `/usr/local/include`.
- Extract the `libduckdb.so` file to `/usr/local/lib`.```sh
unzip libduckdb-linux-amd64.zip -d libduckdb
sudo mv libduckdb/duckdb.* /usr/local/include/
sudo mv libduckdb/libduckdb.so /usr/local/lib
```4. To create the necessary link, run `ldconfig` as root:
```sh
sudo ldconfig /usr/local/lib # adding a --verbose flag is optional - but this will let you know if the libduckdb.so library has been linked
```## Pre-requisite setup (macOS):
Using `brew install` is recommended.
```sh
brew install duckdb
```## Pre-requisite setup (Windows):
Using [Ruby + Devkit](https://rubyinstaller.org/downloads/) is recommended.
1. Download libduckdb-windows-amd64.zip from [DuckDB](https://github.com/duckdb/duckdb/releases) and extranct it.
2. Copy `duckdb.dll` into `C:\Windows\System32`## How to install
```sh
gem install duckdb
```After you've run the above pre-requisite setup, this should work fine.
If it doesn't, you may have to specify the location of the C header and library files:
```sh
gem install duckdb -- --with-duckdb-include=/duckdb_header_directory --with-duckdb-lib=/duckdb_library_directory
```## Usage
```ruby
require 'duckdb'db = DuckDB::Database.open # database in memory
con = db.connectcon.query('CREATE TABLE users (id INTEGER, name VARCHAR(30))')
con.query("INSERT into users VALUES(1, 'Alice')")
con.query("INSERT into users VALUES(2, 'Bob')")
con.query("INSERT into users VALUES(3, 'Cathy')")result = con.query('SELECT * from users')
result.each do |row|
puts row
end
```Or, you can use block.
```ruby
require 'duckdb'DuckDB::Database.open do |db|
db.connect do |con|
con.query('CREATE TABLE users (id INTEGER, name VARCHAR(30))')con.query("INSERT into users VALUES(1, 'Alice')")
con.query("INSERT into users VALUES(2, 'Bob')")
con.query("INSERT into users VALUES(3, 'Cathy')")result = con.query('SELECT * from users')
result.each do |row|
puts row
end
end
end
```### Using bind variables
You can use bind variables.
```ruby
con.query('SELECT * FROM users WHERE name = ? AND email = ?', 'Alice', '[email protected]')
# or
con.query('SELECT * FROM users WHERE name = $name AND email = $email', name: 'Alice', email: '[email protected]')
```
### Using prepared statementYou can use prepared statement.
```ruby
stmt = con.prepare('SELECT * FROM users WHERE name = $name AND email = $email')
# or
# stmt = con.prepared_statement('SELECT * FROM users WHERE name = $name AND email = $email')
# or
# stmt = DuckDB::PreparedStatement.new(con, 'SELECT * FROM users WHERE name = $name AND email = $email')
stmt.bind(name: 'Alice', email: '[email protected]')
result = stmt.execute
```
### Using async queryYou can use async query.
```ruby
pending_result = con.async_query_stream('SLOW QUERY')
pending_result.execute_task while pending_result.state == :not_readyresult = pending_result.execute_pending
result.each.first
```Here is [the benchmark](./benchmark/async_query.rb).
### Using BLOB column
Use `DuckDB::Blob.new` or `my_string.force_encoding(Encoding::BINARY)`.
```ruby
require 'duckdb'DuckDB::Database.open do |db|
db.connect do |con|
con.query('CREATE TABLE blob_table (binary_data BLOB)')
stmt = DuckDB::PreparedStatement.new(con, 'INSERT INTO blob_table VALUES ($1)')stmt.bind(1, DuckDB::Blob.new("\0\1\2\3\4\5"))
# or
# stmt.bind(1, "\0\1\2\3\4\5".force_encoding(Encoding::BINARY))
stmt.executeresult = con.query('SELECT binary_data FROM blob_table')
puts result.first.first
end
end
```### Appender
Appender class provides Ruby interface of [DuckDB Appender](https://duckdb.org/docs/data/appender)
```ruby
require 'duckdb'
require 'benchmark'def insert
DuckDB::Database.open do |db|
db.connect do |con|
con.query('CREATE TABLE users (id INTEGER, name VARCHAR(30))')10000.times do
con.query("INSERT into users VALUES(1, 'Alice')")
end
end
end
enddef prepare
DuckDB::Database.open do |db|
db.connect do |con|
con.query('CREATE TABLE users (id INTEGER, name VARCHAR(30))')
stmt = con.prepared_statement('INSERT INTO users VALUES($1, $2)')10000.times do
stmt.bind(1, 1)
stmt.bind(2, 'Alice')
stmt.execute
end
end
end
enddef append
DuckDB::Database.open do |db|
db.connect do |con|
con.query('CREATE TABLE users (id INTEGER, name VARCHAR(30))')
appender = con.appender('users')10000.times do
appender.begin_row
appender.append(1)
appender.append('Alice')
appender.end_row
endappender.flush
end
end
endBenchmark.bm(8) do |x|
x.report('insert') { insert }
x.report('prepare') { prepare }
x.report('append') { append }
end# =>
# user system total real
# insert 0.637439 0.000000 0.637439 ( 0.637486 )
# prepare 0.230457 0.000000 0.230457 ( 0.230460 )
# append 0.012666 0.000000 0.012666 ( 0.012670 )
```### Configuration
Config class provides Ruby interface of [DuckDB configuration](https://duckdb.org/docs/api/c/config).
```ruby
require 'duckdb'config = DuckDB::Config.new
config['default_order'] = 'DESC'db = DuckDB::Database.open(nil, config)
con = db.connect
con.query('CREATE TABLE numbers (number INTEGER)')
con.query('INSERT INTO numbers VALUES (2), (1), (4), (3)')# number is ordered by descending
res = con.query('SELECT number FROM numbers ORDER BY number')
res.first.first # => 4
```