Cache Body toBuffer
/toString
Results (#189)
* cache `body` processing results (Buffer, String) with `Ref` * add `readBodyAs(Buffer|Stream|String)` for accessing `body` * fix tests * add tests for `readBodyAsBuffer` and `readBodyAsString` * move Body to HTTPure.Body and rename it to RequestBody * add HTTPure.Body.toStream * consolidate `readBodyAs(Buffer|String)` into `to(Buffer|String)` and move `Ref` from top level `body` down to `buffer` and `string` fields * fix tests * import constructors explicitly * revert changes This reverts commit e53188c3e6d74ca00d3d891829ca91f0803b870b. * update `Body.read` to return `RequestBody`
This commit is contained in:
parent
3f20f1cff8
commit
4319cffed6
@ -11,7 +11,7 @@ module HTTPure
|
|||||||
, module HTTPure.Status
|
, module HTTPure.Status
|
||||||
) where
|
) where
|
||||||
|
|
||||||
import HTTPure.Body (toString, toBuffer)
|
import HTTPure.Body (toBuffer, toStream, toString)
|
||||||
import HTTPure.Headers (Headers, empty, header, headers)
|
import HTTPure.Headers (Headers, empty, header, headers)
|
||||||
import HTTPure.Lookup (at, (!@), has, (!?), lookup, (!!))
|
import HTTPure.Lookup (at, (!@), has, (!?), lookup, (!!))
|
||||||
import HTTPure.Method (Method(..))
|
import HTTPure.Method (Method(..))
|
||||||
|
@ -1,47 +1,99 @@
|
|||||||
module HTTPure.Body
|
module HTTPure.Body
|
||||||
( class Body
|
( class Body
|
||||||
|
, RequestBody
|
||||||
, defaultHeaders
|
, defaultHeaders
|
||||||
, write
|
, write
|
||||||
, read
|
, read
|
||||||
, toString
|
|
||||||
, toBuffer
|
, toBuffer
|
||||||
|
, toStream
|
||||||
|
, toString
|
||||||
) where
|
) where
|
||||||
|
|
||||||
import Prelude
|
import Prelude
|
||||||
import Data.Either (Either(Right))
|
import Data.Either (Either(Right))
|
||||||
|
import Data.Maybe (Maybe(Just, Nothing))
|
||||||
import Effect (Effect)
|
import Effect (Effect)
|
||||||
import Effect.Class (liftEffect)
|
|
||||||
import Effect.Aff (Aff, makeAff, nonCanceler)
|
import Effect.Aff (Aff, makeAff, nonCanceler)
|
||||||
import Effect.Ref (read) as Ref
|
import Effect.Class (liftEffect)
|
||||||
import Effect.Ref (new, modify)
|
import Effect.Ref (Ref)
|
||||||
|
import Effect.Ref (read, modify, new, write) as Ref
|
||||||
import HTTPure.Headers (Headers, header)
|
import HTTPure.Headers (Headers, header)
|
||||||
import Node.Buffer (toString) as Buffer
|
|
||||||
import Node.Buffer (Buffer, concat, fromString, size)
|
import Node.Buffer (Buffer, concat, fromString, size)
|
||||||
|
import Node.Buffer (toString) as Buffer
|
||||||
import Node.Encoding (Encoding(UTF8))
|
import Node.Encoding (Encoding(UTF8))
|
||||||
import Node.HTTP (Request, Response, requestAsStream, responseAsStream)
|
import Node.HTTP (Request, Response, requestAsStream, responseAsStream)
|
||||||
import Node.Stream (write) as Stream
|
|
||||||
import Node.Stream (Stream, Readable, onData, onEnd, writeString, pipe, end)
|
import Node.Stream (Stream, Readable, onData, onEnd, writeString, pipe, end)
|
||||||
|
import Node.Stream (write) as Stream
|
||||||
import Type.Equality (class TypeEquals, to)
|
import Type.Equality (class TypeEquals, to)
|
||||||
|
|
||||||
|
type RequestBody =
|
||||||
|
{ buffer :: Ref (Maybe Buffer)
|
||||||
|
, stream :: Readable ()
|
||||||
|
, string :: Ref (Maybe String)
|
||||||
|
}
|
||||||
|
|
||||||
-- | Read the body `Readable` stream out of the incoming request
|
-- | Read the body `Readable` stream out of the incoming request
|
||||||
read :: Request -> Readable ()
|
read :: Request -> Effect RequestBody
|
||||||
read = requestAsStream
|
read request = do
|
||||||
|
buffer <- Ref.new Nothing
|
||||||
|
string <- Ref.new Nothing
|
||||||
|
pure
|
||||||
|
{ buffer
|
||||||
|
, stream: requestAsStream request
|
||||||
|
, string
|
||||||
|
}
|
||||||
|
|
||||||
-- | Slurp the entire `Readable` stream into a `String`
|
-- | Turn `RequestBody` into a `String`
|
||||||
toString :: Readable () -> Aff String
|
-- |
|
||||||
toString = toBuffer >=> Buffer.toString UTF8 >>> liftEffect
|
-- | This drains the `Readable` stream in `RequestBody` for the first time
|
||||||
|
-- | and returns cached result from then on.
|
||||||
|
toString :: RequestBody -> Aff String
|
||||||
|
toString requestBody = do
|
||||||
|
maybeString <-
|
||||||
|
liftEffect
|
||||||
|
$ Ref.read requestBody.string
|
||||||
|
case maybeString of
|
||||||
|
Nothing -> do
|
||||||
|
buffer <- toBuffer requestBody
|
||||||
|
string <- liftEffect
|
||||||
|
$ Buffer.toString UTF8 buffer
|
||||||
|
liftEffect
|
||||||
|
$ Ref.write (Just string) requestBody.string
|
||||||
|
pure string
|
||||||
|
Just string -> pure string
|
||||||
|
|
||||||
-- | Slurp the entire `Readable` stream into a `Buffer`
|
-- | Turn `RequestBody` into a `Buffer`
|
||||||
toBuffer :: Readable () -> Aff Buffer
|
-- |
|
||||||
toBuffer stream =
|
-- | This drains the `Readable` stream in `RequestBody` for the first time
|
||||||
|
-- | and returns cached result from then on.
|
||||||
|
toBuffer :: RequestBody -> Aff Buffer
|
||||||
|
toBuffer requestBody = do
|
||||||
|
maybeBuffer <-
|
||||||
|
liftEffect
|
||||||
|
$ Ref.read requestBody.buffer
|
||||||
|
case maybeBuffer of
|
||||||
|
Nothing -> do
|
||||||
|
buffer <- streamToBuffer requestBody.stream
|
||||||
|
liftEffect
|
||||||
|
$ Ref.write (Just buffer) requestBody.buffer
|
||||||
|
pure buffer
|
||||||
|
Just buffer -> pure buffer
|
||||||
|
where
|
||||||
|
-- | Slurp the entire `Readable` stream into a `Buffer`
|
||||||
|
streamToBuffer :: Readable () -> Aff Buffer
|
||||||
|
streamToBuffer stream =
|
||||||
makeAff \done -> do
|
makeAff \done -> do
|
||||||
bufs <- new []
|
bufs <- Ref.new []
|
||||||
onData stream \buf -> void $ modify (_ <> [ buf ]) bufs
|
onData stream \buf -> void $ Ref.modify (_ <> [ buf ]) bufs
|
||||||
onEnd stream do
|
onEnd stream do
|
||||||
body <- Ref.read bufs >>= concat
|
body <- Ref.read bufs >>= concat
|
||||||
done $ Right body
|
done $ Right body
|
||||||
pure nonCanceler
|
pure nonCanceler
|
||||||
|
|
||||||
|
-- | Return the `Readable` stream directly from `RequestBody`
|
||||||
|
toStream :: RequestBody -> Readable ()
|
||||||
|
toStream = _.stream
|
||||||
|
|
||||||
-- | Types that implement the `Body` class can be used as a body to an HTTPure
|
-- | Types that implement the `Body` class can be used as a body to an HTTPure
|
||||||
-- | response, and can be used with all the response helpers.
|
-- | response, and can be used with all the response helpers.
|
||||||
class Body b where
|
class Body b where
|
||||||
|
@ -5,12 +5,11 @@ module HTTPure.Request
|
|||||||
) where
|
) where
|
||||||
|
|
||||||
import Prelude
|
import Prelude
|
||||||
import Effect.Aff (Aff)
|
|
||||||
import Data.String (joinWith)
|
import Data.String (joinWith)
|
||||||
|
import Effect.Aff (Aff)
|
||||||
|
import Effect.Class (liftEffect)
|
||||||
import Foreign.Object (isEmpty, toArrayWithKey)
|
import Foreign.Object (isEmpty, toArrayWithKey)
|
||||||
import Node.HTTP (requestURL)
|
import HTTPure.Body (RequestBody)
|
||||||
import Node.HTTP (Request) as HTTP
|
|
||||||
import Node.Stream (Readable)
|
|
||||||
import HTTPure.Body (read) as Body
|
import HTTPure.Body (read) as Body
|
||||||
import HTTPure.Headers (Headers)
|
import HTTPure.Headers (Headers)
|
||||||
import HTTPure.Headers (read) as Headers
|
import HTTPure.Headers (read) as Headers
|
||||||
@ -23,6 +22,8 @@ import HTTPure.Query (read) as Query
|
|||||||
import HTTPure.Utils (encodeURIComponent)
|
import HTTPure.Utils (encodeURIComponent)
|
||||||
import HTTPure.Version (Version)
|
import HTTPure.Version (Version)
|
||||||
import HTTPure.Version (read) as Version
|
import HTTPure.Version (read) as Version
|
||||||
|
import Node.HTTP (Request) as HTTP
|
||||||
|
import Node.HTTP (requestURL)
|
||||||
|
|
||||||
-- | The `Request` type is a `Record` type that includes fields for accessing
|
-- | The `Request` type is a `Record` type that includes fields for accessing
|
||||||
-- | the different parts of the HTTP request.
|
-- | the different parts of the HTTP request.
|
||||||
@ -31,7 +32,7 @@ type Request =
|
|||||||
, path :: Path
|
, path :: Path
|
||||||
, query :: Query
|
, query :: Query
|
||||||
, headers :: Headers
|
, headers :: Headers
|
||||||
, body :: Readable ()
|
, body :: RequestBody
|
||||||
, httpVersion :: Version
|
, httpVersion :: Version
|
||||||
, url :: String
|
, url :: String
|
||||||
}
|
}
|
||||||
@ -51,12 +52,15 @@ fullPath request = "/" <> path <> questionMark <> queryParams
|
|||||||
-- | Given an HTTP `Request` object, this method will convert it to an HTTPure
|
-- | Given an HTTP `Request` object, this method will convert it to an HTTPure
|
||||||
-- | `Request` object.
|
-- | `Request` object.
|
||||||
fromHTTPRequest :: HTTP.Request -> Aff Request
|
fromHTTPRequest :: HTTP.Request -> Aff Request
|
||||||
fromHTTPRequest request = pure
|
fromHTTPRequest request = do
|
||||||
|
body <- liftEffect $ Body.read request
|
||||||
|
pure
|
||||||
{ method: Method.read request
|
{ method: Method.read request
|
||||||
, path: Path.read request
|
, path: Path.read request
|
||||||
, query: Query.read request
|
, query: Query.read request
|
||||||
, headers: Headers.read request
|
, headers: Headers.read request
|
||||||
, body: Body.read request
|
, body
|
||||||
, httpVersion: Version.read request
|
, httpVersion: Version.read request
|
||||||
, url: requestURL request
|
, url: requestURL request
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1,39 +1,69 @@
|
|||||||
module Test.HTTPure.BodySpec where
|
module Test.HTTPure.BodySpec where
|
||||||
|
|
||||||
import Prelude
|
import Prelude
|
||||||
|
|
||||||
import Data.Maybe (Maybe(Nothing), fromMaybe)
|
import Data.Maybe (Maybe(Nothing), fromMaybe)
|
||||||
|
import Effect.Aff (Aff)
|
||||||
import Effect.Class (liftEffect)
|
import Effect.Class (liftEffect)
|
||||||
import Node.Buffer (toString) as Buffer
|
import Effect.Ref (new) as Ref
|
||||||
|
import HTTPure.Body (RequestBody, defaultHeaders, read, toBuffer, toStream, toString, write)
|
||||||
|
import HTTPure.Headers (header)
|
||||||
import Node.Buffer (Buffer, fromString)
|
import Node.Buffer (Buffer, fromString)
|
||||||
|
import Node.Buffer (toString) as Buffer
|
||||||
import Node.Encoding (Encoding(UTF8))
|
import Node.Encoding (Encoding(UTF8))
|
||||||
import Node.Stream (readString)
|
import Node.Stream (readString)
|
||||||
import Test.Spec (describe, it)
|
|
||||||
import HTTPure.Body (read, toString, toBuffer, defaultHeaders, write)
|
|
||||||
import HTTPure.Headers (header)
|
|
||||||
import Test.HTTPure.TestHelpers (Test, (?=), mockRequest, mockResponse, getResponseBody, stringToStream)
|
import Test.HTTPure.TestHelpers (Test, (?=), mockRequest, mockResponse, getResponseBody, stringToStream)
|
||||||
|
import Test.Spec (describe, it)
|
||||||
|
|
||||||
|
mockRequestBody :: String -> Aff RequestBody
|
||||||
|
mockRequestBody body =
|
||||||
|
liftEffect do
|
||||||
|
buffer <- Ref.new Nothing
|
||||||
|
string <- Ref.new Nothing
|
||||||
|
pure
|
||||||
|
{ buffer
|
||||||
|
, stream: stringToStream body
|
||||||
|
, string
|
||||||
|
}
|
||||||
|
|
||||||
readSpec :: Test
|
readSpec :: Test
|
||||||
readSpec =
|
readSpec =
|
||||||
describe "read" do
|
describe "read" do
|
||||||
it "is the body of the Request" do
|
it "is the body of the Request" do
|
||||||
body <- read <$> mockRequest "" "GET" "" "test" []
|
body <- (liftEffect <<< read) =<< mockRequest "" "GET" "" "test" []
|
||||||
string <- liftEffect $ fromMaybe "" <$> readString body Nothing UTF8
|
string <- liftEffect $ fromMaybe "" <$> readString (toStream body) Nothing UTF8
|
||||||
string ?= "test"
|
string ?= "test"
|
||||||
|
|
||||||
toStringSpec :: Test
|
toStringSpec :: Test
|
||||||
toStringSpec =
|
toStringSpec =
|
||||||
describe "toString" do
|
describe "toString" do
|
||||||
it "slurps Streams into Strings" do
|
it "turns RequestBody into a String" do
|
||||||
string <- toString $ stringToStream "foobar"
|
requestBody <- mockRequestBody "foobar"
|
||||||
|
string <- toString requestBody
|
||||||
string ?= "foobar"
|
string ?= "foobar"
|
||||||
|
it "is idempotent" do
|
||||||
|
requestBody <- mockRequestBody "foobar"
|
||||||
|
string1 <- toString requestBody
|
||||||
|
string2 <- toString requestBody
|
||||||
|
string1 ?= string2
|
||||||
|
|
||||||
toBufferSpec :: Test
|
toBufferSpec :: Test
|
||||||
toBufferSpec =
|
toBufferSpec =
|
||||||
describe "toBuffer" do
|
describe "toBuffer" do
|
||||||
it "slurps Streams into Buffers" do
|
it "turns RequestBody into a Buffer" do
|
||||||
buf <- toBuffer $ stringToStream "foobar"
|
requestBody <- mockRequestBody "foobar"
|
||||||
|
buf <- toBuffer requestBody
|
||||||
string <- liftEffect $ Buffer.toString UTF8 buf
|
string <- liftEffect $ Buffer.toString UTF8 buf
|
||||||
string ?= "foobar"
|
string ?= "foobar"
|
||||||
|
it "is idempotent" do
|
||||||
|
requestBody <- mockRequestBody "foobar"
|
||||||
|
buffer1 <- toBuffer requestBody
|
||||||
|
buffer2 <- toBuffer requestBody
|
||||||
|
string1 <- bufferToString buffer1
|
||||||
|
string2 <- bufferToString buffer2
|
||||||
|
string1 ?= string2
|
||||||
|
where
|
||||||
|
bufferToString = liftEffect <<< Buffer.toString UTF8
|
||||||
|
|
||||||
defaultHeadersSpec :: Test
|
defaultHeadersSpec :: Test
|
||||||
defaultHeadersSpec =
|
defaultHeadersSpec =
|
||||||
|
Loading…
Reference in New Issue
Block a user