org.scalaprimavera.jdbc.simple.dsl.DSLJdbcTemplate
is template that expose a clean fluent interface to make JDBC Operations
- Initial version.
- Support for inject
scala.collection.inmutable.List
andscala.collection.inmutable.Map
in xml configuration. org.scalaprimavera.jdbc.simple.ScalaSimpleJdbcTemplate
is a wrapper fororg.springframework.jdbc.core.simple.SimpleJdbcTemplate
exposing his methods using Scala Collections and others goodies like High-Order Functions and Options
The jars will be uploaded to Maven repositories in the next few weeks. You can download the distribution here
The Jar names in ScalaPrimavera have a notation to make clear with versions of Scala and Spring are compiled
The notation is org.scalaprimavera.[module]-[versions]_[scalaVersion]_[springVersion].jar
Ex: org.scalaprimavera.jdbc-0.1_291_306.jar
for the jdbc module jar compiled with Scala 2.9.1 and Spring 3.0.6
The package structure in ScalaPrimavera tries to mirroring the structure of Spring, so the classes for jdbc are in the package org.scalaprimavera.jdbc
. We make this in order to make more easy to find that classes that you want.
Also in order to avoid classes with the same name, we prefix the classes that have the same name with Scala
Ex:ScalaSimpleJdbcTemplate
For logging we're following the SpringSource recomendations.
This module have all the classes that we need to build and inject beans. Spring Framework already have all the classes to build instances of Scala classes, but there's no way to build Scala collections.
ScalaPrimavera comes with a FactoryBean to build scala.collections.inmutable.List
<bean id="scalaStringsList" class="org.scalaprimavera.beans.factory.config.ScalaListFactoryBean">
<property name="sourceList">
<list>
<value>foo</value>
<value>bar</value>
</list>
</property>
</bean>
This will create a bean with type scala.collections.inmutable.List[String]
If you want to specify another type rather than String
you need to add this on the attribute value-type
of the tag list
. Ex:
<bean id="scalaIntList" class="org.scalaprimavera.beans.factory.config.ScalaListFactoryBean">
<property name="sourceList">
<list value-type="java.lang.Integer">
<value>foo</value>
<value>bar</value>
</list>
</property>
</bean>
This will create a bean of type scala.collections.inmutable.List[Int]
.
Also you can build lists of your own Scala or Java classes
<bean id="scalaUserList" class="org.scalaprimavera.beans.factory.config.ScalaListFactoryBean">
<property name="sourceList">
<list value-type="org.scalaprimavera.beans.User">
<bean class="org.scalaprimavera.beans.User">
<property name="name" value="John"/>
</bean>
</list>
</property>
</bean>
In the same way as lists, ScalaPrimavera also have a FactoryBean to build scala.collections.inmutable.Map
<bean id="stringIntMap" class="org.scalaprimavera.beans.factory.config.ScalaMapFactoryBean">
<property name="sourceMap">
<map key-type="java.lang.String" value-type="java.lang.Integer">
<entry key="foo" value="123"/>
<entry key="bar" value="456"/>
</map>
</property>
</bean>
ScalaPrimavera comes with a xml namespace to build Scala Collections
You only need to configure the namespace 'sp-collections' on your xml, in the same way that the namespaces provided with Spring
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xmlns:sp-collections="http://www.scalaprimavera.org/schema/sp-collections"
xsi:schemaLocation="http://www.springframework.org/schema/beans
http://www.springframework.org/schema/beans/spring-beans-3.0.xsd
http://www.scalaprimavera.org/schema/sp-collections
http://www.scalaprimavera.org/schema/collections/sp-collections-1.0.xsd">
<sp-collections:list id="scalaStringsList" value-type="java.lang.String">
<value>foo</value>
<value>bar</value>
</sp-collections:list>
<sp-collections:list id="scalaIntList" value-type="java.lang.Integer">
<value>123</value>
<value>456</value>
</sp-collections:list>
<sp-collections:map id="stringStringMap" key-type="java.lang.String" value-type="java.lang.String">
<entry key="foo" value="bar"/>
</sp-collections:map>
<sp-collections:map id="stringIntMap" key-type="java.lang.String" value-type="java.lang.Integer">
<entry key="foo" value="123"/>
<entry key="bar" value="456"/>
</sp-collections:map>
</beans>
Right now we're only supporting this two collections, if you want support for other collections, please add an issue on GitHub
This modules offer support for using SimpleJdbcTemplate
inside Scala. Also offer a new class DSLJdbcTemplate
that expose a clean fluent interface
Support for other classes will come in next releases
org.scalaprimavera.jdbc.core.simple.ScalaSimpleJdbcTemplate
is a wrapper for org.springframework.jdbc.core.simple.SimpleJdbcTemplate
.
This wrapper add support for receive and return Scala Collections, also use High-Order functions and Options
In order to create a ScalaSimpleJdbcTemplate
you need to inject a bean of type SimpleJdbcTemplate
In the next example we'll add an anonymous bean:
<bean id="template" class="org.scalaprimavera.jdbc.core.simple.ScalaSimpleJdbcTemplate">
<constructor-arg>
<bean class="org.springframework.jdbc.core.simple.SimpleJdbcTemplate">
<constructor-arg type="javax.sql.DataSource" ref="dataSource"/>
</bean>
</constructor-arg>
</bean>
To make a query that returns an Int/Long you must use the method queryForInt
to Int
or queryForLong
to Long
. Ex:
val count = template.queryForInt("select count(id) from users")
If you want to add parameters to the query you have three options:
First you can add more parameters to this function (The last parameter of this function is Any*
)
val count = template.queryForInt("select count(id) from users where age > ?",18)
The symbol ?
act as placeholder and will be replaced with 18
just like a PreparedStatement
. If there's more placeholders they will be replaced in order
val count = template.queryForInt("select count(id) from users where age > ? and first_name = ?",18,"John")
But also you can use a Map
val count = template.queryForInt("select count(id) from users where age > :age and first_name = :name",Map("age" -> 18,"name" -> "John"))
Now the placeholders have a name with a prefix :
, in this case :age
and :name
; and they will be replaced for the values in the map with the same key
The last option is to use an instance of org.springframework.jdbc.core.namedparam.SqlParameterSource
val count = template.queryForInt("select count(id) from users where age > :age", new MapSqlParameterSource("age",18))
Is a common scenario that some queries don't return any result, in this cases in Spring don't will return a null
but a org.springframework.dao.EmptyResultDataAccessException
will be thrown
try {
val age = template.queryForInt("select age from users where id > ?", 4567)
println("The age for user with id:" + 4567 + " is " + age)
}
catch {
case e:EmptyResultDataAccessException => println("Don't found data for user with id:" + 4567)
}
ScalaSimpleJdbcTemplate
provide a more elegant way to manage this situations using to the Scala Option[T]
template.queryForOptionInt("select age from users where id > ?", 4567) match {
case Some(age) => println("The age for user with id:" + 4567 + " is " + age)
case None => println("Don't found data for user with id:" + 4567)
}
In ScalaSimpleJdbcTemplate
every function that could be throw an EmptyResultDataAccessException
has a 'mirror' function that return an Option[T]
We really love Int
's but our databases have more types of fields. If we want to retrieve a String
we can use the queryForObject
function
val firstName = template.queryForObject("select first_name from users where id = ?", classOf[String], 4567)
You can extract any type that can be converted from SQL types to Java/Scala types
This function is overrided to receiving the query parameters as Map
, SqlParameterSource
, and Any*
.
If you want a function that returns a Option[T]
you can use queryForOption
The most common case is to extract an entity for our domain/model, so you can pass a function to map the data on a ResultSet
to your object
val user = template.queryForObject("select * from users where id = ?",
(set: ResultSet, i: Int) => {
new User(set.getInt("id"),set.getString("first_name"),set.getString("last_name"),set.getInt("age"))
}, 4567)
The first argument is our query, the second argument is a function of type (ResultSet, Int) => T
, the third argument is the query parameters; in this case is an Any*
but could be a Map
or a SqlParameterSource
As you can see the function type is very similar to the method mapRow
from the class org.springframework.jdbc.core.RowMapper
.
Behind de scenes ScalaPrimavera convert this function into an instance of RowMapper
If you want to reuse the function, you can declare it as a val
val mapperFunction = (set: ResultSet, i: Int) => {
new User(set.getInt("id"), set.getString("first_name"), set.getString("last_name"), set.getInt("age"))
}
val user = template.queryForObject("select * from users where id = ?",mapperFunction, 4567)
With some Scala tricks you can make the function more concise:
val mapperFunction = (set: ResultSet, i: Int) => {
import set._
new User(getInt("id"), getString("first_name"), getString("last_name"), getInt("age"))
}
If you want to retrieve a List[T]
you can use the function query
val mapperFunction = (set: ResultSet, i: Int) => {
import set._
new User(getInt("id"), getString("first_name"), getString("last_name"), getInt("age"))
}
val users:List[User] = template.query("select * from users where",mapperFunction)
Sometimes in complex queries the return type don't match any Entity of your domain. In those cases you can use queryForMap
val user = template.queryForMap("select firs_name, age as years from users where id = ?",4567)
println(user("FIRST_NAME") + " is " + user("YEARS") + " years old")
queryForMap
will return a Map[String,Any]
where the key is the column name
For returning more than one value you can use queryForList
that will return a List[Map[String,Any]]
The next table will help you to choose the right function for your query
Return Type | Method | Option Method |
---|---|---|
Int |
queryForInt |
queryForOptionInt |
Long |
queryForLong |
queryForOptionLong |
T |
queryForObject |
queryForOption |
List[T] |
query |
Non exists |
Map[String,Any] |
queryForMap |
queryForOptionMap |
List[Map[String,Any]] |
queryForList |
Non exists |
For operations like update, delete, insert and others you can use the function update
template.update("insert into users(first_name,last_name,age) values(?,?,?)", "John", "Doe", 24);
Also you can use a Map
to pass the query parameters
template.update("insert into users(first_name,last_name,age) values(:name,:lastName,:age)",
Map("name" -> "John","lastName" -> "Doe", "age" -> 24);
This functions will return an Int
with the count of modified rows
For batch operations you could use the batchUpdate
function
template.batchUpdate("delete from users where id = :id",
Seq(new MapSqlParameterSource("id", 1),
new MapSqlParameterSource("id", 3),
new MapSqlParameterSource("id", 4567),
new MapSqlParameterSource("id", 9078)))
Or like this
template.batchUpdate("insert into users(first_name,last_name,age) values(?,?,?)",
Seq(Seq("John", "Doe", 24),
Seq("Jane", "Doe", 22),
Seq("Fulano", "de Tal", 50),
Seq("Mike", "Foo", 24)),
Seq(java.sql.Types.VARCHAR, java.sql.Types.VARCHAR, java.sql.Types.INTEGER))
In the last argument you specify the SQL Type of the parameters, if the parameters have the default type (Like in the past example) you could pass an empty Seq
template.batchUpdate("insert into users(first_name,last_name,age) values(?,?,?)",
Seq(Seq("John", "Doe", 24),
Seq("Jane", "Doe", 22),
Seq("Fulano", "de Tal", 50),
Seq("Mike", "Foo", 24)),
Seq.empty)
org.scalaprimavera.jdbc.core.simple.dsl.DSLJdbcTemplate
is a class that offers a clean fluent interface to make JDBC Operations
In order to create a DSLJdbcTemplate
you need to inject a bean of type ScalaSimpleJdbcTemplate
In the next example we'll add an anonymous bean:
<bean id="template" class="org.scalaprimavera.jdbc.core.simple.dsl.DSLJdbcTemplate">
<constructor-arg>
<bean class="org.scalaprimavera.jdbc.core.simple.ScalaSimpleJdbcTemplate">
<constructor-arg>
<bean class="org.springframework.jdbc.core.simple.SimpleJdbcTemplate">
<constructor-arg type="javax.sql.DataSource" ref="dataSource"/>
</bean>
</constructor-arg>
</bean>
</constructor-arg>
</bean>
To make a query that returns an Int/Long you must use the method queryForInt
to Int
or queryForLong
to Long
. Ex:
val count = template.queryForInt("select count(id) from users where age > :age")
.paramaters("age" -> 18)
You can also use infix notation
val count = template queryForInt "select count(id) from users where age > :age" paramaters "age" -> 18
If you need to add more parameters
val count = template queryForInt "select count(id) from users where age > :age and first_name = :name" parameters("id" -> 18,"name" -> "John")
'DSLJdbcTemplate' also offer functions that return Option
like ScalaSimpleJdbcTemplate
If we want to retrieve a String
we can use the queryForObject
function
val firstName = template queryForObject "select first_name from users where id = :id" parameters "id" -> 4567 as classOf[String]
The most common case is to extract an entity for our domain/model
val user = template queryForObject "select * from users where id = :id" parameters "id" -> 4567 mappedWith {
(set, i) => {
new User(set.getInt("id"),set.getString("first_name"),set.getString("last_name"),set.getInt("age"))
}
}
If you want to reuse the mappedWith
argument function, you can declare it as a val
val mapperFunction = (set: ResultSet, i: Int) => {
new User(set.getInt("id"), set.getString("first_name"), set.getString("last_name"), set.getInt("age"))
}
val user = template queryForObject "select * from users where id = :id" parameters "id" -> 4567 mappedWith mapperFunction
If you want to retrieve a List[T]
you can use the function query
Sometimes in complex queries the return type don't match any Entity of your domain. In those cases you can use queryForMap
val user = template queryForMap "select firs_name, age as years from users where id = :id" parameters "id" -> 4567
println(user("FIRST_NAME") + " is " + user("YEARS") + " years old")
queryForMap
will return a Map[String,Any]
where the key is the column name
For returning more than one value you can use queryForList
that will return a List[Map[String,Any]]
For operations like update, delete, insert and others you can use the function update
template.update "insert into users(first_name,last_name,age) values(:name,:lastName,:age)" parameters("name" -> "John","lastName" -> "Doe", "age" -> 24)
Also you can use an instance of your's model class (Very useful on most MVC applications)
val user = new User("John", "Doe", 18)
template.update "insert into users(first_name,last_name,age) values(:firstName,:lastName,:age)" withThis user
In this case your model class need to have properties with the same name as the query parameters. In this example the 'User' class need to have properties with the name firstName
, lastName
and age
; and, of course, this properties need to have a type that could be converted to SQL types
This functions will return an Int
with the count of modified rows
If the functions that offer DSLJdbcTemplate
aren't enough for your needs, you always could fallback to ScalaSimpleJdbcTemplate
with the method noDSL
val count = template.noDSL.queryForInt("select count(id) from users")
ScalaPrimavera is a project to bring the Scala Language power to the Spring Framework world.
Some companies have a heavy investment on the Spring Framework and related projects, but also want to try Scala. ScalaPrimavera let's they try Scala in a safer way, while learn functional programming concepts that could improve code quality.
Cobalto Labs SAS is a company based in Bogotá, Colombia, with a broad experience in Spring development, we deliver SpringSource certified training and consulting across the Americas.
We develop succesful projects using Spring and Scala, and we know that huge impact that this combination could bring to a project.
ScalaPrimavera was developed as our internal library, now we are sharing this with the development community with an open source license (Apache 2.0)
Not a detailed one, but we have broad plan in mind:
From the version 0.1.* to the 0.9.* ScalaPrimavera will don't have a stable API. We'll could change the API, in a way that will don't be backwards compatible. This versions are designed to early adopters and application prototypes (Altough we already use this on production on our projects).
In the version 1.0.* we plan to have a stable API, that could be used in production
The versions 1.*.* we like to add some famous Scala libraries to the Spring world
Primavera is the spanish word for "Spring", in a serie of fortunate events this also a valid word in Italian, Catalan, Portuguese and other languages. We're very proud of our hispanic heritage, so we want to reflect this in the name of our project