Module mysql
ballerinax/mysql Ballerina library
Overview
This module provides the functionality required to access and manipulate data stored in a MySQL database.
Prerequisite
Add the MySQL driver as a dependency to the Ballerina project.
Note:
ballerinax/mysql
supports MySQL driver versions above 8.0.13.
You can achieve this by importing the ballerinax/mysql.driver
module,
import ballerinax/mysql.driver as _;
ballerinax/mysql.driver
package bundles the latest MySQL driver JAR.
Tip: GraalVM native build is supported when
ballerinax/mysql
is used along with theballerinax/mysql.driver
If you want to add a MySQL driver of a specific version, you can add it as a dependency in Ballerina.toml. Follow one of the following ways to add the JAR in the file:
-
Download the JAR and update the path
[[platform.java17.dependency]] path = "PATH"
-
Add JAR with a maven dependency params
[[platform.java17.dependency]] groupId = "mysql" artifactId = "mysql-connector-java" version = "8.0.20"
Client
To access a database, you must first create a
mysql:Client
object.
The samples for creating a MySQL client can be found below.
Tip: The client should be used throughout the application lifetime.
Create a client
This sample shows the different ways of creating the mysql:Client
.
The client can be created with an empty constructor, and thereby, the client will be initialized with the default properties.
mysql:Client|sql:Error dbClient = new ();
The dbClient
receives the host, username, and password. Since the properties are passed in the same order as they are defined
in the mysql:Client
, they can be passed without named parameters.
mysql:Client|sql:Error dbClient = new ("localhost", "rootUser", "rootPass", "information_schema", 3306);
The sample below shows a mysql:Client
using named parameters to pass the attributes since some parameters are skipped in the constructor.
Further, the mysql:Options
property is passed to configure SSL and connection timeout properties in the MySQL client.
mysql:Options mysqlOptions = { ssl: { mode: mysql:SSL_PREFERRED }, connectTimeout: 10 }; mysql:Client|sql:Error dbClient = new (user = "rootUser", password = "rootPass", options = mysqlOptions);
Similarly in the sample below, the mysql:Client
uses named parameters, and it provides an unshared connection pool of type
sql:ConnectionPool
to be used within the client.
For more details about connection pooling, see the sql
Module.
mysql:Client|sql:Error dbClient = new (user = "rootUser", password = "rootPass", connectionPool = {maxOpenConnections: 5});
SSL usage
To connect to the MySQL database using an SSL connection, you must add the SSL configurations to mysql:Options
when creating the mysql:Client
.
For the SSL mode, you can select one of the modes: mysql:SSL_PREFERRED
, mysql:SSL_REQUIRED
, mysql:SSL_VERIFY_CA
, or mysql:SSL_VERIFY_IDENTITY
according to the requirement.
The key and cert files must be provided in the .p12
format.
string clientStorePath = "/path/to/keystore.p12"; string turstStorePath = "/path/to/truststore.p12"; mysql:Options mysqlOptions = { ssl: { mode: mysql:SSL_PREFERRED, key: { path: clientStorePath, password: "password" }, cert: { path: turstStorePath, password: "password" } } };
Handle connection pools
All database modules share the same connection pooling concept and there are three possible scenarios for
connection pool handling. For its properties and possible values, see sql:ConnectionPool
.
Note: Connection pooling is used to optimize opening and closing connections to the database. However, the pool comes with an overhead. It is best to configure the connection pool properties as per the application need to get the best performance.
-
Global, shareable, default connection pool
If you do not provide the
connectionPool
field when creating the client, a globally-shareable pool will be created for your database unless a connection pool matching with the properties you provided already exists.mysql:Client|sql:Error dbClient = new ("localhost", "rootUser", "rootPass");
-
Client-owned, unsharable connection pool
If you define the
connectionPool
field inline when creating the client with thesql:ConnectionPool
type, an unsharable connection pool will be created.mysql:Client|sql:Error dbClient = new ("localhost", "rootUser", "rootPass", connectionPool = { maxOpenConnections: 5 });
-
Local, shareable connection pool
If you create a record of the
sql:ConnectionPool
type and reuse that in the configuration of multiple clients, for each set of clients that connects to the same database instance with the same set of properties, a shared connection pool will be created.sql:ConnectionPool connPool = {maxOpenConnections: 5}; mysql:Client|sql:Error dbClient1 = new ("localhost", "rootUser", "rootPass", connectionPool = connPool); mysql:Client|sql:Error dbClient2 = new ("localhost", "rootUser", "rootPass", connectionPool = connPool); mysql:Client|sql:Error dbClient3 = new ("localhost", "rootUser", "rootPass", connectionPool = connPool);
For more details about each property, see the mysql:Client
constructor.
The mysql:Client
references
the sql:Client
and all the operations
defined by the sql:Client
will be supported by the mysql:Client
as well.
Close the client
Once all the database operations are performed, you can close the client you have created by invoking the close()
operation. This will close the corresponding connection pool if it is not shared by any other clients.
Note: The client must be closed only at the end of the application lifetime (or closed for graceful stops in a service).
error? e = dbClient.close();
Or
check dbClient.close();
Database operations
Once the client is created, database operations can be executed through that client. This module defines the interface and common properties that are shared among multiple database clients. It also supports querying, inserting, deleting, updating, and batch updating data.
Parameterized query
The sql:ParameterizedQuery
is used to construct the SQL query to be executed by the client.
You can create a query with constant or dynamic input data as follows.
Query with constant values
sql:ParameterizedQuery query = `SELECT * FROM students WHERE id < 10 AND age > 12`;
Query with dynamic values
int[] ids = [10, 50]; int age = 12; sql:ParameterizedQuery query = `SELECT * FROM students WHERE id < ${ids[0]} AND age > ${age}`;
Moreover, the SQL package has sql:queryConcat()
and sql:arrayFlattenQuery()
util functions which make it easier
to create a dynamic/constant complex query.
sql:queryConcat()
is used to create a single parameterized query by concatenating a set of parameterized queries.
The sample below shows how to concatenate queries.
int id = 10; int age = 12; sql:ParameterizedQuery query = `SELECT * FROM students`; sql:ParameterizedQuery query1 = ` WHERE id < ${id} AND age > ${age}`; sql:ParameterizedQuery sqlQuery = sql:queryConcat(query, query1);
A query with the IN
operator can be created using the sql:ParameterizedQuery
as shown below. Here you need to flatten the array and pass each element separated by a comma.
int[] ids = [1, 2, 3]; sql:ParameterizedQuery query = `SELECT count(*) as total FROM DataTable WHERE row_id IN (${ids[0]}, ${ids[1]}, ${ids[2]})`;
The util function sql:arrayFlattenQuery()
is used to make array flattening easier. It makes the inclusion of varying array elements into the query easier by flattening the array to return a parameterized query. You can construct a complex dynamic query with the IN
operator by using both functions as shown below.
int[] ids = [1, 2]; sql:ParameterizedQuery sqlQuery = sql:queryConcat(`SELECT * FROM DataTable WHERE id IN (`, sql:arrayFlattenQuery(ids), `)`);
Create tables
This sample creates a table with three columns. The first column is a primary key of type int
, while the second
column is of type int
and the other is of type varchar
.
The CREATE
statement is executed via the execute
remote method of the client.
// Create the ‘Students’ table with the ‘id’, 'age', and ‘name’ fields. sql:ExecutionResult result = check dbClient->execute(`CREATE TABLE student ( id INT AUTO_INCREMENT, age INT, name VARCHAR(255), PRIMARY KEY (id) )`); // A value of the sql:ExecutionResult type is returned for 'result'.
Insert data
These samples show the data insertion by executing an INSERT
statement using the execute
remote method
of the client.
In this sample, the query parameter values are passed directly into the query statement of the execute
remote method.
sql:ExecutionResult result = check dbClient->execute(`INSERT INTO student(age, name) VALUES (23, 'john')`);
In this sample, the parameter values, which are assigned to local variables are used to parameterize the SQL query in
the execute
remote method. This type of parameterized SQL query can be used with any primitive Ballerina type
such as string
, int
, float
, or boolean
and in that case, the corresponding SQL type of the parameter is derived
from the type of the Ballerina variable that is passed.
string name = "Anne"; int age = 8; sql:ParameterizedQuery query = `INSERT INTO student(age, name) VALUES (${age}, ${name})`; sql:ExecutionResult result = check dbClient->execute(query);
In this sample, the parameter values are passed as a sql:TypedValue
to the execute
remote method. Use the
corresponding subtype of the sql:TypedValue
such as sql:VarcharValue
, sql:CharValue
, sql:IntegerValue
, etc., when you need to
provide more details such as the exact SQL type of the parameter.
sql:VarcharValue name = new ("James"); sql:IntegerValue age = new (10); sql:ParameterizedQuery query = `INSERT INTO student(age, name) VALUES (${age}, ${name})`; sql:ExecutionResult result = check dbClient->execute(query);
Insert data with auto-generated keys
This sample demonstrates inserting data while returning the auto-generated keys. It achieves this by using the
execute
remote method to execute the INSERT
statement.
int age = 31; string name = "Kate"; sql:ParameterizedQuery query = `INSERT INTO student(age, name) VALUES (${age}, ${name})`; sql:ExecutionResult result = check dbClient->execute(query); // Number of rows affected by the execution of the query. int? count = result.affectedRowCount; // The integer or string generated by the database in response to a query execution. string|int? generatedKey = result.lastInsertId;
Query data
These samples show how to demonstrate the different usages of the query
operation to query the
database table and obtain the results as a stream.
Note: When processing the stream, make sure to consume all fetched data or close the stream.
This sample demonstrates querying data from a table in a database.
First, a type is created to represent the returned result set. This record can be defined as an open or a closed record
according to the requirement. If an open record is defined, the returned stream type will include both defined fields
in the record and additional database columns fetched by the SQL query which are not defined in the record.
Note the mapping of the database column to the returned record's property is case-insensitive if it is defined in the
record(i.e., the ID
column in the result can be mapped to the id
property in the record). Additional column names are
added to the returned record as in the SQL query. If the record is defined as a closed record, only defined fields in the
record are returned or gives an error when additional columns present in the SQL query. Next, the SELECT
query is executed
via the query
remote method of the client. Once the query is executed, each data record can be retrieved by iterating through
the result set. The stream
returned by the SELECT
operation holds a pointer to the actual data in the database, and it
loads data from the table only when it is accessed. This stream can be iterated only once.
// Define an open record type to represent the results. type Student record { int id; int age; string name; }; // Select the data from the database table. The query parameters are passed // directly. Similar to the `execute` samples, parameters can be passed as // sub types of `sql:TypedValue` as well. int id = 10; int age = 12; sql:ParameterizedQuery query = `SELECT * FROM students WHERE id < ${id} AND age > ${age}`; stream<Student, sql:Error?> resultStream = dbClient->query(query); // Iterating the returned table. check from Student student in resultStream do { // Can perform operations using the record 'student' of type `Student`. };
Defining the return type is optional, and you can query the database without providing the result type. Hence, the above sample can be modified as follows with an open record type as the return type. The property name in the open record type will be the same as how the column is defined in the database.
// Select the data from the database table. The query parameters are passed // directly. Similar to the `execute` samples, parameters can be passed as // sub types of `sql:TypedValue` as well. int id = 10; int age = 12; sql:ParameterizedQuery query = `SELECT * FROM students WHERE id < ${id} AND age > ${age}`; stream<record{}, sql:Error?> resultStream = dbClient->query(query); // Iterating the returned table. check from record{} student in resultStream do { // Can perform operations using the record 'student'. io:println("Student name: ", student.value["name"]); };
There are situations in which you may not want to iterate through the database and in that case, you may decide
to use the sql:queryRow()
operation. If the provided return type is a record, this method returns only the first row
retrieved by the query as a record.
int id = 10; sql:ParameterizedQuery query = `SELECT * FROM students WHERE id = ${id}`; Student retrievedStudent = check dbClient->queryRow(query);
The sql:queryRow()
operation can also be used to retrieve a single value from the database (e.g., when querying using
COUNT()
and other SQL aggregation functions). If the provided return type is not a record (i.e., a primitive data type)
, this operation will return the value of the first column of the first row retrieved by the query.
int age = 12; sql:ParameterizedQuery query = `SELECT COUNT(*) FROM students WHERE age < ${age}`; int youngStudents = check dbClient->queryRow(query);
Update data
This sample demonstrates modifying data by executing an UPDATE
statement via the execute
remote method of
the client.
int age = 23; sql:ParameterizedQuery query = `UPDATE students SET name = 'John' WHERE age = ${age}`; sql:ExecutionResult result = check dbClient->execute(query);
Delete data
This sample demonstrates deleting data by executing a DELETE
statement via the execute
remote method of
the client.
string name = "John"; sql:ParameterizedQuery query = `DELETE from students WHERE name = ${name}`; sql:ExecutionResult result = check dbClient->execute(query);
Batch update data
This sample demonstrates how to insert multiple records with a single INSERT
statement that is executed via the
batchExecute
remote method of the client. This is done by creating a table
with multiple records and
parameterized SQL query as same as the above execute
operations.
// Create the table with the records that need to be inserted. var data = [ { name: "John", age: 25 }, { name: "Peter", age: 24 }, { name: "jane", age: 22 } ]; // Do the batch update by passing the batches. sql:ParameterizedQuery[] batch = from var row in data select `INSERT INTO students ('name', 'age') VALUES (${row.name}, ${row.age})`; sql:ExecutionResult[] result = check dbClient->batchExecute(batch);
Execute stored procedures
This sample demonstrates how to execute a stored procedure with a single INSERT
statement that is executed via the
call
remote method of the client.
int uid = 10; sql:IntegerOutParameter insertId = new; sql:ProcedureCallResult result = check dbClient->call(`call InsertPerson(${uid}, ${insertId})`); stream<record{}, sql:Error?>? resultStr = result.queryResult; if resultStr is stream<record{}, sql:Error?> { check from record{} value in resultStr do { // Can perform operations using the record 'value'. }; } check result.close();
Note: Once the results are processed, the
close
method on thesql:ProcedureCallResult
must be called.
Note: The default thread pool size used in Ballerina is:
the number of processors available * 2
. You can configure the thread pool size by using theBALLERINA_MAX_POOL_SIZE
environment variable.