-
Notifications
You must be signed in to change notification settings - Fork 11
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Loading status checks…
Bring back the akka module
Showing
8 changed files
with
2,540 additions
and
1 deletion.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
183 changes: 183 additions & 0 deletions
183
akka/src/main/scala/com/qvantel/jsonapi/akka/AkkaExceptionHandler.scala
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,183 @@ | ||
/* | ||
Copyright (c) 2017, Qvantel | ||
All rights reserved. | ||
Redistribution and use in source and binary forms, with or without | ||
modification, are permitted provided that the following conditions are met: | ||
* Redistributions of source code must retain the above copyright | ||
notice, this list of conditions and the following disclaimer. | ||
* Redistributions in binary form must reproduce the above copyright | ||
notice, this list of conditions and the following disclaimer in the | ||
documentation and/or other materials provided with the distribution. | ||
* Neither the name of the Qvantel nor the | ||
names of its contributors may be used to endorse or promote products | ||
derived from this software without specific prior written permission. | ||
THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND | ||
ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED | ||
WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE | ||
DISCLAIMED. IN NO EVENT SHALL Qvantel BE LIABLE FOR ANY | ||
DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES | ||
(INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; | ||
LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND | ||
ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT | ||
(INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS | ||
SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. | ||
*/ | ||
package com.qvantel.jsonapi.akka | ||
|
||
import com.qvantel.jsonapi.model.ErrorObject | ||
|
||
import _root_.spray.json.DefaultJsonProtocol._ | ||
import _root_.spray.json._ | ||
|
||
import akka.event.LoggingAdapter | ||
import akka.http.scaladsl.model.StatusCodes._ | ||
import akka.http.scaladsl.model.{StatusCode, IllegalRequestException, ContentType} | ||
import akka.http.scaladsl.settings.RoutingSettings | ||
import akka.http.scaladsl.model.{HttpEntity, HttpResponse, MediaTypes} | ||
import akka.http.scaladsl.server.Directives._ | ||
import akka.http.scaladsl.server._ | ||
import akka.http.scaladsl.server.AuthenticationFailedRejection._ | ||
|
||
import scala.util.control.NonFatal | ||
|
||
trait AkkaExceptionHandlerTrait { | ||
|
||
import AkkaExceptionHandlerObject._ | ||
|
||
val defaultAkkaRejectionHandler: RejectionHandler = RejectionHandler | ||
.newBuilder() | ||
.handle { | ||
case AuthenticationFailedRejection(cause, _) => | ||
val rejectionMessage = cause match { | ||
case CredentialsMissing => "The resource requires authentication, which was not supplied with the request" | ||
case CredentialsRejected => "The supplied authentication is invalid" | ||
} | ||
completeJsonApiError(Unauthorized, "Authentication Failed", rejectionMessage) | ||
|
||
case AuthorizationFailedRejection => | ||
completeJsonApiError(Forbidden, | ||
"Authorization Failed", | ||
"The supplied authentication is not authorized to access this resource") | ||
|
||
case MalformedFormFieldRejection(name, msg, _) => | ||
completeJsonApiError(BadRequest, "Malformed Form Field", "The form field '" + name + "' was malformed:\n" + msg) | ||
|
||
case MalformedHeaderRejection(headerName, msg, _) => | ||
completeJsonApiError(BadRequest, | ||
"Malformed Header", | ||
s"The value of HTTP header '$headerName' was malformed:\n" + msg) | ||
|
||
case MalformedQueryParamRejection(name, msg, _) => | ||
completeJsonApiError(BadRequest, | ||
"Malformed Query Param", | ||
"The query parameter '" + name + "' was malformed:\n" + msg) | ||
|
||
case MalformedRequestContentRejection(msg, _) => | ||
completeJsonApiError(BadRequest, "Malformed Request Content", "The request content was malformed:\n" + msg) | ||
|
||
case MethodRejection(supported) => | ||
completeJsonApiError(MethodNotAllowed, | ||
"HTTP method not allowed", | ||
"HTTP method not allowed, supported methods: " + supported.toString) | ||
|
||
case SchemeRejection(supported) => | ||
completeJsonApiError(BadRequest, | ||
"Uri scheme not allowed", | ||
"Uri scheme not allowed, supported schemes: " + supported) | ||
|
||
case MissingCookieRejection(cookieName) => | ||
completeJsonApiError(BadRequest, "Missing Cookie", s"Request is missing required cookie '$cookieName'") | ||
|
||
case MissingFormFieldRejection(fieldName) => | ||
completeJsonApiError(BadRequest, "Missing Form Field", s"Request is missing required form field '$fieldName'") | ||
|
||
case MissingHeaderRejection(headerName) => | ||
completeJsonApiError(BadRequest, "Missing Header", s"Request is missing required HTTP header '$headerName'") | ||
|
||
case MissingQueryParamRejection(paramName) => | ||
completeJsonApiError(BadRequest, | ||
"Missing Query Param", | ||
s"Request is missing required query parameter '$paramName'") | ||
|
||
case RequestEntityExpectedRejection => | ||
completeJsonApiError(BadRequest, "Request Entity Expected", "Request entity expected but not supplied") | ||
|
||
case TooManyRangesRejection(_) => | ||
completeJsonApiError(RangeNotSatisfiable, "Too Many Ranges", "Request contains too many ranges") | ||
|
||
case UnsatisfiableRangeRejection(unsatisfiableRanges, _) => | ||
completeJsonApiError( | ||
RangeNotSatisfiable, | ||
"Unsatisfiable Range", | ||
unsatisfiableRanges.mkString("None of the following requested Ranges were satisfiable:\n", "\n", "") | ||
) | ||
|
||
case UnacceptedResponseContentTypeRejection(supported) => | ||
completeJsonApiError( | ||
NotAcceptable, | ||
"Unaccepted Response Content Type", | ||
"Resource representation is only available with these Content-Types:\n" + supported.mkString("\n") | ||
) | ||
|
||
case UnacceptedResponseEncodingRejection(supported) => | ||
completeJsonApiError( | ||
NotAcceptable, | ||
"Unaccepted Response Encoding", | ||
"Resource representation is only available with these Content-Encodings:\n" + supported.mkString("\n") | ||
) | ||
|
||
case UnsupportedRequestContentTypeRejection(supported) => | ||
completeJsonApiError(UnsupportedMediaType, | ||
"Unsupported Request Content-Type", | ||
"There was a problem with the requests Content-Type:\n" + supported.mkString(" or ")) | ||
|
||
case UnsupportedRequestEncodingRejection(supported) => | ||
completeJsonApiError(BadRequest, | ||
"Unsupported Request Encoding", | ||
"The request Content-Encoding must be the following:\n" + supported.value) | ||
|
||
case ValidationRejection(msg, _) => | ||
completeJsonApiError(BadRequest, "Validation Rejection", msg) | ||
} | ||
.handleNotFound { | ||
completeJsonApiError(NotFound, NotFound.reason, NotFound.defaultMessage) | ||
} | ||
.result() | ||
|
||
def defaultAkkaExceptionHandler(implicit settings: RoutingSettings, log: LoggingAdapter): ExceptionHandler = | ||
ExceptionHandler { | ||
case e: IllegalRequestException => { | ||
extractRequestContext { ctx => | ||
log.warning("Illegal request {}\n\t{}\n\tCompleting with '{}' response", ctx.request, e.getMessage, e.status) | ||
complete(jsonApiErrorResponse(e.status, "Illegal Request", e.info.format(settings.verboseErrorMessages))) | ||
} | ||
} | ||
case NonFatal(e) => { | ||
extractRequestContext { ctx => | ||
log.error(e, "Error during processing of request {}", ctx.request) | ||
complete( | ||
jsonApiErrorResponse(InternalServerError, | ||
InternalServerError.reason, | ||
if (e.getMessage != null) e.getMessage else InternalServerError.defaultMessage)) | ||
} | ||
} | ||
} | ||
} | ||
|
||
object AkkaExceptionHandlerObject extends Rejection { | ||
|
||
def jsonApiError(code: StatusCode, title: String, detail: String): JsValue = | ||
JsObject("errors" -> List( | ||
ErrorObject(status = Some(code.intValue.toString), title = Some(title), detail = Some(detail))).toJson) | ||
|
||
def jsonApiErrorResponse(code: StatusCode, title: String, detail: String): HttpResponse = | ||
HttpResponse( | ||
status = code, | ||
entity = | ||
HttpEntity(ContentType(MediaTypes.`application/vnd.api+json`), jsonApiError(code, title, detail).prettyPrint)) | ||
|
||
def completeJsonApiError(code: StatusCode, title: String, detail: String): Route = | ||
complete(jsonApiErrorResponse(code, title, detail)) | ||
} |
195 changes: 195 additions & 0 deletions
195
akka/src/main/scala/com/qvantel/jsonapi/akka/JsonApiSupport.scala
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,195 @@ | ||
/* | ||
Copyright (c) 2017, Qvantel | ||
All rights reserved. | ||
Redistribution and use in source and binary forms, with or without | ||
modification, are permitted provided that the following conditions are met: | ||
* Redistributions of source code must retain the above copyright | ||
notice, this list of conditions and the following disclaimer. | ||
* Redistributions in binary form must reproduce the above copyright | ||
notice, this list of conditions and the following disclaimer in the | ||
documentation and/or other materials provided with the distribution. | ||
* Neither the name of the Qvantel nor the | ||
names of its contributors may be used to endorse or promote products | ||
derived from this software without specific prior written permission. | ||
THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND | ||
ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED | ||
WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE | ||
DISCLAIMED. IN NO EVENT SHALL Qvantel BE LIABLE FOR ANY | ||
DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES | ||
(INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; | ||
LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND | ||
ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT | ||
(INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS | ||
SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. | ||
*/ | ||
package com.qvantel.jsonapi.akka | ||
|
||
import _root_.akka.http.scaladsl.Http | ||
import _root_.akka.http.scaladsl.client.RequestBuilding | ||
import _root_.akka.http.scaladsl.marshalling._ | ||
import _root_.akka.http.scaladsl.model._ | ||
import _root_.akka.http.scaladsl.model.headers._ | ||
import _root_.akka.http.scaladsl.unmarshalling._ | ||
import _root_.akka.stream.Materializer | ||
import _root_.akka.stream.scaladsl._ | ||
import _root_.akka.util.{ByteString, Timeout} | ||
import _root_.spray.json._ | ||
|
||
import scala.concurrent.{ExecutionContext, Future} | ||
import scala.concurrent.duration._ | ||
|
||
import com.qvantel.jsonapi._ | ||
import com.qvantel.jsonapi.model.TopLevel | ||
|
||
trait JsonApiSupport extends JsonApiSupport0 { | ||
|
||
implicit def jsonApiCollectionMarshaller[T]( | ||
implicit writer: JsonApiWriter[T], | ||
printer: JsonPrinter = PrettyPrinter, | ||
metaProfiles: Set[MetaProfile] = Set.empty, | ||
sorting: JsonApiSorting = JsonApiSorting.Unsorted, | ||
sparseFields: Map[String, List[String]] = Map.empty, | ||
pagination: JsonApiPagination.PaginationFunc = JsonApiPagination.EmptyFunc): ToEntityMarshaller[Iterable[T]] = | ||
Marshaller.withFixedContentType(ct) { as => | ||
HttpEntity(ct, rawCollection(as)) | ||
} | ||
|
||
implicit def jsonApiCollectionRequestUnmarshaller[T]( | ||
implicit reader: JsonApiReader[T]): FromRequestUnmarshaller[Iterable[T]] = | ||
new FromRequestUnmarshaller[Iterable[T]] { | ||
override def apply(value: HttpRequest)(implicit ec: ExecutionContext, | ||
materializer: Materializer): Future[Iterable[T]] = { | ||
val include = value.uri.query().get("include").map(_.split(',').toSet).getOrElse(Set.empty[String]) | ||
value.entity.toStrict(10.seconds).flatMap(strictEntity => extractEntities(strictEntity.data, include)) | ||
} | ||
} | ||
|
||
implicit def jsonApiCollectionResponseUnmarshaller[T]( | ||
implicit reader: JsonApiReader[T]): FromResponseUnmarshaller[Iterable[T]] = | ||
new FromResponseUnmarshaller[Iterable[T]] { | ||
override def apply(value: HttpResponse)(implicit ec: ExecutionContext, | ||
materializer: Materializer): Future[Iterable[T]] = { | ||
val include = value.headers | ||
.find(_.name == JsonApiSupport.JsonApiIncludeHeader) | ||
.map(_.value.split(',').toSet) | ||
.getOrElse(Set.empty[String]) | ||
value.entity.toStrict(10.seconds).flatMap(strictEntity => extractEntities(strictEntity.data, include)) | ||
} | ||
} | ||
|
||
} | ||
|
||
trait JsonApiSupport0 { | ||
val ct = MediaTypes.`application/vnd.api+json` | ||
|
||
implicit val jsonApiTopLevelSingle: Unmarshaller[HttpEntity, TopLevel.Single] = { | ||
Unmarshaller.byteStringUnmarshaller.map { data => | ||
JsonParser(data.utf8String).asJsObject.convertTo[TopLevel.Single] | ||
} | ||
} | ||
|
||
implicit val jsonApiTopLevelCollection: Unmarshaller[HttpEntity, TopLevel.Collection] = { | ||
Unmarshaller.byteStringUnmarshaller.map { data => | ||
JsonParser(data.utf8String).asJsObject.convertTo[TopLevel.Collection] | ||
} | ||
} | ||
|
||
implicit val jsonApiErrorObject: Unmarshaller[HttpEntity, TopLevel.Errors] = { | ||
Unmarshaller.byteStringUnmarshaller.map { data => | ||
JsonParser(data.utf8String).asJsObject.convertTo[TopLevel.Errors] | ||
} | ||
} | ||
|
||
implicit def jsonApiOneMarshaller[T](implicit writer: JsonApiWriter[T], | ||
printer: JsonPrinter = PrettyPrinter, | ||
metaProfiles: Set[MetaProfile] = Set.empty, | ||
sorting: JsonApiSorting = JsonApiSorting.Unsorted, | ||
sparseFields: Map[String, List[String]] = Map.empty): ToEntityMarshaller[T] = | ||
Marshaller.withFixedContentType(ct) { a => | ||
HttpEntity(ct, rawOne(a)) | ||
} | ||
|
||
implicit def relatedResponseMarshaller[A]( | ||
implicit writer: JsonApiWriter[A], | ||
printer: JsonPrinter = PrettyPrinter, | ||
sorting: JsonApiSorting = JsonApiSorting.Unsorted, | ||
sparseFields: Map[String, List[String]] = Map.empty): ToEntityMarshaller[com.qvantel.jsonapi.RelatedResponse[A]] = | ||
PredefinedToEntityMarshallers.StringMarshaller.wrap(ct) { value => | ||
printer.apply(value.toResponse) | ||
} | ||
|
||
implicit def jsonApiOneRequestUnmarshaller[T](implicit reader: JsonApiReader[T]): FromRequestUnmarshaller[T] = | ||
new FromRequestUnmarshaller[T] { | ||
override def apply(value: HttpRequest)(implicit ec: ExecutionContext, materializer: Materializer): Future[T] = { | ||
val include = value.uri.query().get("include").map(_.split(',').toSet).getOrElse(Set.empty[String]) | ||
value.entity.toStrict(10.seconds).flatMap(strictEntity => extractEntity(strictEntity.data, include)) | ||
} | ||
} | ||
|
||
implicit def jsonApiOneResponseUnmarshaller[T](implicit reader: JsonApiReader[T]): FromResponseUnmarshaller[T] = | ||
new FromResponseUnmarshaller[T] { | ||
override def apply(value: HttpResponse)(implicit ec: ExecutionContext, materializer: Materializer): Future[T] = { | ||
val include = value.headers | ||
.find(_.name == JsonApiSupport.JsonApiIncludeHeader) | ||
.map(_.value.split(',').toSet) | ||
.getOrElse(Set.empty[String]) | ||
value.entity.toStrict(10.seconds).flatMap(strictEntity => extractEntity(strictEntity.data, include)) | ||
} | ||
} | ||
|
||
def extractEntity[T](data: ByteString, include: Set[String])(implicit reader: JsonApiReader[T], | ||
ec: ExecutionContext): Future[T] = | ||
Future { | ||
val json = JsonParser(data.decodeString("UTF-8")).asJsObject | ||
readOne[T](json, include) | ||
} | ||
|
||
def extractEntity[T](data: Source[ByteString, Any], include: Set[String])(implicit materializer: Materializer, | ||
reader: JsonApiReader[T], | ||
ec: ExecutionContext): Future[T] = | ||
data.runFold(ByteString(""))(_ ++ _).flatMap(extractEntity[T](_, include)) | ||
|
||
def extractEntities[T](data: ByteString, include: Set[String])(implicit reader: JsonApiReader[T], | ||
ec: ExecutionContext): Future[Iterable[T]] = | ||
Future { | ||
val json = JsonParser(data.decodeString("UTF-8")).asJsObject | ||
readCollection[T](json, include).toList | ||
} | ||
|
||
def extractEntities[T](data: Source[ByteString, Any], include: Set[String])( | ||
implicit materializer: Materializer, | ||
reader: JsonApiReader[T], | ||
ec: ExecutionContext): Future[Iterable[T]] = | ||
data.runFold(ByteString(""))(_ ++ _).flatMap(extractEntities[T](_, include)) | ||
} | ||
|
||
/** Custom SendReceive that adds the include params into X-Internal-Include | ||
* header that can be read by FromResponseUnmarshaller | ||
*/ | ||
object JsonApiClientAkka extends RequestBuilding { | ||
import _root_.akka.actor._ | ||
import _root_.akka.http.scaladsl.settings.{ClientConnectionSettings, ConnectionPoolSettings} | ||
|
||
import scala.concurrent.duration._ | ||
|
||
def jsonApiSendReceive(implicit refFactory: ActorRefFactory, | ||
executionContext: ExecutionContext, | ||
system: ActorSystem, | ||
futureTimeout: Timeout = 60.seconds): HttpRequest => Future[HttpResponse] = { | ||
|
||
val conSettings = ClientConnectionSettings(system.settings.config).withIdleTimeout(futureTimeout.duration) | ||
val timeoutSettings = ConnectionPoolSettings(system.settings.config).withConnectionSettings(conSettings) | ||
req => | ||
val response = Http().singleRequest(request = req, settings = timeoutSettings) | ||
req.uri.query().get("include") match { | ||
case Some(include) => response.map(_.withHeaders(RawHeader(JsonApiSupport.JsonApiIncludeHeader, include))) | ||
case None => response | ||
} | ||
} | ||
} | ||
|
||
object JsonApiSupport extends JsonApiSupport { | ||
val JsonApiIncludeHeader: String = "X-Internal-Include" | ||
} |
617 changes: 617 additions & 0 deletions
617
akka/src/test/scala/com/qvantel/jsonapi/JsonApiSortingAkkaSpec.scala
Large diffs are not rendered by default.
Oops, something went wrong.
106 changes: 106 additions & 0 deletions
106
akka/src/test/scala/com/qvantel/jsonapi/RelatedResponseAkkaSpec.scala
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,106 @@ | ||
/* | ||
Copyright (c) 2017, Qvantel | ||
All rights reserved. | ||
Redistribution and use in source and binary forms, with or without | ||
modification, are permitted provided that the following conditions are met: | ||
* Redistributions of source code must retain the above copyright | ||
notice, this list of conditions and the following disclaimer. | ||
* Redistributions in binary form must reproduce the above copyright | ||
notice, this list of conditions and the following disclaimer in the | ||
documentation and/or other materials provided with the distribution. | ||
* Neither the name of the Qvantel nor the | ||
names of its contributors may be used to endorse or promote products | ||
derived from this software without specific prior written permission. | ||
THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND | ||
ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED | ||
WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE | ||
DISCLAIMED. IN NO EVENT SHALL Qvantel BE LIABLE FOR ANY | ||
DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES | ||
(INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; | ||
LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND | ||
ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT | ||
(INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS | ||
SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. | ||
*/ | ||
package com.qvantel.jsonapi | ||
|
||
import com.qvantel.jsonapi.akka.JsonApiSupport._ | ||
|
||
import org.specs2.mutable._ | ||
import _root_.spray.json._ | ||
import _root_.spray.json.DefaultJsonProtocol._ | ||
import _root_.akka.http.scaladsl.testkit.Specs2RouteTest | ||
import _root_.akka.http.scaladsl.model._ | ||
import _root_.akka.http.scaladsl.server.Directives._ | ||
|
||
class RelatedResponseAkkaSpec extends Specification with Specs2RouteTest { | ||
def actorRefFactory = system | ||
implicit val apiRoot: com.qvantel.jsonapi.ApiRoot = ApiRoot(None) | ||
@jsonApiResource final case class Test(id: String, name: String) | ||
|
||
val test: Option[Test] = Some(Test("teståöä•Ωé®", "name")) // test UTF-8 | ||
val emptyTest: Option[Test] = None | ||
val tests: List[Test] = List(Test("test 1", "name 1"), Test("test 2", "name 2")) | ||
val emptyTests: List[Test] = List.empty | ||
|
||
val route = get { | ||
complete { | ||
RelatedResponse(test) | ||
} | ||
} | ||
|
||
"correctly write to one none case" in { | ||
RelatedResponse(emptyTest).toResponse must be equalTo JsObject( | ||
"data" -> JsNull | ||
) | ||
} | ||
|
||
"correctly write to one some case" in { | ||
val answer = rawOne(test.get) | ||
|
||
RelatedResponse(test).toResponse must be equalTo answer | ||
RelatedResponse(test.get).toResponse must be equalTo answer | ||
} | ||
|
||
"correctly write to one some case with sparse fields defined" in { | ||
implicit val sparseFields: Map[String, List[String]] = Map("tests" -> List("someFieldThatDoesNotExist")) | ||
val answer = rawOne(test.get) | ||
|
||
RelatedResponse(test).toResponse must be equalTo answer | ||
RelatedResponse(test.get).toResponse must be equalTo answer | ||
} | ||
|
||
"correctly write to many empty case" in { | ||
RelatedResponse(emptyTests).toResponse must be equalTo JsObject( | ||
"data" -> JsArray.empty | ||
) | ||
} | ||
|
||
"correctly write to many non-empty case" in { | ||
val answer = rawCollection(tests) | ||
|
||
RelatedResponse(tests).toResponse must be equalTo answer | ||
RelatedResponse(tests.toSeq).toResponse must be equalTo answer | ||
RelatedResponse(tests.toIterable).toResponse must be equalTo answer | ||
RelatedResponse(tests.toSet).toResponse must be equalTo answer | ||
} | ||
|
||
"correctly write to many non-empty case with sparse fields defined" in { | ||
implicit val sparseFields: Map[String, List[String]] = Map("tests" -> List("someFieldThatDoesNotExist")) | ||
|
||
val answer = rawCollection(tests) | ||
|
||
RelatedResponse(tests).toResponse must be equalTo answer | ||
RelatedResponse(tests.toSeq).toResponse must be equalTo answer | ||
RelatedResponse(tests.toIterable).toResponse must be equalTo answer | ||
RelatedResponse(tests.toSet).toResponse must be equalTo answer | ||
} | ||
|
||
"make sure that correct content type is given" in { | ||
Get("/") ~> route ~> check { | ||
contentType must be equalTo ContentType(MediaTypes.`application/vnd.api+json`) | ||
} | ||
} | ||
} |
424 changes: 424 additions & 0 deletions
424
akka/src/test/scala/com/qvantel/jsonapi/akka/AkkaExceptionHandlerSpec.scala
Large diffs are not rendered by default.
Oops, something went wrong.
942 changes: 942 additions & 0 deletions
942
akka/src/test/scala/com/qvantel/jsonapi/akka/JsonApiSupportSpec.scala
Large diffs are not rendered by default.
Oops, something went wrong.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters