The complete Scala example is here
A fauna query returns a future of a result. In real production code you should never block on a future and instead handle it async with map, flatMap, etc.
Only for demo purposes we are wrapping queries with await which blocks for the query to return and gets the result back. Never do this in production code.
This small helper method is used to make wrapping every call in await easier:
def await[T](f: Future[T]): T = Await.result(f, 5.second)
An admin connection should only be used to create top level databases. After the database is created, a separate client connection should be created.
If you are using the FaunaDB-Cloud version:
- remove the 'endpoint' argument below
- substitute "secret" for your authentication key's secret
val adminClient = FaunaClient("secret", "http://127.0.0.1:8443")
val DB_NAME = "demo"
val dbResults = await(adminClient.query(
Do(
If(
Exists(Database(DB_NAME)),
Delete(Database(DB_NAME)),
true),
CreateDatabase(Obj("name" -> DB_NAME)
)
)
))
println(s"Successfully created database ${dbResults("name").to[String].get} :\n $dbResults \n")
After the database is created, a new key specific to that database can be used to create a client connection to that database.
val keyResults = await(adminClient.query(
CreateKey(Obj("database" -> Database(DB_NAME), "role" -> "server"))
))
val key: String = keyResults("secret").to[String].get
val client = adminClient.sessionClient(key)
val SPELLS_COLLECTION = "spells"
val INDEX_NAME = "spells_index"
val collectionResults: Value = await(client.query(CreateCollection(Obj("name" -> SPELLS_COLLECTION))))
println(s"Create Collection for $DB_NAME:\n $collectionResults\n")
val indexResults: Value = await(client.query(
CreateIndex(
Obj("name" -> INDEX_NAME,
"source" -> Collection(SPELLS_COLLECTION)
)
)
))
println(s"Create Index for $DB_NAME:\n $indexResults\n")
val addFireResults = await(client.query(
Create(Collection(Value(SPELLS_COLLECTION)),
Obj("data" ->
Obj("name" -> "Fire Beak", "element" -> "water", "cost" -> 15)))
))
println(s"Added spell to collection $SPELLS_COLLECTION: \n $addFireResults \n")
val addHippoResults = await(client.query(
Create(Collection(Value(SPELLS_COLLECTION)),
Obj("data" ->
Obj("name" -> "Hippo's Wallow", "element" -> "water", "cost" -> 35)))
))
println(s"Added spell to collection $SPELLS_COLLECTION:\n $addHippoResults \n")
Adding data to a collection returns a reference to the resource with the reference, a timestamp and the corresponding object in a json structure like:
{
"object" : {
"ref" : {
....
},
"ts" : 1528414251402950,
"data" : {
....
}
}
}
Objects fields are accessed through the default method of Value
class. It's possible to access fields by names if the value represents an object or by index if it represents an array. For example to retrieve the resource reference of the returned Value use the following to get the ref
field:
//The results at 'ref' are a pointer to the document of the collection that was just created.
val hippoRef = addHippoResults("ref")
println(s"hippoRef = $hippoRef \n")
The query
method takes an Expr
object. Expr
objects can be composed with others Expr
to create complex query objects. faunadb.query
is a helper package where you can find all available expressions in the library.
Value readHippoResults = client.query(
Select(Value("data"),Get(hippoRef))
).get()
println("Hippo Spells:\n " + readHippoResults + "\n")
The query
method also accepts a timeout
parameter. The timeout
value defines the maximum time a query
will be allowed to run on the server. If the value is exceeded, the query is aborted. If no timeout
is defined in scope, a default value is assigned on the server side.
import scala.concurrent.duration._
val timeout = 500 millis
client.query(
Select(Value("data"), Get(hippoRef)),
timeout
)
That query returns the data in the form of a json object. It's possible to convert Value
class to its primitive correspondent using to
methods specifying a type. For example the data can be extracted from the results by using:
//convert the hippo results into primitive elements
val name: String = getHippoResults("name").to[String].get
val cost: Int = getHippoResults("cost").to[Int].get
val element: String = getHippoResults("element").to[String].get
println(s"Spell Details: Name=$name, Cost=$cost, Element=$element")
Later on we will show a better method for converting to native types with User Defined types that do this transformation automatically.
This object represents the result of an operation and it might be success or a failure. All conversion operations returns an object like this. This way it's possible to avoid check for nullability everywhere in the code.
//This would return an empty option if the field is not found or if the conversion fails
val optSpellElement: Option[String] = getHippoResults("element").to[String].toOpt
optSpellElement match {
case Some(element2) => println(s"optional spell element $element2")
case None => println("Something went wrong reading the spell")
}
Optionally it's possible transform one Result[T]
into another Result[T]
of different type using map
and flatMap
. If the result
represents an failure all calls to map
and flatMap
are ignored and it returns a new failure with the same error message. See faunadb.values.Result
for details.
The query
method takes an Expr
object. Expr
objects can be composed with others Expr
to create complex query objects. faunadb.query
is a helper package where you can find all available expressions in the library.
/*
* Query for all the spells in the index
*/
val queryIndexResults: Value = await(client.query(
SelectAll(Path("data", "id"),
Paginate(
Match(Index(Value(INDEX_NAME)))
)
)
))
That query returns a list of resource references to all the spells in the index. The list of references can be extracted from the results by using:
val spellsRefIds: Seq[String] = queryIndexResults.to[Seq[String]].getOrElse(Seq.empty)
println(s"spellsRefIds = $spellsRefIds \n")
Instead of manually creating your objects via the DSL (e.g. the Obj()), you can use case classes and codec to automatically encode and decode the class to user-defined types. These transform the types into the equivalent Value
types.
For example a Spell case class could be used that defines the fields and constructor:
case class Spell(name: String, element: String, cost: Option[Int])
object Spell {
implicit val spellCodec: Codec[Spell] = Codec.caseClass[Spell]
}
Option is used to mark a field that is optional and might not have a value
To persist a document of Spell
in FaunaDB:
val newSpell = Spell("Water Dragon's Claw", "water", Option(25))
val storeSpellResult = await(client.query(
Create(
Collection(SPELLS_COLLECTION),
Obj("data" -> newSpell))
))
println(s"Stored spell:\n $storeSpellResult \n")
Read the spell we just created and convert from a Value
type back to the Spell
type:
val dragonRef = storeSpellResult("ref")
val getDragonResult = await(client.query(
Select(
Value("data"),
Get(dragonRef)
)
))
val spell = getDragonResult.to[Spell].get
println(s"dragon spell: $spell \n")
Example of using a proper Future mapping to handle results async. Could be map, flatMap or a for expression.
for {
dragonSpellVal <- client.query(Select(Value("data"), Get(dragonRef)))
dragonSpell = dragonSpellVal.to[Spell].get
hippoSpellVal <- client.query(Select(Value("data"), Get(hippoRef)))
hippoSpell = hippoSpellVal.to[Spell].get
}
yield {
//process all spells retrieved
println(s"Retrieved spells of $dragonSpell and $hippoSpell")
}
To persist a Scala sequence of Spell
to FaunaDB because a codec is defined for the list it can be directly passed to the query Foreach. This will convert it into a Value
type:
val spellOne = Spell("Chill Touch", "ice", Option(18))
val spellTwo = Spell("Dancing Lights", "fire", Option(45))
val spellThree = Spell("Fire Bolt", "fire", Option(32))
val spellList = Seq(spellOne, spellTwo, spellThree)
//This query can be approximately read as for each spell in the list of spells evaluate the lambda function.
//That lambda function creates a temporary variable with each spell in the list and passes it to the create function.
//The create function then stores that spell in the database
val spellsListSave = await(client.query(
Foreach(spellList,
Lambda { nextSpell =>
Create(
Collection(Value(SPELLS_COLLECTION)),
Obj("data" -> nextSpell))
})
))
println(s"Created list of spells from java list: \n $spellsListSave \n")
val spellCollection = spellsListSave.to[Seq[Spell]].get
println(s"saved ${spellCollection.size} spells:")
spellCollection.foreach((nextSpell: Spell) => println(s" $nextSpell"))
println("\n")
Read a list of all Spells
out of the Spells
index and decode back to a Scala sequence of Spells
:
val findAllSpells = await(client.query(
SelectAll("data" / "data",
Map(
Paginate(Match(Index(Value(INDEX_NAME)))),
Lambda { x => Get(x) }
)
)
))
println(s"findAllSpells = $findAllSpells\n")
val allSpellsCollection = findAllSpells.to[Seq[Spell]].get
println(s"read ${allSpellsCollection.size} spells:")
allSpellsCollection.foreach((nextSpell: Spell) => println(s"$nextSpell"))