Simple queries

Finagle-postgres can use PostgreSQL's simple query interface to perform queries. These queries are limited, because they cannot be parameterized - any data that will be provided by the client must be included in the query string itself. Therefore, the simple query interface should not be used when any user input is going to be used in the query, as that could be a vector for SQL injection attacks.

Still, it can be useful to try them out.

import com.twitter.finagle.Postgres
import com.twitter.util.Await
// create the client based on environment variables
val client = {
  Postgres.Client()
    .withCredentials(sys.env("PG_USER"), sys.env.get("PG_PASSWORD"))
    .database(sys.env("PG_DBNAME"))
    .withSessionPool.maxSize(1)
    .withBinaryResults(true)
    .withBinaryParams(true)
    .newRichClient(sys.env("PG_HOST_PORT"))
}
  
// execute a query that has no results - i.e. CREATE TABLE, UPDATE, INSERT, DELETE, etc.
val create = Await.result {
  client.execute("CREATE TABLE demo(id serial PRIMARY KEY, foo text)")
}

val insert = Await.result {
  client.execute("INSERT INTO demo(foo) VALUES ('foo')")
}

// execute a query that has results - a function is given to treat the rows
val select = client.select("SELECT * FROM demo") {
  row => row.get[String]("foo")
}

Await.result(select)

Since the results of any operation in finagle are typically a Future, we use Await.result to block and wait for the result to materialize. This is handy for a demo, but in production code you wouldn't use await. Instead, you would sequence operations using flatMap, and parallelize them using Future.join or Future.sequence. See the Finagle User Guide for more information about concurrent programming with Futures.

As you can see, the SELECT query includes a function body in a second argument list. This function will receive each Row that results from the query, and the results of the function will be accumulated into the resulting Future[Seq]. Row is an interface that allows retrieving values from Postgres rows in a type-safe way using the following interface:

Each of these methods additionally has an overloaded version that takes Int, specifying the index of the column rather than its name.

ValueDecoder

All of the typed methods require an implicit ValueDecoder[T], which is a typeclass that tells finagle-postgres how to decode the column. Currently, instances are supplied for the following Scala types:

New ValueDecoder instances can be specified for other types; take a look at the existing instances for guidance. We are happy to accept instances for built-in Scala or Java types into finagle-postgres!

Next, read about Parameterized Queries