soarjs was originally a Java project and I converted it to a node module a year ago. Back to the Java days, writing JDBC programs is really a pain, especially when I have to convert a result set into objects.
When rewriting soar to work for node.js, I quickly found out object wrapping is not that a big problem on node as the node-mysql module has already done that. On the other hand, generating SQL statements is still a heck of a job and that's where the soar project can help developers a lot.
It seems using JSON to represent table schema is much more straight forward than using XML, so it may be a better idea to just choose JSON over XML. The soarjs module is clogged with the requirements to support both and it started to bug me that I cannot make the API really clean and fun to use.
So I created a new project, sql-soar which is based on the original soarjs project. With the new project, I got the freedom to make its API really clean and fun to use. If you're a soarjs developer, please switch to sql-soar and have fun!
SOAR (Simple Object Adapter for Relational database) is a relational database access tool. It allows developers to access database with/as Javascript objects. Unlike most ORM solutions, SOAR gives back to developers the full control of how SQL statements are generated. SOAR offers some benefits of ORM and tries to avoid its overhead and problems. Also, if you need to access multiple databases in an application, SOAR would greatly simplify the task for you.
Most developers would agree it's not a good idea to directly compose SQL statements inside programs. It's tedious and error-prone. It would be nice to have a light-weight tool to harness SQL with the following feaures:
-
Reusable: you can formulate a SQL statement into a template. You can later invoke that SQL template with various query conditions.
-
Less tedious: you don't have to hand code the SQL where clause any more. Just specify the query values and SOAR will do the tedious works for you.
-
Multiple database access: a simple way to access multiple databases within an application.
-
Full control: unlike most ORM solutions, you have full control of how SQL is generated and applied.
The following example shows how easily to access DB using SOAR. Assuming you have a table called 'Person', consider the sample code below:
var soar = requrie('soarjs');
var cmd = {
op: 'list',
expr: soar.sqlTemplate('Person')
};
soar.execute(cmd, {age: 25}, function(err, list) {
// list will include any person whose age is 25.
});
That's similar to issuing a SQL statement like:
SELECT * FROM Person WHERE age = 25;
Unlike sql statements, your can easily resue the command (this time with different query conditions):
soar.execute(cmd, {hobby: "coding"}, function(err, list) {
// list persons whose hobby is to do coding.
});
- It's no longer necessary to convert a sql template into a sql expression by the value() function as: var expr = sqlTemplate.value(). Actually, Sql templates and expressions will be treated as the same (v.1.2.0).
Please refer to the release notes for details.
npm install soarjs
Below are short cuts to major sections of this guide:
There are two ways to setup the database configuration in SOAR: using a config file or doing it programmatically.
Right beneath the SOAR installation directory, there is a config.json file which would look like:
{
"dbConfig": {
"host" : "127.0.0.1",
"database" : "soar",
"user" : "myDB_acc_name",
"password" : "xxxx",
"supportBigNumbers" : true,
"connectionLimit" : 32
},
"defPath": "file_path_to_the_data_view_files"
}
where host is the database host and database is the database name. user and password are the database user name and password respectively. SOAR ueses the mysql node module as its mySQL driver and the connection pool feature is turned on by default.
defPath is the file directory where data view files of a database are saved. For details about what data view files are, please refer to this article. If you'll use SQL templates instead of data views to access databases, you can omit this property.
You can configure the database connection settings right inside your node program. Here is how:
var soar = require('soarjs');
var options = {
dbConfig: {
"host" : "127.0.0.1",
"database" : "soar",
"user" : "myDB_acc_name",
"password" : "xxxx",
"supportBigNumbers" : true,
"connectionLimit" : 32
}
};
soar.config( options );
Using SOAR to access multiple databases can be extremely easy, but first you have to configure SOAR to connect to multiple databases. It turns out that is quite simple, too.
In your config.json file, use an array of options instead of a single configuration option with each option specifying the settings of each database. Below is an example:
[
{
"dbConfig": {
"host" : "127.0.0.1",
"database" : "db_1",
"user" : "db1_acc_name",
"password" : "xxxx",
"supportBigNumbers" : true,
"connectionLimit" : 32
},
"defPath": "file_path_to_the_data_view_files_of_db1"
},
{
"dbConfig": {
"host" : "127.0.0.1",
"database" : "db_2",
"user" : "db2_acc_name",
"password" : "xxxx",
"supportBigNumbers" : true,
"connectionLimit" : 32
},
"defPath": "file_path_to_the_data_view_files_of_db2"
}
]
If you need to connect to 10 databases in your application, then the configuration array should have 10 elements. Configuring multiple databases programmatically can be done in a similar way.
How to access each database in a multi-databases scenario will be explained in each database access method (query, list, create, update and delete) below.
If you'll use SQL templates instead of data views to access databases, you can omit the defPath property.
SOAR offers two types of programming styles to access databases: using sql templates or "Data View" XML definitions.
SQL templates allow you to compose and reuse SQL queries in a clean and managable way. Let's start with an example:
var soar = require('soarjs');
var stmp = soar.sqlTemplate('Person')
.column(['id', 'addr AS address', 'age'])
.filter( {name: 'age', op: '>='} )
.extra( 'ORDER BY id' );
var cmd = {
op: list,
expr: stmp
},
query = {age: 18};
soar.execute(cmd, query, function(err, list) {
// 'list' is the query result
});
soar.sqlTempalte(tableName) takes a table name as its input and returns a SQL Template object. With SQL templates, you can add columns, set query conditions and specify addtional options. Most SQL template functions will return the template object itself, so you can chain funcion calls such that SQL queries can be composed succintly.
Belows are APIs related to SQL templates:
This function can be used to execute SQL queries (query, list, insert, update and delete). The data parameter is a JSON object which contains data to be inserted or updated to a table. The query parameter is a JSON object which specifies actual query values. The cmd parameter has the following properties:
-
op: should be one of the following: 'query', 'list', 'insert', 'update' and 'delete'.
-
expr: a SQL expression which can be built using SQL templates as shown in the above sample code.
-
range: specifies the window of a result set. The range object can be created using the soar.range() function.
-
conn: a database connection object. You usually don't have to specify this property unless you want to do transactions.
(Note that the data and query parameters were used to be included as the properties of the cmd parameter. However, including data and query in the cmd paramater would reduce the possibilities of reusing the cmd parameter which actually defines how to access DBMS. As a result, data and query are extracted from the cmd parameter sinece v1.1.4. The old signature still works, but it's deprecated.)
If the data parameter is not needed, the function signature can be simplified to execute(cmd, query, cb).
cb is the callback function which receives an error and sometimes a result object (when it's a query, list or insert operation).
This function returns a SQL template object which can be used to build parameterized SQL statements. tableName is the name of a table. If you'll access multiple databases in an application, tableName has to be in the form of dbName.tableName so that SOAR knows which database to talk to.
This template function can be used to specify join conditions. Below is a sample code:
var stemp = soar.sqlTemplate('myTable AS myT');
stemp.join( {table: 'Location AS loc', onWhat: 'myT.locID=loc.locID'} );
If you want to make multiple joins, just call join() as many times as you need. The join information is also a plain object with the following properties:
- table: name of the joined table.
- type: if you want to make a left join, you can set this property to 'LEFT'.
- use: the common column name to join two tables.
- onWhat: the join clause. If the use property is specified, this property will be ignored.
This function can be used to add table columns. If column is a string, the specified column will be added to the existing column collection. You can also add all columns at once by specifying an array of column name strings.
This function is used to set query conditions (filter). The primitive format of a filter is a plain object with the following properties:
- name: name of the filter. It's also used as the key to retrieve filter value from a query object. This property is required.
- field: the real column name in a table. If this property is missing, the name property will be used instead.
- op: what comparator to be used. It can be '>', '=' or 'IS NULL', etc.
- noArg: when a query operation does not require argument (e.g. IS NULL), this property should be set to true.
Note that this function should be called just once for each SQL templat. Otherwise the new setting will replace the old one.
If you want to make a compound filter (ANDed or ORed filters), this is the function you need. op should be 'AND' or 'OR', and filters is an array of filters. Below is an example:
var stemp = soar.sqlTemplate('myTable'),
orFilters = stemp.chainFilters('OR', [
{name: 'region', op: '='},
{name: 'age', op: '>'}
]);
The resulting filter (orFilters) is a compound filter ORing two filters (region and age).
This function can add extra options to a SQL statement. extra is a string with possible values like 'GROUP BY col_name' or 'ORDER BY col_name'.
Below we'll show how to use soar.execute() to do query, list, insert, update and delete. We'll assume a 'Person' table and use it in the following sample codes.
If you expect a table query should return only one entity (even though there maybe multiple matches to your query), you can ask SOAR to do query. Below is a sample code to query a person from the Person table:
var stemp = soar.sqlTemplate('Person');
// 'expr' is a SQL expression equivalent to
// SELECT * FROM Person WHERE psnID=?
var expr = stemp.filter( {name: 'psnID', op: '='} );
// 'cmd' below is like a command to SOAR
// It will query the person whose psnID is 1
var cmd = {
op: 'query',
expr: expr
},
query = {psnID: 1};
// Invoke soar.execute() to get the result
soar.execute(cmd, query, function(err, data) {
// 'data' is the query result
});
If your query will return multiple entities, you should do a list. Below is a sample code to list persons whose age are over 25 from the Person table:
var stemp = soar.sqlTemplate('Person');
// 'expr' is a SQL expression equivalent to
// SELECT * FROM Person WHERE age > ?
var expr = stemp.filter( {name: 'age', op: '>'} );
// 'cmd' below is like a command to SOAR
// It will list all persons whose age is greater than 25
var cmd = {
op: 'list',
expr: expr
},
query = {age: 25};
// Invoke soar.execute() to get the result
soar.execute(cmd, query, function(err, list) {
// 'list' is the query result
});
Below is a sample code to insert to a (Person) table:
// 'expr' is a SQL expression equivalent to
// INSERT INTO Person VALUES (...)
var expr = soar.sqlTemplate('Person');
// 'cmd' below is like a command to SOAR
// It will isnert a person whose name is 'Scott Copper'
var cmd = {
op: 'insert',
expr: expr
},
data = {name: 'Scott Cooper'};
// Invoke soar.execute() to do the insert
soar.execute(cmd, data, null, function(err, value) {
// 'value' contains the primary key value of the inserted
// in this case, it will be something like:
// {psnID: _the_psnID_of_the_newly_inserted_entity}
// where 'psnID' is the primary key of the Person table
});
Below is a sample code to update a (Person) table:
var stemp = soar.sqlTemplate('Person');
// 'expr' is a SQL expression equivalent to
// Update Person set ... WHERE psnID=?
var expr = stemp.filter( {name: 'psnID', op: '='} );
// 'cmd' below is like a command to SOAR
// It will update the person whose psnID is 1
// 'data' specify the new value of the column 'name'
var cmd = {
op: 'update',
expr: expr
},
data = {name: 'John Mayer'},
query = {psnID: 1};
// Invoke soar.execute() to do the update
soar.execute(cmd, data, query, function(err) {
// you should check 'err' to see if anything goes wrong.
});
Below is a sample code to do delete from a (Person) table:
var stemp = soar.sqlTemplate('Person');
// 'expr' is a SQL expression equivalent to
// DELETE FROM Person WHERE psnID=?
var expr = stemp.filter( {name: 'psnID'} );
// 'cmd' below is like a command to SOAR
// It will delete the person whose psnID is 1
var cmd = {
op: 'delete',
expr: expr
},
query = {psnID: 1};
// Invoke soar.execute() to do the delete
soar.execute(cmd, query, function(err) {
// you should check 'err' to see if anything goes wrong.
});
Doing transaction is faily simple. All you need to do is to obtain a database connection and pass it to soar.execute(). Below is the sample code:
var expr = soar.sqlTemplate('Perons');
soar.getConnection( function(err, conn) {
// remember to specify database connection in 'cmd'
var cmd = {
op: 'insert',
expr: expr,
conn: conn
},
data = {name: 'Scott Cooper'};
conn.beginTransaction(function(err) {
soar.execute(option, data, null, function(err, data) {
if (err)
conn.rollback(function() {
// remember to release the connection
conn.release();
});
else
conn.commit(function(err) {
if (err)
conn.rollback(function() {
// remember to release the connection
conn.release();
});
else
conn.release();
});
});
};
});
Data view is a XML file used to formulate SQL queries. By parameterizing and formulating SQL queries in XML format, SQL queries can be easily reused and managed. For details about how to use "data view" to access databases, please refer to Using Data View To Access Database.
Besides accessing data, you can also use SOAR to manage table schema. Below are what you can do:
This function will create a database table. schema is a schema notation object which defines a table schema. Please refer to schema notation to know about what it is and how to create a schema notation. cb is a callback function when table creation is successful or erred.
If you want to call createTable() with a specific database conection object, you can do createTable(conn, schema, cb).
This function can be used to alter table schema. schema is a schema notation object which defines a table schema. Please refer to schema notation to know about what it is and how to create a schema notation. cb is a callback function when altering table is successfully done or erred.
If you want to call alterTable() with a specific database conection object, you can do alterTable(conn, schema, cb).
This function can be used to delete (drop) a table. tableName is the name of the table to be dropped. cb is a callback function when deleting table is successfully done or erred.
If you want to call deleteTable() with a specific database conection object, you can do deleteTable(conn, schema, cb).
This function can be used to derive schema from an existing table. tableName is the name of the table to be derived. cb(err, schema) is the callback function to return the derived schema. The returned schema object is the same as the schema notation as described in this document.
If you want to call describeTable() with a specific database conection object, you can do describeTable(conn, schema, cb).
If you want to know what SQLs are actually generated by SOAR, you can turn on debug messages as shown below:
soar.setDebug( true );
That will display generated SQL along with other debug information in console.
The SOAR package comes with some test files. To run those tests, sample data have to be built first. Inside the SOAR istallation, there is a "def" directory which includes schema.sql and sampleData.sql. Those two files can be used to build the sample data. In addition, remember to change the user name and password in your config.json file and the related database settings in the test programs.
In the current release, SOAR only supports mySQL. If you want to use SOAR for other databases such as Postgre, MS SQL server or Oracle DB, etc, you'll have to write your own SQL generator. Right now SQL generation is implemented by ./lib/sqlGenMySql.js.