2017-10-26 21:19:30 +00:00
|
|
|
module Test.HTTPure.TestHelpers where
|
2017-07-10 10:17:13 +00:00
|
|
|
|
2017-07-18 05:31:46 +00:00
|
|
|
import Prelude
|
2017-07-10 10:17:13 +00:00
|
|
|
|
2018-07-08 23:16:48 +00:00
|
|
|
import Effect as Effect
|
|
|
|
import Effect.Aff as Aff
|
|
|
|
import Effect.Class as EffectClass
|
|
|
|
import Effect.Ref as Ref
|
2018-08-20 02:50:07 +00:00
|
|
|
import Data.Array as Array
|
2018-02-25 02:51:07 +00:00
|
|
|
import Data.Either as Either
|
2018-08-20 02:50:07 +00:00
|
|
|
import Data.List as List
|
2017-07-17 23:42:13 +00:00
|
|
|
import Data.Maybe as Maybe
|
2017-07-23 06:13:47 +00:00
|
|
|
import Data.Options ((:=))
|
2017-07-17 23:42:13 +00:00
|
|
|
import Data.String as StringUtil
|
2017-09-26 06:08:07 +00:00
|
|
|
import Data.Tuple as Tuple
|
2018-07-08 23:16:48 +00:00
|
|
|
import Foreign.Object as Object
|
2018-08-20 02:50:07 +00:00
|
|
|
import Node.Buffer as Buffer
|
2017-07-10 10:17:13 +00:00
|
|
|
import Node.Encoding as Encoding
|
|
|
|
import Node.HTTP as HTTP
|
|
|
|
import Node.HTTP.Client as HTTPClient
|
|
|
|
import Node.Stream as Stream
|
|
|
|
import Test.Spec as Spec
|
2017-07-19 05:21:36 +00:00
|
|
|
import Test.Spec.Assertions as Assertions
|
2017-07-10 10:17:13 +00:00
|
|
|
import Unsafe.Coerce as Coerce
|
|
|
|
|
2017-07-19 05:21:36 +00:00
|
|
|
infix 1 Assertions.shouldEqual as ?=
|
|
|
|
|
2017-07-10 10:17:13 +00:00
|
|
|
-- | The type for integration tests.
|
2018-07-08 23:16:48 +00:00
|
|
|
type Test = Spec.Spec Unit
|
2017-07-10 10:17:13 +00:00
|
|
|
|
|
|
|
-- | The type for the entire test suite.
|
2018-07-08 23:16:48 +00:00
|
|
|
type TestSuite = Effect.Effect Unit
|
2017-07-10 10:17:13 +00:00
|
|
|
|
|
|
|
-- | Given a URL, a failure handler, and a success handler, create an HTTP
|
|
|
|
-- | client request.
|
2018-07-08 23:16:48 +00:00
|
|
|
request :: Boolean ->
|
2017-07-18 17:09:03 +00:00
|
|
|
Int ->
|
|
|
|
String ->
|
2018-07-08 23:16:48 +00:00
|
|
|
Object.Object String ->
|
2017-07-18 17:09:03 +00:00
|
|
|
String ->
|
|
|
|
String ->
|
2018-07-08 23:16:48 +00:00
|
|
|
Aff.Aff HTTPClient.Response
|
2018-02-25 02:51:07 +00:00
|
|
|
request secure port method headers path body = Aff.makeAff \done -> do
|
|
|
|
req <- HTTPClient.request options $ Either.Right >>> done
|
2017-07-18 17:09:03 +00:00
|
|
|
let stream = HTTPClient.requestAsStream req
|
2017-07-23 06:13:47 +00:00
|
|
|
_ <- Stream.writeString stream Encoding.UTF8 body $ pure unit
|
|
|
|
Stream.end stream $ pure unit
|
2018-02-25 02:51:07 +00:00
|
|
|
pure Aff.nonCanceler
|
2017-07-18 17:09:03 +00:00
|
|
|
where
|
|
|
|
options =
|
2017-07-23 19:17:02 +00:00
|
|
|
HTTPClient.protocol := (if secure then "https:" else "http:") <>
|
2017-07-23 06:13:47 +00:00
|
|
|
HTTPClient.method := method <>
|
|
|
|
HTTPClient.hostname := "localhost" <>
|
|
|
|
HTTPClient.port := port <>
|
|
|
|
HTTPClient.path := path <>
|
2017-07-23 19:17:02 +00:00
|
|
|
HTTPClient.headers := HTTPClient.RequestHeaders headers <>
|
|
|
|
HTTPClient.rejectUnauthorized := false
|
2017-07-10 10:17:13 +00:00
|
|
|
|
2018-08-20 02:50:07 +00:00
|
|
|
-- | Convert a request to an Aff containing the `Buffer with the response body.
|
|
|
|
toBuffer :: HTTPClient.Response -> Aff.Aff Buffer.Buffer
|
|
|
|
toBuffer response = Aff.makeAff \done -> do
|
|
|
|
let stream = HTTPClient.responseAsStream response
|
|
|
|
chunks <- Ref.new List.Nil
|
|
|
|
Stream.onData stream $ \new -> Ref.modify_ (List.Cons new) chunks
|
|
|
|
Stream.onEnd stream $
|
|
|
|
Ref.read chunks
|
|
|
|
>>= List.reverse >>> Array.fromFoldable >>> Buffer.concat
|
|
|
|
>>= Either.Right >>> done
|
2018-08-27 04:54:04 +00:00
|
|
|
pure Aff.nonCanceler
|
2017-07-10 10:17:13 +00:00
|
|
|
|
|
|
|
-- | Convert a request to an Aff containing the string with the response body.
|
2018-07-08 23:16:48 +00:00
|
|
|
toString :: HTTPClient.Response -> Aff.Aff String
|
2018-08-20 02:50:07 +00:00
|
|
|
toString resp = do
|
|
|
|
buf <- toBuffer resp
|
|
|
|
EffectClass.liftEffect $ Buffer.toString Encoding.UTF8 buf
|
2017-07-10 10:17:13 +00:00
|
|
|
|
|
|
|
-- | Run an HTTP GET with the given url and return an Aff that contains the
|
|
|
|
-- | string with the response body.
|
2018-07-08 23:16:48 +00:00
|
|
|
get :: Int ->
|
|
|
|
Object.Object String ->
|
2017-07-18 17:09:03 +00:00
|
|
|
String ->
|
2018-07-08 23:16:48 +00:00
|
|
|
Aff.Aff String
|
2017-07-23 19:17:02 +00:00
|
|
|
get port headers path = request false port "GET" headers path "" >>= toString
|
|
|
|
|
2018-08-20 02:50:07 +00:00
|
|
|
-- | Like `get` but return a response body in a `Buffer`
|
|
|
|
getBinary :: Int ->
|
|
|
|
Object.Object String ->
|
|
|
|
String ->
|
|
|
|
Aff.Aff Buffer.Buffer
|
|
|
|
getBinary port headers path =
|
|
|
|
request false port "GET" headers path "" >>= toBuffer
|
|
|
|
|
2017-07-23 19:17:02 +00:00
|
|
|
-- | Run an HTTPS GET with the given url and return an Aff that contains the
|
|
|
|
-- | string with the response body.
|
2018-07-08 23:16:48 +00:00
|
|
|
get' :: Int ->
|
|
|
|
Object.Object String ->
|
2017-07-23 19:17:02 +00:00
|
|
|
String ->
|
2018-07-08 23:16:48 +00:00
|
|
|
Aff.Aff String
|
2017-07-23 19:17:02 +00:00
|
|
|
get' port headers path = request true port "GET" headers path "" >>= toString
|
2017-07-18 17:09:03 +00:00
|
|
|
|
|
|
|
-- | Run an HTTP POST with the given url and body and return an Aff that
|
|
|
|
-- | contains the string with the response body.
|
2018-07-08 23:16:48 +00:00
|
|
|
post :: Int ->
|
|
|
|
Object.Object String ->
|
2017-07-18 17:09:03 +00:00
|
|
|
String ->
|
|
|
|
String ->
|
2018-07-08 23:16:48 +00:00
|
|
|
Aff.Aff String
|
2017-07-23 19:17:02 +00:00
|
|
|
post port headers path = request false port "POST" headers path >=> toString
|
2017-07-17 23:42:13 +00:00
|
|
|
|
|
|
|
-- | Convert a request to an Aff containing the string with the given header
|
|
|
|
-- | value.
|
|
|
|
extractHeader :: String -> HTTPClient.Response -> String
|
|
|
|
extractHeader header = unmaybe <<< lookup <<< HTTPClient.responseHeaders
|
|
|
|
where
|
|
|
|
unmaybe = Maybe.fromMaybe ""
|
2018-07-08 23:16:48 +00:00
|
|
|
lookup = Object.lookup $ StringUtil.toLower header
|
2017-07-17 23:42:13 +00:00
|
|
|
|
|
|
|
-- | Run an HTTP GET with the given url and return an Aff that contains the
|
|
|
|
-- | string with the header value for the given header.
|
2018-07-08 23:16:48 +00:00
|
|
|
getHeader :: Int ->
|
|
|
|
Object.Object String ->
|
2017-07-17 23:42:13 +00:00
|
|
|
String ->
|
|
|
|
String ->
|
2018-07-08 23:16:48 +00:00
|
|
|
Aff.Aff String
|
2017-07-18 17:09:03 +00:00
|
|
|
getHeader port headers path header =
|
2017-07-23 19:17:02 +00:00
|
|
|
extractHeader header <$> request false port "GET" headers path ""
|
2017-07-10 10:17:13 +00:00
|
|
|
|
2018-10-09 17:37:23 +00:00
|
|
|
getStatus :: Int ->
|
|
|
|
Object.Object String ->
|
|
|
|
String ->
|
|
|
|
Aff.Aff Int
|
|
|
|
getStatus port headers path =
|
|
|
|
HTTPClient.statusCode <$> request false port "GET" headers path ""
|
|
|
|
|
2017-07-10 10:17:13 +00:00
|
|
|
-- | Mock an HTTP Request object
|
2017-09-26 06:08:07 +00:00
|
|
|
foreign import mockRequestImpl ::
|
2017-07-18 05:25:14 +00:00
|
|
|
String ->
|
|
|
|
String ->
|
|
|
|
String ->
|
2018-07-08 23:16:48 +00:00
|
|
|
Object.Object String ->
|
|
|
|
Effect.Effect HTTP.Request
|
2017-07-10 10:17:13 +00:00
|
|
|
|
2017-09-26 06:08:07 +00:00
|
|
|
-- | Mock an HTTP Request object
|
2018-07-08 23:16:48 +00:00
|
|
|
mockRequest :: String ->
|
2017-09-26 06:08:07 +00:00
|
|
|
String ->
|
|
|
|
String ->
|
|
|
|
Array (Tuple.Tuple String String) ->
|
2018-07-08 23:16:48 +00:00
|
|
|
Aff.Aff HTTP.Request
|
2017-09-26 06:08:07 +00:00
|
|
|
mockRequest method url body =
|
2018-07-08 23:16:48 +00:00
|
|
|
EffectClass.liftEffect <<< mockRequestImpl method url body <<< Object.fromFoldable
|
2017-07-17 23:42:13 +00:00
|
|
|
|
|
|
|
-- | Mock an HTTP Response object
|
2018-08-27 04:54:04 +00:00
|
|
|
foreign import mockResponse :: Effect.Effect HTTP.Response
|
2017-07-17 23:42:13 +00:00
|
|
|
|
|
|
|
-- | Get the current body from an HTTP Response object (note this will only work
|
|
|
|
-- | with an object returned from mockResponse).
|
|
|
|
getResponseBody :: HTTP.Response -> String
|
|
|
|
getResponseBody = _.body <<< Coerce.unsafeCoerce
|
|
|
|
|
|
|
|
-- | Get the currently set status from an HTTP Response object.
|
|
|
|
getResponseStatus :: HTTP.Response -> Int
|
|
|
|
getResponseStatus = _.statusCode <<< Coerce.unsafeCoerce
|
|
|
|
|
|
|
|
-- | Get all current headers on the HTTP Response object.
|
2018-07-08 23:16:48 +00:00
|
|
|
getResponseHeaders :: HTTP.Response -> Object.Object String
|
2017-07-17 23:42:13 +00:00
|
|
|
getResponseHeaders = Coerce.unsafeCoerce <<< _.headers <<< Coerce.unsafeCoerce
|
|
|
|
|
|
|
|
-- | Get the current value for the header on the HTTP Response object.
|
|
|
|
getResponseHeader :: String -> HTTP.Response -> String
|
|
|
|
getResponseHeader header =
|
2018-07-08 23:16:48 +00:00
|
|
|
Maybe.fromMaybe "" <<< Object.lookup header <<< getResponseHeaders
|
2018-08-30 22:01:49 +00:00
|
|
|
|
|
|
|
-- | Create a stream out of a string.
|
|
|
|
foreign import stringToStream :: String -> Stream.Readable ()
|