Mysql_Framework
Welcome to Mysql_Framework, this is a lightweight framework that provides managers to help with interacting with mysql.
Installation
Add this line to your application's Gemfile:
gem 'mysql_framework'Usage
Environment Variables
MySQL Connection Variables
-
MYSQL_HOST- MySQL Host -
MYSQL_PORT- MySQL Port -
MYSQL_DATABASE- MySQL database name -
MYSQL_USERNAME- MySQL username -
MYSQL_PASSWORD- MySQL password
MySQL Timeout Variables
-
MYSQL_READ_TIMEOUT- how long before connections time out when reading information from the DB (default:30seconds) -
MYSQL_WRITE_TIMEOUT- how long before connections time out when writing information to the DB (default:10seconds)
MySQL Connection Pooling Variables
-
MYSQL_START_POOL_SIZE- how many connections should be created by default (default:1) -
MYSQL_MAX_POOL_SIZE- how many connections should the pool be allowed to grow to (default:5)
MySQL Migration Variables
-
MYSQL_MIGRATION_TABLE- the name of the table that holds a record of applied migrations (default:migration_script_history) -
MYSQL_MIGRATION_LOCK_TTL- how long the tables should be locked for whilst performing migrations (default:2000/2 seconds) -
MYSQL_MIGRATION_LOCK_MAX_ATTEMPTS- how many times the lock manager should attempt to acquire the lock before failing (default:300) -
MYSQL_MIGRATION_LOCK_RETRY_DELAY_S- how long the lock manager should sleep between lock request attempts (default:1 second) -
REDIS_URL- The URL for redis - used for managing locks for DB migrations
Miscellaneous Variables
-
MYSQL_PARTITIONS- if a table is partitioned, how many partitions should be created (default:500)
Migration Scripts
Migration scripts need to be in the following format:
class CreateDemoTable < MysqlFramework::Scripts::Base
def initialize
@identifier = 201806021520 # 15:20 02/06/2018
end
def apply(client)
client.query(<<~SQL)
CREATE TABLE IF NOT EXISTS `#{table_name}` (
`id` CHAR(36) NOT NULL,
`name` VARCHAR(255) NULL,
`created_at` DATETIME NOT NULL,
`updated_at` DATETIME NOT NULL,
PRIMARY KEY (`id`)
)
SQL
end
def rollback(client)
client.query(<<~SQL)
DROP TABLE IF EXISTS `#{table_name}`
SQL
end
def tags
[table_name]
end
private
def table_name
DemoTable::NAME
end
end#initialize
The initialize method should set the @identifier value, which should be a timestamp:
@identifier = 201806021520 # 15:20 02/06/2018Make sure @identifier is an integer too, otherwise MysqlFramework::Scripts::Manager may struggle to determine which are your pending migrations.
#apply
The apply method should action the migration. An instance of Mysql2::Client is
available as client to use.
#rollback
The rollback method should action the migration. An instance of Mysql2::Client is
available as client to use.
#tags
Tags are used for when we want to specify which migrations to run based on a tag. This is useful for tests where you don't need to run all migrations to assert something is working or not.
Running migrations
Use the MysqlFramework::Scripts::Manager#execute method to run all pending migrations.
MysqlFramework::Scripts::Table
Used to register tables. This is used as part of the all_tables method in the script manager for
awareness of tables to drop.
class DemoTable
extend MysqlFramework::Scripts::Table
NAME = 'demo'
register_table NAME
endMysqlFramework::Connector
The connector deals with the connection pooling of MySQL2::Client instances, providing a wrapper for queries and transactions.
connector = MysqlFramework::Connector.new
connector.setup
connector.query(<<~SQL)
SELECT * FROM gems
SQLOptions can be provided to override the defaults as follows:
options = {
host: ENV.fetch('MYSQL_HOST'),
port: ENV.fetch('MYSQL_PORT'),
database: ENV.fetch('MYSQL_DATABASE'),
username: ENV.fetch('MYSQL_USERNAME'),
password: ENV.fetch('MYSQL_PASSWORD'),
reconnect: true
}
MysqlFramework::Connector.new(options)#setup
Sets up the connection pooling. Creates ENV['MYSQL_START_POOL_SIZE'] Mysql2::Client instances up front. This is provided as a separate method to allow for use within process forking where connections would need to be created after forking the process.
connector.setup#dispose
Closes all the Mysql2::Client connections and removes the connection pool. Intended as a clean-up method to be used on process fork shutdown.
connector.dispose#check_out
Check out a client from the connection pool. Will create new Mysql2::Client instances up-to ENV['MYSQL_MAX_POOL_SIZE'] times if no idle connections are available.
client = connector.check_out#check_in
Check in a client to the connection pool
client = connector.check_out
# ...
connector.check_in(client)#with_client
Called with a block. The method checks out a client from the pool and yields it to the block. Finally it ensures that the client is always checked back into the pool.
connector.with_client do |client|
client.query(<<~SQL)
SELECT * FROM gems
SQL
endIt can optionally accept an existing client to avoid starting new connections in the middle of a transaction. This can be used to ensure that a series of queries are wrapped by the same transaction.
connector.with_client(existing_client) do |client|
client.query(<<~SQL)
SELECT * FROM gems
SQL
end#execute
This method is called when executing a prepared statement where value substitution is required:
insert = MysqlFramework::SqlQuery.new.insert(gems)
.into(gems[:id],gems[:name],gems[:author],gems[:created_at],gems[:updated_at])
.values(SecureRandom.uuid,'mysql_framework','sage',Time.now,Time.now)
connector.execute(insert)It can optionally accept an existing client to avoid checking out a new client.
insert = MysqlFramework::SqlQuery.new.insert(gems)
.into(gems[:id],gems[:name],gems[:author],gems[:created_at],gems[:updated_at])
.values(SecureRandom.uuid,'mysql_framework','sage',Time.now,Time.now)
connector.execute(insert, existing_client)#query
This method is called to execute a query without having to worry about obtaining a client
connector.query(<<~SQL)
SELECT * FROM versions
SQLIt can optionally accept an existing client to avoid checking out a new client.
connector.query(<<~SQL, existing_client)
SELECT * FROM versions
SQL#transaction
This method requires a block and yields a client obtained from the pool. It wraps the yield in a BEGIN and COMMIT query. If an exception is raised then it will submit a ROLLBACK query and re-raise the exception.
insert = MysqlFramework::SqlQuery.new.insert(gems)
.into(gems[:id],gems[:name],gems[:author],gems[:created_at],gems[:updated_at])
.values(SecureRandom.uuid,'mysql_framework','sage',Time.now,Time.now)
connector.transaction do |client|
client.query(insert)
end#default_options
The default options used to initialise MySQL2::Client instances:
{
host: ENV.fetch('MYSQL_HOST'),
port: ENV.fetch('MYSQL_PORT'),
database: ENV.fetch('MYSQL_DATABASE'),
username: ENV.fetch('MYSQL_USERNAME'),
password: ENV.fetch('MYSQL_PASSWORD'),
reconnect: true
}MysqlFramework::SqlCondition
A representation of a MySQL Condition for a column. Created automatically by SqlColumn
# eq condition
SqlCondition.new(column: 'name', comparison: '=', value: 'mysql_framework')MysqlFramework::SqlColumn
A representation of a MySQL column within a table. Created automatically by SqlTable.
SqlCondition.new(table: 'gems', column: 'name')MysqlFramework::SqlQuery
A representation of a MySQL Query.
gems = MysqlFramework::SqlTable.new('gems')
guid = SecureRandom.uuid
# Insert Query
insert = MysqlFramework::SqlQuery.new.insert(gems)
.into(gems[:id],gems[:name],gems[:author],gems[:created_at],gems[:updated_at])
.values(guid,'mysql_framework','sage',Time.now,Time.now)
# Update Query
update = MysqlFramework::SqlQuery.new.update(gems)
.set(updated_at: Time.now)
.where(gems[:id].eq(guid))
# Delete Query
delete = MysqlFramework::SqlQuery.new.delete
.from(gems)
.where(gems[:id].eq(guid))
# Bulk Values Query
bulk_insert = MysqlFramework::SqlQuery.new.insert(gems)
.into(gems[:id],gems[:name],gems[:author],gems[:created_at],gems[:updated_at])
.bulk_values([[guid,'mysql_framework','sage',Time.now,Time.now], [guid,'mysql_framework','sage',Time.now,Time.now]])
# Bulk On Duplicate Query
bulk_upsert = MysqlFramework::SqlQuery.new.insert(gems)
.into(gems[:id],gems[:name],gems[:author],gems[:created_at],gems[:updated_at])
.bulk_values([[guid,'mysql_framework','sage',Time.now,Time.now], [guid,'mysql_framework','sage',Time.now,Time.now]])
.on_duplicate(gems[:id] => nil,gems[:name] => nil,gems[:author] => nil,gems[:created_at] => nil,gems[:updated_at] => nil)MysqlFramework::SqlTable
A representation of a MySQL table.
MysqlFramework::SqlTable.new('gems')Configuring Logs
As a default, MysqlFramework will log to STDOUT. You can provide your own logger using the logger= method:
MysqlFramework.logger = Logger.new('development.log')Contributing
Bug reports and pull requests are welcome on GitHub at https://github.com/sage/mysql_framework. This project is intended to be a safe, welcoming space for collaboration, and contributors are expected to adhere to the Contributor Covenant code of conduct.
Testing (with Docker)
A compose file is provided for running specs.
Setup
docker-compose up -d
docker-compose exec test-runner bash
# Once the shell opens in the container
bundle
Running specs
bundle exec rspec
Exit out of the shell when finished.
Cleanup
docker-compose down
License
This gem is available as open source under the terms of the MIT licence.
Copyright (c) 2018 Sage Group Plc. All rights reserved.