mirror of
https://github.com/dillonkearns/elm-pages-v3-beta.git
synced 2025-01-04 09:56:44 +03:00
918 lines
36 KiB
Elm
918 lines
36 KiB
Elm
module StaticHttpRequestsTests exposing (all)
|
|
|
|
import ApiRoute
|
|
import BackendTask exposing (BackendTask)
|
|
import BackendTask.Http
|
|
import Bytes.Decode
|
|
import Bytes.Encode
|
|
import Codec
|
|
import Dict
|
|
import Expect
|
|
import FatalError exposing (FatalError)
|
|
import Html
|
|
import Json.Decode as JD exposing (Decoder)
|
|
import Json.Encode as Encode
|
|
import Pages.Internal.Platform.Cli exposing (..)
|
|
import Pages.Internal.Platform.Effect as Effect exposing (Effect)
|
|
import Pages.Internal.Platform.ToJsPayload as ToJsPayload
|
|
import Pages.ProgramConfig exposing (ProgramConfig)
|
|
import Pages.SiteConfig exposing (SiteConfig)
|
|
import Pages.StaticHttp.Request as Request
|
|
import Path
|
|
import ProgramTest exposing (ProgramTest)
|
|
import Regex
|
|
import RenderRequest
|
|
import RequestsAndPending exposing (ResponseBody(..))
|
|
import Server.Response as Response
|
|
import SimulatedEffect.Cmd
|
|
import SimulatedEffect.Ports
|
|
import Test exposing (Test, describe, test)
|
|
|
|
|
|
all : Test
|
|
all =
|
|
describe "Static Http Requests"
|
|
[ test "initial requests are sent out" <|
|
|
\() ->
|
|
startSimple []
|
|
(BackendTask.Http.getJson "https://api.github.com/repos/dillonkearns/elm-pages" starDecoder |> BackendTask.allowFatal)
|
|
|> simulateHttp
|
|
(get "https://api.github.com/repos/dillonkearns/elm-pages")
|
|
(JsonBody
|
|
(Encode.object [ ( "stargazer_count", Encode.int 86 ) ])
|
|
)
|
|
|> expectSuccess []
|
|
, test "StaticHttp request for initial are resolved" <|
|
|
\() ->
|
|
startSimple
|
|
[ "post-1" ]
|
|
(BackendTask.Http.getJson "https://api.github.com/repos/dillonkearns/elm-pages" starDecoder |> BackendTask.allowFatal)
|
|
|> simulateHttp
|
|
(get "https://api.github.com/repos/dillonkearns/elm-pages")
|
|
(JsonBody
|
|
(Encode.object [ ( "stargazer_count", Encode.int 86 ) ])
|
|
)
|
|
|> expectSuccess []
|
|
, describe "single page renders"
|
|
[ test "single pages that are pre-rendered" <|
|
|
\() ->
|
|
startSimple [ "post-1" ]
|
|
(BackendTask.Http.getJson "https://api.github.com/repos/dillonkearns/elm-pages" starDecoder |> BackendTask.allowFatal)
|
|
|> simulateHttp
|
|
(get "https://api.github.com/repos/dillonkearns/elm-pages")
|
|
(JsonBody
|
|
(Encode.object [ ( "stargazer_count", Encode.int 86 ) ])
|
|
)
|
|
|> ProgramTest.expectOutgoingPortValues
|
|
"toJsPort"
|
|
(Codec.decoder (ToJsPayload.successCodecNew2 "" ""))
|
|
(\actualPorts ->
|
|
case actualPorts of
|
|
[ ToJsPayload.PageProgress portData ] ->
|
|
if portData.is404 then
|
|
Expect.fail "Expected page to be found and rendered"
|
|
|
|
else
|
|
Expect.pass
|
|
|
|
_ ->
|
|
Expect.fail <| "Expected exactly 1 port of type PageProgress. Instead, got \n" ++ Debug.toString actualPorts
|
|
)
|
|
]
|
|
, test "the stripped JSON from the same request with different decoders is merged so the decoders succeed" <|
|
|
\() ->
|
|
startSimple
|
|
[ "post-1" ]
|
|
(BackendTask.map2 Tuple.pair
|
|
(BackendTask.Http.getJson "https://api.github.com/repos/dillonkearns/elm-pages"
|
|
(JD.field "stargazer_count" JD.int)
|
|
|> BackendTask.allowFatal
|
|
)
|
|
(BackendTask.Http.getJson "https://api.github.com/repos/dillonkearns/elm-pages"
|
|
(JD.field "language" JD.string)
|
|
|> BackendTask.allowFatal
|
|
)
|
|
)
|
|
|> simulateHttp
|
|
(get "https://api.github.com/repos/dillonkearns/elm-pages")
|
|
(JsonBody
|
|
(Encode.object
|
|
[ ( "stargazer_count", Encode.int 86 )
|
|
, ( "language", Encode.string "Elm" )
|
|
]
|
|
)
|
|
)
|
|
|> expectSuccess []
|
|
, test "andThen" <|
|
|
\() ->
|
|
startSimple
|
|
[ "elm-pages" ]
|
|
(BackendTask.Http.getJson "https://api.github.com/repos/dillonkearns/elm-pages" (JD.succeed ())
|
|
|> BackendTask.allowFatal
|
|
|> BackendTask.andThen
|
|
(\_ ->
|
|
BackendTask.Http.getJson "NEXT-REQUEST" (JD.succeed ())
|
|
|> BackendTask.allowFatal
|
|
)
|
|
)
|
|
|> simulateHttp
|
|
(get "https://api.github.com/repos/dillonkearns/elm-pages")
|
|
(JsonBody Encode.null)
|
|
|> simulateHttp
|
|
(get "NEXT-REQUEST")
|
|
(JsonBody Encode.null)
|
|
|> expectSuccess []
|
|
|
|
--, test "andThen chain avoids repeat requests" <|
|
|
-- TODO is this test case still relevant? Need to think about the new desired functionality with caching HTTP requests given that
|
|
-- BackendTask's can perform non-deterministic effects now.
|
|
-- \() ->
|
|
-- let
|
|
-- pokemonDetailRequest : BackendTask ()
|
|
-- pokemonDetailRequest =
|
|
-- JsonBackendTask.Http.get
|
|
-- "https://pokeapi.co/api/v2/pokemon/"
|
|
-- (JD.list
|
|
-- (JD.field "url" JD.string
|
|
-- |> JD.map
|
|
-- (\url ->
|
|
-- BackendTask.Http.getJson url
|
|
-- (JD.field "image" JD.string)
|
|
-- )
|
|
-- )
|
|
-- )
|
|
-- |> BackendTask.resolve
|
|
-- |> BackendTask.map (\_ -> ())
|
|
-- in
|
|
-- startSimple
|
|
-- [ "elm-pages" ]
|
|
-- pokemonDetailRequest
|
|
-- |> simulateMultipleHttp
|
|
-- [ ( get "https://pokeapi.co/api/v2/pokemon/"
|
|
-- , jsonBody """[
|
|
-- {"url": "url1"},
|
|
-- {"url": "url2"},
|
|
-- {"url": "url3"},
|
|
-- {"url": "url4"},
|
|
-- {"url": "url5"},
|
|
-- {"url": "url6"},
|
|
-- {"url": "url7"},
|
|
-- {"url": "url8"},
|
|
-- {"url": "url9"},
|
|
-- {"url": "url10"}
|
|
-- ]"""
|
|
-- )
|
|
-- , ( get "url1"
|
|
-- , jsonBody """{"image": "image1.jpg"}"""
|
|
-- )
|
|
-- , ( get "url2"
|
|
-- , jsonBody """{"image": "image2.jpg"}"""
|
|
-- )
|
|
-- , ( get "url3"
|
|
-- , jsonBody """{"image": "image3.jpg"}"""
|
|
-- )
|
|
-- , ( get "url4"
|
|
-- , jsonBody """{"image": "image4.jpg"}"""
|
|
-- )
|
|
-- , ( get "url5"
|
|
-- , jsonBody """{"image": "image5.jpg"}"""
|
|
-- )
|
|
-- , ( get "url6"
|
|
-- , jsonBody """{"image": "image6.jpg"}"""
|
|
-- )
|
|
-- , ( get "url7"
|
|
-- , jsonBody """{"image": "image7.jpg"}"""
|
|
-- )
|
|
-- , ( get "url8"
|
|
-- , jsonBody """{"image": "image8.jpg"}"""
|
|
-- )
|
|
-- , ( get "url9"
|
|
-- , jsonBody """{"image": "image9.jpg"}"""
|
|
-- )
|
|
-- , ( get "url10"
|
|
-- , jsonBody """{"image": "image10.jpg"}"""
|
|
-- )
|
|
-- ]
|
|
-- |> expectSuccess []
|
|
--
|
|
--, test "port is sent out once all requests are finished" <|
|
|
-- \() ->
|
|
-- start
|
|
-- [ ( [ "elm-pages" ]
|
|
-- , BackendTask.Http.getJson (Secrets.succeed "https://api.github.com/repos/dillonkearns/elm-pages") starDecoder
|
|
-- )
|
|
-- , ( [ "elm-pages-starter" ]
|
|
-- , BackendTask.Http.getJson (Secrets.succeed "https://api.github.com/repos/dillonkearns/elm-pages-starter") starDecoder
|
|
-- )
|
|
-- ]
|
|
-- |> ProgramTest.simulateHttpOk
|
|
-- "GET"
|
|
-- "https://api.github.com/repos/dillonkearns/elm-pages"
|
|
-- """{ "stargazer_count": 86 }"""
|
|
-- |> ProgramTest.simulateHttpOk
|
|
-- "GET"
|
|
-- "https://api.github.com/repos/dillonkearns/elm-pages-starter"
|
|
-- """{ "stargazer_count": 22 }"""
|
|
-- |> expectSuccess
|
|
-- [ ( "elm-pages"
|
|
-- , [ ( get "https://api.github.com/repos/dillonkearns/elm-pages"
|
|
-- , """{"stargazer_count":86}"""
|
|
-- )
|
|
-- ]
|
|
-- )
|
|
-- , ( "elm-pages-starter"
|
|
-- , [ ( get "https://api.github.com/repos/dillonkearns/elm-pages-starter"
|
|
-- , """{"stargazer_count":22}"""
|
|
-- )
|
|
-- ]
|
|
-- )
|
|
-- ]
|
|
, test "reduced JSON is sent out" <|
|
|
\() ->
|
|
startSimple []
|
|
(BackendTask.Http.getJson "https://api.github.com/repos/dillonkearns/elm-pages" (JD.field "stargazer_count" JD.int)
|
|
|> BackendTask.allowFatal
|
|
)
|
|
|> simulateHttp
|
|
(get "https://api.github.com/repos/dillonkearns/elm-pages")
|
|
(jsonBody """{ "stargazer_count": 86, "unused_field": 123 }""")
|
|
|> expectSuccess []
|
|
, test "you can use elm/json decoders with StaticHttp.unoptimizedRequest" <|
|
|
\() ->
|
|
startSimple []
|
|
(BackendTask.Http.request
|
|
{ url = "https://api.github.com/repos/dillonkearns/elm-pages"
|
|
, method = "GET"
|
|
, headers = []
|
|
, body = BackendTask.Http.emptyBody
|
|
, retries = Nothing
|
|
, timeoutInMs = Nothing
|
|
}
|
|
(BackendTask.Http.expectJson
|
|
(JD.field "stargazer_count" JD.int)
|
|
)
|
|
|> BackendTask.allowFatal
|
|
)
|
|
|> simulateHttp
|
|
(get "https://api.github.com/repos/dillonkearns/elm-pages")
|
|
(jsonBody """{ "stargazer_count": 86, "unused_field": 123 }""")
|
|
|> expectSuccess []
|
|
, test "plain string" <|
|
|
\() ->
|
|
startSimple []
|
|
(BackendTask.Http.request
|
|
{ url = "https://example.com/file.txt"
|
|
, method = "GET"
|
|
, headers = []
|
|
, body = BackendTask.Http.emptyBody
|
|
, retries = Nothing
|
|
, timeoutInMs = Nothing
|
|
}
|
|
BackendTask.Http.expectString
|
|
|> BackendTask.allowFatal
|
|
)
|
|
|> simulateHttp
|
|
{ method = "GET"
|
|
, url = "https://example.com/file.txt"
|
|
, headers =
|
|
[]
|
|
, body = BackendTask.Http.emptyBody
|
|
, cacheOptions = Nothing
|
|
}
|
|
(StringBody "This is a raw text file.")
|
|
|> expectSuccess []
|
|
, test "Err in String to Result function turns into decode error" <|
|
|
\() ->
|
|
startSimple []
|
|
(BackendTask.Http.request
|
|
{ url = "https://example.com/file.txt"
|
|
, method = "GET"
|
|
, headers = []
|
|
, body = BackendTask.Http.emptyBody
|
|
, retries = Nothing
|
|
, timeoutInMs = Nothing
|
|
}
|
|
BackendTask.Http.expectString
|
|
|> BackendTask.allowFatal
|
|
|> BackendTask.map
|
|
(\string ->
|
|
if String.toUpper string == string then
|
|
Ok string
|
|
|
|
else
|
|
Err "String was not uppercased"
|
|
)
|
|
|> BackendTask.andThen (\result -> result |> Result.mapError FatalError.fromString |> BackendTask.fromResult)
|
|
)
|
|
|> simulateHttp
|
|
(get "https://example.com/file.txt")
|
|
(StringBody "This is a raw text file.")
|
|
|> ProgramTest.expectOutgoingPortValues
|
|
"toJsPort"
|
|
(Codec.decoder (ToJsPayload.successCodecNew2 "" ""))
|
|
(expectErrorsPort
|
|
"""-- CUSTOM ERROR ----------------------------------------------------- elm-pages
|
|
|
|
|
|
|
|
String was not uppercased"""
|
|
)
|
|
, test "POST method works" <|
|
|
\() ->
|
|
startSimple []
|
|
(BackendTask.Http.request
|
|
{ method = "POST"
|
|
, url = "https://api.github.com/repos/dillonkearns/elm-pages"
|
|
, headers = []
|
|
, body = BackendTask.Http.emptyBody
|
|
, retries = Nothing
|
|
, timeoutInMs = Nothing
|
|
}
|
|
(BackendTask.Http.expectJson
|
|
(JD.field "stargazer_count" JD.int)
|
|
)
|
|
|> BackendTask.allowFatal
|
|
)
|
|
|> simulateHttp
|
|
(post "https://api.github.com/repos/dillonkearns/elm-pages")
|
|
(jsonBody """{ "stargazer_count": 86, "unused_field": 123 }""")
|
|
|> expectSuccess []
|
|
, test "json is reduced from andThen chains" <|
|
|
\() ->
|
|
startSimple []
|
|
(BackendTask.Http.getJson "https://api.github.com/repos/dillonkearns/elm-pages" (JD.field "stargazer_count" JD.int)
|
|
|> BackendTask.andThen
|
|
(\_ ->
|
|
BackendTask.Http.getJson "https://api.github.com/repos/dillonkearns/elm-pages-starter" (JD.field "stargazer_count" JD.int)
|
|
)
|
|
|> BackendTask.allowFatal
|
|
)
|
|
|> simulateHttp
|
|
(get "https://api.github.com/repos/dillonkearns/elm-pages")
|
|
(jsonBody """{ "stargazer_count": 100, "unused_field": 123 }""")
|
|
|> simulateHttp
|
|
(get "https://api.github.com/repos/dillonkearns/elm-pages-starter")
|
|
(jsonBody """{ "stargazer_count": 50, "unused_field": 456 }""")
|
|
|> expectSuccess []
|
|
, test "reduced json is preserved by StaticHttp.map2" <|
|
|
\() ->
|
|
startSimple []
|
|
(BackendTask.map2 (\_ _ -> ())
|
|
(BackendTask.Http.getJson "https://api.github.com/repos/dillonkearns/elm-pages" (JD.field "stargazer_count" JD.int) |> BackendTask.allowFatal)
|
|
(BackendTask.Http.getJson "https://api.github.com/repos/dillonkearns/elm-pages-starter" (JD.field "stargazer_count" JD.int) |> BackendTask.allowFatal)
|
|
)
|
|
|> simulateMultipleHttp
|
|
[ ( get "https://api.github.com/repos/dillonkearns/elm-pages"
|
|
, jsonBody """{ "stargazer_count": 100, "unused_field": 123 }"""
|
|
)
|
|
, ( get "https://api.github.com/repos/dillonkearns/elm-pages-starter"
|
|
, jsonBody """{ "stargazer_count": 50, "unused_field": 456 }"""
|
|
)
|
|
]
|
|
|> expectSuccess []
|
|
, test "the port sends out even if there are no http requests" <|
|
|
\() ->
|
|
start
|
|
[ ( []
|
|
, BackendTask.succeed ()
|
|
)
|
|
]
|
|
|> expectSuccess []
|
|
, test "the port sends out when there are duplicate http requests for the same page" <|
|
|
\() ->
|
|
startSimple []
|
|
(BackendTask.map2 (\_ _ -> ())
|
|
(BackendTask.Http.getJson "http://example.com" (JD.succeed ()) |> BackendTask.allowFatal)
|
|
(BackendTask.Http.getJson "http://example.com" (JD.succeed ()) |> BackendTask.allowFatal)
|
|
)
|
|
|> simulateHttp
|
|
(get "http://example.com")
|
|
(jsonBody """null""")
|
|
|> expectSuccess []
|
|
, test "an error is sent out for decoder failures" <|
|
|
\() ->
|
|
startSimple [ "elm-pages" ]
|
|
(BackendTask.Http.getJson "https://api.github.com/repos/dillonkearns/elm-pages" (JD.fail "The user should get this message from the CLI.") |> BackendTask.allowFatal)
|
|
|> simulateHttp
|
|
(get "https://api.github.com/repos/dillonkearns/elm-pages")
|
|
(jsonBody """{ "stargazer_count": 86 }""")
|
|
|> ProgramTest.expectOutgoingPortValues
|
|
"toJsPort"
|
|
(Codec.decoder (ToJsPayload.successCodecNew2 "" ""))
|
|
(expectErrorsPort
|
|
"""-- HTTP ERROR ----------------------------------------------------- elm-pages
|
|
|
|
BadBody: Problem with the given value:
|
|
|
|
{
|
|
"stargazer_count": 86
|
|
}
|
|
|
|
The user should get this message from the CLI."""
|
|
)
|
|
]
|
|
|
|
|
|
type Route
|
|
= Route String
|
|
|
|
|
|
start : List ( List String, BackendTask FatalError a ) -> ProgramTest (Model Route) Msg Effect
|
|
start pages =
|
|
startWithHttpCache [] pages
|
|
|
|
|
|
startWithHttpCache :
|
|
List ( Request.Request, String )
|
|
-> List ( List String, BackendTask FatalError a )
|
|
-> ProgramTest (Model Route) Msg Effect
|
|
startWithHttpCache =
|
|
startLowLevel []
|
|
|
|
|
|
startLowLevel :
|
|
List (ApiRoute.ApiRoute ApiRoute.Response)
|
|
-> List ( Request.Request, String )
|
|
-> List ( List String, BackendTask FatalError a )
|
|
-> ProgramTest (Model Route) Msg Effect
|
|
startLowLevel apiRoutes _ pages =
|
|
let
|
|
pageToLoad : List String
|
|
pageToLoad =
|
|
case pages |> List.head |> Maybe.map Tuple.first of
|
|
Just justPageToLoad ->
|
|
justPageToLoad
|
|
|
|
Nothing ->
|
|
Debug.todo "Error - no pages"
|
|
|
|
encodedFlags : Encode.Value
|
|
encodedFlags =
|
|
--{"secrets":
|
|
-- {"API_KEY": "ABCD1234","BEARER": "XYZ789"}, "mode": "prod", "staticHttpCache": {}
|
|
-- }
|
|
Encode.object
|
|
[ ( "secrets"
|
|
, [ ( "API_KEY", "ABCD1234" )
|
|
, ( "BEARER", "XYZ789" )
|
|
]
|
|
|> Dict.fromList
|
|
|> Encode.dict identity Encode.string
|
|
)
|
|
, ( "mode", Encode.string "prod" )
|
|
, ( "compatibilityKey", Encode.int currentCompatibilityKey )
|
|
]
|
|
in
|
|
{-
|
|
(Model -> model)
|
|
-> ContentCache.ContentCache metadata view
|
|
-> Result (List BuildError) (List ( PagePath, metadata ))
|
|
-> Config pathKey userMsg userModel metadata view
|
|
-> Decode.Value
|
|
-> ( model, Effect pathKey )
|
|
-}
|
|
ProgramTest.createDocument
|
|
{ init =
|
|
init
|
|
site
|
|
(RenderRequest.SinglePage
|
|
RenderRequest.OnlyJson
|
|
(RenderRequest.Page
|
|
{ path = Path.fromString (pageToLoad |> String.join "/")
|
|
, frontmatter = Route (pageToLoad |> String.join "/")
|
|
}
|
|
)
|
|
(Encode.object [])
|
|
)
|
|
(config apiRoutes pages)
|
|
, update = update
|
|
, view = \_ -> { title = "", body = [] }
|
|
}
|
|
|> ProgramTest.withSimulatedEffects simulateEffects
|
|
|> ProgramTest.start (flags (Encode.encode 0 encodedFlags))
|
|
|
|
|
|
site : SiteConfig
|
|
site =
|
|
{ canonicalUrl = "canonical-site-url"
|
|
, head = BackendTask.succeed []
|
|
}
|
|
|
|
|
|
startSimple : List String -> BackendTask FatalError a -> ProgramTest (Model Route) Msg Effect
|
|
startSimple route backendTasks =
|
|
startWithRoutes route [ route ] [] [ ( route, backendTasks ) ]
|
|
|
|
|
|
config : List (ApiRoute.ApiRoute ApiRoute.Response) -> List ( List String, BackendTask FatalError a ) -> ProgramConfig Msg () Route () () () Effect mappedMsg ()
|
|
config apiRoutes pages =
|
|
{ toJsPort = toJsPort
|
|
, fromJsPort = fromJsPort
|
|
, init = \_ _ _ _ _ -> ( (), Effect.NoEffect )
|
|
, getStaticRoutes =
|
|
--StaticHttp.get (Secrets.succeed "https://my-cms.com/posts")
|
|
-- (Decode.field "posts" (Decode.list (Decode.string |> Decode.map Route)))
|
|
pages
|
|
|> List.map Tuple.first
|
|
|> List.map (String.join "/")
|
|
|> List.map Route
|
|
|> BackendTask.succeed
|
|
, handleRoute = \_ -> BackendTask.succeed Nothing
|
|
, urlToRoute = .path >> Route
|
|
, update = \_ _ _ _ _ _ _ _ -> ( (), Effect.NoEffect )
|
|
, basePath = []
|
|
, onActionData = \() -> Nothing
|
|
, data =
|
|
\_ (Route pageRoute) ->
|
|
let
|
|
thing : Maybe (BackendTask FatalError a)
|
|
thing =
|
|
pages
|
|
|> Dict.fromList
|
|
|> Dict.get
|
|
(pageRoute
|
|
|> String.split "/"
|
|
|> List.filter (\pathPart -> pathPart /= "")
|
|
)
|
|
in
|
|
case thing of
|
|
Just request ->
|
|
request |> BackendTask.map (\_ -> Response.render ())
|
|
|
|
Nothing ->
|
|
Debug.todo <| "Couldn't find page: " ++ pageRoute ++ "\npages: " ++ Debug.toString pages
|
|
, site = Just site
|
|
, view =
|
|
\_ _ _ page _ _ _ _ ->
|
|
let
|
|
thing : Maybe (BackendTask FatalError a)
|
|
thing =
|
|
pages
|
|
|> Dict.fromList
|
|
|> Dict.get
|
|
(page.path |> Path.toSegments)
|
|
in
|
|
case thing of
|
|
Just _ ->
|
|
{ view = \_ -> { title = "Title", body = [ Html.text "" ] }, head = [] }
|
|
|
|
Nothing ->
|
|
Debug.todo <| "Couldn't find page: " ++ Debug.toString page ++ "\npages: " ++ Debug.toString pages
|
|
, subscriptions = \_ _ _ -> Sub.none
|
|
, routeToPath = \(Route route) -> route |> String.split "/"
|
|
, sharedData = BackendTask.succeed ()
|
|
, onPageChange = \_ -> GotDataBatch (Encode.object [])
|
|
, apiRoutes = \_ -> apiRoutes
|
|
, pathPatterns = []
|
|
, byteDecodePageData = \_ -> Bytes.Decode.fail
|
|
, sendPageData = \_ -> Cmd.none
|
|
, encodeResponse = \_ -> Bytes.Encode.signedInt8 0
|
|
, hotReloadData = Sub.none
|
|
, decodeResponse = Bytes.Decode.fail
|
|
, byteEncodePageData = \_ -> Bytes.Encode.signedInt8 0
|
|
, gotBatchSub = Sub.none
|
|
, globalHeadTags = Nothing
|
|
, perform = \_ _ -> Cmd.none
|
|
, cmdToEffect = \_ -> Effect.NoEffect
|
|
, errorStatusCode = \_ -> 404
|
|
, notFoundPage = ()
|
|
, notFoundRoute = Route "not-found"
|
|
, internalError = \_ -> ()
|
|
, errorPageToData = \_ -> ()
|
|
, action = \_ _ -> BackendTask.fail (FatalError.fromString "No action.")
|
|
, encodeAction = \_ -> Bytes.Encode.signedInt8 0
|
|
}
|
|
|
|
|
|
startWithRoutes :
|
|
List String
|
|
-> List (List String)
|
|
-> List ( Request.Request, String )
|
|
-> List ( List String, BackendTask FatalError a )
|
|
-> ProgramTest (Model Route) Msg Effect
|
|
startWithRoutes pageToLoad _ _ pages =
|
|
let
|
|
encodedFlags : Encode.Value
|
|
encodedFlags =
|
|
--{"secrets":
|
|
-- {"API_KEY": "ABCD1234","BEARER": "XYZ789"}, "mode": "prod", "staticHttpCache": {}
|
|
-- }
|
|
Encode.object
|
|
[ ( "secrets"
|
|
, [ ( "API_KEY", "ABCD1234" )
|
|
, ( "BEARER", "XYZ789" )
|
|
]
|
|
|> Dict.fromList
|
|
|> Encode.dict identity Encode.string
|
|
)
|
|
, ( "staticHttpCache", encodedStaticHttpCache )
|
|
, ( "mode", Encode.string "dev-server" )
|
|
, ( "compatibilityKey", Encode.int currentCompatibilityKey )
|
|
]
|
|
|
|
encodedStaticHttpCache : Encode.Value
|
|
encodedStaticHttpCache =
|
|
[] |> Encode.object
|
|
in
|
|
{-
|
|
(Model -> model)
|
|
-> ContentCache.ContentCache metadata view
|
|
-> Result (List BuildError) (List ( PagePath, metadata ))
|
|
-> Config pathKey userMsg userModel metadata view
|
|
-> Decode.Value
|
|
-> ( model, Effect pathKey )
|
|
-}
|
|
ProgramTest.createDocument
|
|
{ init =
|
|
init
|
|
site
|
|
(RenderRequest.SinglePage
|
|
RenderRequest.OnlyJson
|
|
(RenderRequest.Page
|
|
{ path = Path.fromString (pageToLoad |> String.join "/")
|
|
, frontmatter = Route (pageToLoad |> String.join "/")
|
|
}
|
|
)
|
|
(Encode.object [])
|
|
)
|
|
(config [] pages)
|
|
, update = update
|
|
, view = \_ -> { title = "", body = [] }
|
|
}
|
|
|> ProgramTest.withSimulatedEffects simulateEffects
|
|
|> ProgramTest.withSimulatedSubscriptions simulateSubscriptions
|
|
|> ProgramTest.start (flags (Encode.encode 0 encodedFlags))
|
|
|
|
|
|
flags : String -> JD.Value
|
|
flags jsonString =
|
|
case JD.decodeString JD.value jsonString of
|
|
Ok value ->
|
|
value
|
|
|
|
Err _ ->
|
|
Debug.todo "Invalid JSON value."
|
|
|
|
|
|
sendToJsPort : ToJsPayload.ToJsSuccessPayloadNewCombined -> ProgramTest.SimulatedEffect msg
|
|
sendToJsPort value =
|
|
SimulatedEffect.Ports.send "toJsPort" (value |> Codec.encoder (ToJsPayload.successCodecNew2 "" ""))
|
|
|
|
|
|
simulateEffects : Effect -> ProgramTest.SimulatedEffect Msg
|
|
simulateEffects effect =
|
|
case effect of
|
|
Effect.NoEffect ->
|
|
SimulatedEffect.Cmd.none
|
|
|
|
-- toJsPort value |> Cmd.map never
|
|
Effect.Batch list ->
|
|
list
|
|
|> List.map simulateEffects
|
|
|> SimulatedEffect.Cmd.batch
|
|
|
|
Effect.FetchHttp requests ->
|
|
requests
|
|
|> List.map
|
|
(\request -> ( Request.hash request, request ))
|
|
|> ToJsPayload.DoHttp
|
|
|> sendToJsPort
|
|
|> SimulatedEffect.Cmd.map never
|
|
|
|
Effect.SendSinglePage info ->
|
|
SimulatedEffect.Cmd.batch
|
|
[ info
|
|
|> Codec.encoder (ToJsPayload.successCodecNew2 "" "")
|
|
|> SimulatedEffect.Ports.send "toJsPort"
|
|
, SimulatedEffect.Cmd.none
|
|
]
|
|
|
|
Effect.SendSinglePageNew _ toJsSuccessPayloadNewCombined ->
|
|
SimulatedEffect.Cmd.batch
|
|
[ toJsSuccessPayloadNewCombined
|
|
|> Codec.encoder (ToJsPayload.successCodecNew2 "" "")
|
|
|> SimulatedEffect.Ports.send "toJsPort"
|
|
, SimulatedEffect.Cmd.none
|
|
]
|
|
|
|
|
|
expectErrorsPort : String -> List ToJsPayload.ToJsSuccessPayloadNewCombined -> Expect.Expectation
|
|
expectErrorsPort expectedPlainString actualPorts =
|
|
case actualPorts |> List.reverse |> List.head of
|
|
Just (ToJsPayload.Errors actualRichTerminalString) ->
|
|
actualRichTerminalString
|
|
|> List.map .title
|
|
|> String.join "\n"
|
|
|> normalizeErrorExpectEqual expectedPlainString
|
|
|
|
Nothing ->
|
|
Expect.fail "Expected single error port. Didn't receive any ports."
|
|
|
|
_ ->
|
|
Expect.fail <| "Expected single error port. Got\n" ++ String.join "\n\n" (List.map Debug.toString actualPorts)
|
|
|
|
|
|
normalizeErrorExpectEqual : String -> String -> Expect.Expectation
|
|
normalizeErrorExpectEqual expectedPlainString actualRichTerminalString =
|
|
actualRichTerminalString
|
|
|> Regex.replace
|
|
-- strip out all possible ANSI sequences
|
|
(Regex.fromString "(\\x9B|\\x1B\\[)[0-?]*[ -/]*[@-~]"
|
|
|> Maybe.withDefault Regex.never
|
|
)
|
|
(\_ -> "")
|
|
|> String.replace "\u{001B}" ""
|
|
|> normalizeNewlines
|
|
|> Expect.equal
|
|
(expectedPlainString |> normalizeNewlines)
|
|
|
|
|
|
normalizeNewlines : String -> String
|
|
normalizeNewlines string =
|
|
string
|
|
|> Regex.replace
|
|
(Regex.fromString "(\n)+" |> Maybe.withDefault Regex.never)
|
|
(\_ -> "")
|
|
|> Regex.replace
|
|
(Regex.fromString "( )+" |> Maybe.withDefault Regex.never)
|
|
(\_ -> " ")
|
|
|> String.replace "\u{000D}" ""
|
|
|> Regex.replace
|
|
(Regex.fromString "\\s" |> Maybe.withDefault Regex.never)
|
|
(\_ -> "")
|
|
|
|
|
|
toJsPort : a -> Cmd msg
|
|
toJsPort _ =
|
|
Cmd.none
|
|
|
|
|
|
fromJsPort : Sub msg
|
|
fromJsPort =
|
|
Sub.none
|
|
|
|
|
|
starDecoder : Decoder Int
|
|
starDecoder =
|
|
JD.field "stargazer_count" JD.int
|
|
|
|
|
|
expectSuccess : List ( Request.Request, String ) -> ProgramTest model msg effect -> Expect.Expectation
|
|
expectSuccess expectedRequests previous =
|
|
expectSuccessNew expectedRequests [] previous
|
|
|
|
|
|
expectSuccessNew : List ( Request.Request, String ) -> List (ToJsPayload.ToJsSuccessPayloadNew -> Expect.Expectation) -> ProgramTest model msg effect -> Expect.Expectation
|
|
expectSuccessNew expectedRequest expectations previous =
|
|
previous
|
|
|> ProgramTest.expectOutgoingPortValues
|
|
"toJsPort"
|
|
(Codec.decoder (ToJsPayload.successCodecNew2 "" ""))
|
|
(\value ->
|
|
case value of
|
|
(ToJsPayload.PageProgress portPayload) :: _ ->
|
|
let
|
|
singleExpectation : ToJsPayload.ToJsSuccessPayloadNew -> Expect.Expectation
|
|
singleExpectation =
|
|
\subject ->
|
|
subject.contentJson
|
|
|> Expect.equal
|
|
(expectedRequest
|
|
|> List.map
|
|
(\( request, response ) ->
|
|
( Request.hash request, response )
|
|
)
|
|
|> Dict.fromList
|
|
)
|
|
in
|
|
portPayload
|
|
|> Expect.all
|
|
(singleExpectation
|
|
:: expectations
|
|
)
|
|
|
|
[ errorPort ] ->
|
|
Expect.fail <| "Expected success port. Got:\n" ++ Debug.toString errorPort
|
|
|
|
_ ->
|
|
Expect.fail ("Expected ports to be called once, but instead there were " ++ String.fromInt (List.length value) ++ " calls.")
|
|
)
|
|
|
|
|
|
simulateSubscriptions : a -> ProgramTest.SimulatedSub Msg
|
|
simulateSubscriptions _ =
|
|
-- TODO handle build errors or not needed?
|
|
SimulatedEffect.Ports.subscribe "gotBatchSub"
|
|
(JD.value |> JD.map GotDataBatch)
|
|
identity
|
|
|
|
|
|
get : String -> Request.Request
|
|
get url =
|
|
{ method = "GET"
|
|
, url = url
|
|
, headers = []
|
|
, body = BackendTask.Http.emptyBody
|
|
, cacheOptions = Nothing
|
|
}
|
|
|
|
|
|
post : String -> Request.Request
|
|
post url =
|
|
{ method = "POST"
|
|
, url = url
|
|
, headers = []
|
|
, body = BackendTask.Http.emptyBody
|
|
, cacheOptions = Nothing
|
|
}
|
|
|
|
|
|
simulateHttp : Request.Request -> ResponseBody -> ProgramTest model msg effect -> ProgramTest model msg effect
|
|
simulateHttp request response program =
|
|
program
|
|
|> ProgramTest.ensureOutgoingPortValues
|
|
"toJsPort"
|
|
(Codec.decoder (ToJsPayload.successCodecNew2 "" ""))
|
|
(\actualPorts ->
|
|
case actualPorts of
|
|
[ ToJsPayload.DoHttp _ ] ->
|
|
Expect.pass
|
|
|
|
_ ->
|
|
Expect.fail <|
|
|
"Expected an HTTP request, got:\n"
|
|
++ Debug.toString actualPorts
|
|
)
|
|
|> ProgramTest.simulateIncomingPort "gotBatchSub"
|
|
(Encode.object [ encodeBatchEntry ( request, response ) ])
|
|
|
|
|
|
simulateMultipleHttp : List ( Request.Request, ResponseBody ) -> ProgramTest model msg effect -> ProgramTest model msg effect
|
|
simulateMultipleHttp requests program =
|
|
program
|
|
|> ProgramTest.ensureOutgoingPortValues
|
|
"toJsPort"
|
|
(Codec.decoder (ToJsPayload.successCodecNew2 "" ""))
|
|
(\actualPorts ->
|
|
case actualPorts of
|
|
(ToJsPayload.DoHttp _) :: _ ->
|
|
-- TODO check count of HTTP requests, and check the URLs
|
|
Expect.pass
|
|
|
|
_ ->
|
|
Expect.fail <|
|
|
"Expected an HTTP request, got:\n"
|
|
++ Debug.toString actualPorts
|
|
)
|
|
|> ProgramTest.simulateIncomingPort "gotBatchSub"
|
|
(requests
|
|
|> List.map encodeBatchEntry
|
|
|> Encode.object
|
|
)
|
|
|
|
|
|
jsonBody : String -> ResponseBody
|
|
jsonBody jsonString =
|
|
JsonBody
|
|
(jsonString
|
|
|> JD.decodeString JD.value
|
|
|> Result.withDefault Encode.null
|
|
)
|
|
|
|
|
|
encodeBatchEntry : ( Request.Request, ResponseBody ) -> ( String, Encode.Value )
|
|
encodeBatchEntry ( req, response ) =
|
|
( Request.hash (req |> withInternalHeader response)
|
|
, Encode.object
|
|
[ ( "request"
|
|
, Codec.encodeToValue Request.codec
|
|
(withInternalHeader response req)
|
|
)
|
|
, ( "response"
|
|
, RequestsAndPending.bodyEncoder response
|
|
)
|
|
]
|
|
)
|
|
|
|
|
|
withInternalHeader : ResponseBody -> { a | headers : List ( String, String ) } -> { a | headers : List ( String, String ) }
|
|
withInternalHeader res req =
|
|
{ req
|
|
| headers =
|
|
( "elm-pages-internal"
|
|
, case res of
|
|
JsonBody _ ->
|
|
"ExpectJson"
|
|
|
|
BytesBody _ ->
|
|
"ExpectBytes"
|
|
|
|
StringBody _ ->
|
|
"ExpectString"
|
|
|
|
WhateverBody ->
|
|
"ExpectWhatever"
|
|
)
|
|
:: req.headers
|
|
}
|