forked from avostryakov/Socket.IO-Java
-
Notifications
You must be signed in to change notification settings - Fork 18
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Add akka-http backend implementation
- Loading branch information
Showing
16 changed files
with
807 additions
and
2 deletions.
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
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 |
---|---|---|
|
@@ -69,6 +69,14 @@ | |
<role>contributor</role> | ||
</roles> | ||
</contributor> | ||
<contributor> | ||
<name>Brian Tarricone</name> | ||
<email>[email protected]</email> | ||
<timezone>-8</timezone> | ||
<roles> | ||
<role>contributor</role> | ||
</roles> | ||
</contributor> | ||
</contributors> | ||
|
||
<licenses> | ||
|
@@ -94,6 +102,7 @@ | |
<module>socket-io</module> | ||
<module>socket-io-servlet</module> | ||
<module>samples</module> | ||
<module>socket-io-akka-http-scala</module> | ||
</modules> | ||
|
||
<distributionManagement> | ||
|
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,108 @@ | ||
<?xml version="1.0" encoding="UTF-8"?> | ||
<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" | ||
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd"> | ||
<modelVersion>4.0.0</modelVersion> | ||
|
||
<parent> | ||
<artifactId>socketio-parent</artifactId> | ||
<groupId>com.codeminders.socketio</groupId> | ||
<version>2.0.0-SNAPSHOT</version> | ||
</parent> | ||
|
||
<artifactId>socket-io-akka-http-scala_${scala.binary.version}</artifactId> | ||
<packaging>jar</packaging> | ||
|
||
<properties> | ||
<scala.binary.version>2.12</scala.binary.version> | ||
<scala.version>2.12.8</scala.version> | ||
<akka.version>2.5.19</akka.version> | ||
<akka-http.version>10.1.6</akka-http.version> | ||
</properties> | ||
|
||
<dependencies> | ||
<dependency> | ||
<groupId>com.codeminders.socketio</groupId> | ||
<artifactId>socket-io</artifactId> | ||
<version>2.0.0-SNAPSHOT</version> | ||
</dependency> | ||
|
||
<dependency> | ||
<groupId>org.scala-lang</groupId> | ||
<artifactId>scala-library</artifactId> | ||
<version>${scala.version}</version> | ||
</dependency> | ||
|
||
<dependency> | ||
<groupId>com.typesafe.akka</groupId> | ||
<artifactId>akka-actor_${scala.binary.version}</artifactId> | ||
<version>${akka.version}</version> | ||
</dependency> | ||
<dependency> | ||
<groupId>com.typesafe.akka</groupId> | ||
<artifactId>akka-stream_${scala.binary.version}</artifactId> | ||
<version>${akka.version}</version> | ||
</dependency> | ||
<dependency> | ||
<groupId>com.typesafe.akka</groupId> | ||
<artifactId>akka-http-core_${scala.binary.version}</artifactId> | ||
<version>${akka-http.version}</version> | ||
</dependency> | ||
<dependency> | ||
<groupId>com.typesafe.akka</groupId> | ||
<artifactId>akka-http_${scala.binary.version}</artifactId> | ||
<version>${akka-http.version}</version> | ||
</dependency> | ||
</dependencies> | ||
|
||
<build> | ||
<plugins> | ||
<plugin> | ||
<groupId>net.alchim31.maven</groupId> | ||
<artifactId>scala-maven-plugin</artifactId> | ||
<version>3.4.4</version> | ||
<configuration> | ||
<recompileMode>incremental</recompileMode> | ||
<args> | ||
<arg>-deprecation</arg> | ||
<arg>-encoding</arg><arg>utf-8</arg> | ||
<arg>-feature</arg> | ||
<arg>-unchecked</arg> | ||
<arg>-Xlint:infer-any</arg> | ||
<arg>-Xlint:missing-interpolator</arg> | ||
<arg>-Xlint:nullary-unit</arg> | ||
<arg>-Xlint:private-shadow</arg> | ||
<arg>-Xlint:type-parameter-shadow</arg> | ||
<arg>-Xlint:unsound-match</arg> | ||
<arg>-Ypartial-unification</arg> | ||
<arg>-Yno-adapted-args</arg> | ||
<arg>-Ywarn-dead-code</arg> | ||
<arg>-Ywarn-extra-implicit</arg> | ||
<arg>-Ywarn-infer-any</arg> | ||
<arg>-Ywarn-nullary-unit</arg> | ||
<arg>-Ywarn-unused:imports</arg> | ||
<arg>-Ywarn-unused:locals</arg> | ||
<arg>-Ywarn-unused:privates</arg> | ||
</args> | ||
<charset>UTF-8</charset> | ||
</configuration> | ||
<executions> | ||
<execution> | ||
<id>scala-compile-first</id> | ||
<phase>process-resources</phase> | ||
<goals> | ||
<goal>add-source</goal> | ||
<goal>compile</goal> | ||
</goals> | ||
</execution> | ||
<execution> | ||
<id>scala-test-compile</id> | ||
<phase>process-test-resources</phase> | ||
<goals> | ||
<goal>testCompile</goal> | ||
</goals> | ||
</execution> | ||
</executions> | ||
</plugin> | ||
</plugins> | ||
</build> | ||
</project> |
10 changes: 10 additions & 0 deletions
10
socket-io-akka-http-scala/src/main/resources/reference.conf
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,10 @@ | ||
com.codeminders.socket-io.server.akka-http { | ||
# Whether or not to allow all origins via CORS | ||
allow-all-origins = true | ||
# A list of CORS allowed origins (only used if 'allow-all-origins' is false) | ||
allowed-origins = [] | ||
# Interval at which to expect application-level pings | ||
ping-interval = 25 seconds | ||
# Time after which a connection will be considered dead if a ping has not been received | ||
ping-timeout = 60 seconds | ||
} |
27 changes: 27 additions & 0 deletions
27
...tp-scala/src/main/scala/org/spurint/socketio/server/akkahttp/AkkaHttpRequestWrapper.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,27 @@ | ||
package org.spurint.socketio.server.akkahttp | ||
|
||
import akka.http.scaladsl.model.{HttpRequest => AkkaHttpRequest} | ||
import akka.stream.Materializer | ||
import akka.stream.scaladsl.{Keep, StreamConverters} | ||
import com.codeminders.socketio.server.HttpRequest | ||
import java.io.{BufferedReader, InputStream, InputStreamReader} | ||
import java.util.Locale | ||
|
||
private[akkahttp] class AkkaHttpRequestWrapper(request: AkkaHttpRequest)(implicit materializer: Materializer) extends HttpRequest { | ||
private lazy val query = request.uri.query() | ||
private lazy val entityInputStream = request.entity.dataBytes.toMat(StreamConverters.asInputStream())(Keep.right).run | ||
private lazy val entityReader = new BufferedReader(new InputStreamReader(entityInputStream)) | ||
|
||
override def getMethod: String = request.method.value | ||
|
||
override def getHeader(name: String): String = | ||
request.headers.find(_.lowercaseName == name.toLowerCase(Locale.US)).map(_.value).orNull | ||
|
||
override def getContentType: String = request.entity.contentType.value | ||
|
||
override def getParameter(name: String): String = query.get(name).orNull | ||
|
||
override def getInputStream: InputStream = entityInputStream | ||
|
||
override def getReader: BufferedReader = entityReader | ||
} |
60 changes: 60 additions & 0 deletions
60
...p-scala/src/main/scala/org/spurint/socketio/server/akkahttp/AkkaHttpResponseWrapper.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,60 @@ | ||
package org.spurint.socketio.server.akkahttp | ||
|
||
import akka.http.scaladsl.model | ||
import akka.http.scaladsl.model._ | ||
import akka.http.scaladsl.model.headers.RawHeader | ||
import akka.stream.scaladsl.StreamConverters | ||
import com.codeminders.socketio.server.HttpResponse | ||
import java.io.{OutputStream, PipedInputStream, PipedOutputStream} | ||
import java.nio.charset.StandardCharsets | ||
import java.util.concurrent.ConcurrentHashMap | ||
import scala.collection.JavaConverters._ | ||
|
||
private[akkahttp] class AkkaHttpResponseWrapper extends HttpResponse with AutoCloseable { | ||
@volatile private var statusCode: StatusCode = StatusCodes.OK | ||
private val headers = new ConcurrentHashMap[String, String] | ||
@volatile private var contentType: Option[String] = None | ||
|
||
private lazy val inputStream = new PipedInputStream() | ||
private lazy val outputStream = new PipedOutputStream(inputStream) | ||
private lazy val entitySource = StreamConverters.fromInputStream(() => inputStream) | ||
|
||
override def setHeader(name: String, value: String): Unit = headers.put(name, value) | ||
|
||
override def setContentType(contentType: String): Unit = this.contentType = Option(contentType) | ||
|
||
override def getOutputStream: OutputStream = outputStream | ||
|
||
override def sendError(statusCode: Int, message: String): Unit = { | ||
this.statusCode = parseStatusCode(statusCode) | ||
outputStream.write(message.getBytes(StandardCharsets.UTF_8)) | ||
close() | ||
} | ||
|
||
override def sendError(statusCode: Int): Unit = { | ||
this.statusCode = parseStatusCode(statusCode) | ||
close() | ||
} | ||
|
||
override def flushBuffer(): Unit = outputStream.flush() | ||
|
||
override def close(): Unit = { | ||
flushBuffer() | ||
outputStream.close() | ||
} | ||
|
||
lazy val toAkkaHttpResponse: Either[String, model.HttpResponse] = { | ||
close() | ||
this.contentType.map(s => ContentType.parse(s)).getOrElse(Right(ContentTypes.NoContentType)).map { contentType => | ||
model.HttpResponse( | ||
status = statusCode, | ||
headers = headers.asScala.map { case (name, value) => RawHeader(name, value) }.to[scala.collection.immutable.Seq], | ||
entity = HttpEntity(contentType, entitySource) | ||
) | ||
}.swap.map { errors => errors.map(_.summary).mkString("; ") }.swap | ||
} | ||
|
||
private def parseStatusCode(statusCode: Int): StatusCode = { | ||
StatusCodes.getForKey(statusCode).getOrElse(StatusCodes.custom(statusCode, "", "")) | ||
} | ||
} |
Oops, something went wrong.