MariaDB
The MariaDB connector provides a wrapper around MariaDB, allowing interaction between your Perfect Applications and MariaDB databases.
Note: Because of the history of MySQL & MariaDB, MariaDB keeps the same names as MySQL API functions; however, it is not possible to compile MariaDB in a MySQL configuration "as-is" context. This is why Perfect 3.0 includes both MySQL & MariaDB connectors.
System Requirements
macOS
Requires the use of Homebrew’s MariaDB connector.
brew install mariadb-connector-c
If you need Homebrew, you can install it with:
/usr/bin/ruby -e "$(curl -fsSL https://raw.githubusercontent.com/Homebrew/install/master/install)"
You will need to edit the mariadb.pc file:
/usr/local/lib/pkgconfig/mariadb.pc
Remove the occurrence of "-fno-omit-frame-pointer". This file is read-only by default so you will need to change its permissions first.
A typical configuration of mariadb.pc may looks like:
prefix=/usr/local
exec_prefix=${prefix}/bin
libdir=${prefix}/lib/mariadb
includedir=${prefix}/include/mariadb
Name: mariadb
Description: MariaDB Connector/C
Version: 5.5.1
Requires:
Libs: -L${libdir} -lmariadb -ldl -lm -lpthread
Cflags: -I${includedir}
Libs_r: -L${libdir} -lmariadb -ldl -lm -lpthread
Edit your ~/.bash_profile with the following line:
export PKG_CONFIG_PATH="/usr/local/lib/pkgconfig:/usr/lib/pkgconfig"
Linux
Ensure that you have installed libmariadb2 libmariadb-client-lgpl-dev for MariaDB:
sudo apt-get install clang pkg-config libmariadb2 libmariadb-client-lgpl-dev libcurl4-openssl-dev
Ensure the pkg-config file /usr/lib/pkgconfig/mariadb.pc specified for MariaDB should be MANUALLY added and corrected before building. The file will look similar to the following:
prefix=/usr
exec_prefix=${prefix}/bin
libdir=${prefix}/lib/mariadb
includedir=${prefix}/include/mariadb
Name: mariadb
Description: MariaDB Connector/C
Version: 5.5.0
Requires:
Libs: -L${libdir} -lmariadb -ldl -lm -lpthread
Cflags: -I${includedir}
Libs_r: -L${libdir} -lmariadb -ldl -lm -lpthread
Setup
Add the "Perfect-MariaDB" project as a dependency in your Package.swift file:
.Package(url:"https://github.com/PerfectlySoft/Perfect-MariaDB.git", majorVersion: 3)
Import
To use MariaDB within a file, import the required module:
import MariaDB
Quick Start
Access the Database
In order to access the database, set up your credentials:
let testHost = "127.0.0.1" let testUser = "test" let testPassword = "password" let testDB = "schema"
These are example credentials only. Replace with your own.
There are two common ways to connect to MariaDB. First, you can omit the schema, so that you can use a separate selector. This is handy if you have multiple schemas that your program can choose:
func fetchData() { let dataMysql = MySQL() // Create an instance of MySQL to work with let connected = mysql.connect(host: testHost, user: testUser, password: testPassword) guard connected else { // verify we connected successfully print(mysql.errorMessage()) return } defer { mysql.close() //This defer block makes sure we terminate the connection once finished, regardless of the result } //Choose the database to work with guard dataMysql.selectDatabase(named: testDB) else { Log.info(message: "Failure: \(dataMysql.errorCode()) \(dataMysql.errorMessage())") return } }
Alternatively, you can pass the database you would like to access into the connection and skip selection:
func fetchData() { let dataMysql = MySQL() // Create an instance of MariaDB MySQL Object to work with let connected = mysql.connect(host: testHost, user: testUser, password: testPassword, db: testDB) guard connected else { // verify we connected successfully print(mysql.errorMessage()) return } defer { mysql.close() //This defer block makes sure we terminate the connection once finished, regardless of the result } }
Create Tables
Choosing the database is great, but it is much more helpful to run queries, such as adding tables programmatically. Expanding on our connection example above, it is relatively simple to add a query:
func setupMySQLDB() { let dataMysql = MySQL() // Create an instance of MySQL to work with let connected = mysql.connect(host: testHost, user: testUser, password: testPassword, db: testDB) guard connected else { // verify we connected successfully print(mysql.errorMessage()) return } defer { mysql.close() //This defer block makes sure we terminate the connection once finished, regardless of the result } //Run Query to Add Tables let sql = """ CREATE TABLE IF NOT EXISTS ticket ( id VARCHAR(64) PRIMARY KEY NOT NULL, expiration INTEGER) """ guard mysql.query(statement: sql) else { // verify the table was created successfully print(mysql.errorMessage()) return } }
Run Queries
Getting data from your schema is essential, and relatively easy to do. After running a query, save your data and then act on it. In the example below, we’re assuming we have a table called options with a row id, an option name (text) and an option value (text):
func fetchData() { let dataMysql = MySQL() // Create an instance of MySQL to work with let connected = mysql.connect(host: testHost, user: testUser, password: testPassword, db: testDB) guard connected else { // verify we connected successfully print(mysql.errorMessage()) return } defer { mysql.close() //This defer block makes sure we terminate the connection once finished, regardless of the result } // Run the Query (for example all rows in an options table) let querySuccess = mysql.query(statement: "SELECT option_name, option_value FROM options") // make sure the query worked guard querySuccess else { return } // Save the results to use during this session let results = mysql.storeResults()! //We can implicitly unwrap because of the guard on the querySuccess. You’re welcome to use an if-let here if you like. var ary = [[String:Any]]() //Create an array of dictionaries to store our results in for use results.forEachRow { row in let optionName = getRowString(forRow: row[0]) //Store our Option Name, which would be the first item in the row, and therefore row[0]. let optionName = getRowString(forRow: row[1]) //Store our Option Value ary.append("\(optionName)":optionValue]) //store our options } }
MariaDB Server API
The MariaDB server API provides you with a set of tools to connect to and work with MariaDB server instances. This includes basic connections, disconnections, querying the instance for databases/tables, and running queries (which is actually a light wrapper for the full Statements API. Results returned, however, are handled and manipulated with the Results API. Statements also have a Statements API that lets you work with statements in much more detail than simply running queries though the main MySQL class.
init
public init()
Creates an instance of the MySQL class that allows you to interact with MariaDB databases.
NOTE for ⚠️Character Encoding⚠️
If your dataset contains non-ascii characters, please set this option for proper encoding:
setOption(MYSQL_SET_CHARSET_NAME, "utf8")
close
public func close()
Closes a connection to MariaDB. Most commonly used as a defer after guarding a connection, making sure that your session will close no matter what the outcome.
ping
MariaDB connection will go away when idle timeout. The ping()
function
can confirm the connectivity and also reconnect if need.
guard mysql.ping() else { // connection lost }
clientInfo
public static func clientInfo() -> String
This will give you a string of the MariaDB client library version, e.g. "5.5.x" or similar depending on your MariaDB installation.
errorCode & errorMessage
public func errorCode() -> UInt32
public func errorMessage() -> String
Error codes and messages are useful when debugging. These functions retrieve, display, and make use of those in Swift. You can learn more about what those mean here. This is especially useful after connecting or running queries. Example:
let mysql = MySQL() let connected = mysql.connect(host: dbHost, user: dbUser, password: dbPassword, db: dbName) guard connected else { // verify we connected successfully print(mysql.errorMessage()) return }
In this case, the console output would print any error messages that came up during a connection failure.
serverVersion
public func serverVersion() -> Int
Returns an integer representation of the MariaDB server’s version.
connect
public func connect(host hst: String? = nil, user: String? = nil, password: String? = nil, db: String? = nil, port: UInt32 = 0, socket: String? = nil, flag: UInt = 0) -> Bool
Opens a connection to the MariaDB database when supplied with the bare minimum credentials for your server (usually a host, user, and password). As an option, you can specify the port, database, or socket. Specifying the schema is not required, as you can use the selectDatabase() method after the connection has been made.
selectDatabase
public func selectDatabase(named namd: String) -> Bool
Selects a database from the active MariaDB connection.
listTables
public func listTables(wildcard wild: String? = nil) -> [String]
Returns an array of strings representing the different tables available on the selected database.
listDatabases
public func listDatabases(wildcard wild: String? = nil) -> [String]
Returns an array of strings representing the databases available on the MariaDB server currently connected.
commit
public func commit() -> Bool
Commits the transaction.
rollback
public func rollback() -> Bool
Rolls back the transaction.
moreResults
public func moreResults() -> Bool
Checks mysql_more_results
to see if any more results exist.
nextResult
public func nextResult() -> Int
Returns the next result in a multi-result execution. Most commonly used in a while loop to produce an effect similar to running forEachRow(). For example:
var results = [[String?]]() while let row = results?.next() { results.append(row) }
query
public func query(statement stmt: String) -> Bool
Runs an SQL Query given as a string.
storeResults
public func storeResults() -> MySQL.Results?
This retrieves a complete result set from the server and stores it on the client. This should be run after your query and before a function like forEachRow() or next() so that you can ensure that you iterate through all results.
setOption
public func setOption(_ option: MySQLOpt) -> Bool public func setOption(_ option: MySQLOpt, _ b: Bool) -> Bool public func setOption(_ option: MySQLOpt, _ i: Int) -> Bool public func setOption(_ option: MySQLOpt, _ s: String) -> Bool
Sets the options for connecting and returns a Boolean for success or failure. Requires a MySQLOpt and has several versions to support setting options that require Booleans, integers, or strings as values.
MySQLOpt values that are available to use are defined by the following enumeration:
public enum MySQLOpt { case MYSQL_OPT_CONNECT_TIMEOUT, MYSQL_OPT_COMPRESS, MYSQL_OPT_NAMED_PIPE, MYSQL_INIT_COMMAND, MYSQL_READ_DEFAULT_FILE, MYSQL_READ_DEFAULT_GROUP, MYSQL_SET_CHARSET_DIR, MYSQL_SET_CHARSET_NAME, MYSQL_OPT_LOCAL_INFILE, MYSQL_OPT_PROTOCOL, MYSQL_SHARED_MEMORY_BASE_NAME, MYSQL_OPT_READ_TIMEOUT, MYSQL_OPT_WRITE_TIMEOUT, MYSQL_OPT_USE_RESULT, MYSQL_OPT_USE_REMOTE_CONNECTION, MYSQL_OPT_USE_EMBEDDED_CONNECTION, MYSQL_OPT_GUESS_CONNECTION, MYSQL_SET_CLIENT_IP, MYSQL_SECURE_AUTH, MYSQL_REPORT_DATA_TRUNCATION, MYSQL_OPT_RECONNECT, MYSQL_OPT_SSL_VERIFY_SERVER_CERT, MYSQL_PLUGIN_DIR, MYSQL_DEFAULT_AUTH, MYSQL_OPT_BIND, MYSQL_OPT_SSL_KEY, MYSQL_OPT_SSL_CERT, MYSQL_OPT_SSL_CA, MYSQL_OPT_SSL_CAPATH, MYSQL_OPT_SSL_CIPHER, MYSQL_OPT_SSL_CRL, MYSQL_OPT_SSL_CRLPATH, MYSQL_OPT_CONNECT_ATTR_RESET, MYSQL_OPT_CONNECT_ATTR_ADD, MYSQL_OPT_CONNECT_ATTR_DELETE }
MariaDB Results API
The results API set provides a set of tools for working with result sets that are obtained from running queries.
close
public func close()
Closes the result set by releasing the results. Make sure you have a close function that is always executed after each session with a connection, otherwise you are going to have some memory problems. This is most commonly found in the defer block, just like in the examples at the top of this page.
dataSeek
public func dataSeek(_ offset: UInt)
Moves to an arbitrary row number in the results given an unsigned integer as an offset.
numRows
public func numRows() -> Int
Lets you know how many rows are in a result set.
numFields
public func numFields() -> Int
Similar to numRows, but returns the number of columns in a result set instead. Very useful for Select *
type queries where you may need to know how many columns are in the results.
next
public func next() -> Element?
Returns the next row in the result set as long as one exists.
forEachRow
public func forEachRow(callback: (Element) -> ())
Iterates through all rows in query results. Most useful for appending elements to an array or dictionary, just like we did in the quick start guide.
MariaDB Statements API
init
public init(_ mysql: MySQL)
Initializes the MySQL statement structure. This is very commonly used by other API functions to create a statement structure after you’ve passed in a string.
close
public func close()
This frees the MySQL statement structure pointer. Use it or lose valuable memory to the underlying MariaDB C API. Most commonly in a defer block, just like we used in quick start.
reset
public func reset()
Resets the statement buffers that are in the server. This doesn’t affect bindings or stored result sets. Learn more about this feature here.
clearBinds
func clearBinds()
Clears the current bindings.
freeResult
public func freeResult(
Releases memory tied up in with the result set produced by execution of a prepared statement. Also closes a cursor if one is open for the statement.
errorCode & errorMessage
public func errorCode() -> UInt32
public func errorMessage() -> String
Error codes and messages are useful when debugging. These functions retrieve, display, and make use of both in Swift. You can learn more about what those mean here. This is especially useful after connecting or running queries. Example:
let mysql = MySQL() let connected = mysql.connect(host: dbHost, user: dbUser, password: dbPassword, db: dbName) guard connected else { // verify we connected successfully print(mysql.errorMessage()) return }
In this case, the console output would print any error messages that came up during a connection failure.
prepare
public func prepare(statement query: String) -> Bool
Prepares a SQL statement for execution. More commonly called by other functions in the API, but public if you need it.
execute
public func execute() -> Bool
Executes a prepared statement.
results
public func results() -> MySQLStmt.Results
Returns current results from the server.
fetch
public func fetch() -> FetchResult
Fetches the next row of data from the result set.
numRows
public func numRows() -> UInt
Returns the row count in a buffered statement result set.
affectedRows
public func affectedRows() -> UInt
Returns the number of rows that were changed, deleted, or inserted by a prepared statement or an UPDATE, DELETE, or INSERT statement that was executed.
insertId
public func insertId() -> UInt
Returns the row id number for the last row inserted by a prepared statement, as long as id was an auto-increment enabled column.
fieldCount
public func fieldCount() -> UInt
Returns the number of columns in the results for the most recently executed statement.
nextResult
public func nextResult() -> Int
Returns the next result in a multi-result execution.
dataSeek
public func dataSeek(offset: Int)
Given an offset, it will seek to an arbitrary row in a statement result set.
paramCount
public func paramCount() -> Int
Returns the number of parameters in a prepared statement.
bindParam
public func bindParam() func bindParam(_ s: String, type: enum_field_types) public func bindParam(_ d: Double) public func bindParam(_ i: Int) public func bindParam(_ i: UInt64) public func bindParam(_ s: String) public func bindParam(_ b: UnsafePointer<Int8>, length: Int) public func bindParam(_ b: [UInt8])
Variations above on the bindParam() allow binding to statement parameters with different types. If no arguments are passed, it creates a null binding.