Ecosyste.ms: Awesome
An open API service indexing awesome lists of open source software.
https://github.com/ReactiveCouchbase/reactivecouchbase-rs-core
New ReactiveCouchbase driver using reactive-streams
https://github.com/ReactiveCouchbase/reactivecouchbase-rs-core
akka-streams async couchbase json non-blocking nosql reactive-programming reactive-streams scala
Last synced: about 2 months ago
JSON representation
New ReactiveCouchbase driver using reactive-streams
- Host: GitHub
- URL: https://github.com/ReactiveCouchbase/reactivecouchbase-rs-core
- Owner: ReactiveCouchbase
- License: apache-2.0
- Created: 2017-02-08T08:16:45.000Z (almost 8 years ago)
- Default Branch: master
- Last Pushed: 2018-04-23T09:37:17.000Z (over 6 years ago)
- Last Synced: 2024-08-04T00:05:38.127Z (5 months ago)
- Topics: akka-streams, async, couchbase, json, non-blocking, nosql, reactive-programming, reactive-streams, scala
- Language: Scala
- Size: 7.27 MB
- Stars: 26
- Watchers: 7
- Forks: 10
- Open Issues: 8
-
Metadata Files:
- Readme: readme.md
Awesome Lists containing this project
- awesome-scala - reactivecouchbase-rs-core - streams | ![GitHub stars](https://img.shields.io/github/stars/ReactiveCouchbase/reactivecouchbase-rs-core) ![GitHub commit activity](https://img.shields.io/github/commit-activity/y/ReactiveCouchbase/reactivecouchbase-rs-core) (Table of Contents / Database)
README
# ReactiveCouchbase RS (ReactiveStreams edition) [![Build Status](https://travis-ci.org/ReactiveCouchbase/reactivecouchbase-rs-core.svg?branch=master)](https://travis-ci.org/ReactiveCouchbase/reactivecouchbase-rs-core)
Yes, it's happening !!! with **ReactiveStreams** support ;-)
## If you want to try it
add a resolver to your `build.sbt` file
```scala
Resolver.bintrayRepo("mathieuancelin", "reactivecouchbase-maven")or
Resolver.jcenterRepo
```or you can build it to get the nice goodies
```sh
git clone https://github.com/ReactiveCouchbase/reactivecouchbase-rs-core.git
cd reactivecouchbase-rs-core
sbt ';clean;compile;publish-local'
```then in your project add the following dependency
```scala
libraryDependencies += "org.reactivecouchbase" %% "reactivecouchbase-rs-core" % "1.2.1"
```and you're ready to go
## A small example
```scala
import akka.actor.ActorSystem
import akka.stream.ActorMaterializer
import com.typesafe.config.ConfigFactory
import org.reactivecouchbase.rs.scaladsl.{N1qlQuery, ReactiveCouchbase}
import org.reactivecouchbase.rs.scaladsl.json._
import play.api.libs.json._
import akka.stream.scaladsl.Sink
import akka.actor.ActorSystem
import akka.stream.ActorMaterializer
import com.typesafe.config.ConfigFactoryobject ReactiveCouchbaseTest extends App {
val system = ActorSystem("ReactiveCouchbaseSystem")
implicit val materializer = ActorMaterializer.create(system)
implicit val ec = system.dispatcherval driver = ReactiveCouchbase(ConfigFactory.parseString(
"""
|buckets {
| default {
| name = "default"
| hosts = ["127.0.0.1"]
| }
|}
""".stripMargin))val bucket = driver.bucket("default")
val future = for {
_ <- bucket.insert[JsValue]("key1", Json.obj("message" -> "Hello World", "type" -> "doc"))
doc <- bucket.get("key1")
exists <- bucket.exists("key1")
docs <- bucket.search(N1qlQuery("select message from default where type = $type")
.on(Json.obj("type" -> "doc").asQueryParams))
.asSeq
messages <- bucket.search(N1qlQuery("select message from default where type = 'doc'"))
.asSource.map(doc => (doc \ "message").as[String].toUpperCase)
.runWith(Sink.seq[String])
_ <- driver.terminate()
} yield (doc, exists, docs)future.map {
case (_, _, docs) => println(s"found $docs")
}}
```## Example using custom environment and timeout
```scala
import akka.actor.ActorSystem
import akka.stream.ActorMaterializer
import com.typesafe.config.ConfigFactory
import com.couchbase.client.java.env.DefaultCouchbaseEnvironment
import org.reactivecouchbase.rs.scaladsl.{N1qlQuery, ReactiveCouchbase}
import org.reactivecouchbase.rs.scaladsl.json._
import play.api.libs.json._
import akka.stream.scaladsl.Sink
import akka.actor.ActorSystem
import akka.stream.ActorMaterializer
import com.typesafe.config.ConfigFactory
import scala.concurrent.duration._object ReactiveCouchbaseTest extends App {
val system = ActorSystem("ReactiveCouchbaseSystem")
implicit val materializer = ActorMaterializer.create(system)
implicit val ec = system.dispatcherval driver = ReactiveCouchbase(ConfigFactory.parseString(
"""
|buckets {
| default {
| name = "default"
| hosts = ["127.0.0.1"]
| }
|}
""".stripMargin))
// Sets a custom environment builder for the Couchbase bucket so that mutation tokens are enabled
def customEnv: DefaultCouchbaseEnvironment.Builder => DefaultCouchbaseEnvironment.Builder = env => env.mutationTokensEnabled(true)
// Uses the custom environment, sets a default timeout of 10 seconds for all operations
val bucket = driver.bucket("default", customEnv, Some(10.seconds))val future = for {
_ <- bucket.insert[JsValue]("key1", Json.obj("message" -> "Hello World", "type" -> "doc"))
doc <- bucket.get("key1")
exists <- bucket.exists("key1")
docs <- bucket.search(N1qlQuery("select message from default where type = $type")
.on(Json.obj("type" -> "doc").asQueryParams), Some(5.seconds)) // Overrides the default timeout with its own timeout of 5 seconds
.asSeq
messages <- bucket.search(N1qlQuery("select message from default where type = 'doc'"))
.asSource.map(doc => (doc \ "message").as[String].toUpperCase)
.runWith(Sink.seq[String])
_ <- driver.terminate()
} yield (doc, exists, docs)future.map {
case (_, _, docs) => println(s"found $docs")
}}
```## What about the Play Framework plugin ?
I don't think you actually need a plugin, if you want to use it from Play Framework, you can define a service to access your buckets like the following :
```scala
import javax.inject._
import play.api.inject.ApplicationLifecycle
import play.api.Configuration
import org.reactivecouchbase.rs.scaladsl._@Singleton
class Couchbase @Inject()(configuration: Configuration, lifecycle: ApplicationLifecycle) {private val driver = ReactiveCouchbase(configuration.underlying.getConfig("reactivecouchbase"))
def bucket(name: String): Bucket = driver.bucket(name)
lifecycle.addStopHook { () =>
driver.terminate()
}
}
```so you can define a controller like the following
```scala
import javax.inject._
import scala.concurrent.ExecutionContext
import play.api.mvc._
import akka.stream.Materializer
import play.api.libs.json._@Singleton
class ApiController @Inject()(couchbase: Couchbase)(implicit ec: ExecutionContext, materializer: Materializer) extends Controller {def eventsBucket = couchbase.bucket("events")
def events(filter: Option[String] = None) = Action {
val source = eventsBucket
.search(N1qlQuery("select id, payload, date, params, type from events where type = $type")
.on(Json.obj("type" -> filter.getOrElse("doc")).asQueryParams)
.asSource
.map(Json.stringify)
.intersperse("[", ",", "]")
Ok.chunked(source).as("application/json")
}
}
```## What if I want to use a JSON lib other than Play Json ?
you can easily do that, actually everything linked to Play Json is imported from:
```scala
import org.reactivecouchbase.rs.scaladsl.json._
```then you just have to reimplement a few things
```scala
import akka.util.ByteString
import com.couchbase.client.java.document.json.JsonObject
import org.reactivecouchbase.rs.scaladsl.json.{JsonReads, JsonWrites, JsonSuccess, QueryParams}
import foo.bar.jsonlib.{JsonNode, JsonObj}val read: JsonReads[JsonNode] = JsonReads(bs => JsonSuccess(JsonNode.parse(bs.utf8String)))
val write: JsonWrites[JsonNode] = JsonWrites(jsv => ByteString(JsonNode.stringify(jsv)))implicit val defaultByteStringFormat: JsonFormat[JsonNode] = JsonFormat(read, write)
implicit val defaultByteStringConverter: CouchbaseJsonDocConverter[JsonNode] = new CouchbaseJsonDocConverter[JsonNode] {
override def convert(ref: AnyRef): JsonNode = ...
}case class JsonObjQueryParams(query: JsonObj = ByteString.empty) extends QueryParams {
override def isEmpty: Boolean = !query.hasValue
override def toJsonObject: JsonObject = ...
}
```You have a few examples at
* https://github.com/ReactiveCouchbase/reactivecouchbase-rs-core/blob/master/src/main/scala/org/reactivecouchbase/rs/scaladsl/json/package.scala
* https://github.com/ReactiveCouchbase/reactivecouchbase-rs-core/blob/master/src/main/scala/org/reactivecouchbase/rs/scaladsl/json/bytestring.scala
* https://github.com/ReactiveCouchbase/reactivecouchbase-rs-core/blob/master/src/main/scala/org/reactivecouchbase/rs/scaladsl/json/circejson.scala
* https://github.com/ReactiveCouchbase/reactivecouchbase-rs-core/blob/master/src/main/scala/org/reactivecouchbase/rs/scaladsl/json/converter.scala#L21-L31## How about Circe support?
Reactive Couchbase also supports Circe for JSON serialization:
```scala
import io.circe.{ Encoder, Decoder }
import io.circe.syntax._
import io.circe.generic.semiauto._
import org.reactivecouchbase.rs.scaladsl.json._
import org.reactivecouchbase.rs.scaladsl.circejson._sealed case class TestModel(message: String, `type`: Option[String])
implicit val encoder: Encoder[TestModel] = deriveEncoder
implicit val decoder: Decoder[TestModel] = deriveDecoder
implicit val format2: JsonFormat[TestModel] = createCBFormat
```## Todo
* [ ] separate play-json and circe in modules ?
## Release
```
sbt "release with-defaults"
```