úέŞöÿ²      !"#$%&'()*+,-./0123456789:;<=>?@ABCDEFGHIJKLMNOPQRSTUVWXYZ[\]^_`abcdefghijklmnopqrstuvwxyz{|}~€‚ƒ„…†‡ˆ‰Š‹Œ‘’“”•–—˜™š›œŸ ¡¢£¤¥¦§¨©ª«¬­®¯°±²³´µ¶·¸¹º»¼½¾¿ÀÁÂÃÄÅÆÇÈÉÊËÌÍÎÏĞÑÒÓÔÕÖרÙÚÛÜİŞßàáâãäåæçèéêëìíîïğñòóôõö÷øùúûüışÿ      !"#$%&'()*+,-./0123456789:;<=>?@ABCDEFGHIJKLMN O P Q R S T U V W X Y Z [ \ ] ^ _ ` a bcdefghijklmnopqrstuvwxyz{|}~€‚ƒ„…†‡ˆ‰ Š ‹ Œ     ‘ ’ “ ” • – — ˜ ™ š › œ   Ÿ   ¡ ¢ £ ¤ ¥ ¦ § ¨ © ª « ¬ ­ ® ¯ ° ± Safe<V2?Print out the provided debug message prefixed by the thread ID.Example: ?ghci> debug "Some debug message" [ 225] Some debug message 1Print out the error message corresponding to the   value returned by  q together with any additional information provided by the user (usually the location where the error occurred).Example: ghci> debugErrno "pathtoSource.hs:34" [ 323] pathtoSource.hs:34: failed (Success) NoneYß_A key-value map that represents a collection of HTTP header fields. Keys are case-insensitive.*An empty collection of HTTP header fields.Example: ghci> H. H {unH = []} 2Is a given collection of HTTP header fields empty?Example: 'ghci> :set -XOverloadedStrings ghci> H. H. True ghci> H. $ H. [("Host", "localhost")] False ADoes this collection of HTTP header fields contain a given field?Example: 'ghci> :set -XOverloadedStrings ghci> H. "host" $ H.& [("Host", "localhost")] True ghci> H. "Accept" $ H. [("Host", "localhost")] False /Look up the value of a given HTTP header field.Example: 'ghci> :set -XOverloadedStrings ghci> H. "host" $ H.2 [("Host", "localhost")] Just "localhost" ghci> H. "Accept" $ H.! [("Host", "localhost")] Nothing {Look up the value of a given HTTP header field or return the provided default value when that header field is not present.Example: 2ghci> :set -XOverloadedStrings ghci> let hdrs = H.! [("Host", "localhost")] ghci> H./ "host" "127.0.0.1" $ hdrs "localhost" ghci> H.+ "Accept" "text/plain" $ hdrs "text/plain" xInsert a key-value pair into the headers map. If the key already exists in the map, the values are catenated with ", ".Example: 2ghci> :set -XOverloadedStrings ghci> let hdrs = H. "Accept" "text/plain" $ H.8 ghci> hdrs H {unH = [("accept","text/plain")]} ghci> H.K "Accept" "text/html" $ hdrs H {unH = [("accept","text/plain,text/html")]} kInsert a key-value pair into the headers map, without checking whether the header already exists. The key must; be already case-folded, or none of the lookups will work!Example: 2ghci> :set -XOverloadedStrings ghci> let hdrs = H.  "accept" "text/plain" $ H.D ghci> hdrs H {unH = [("accept","text/plain")]} ghci> let hdrs' = H. m "accept" "text/html" $ hdrs ghci> hdrs' H {unH = [("accept","text/html"), ("accept","text/plain")]} ghci> H.! "accept" hdrs' Just "text/html" PSet the value of a HTTP header field to a given value, replacing the old value.Example: 'ghci> :set -XOverloadedStrings ghci> H.  "accept" "text/plain" $ H.- H {unH = [("accept","text/plain")]} ghci> H.  "accept" "text/html" $ H.? [("Accept", "text/plain")] H {unH = [("accept","text/html")]} ODelete all key-value pairs associated with the given key from the headers map.Example: 'ghci> :set -XOverloadedStrings ghci> H.  "accept" $ H.) [("Accept", "text/plain")] H {unH = []} =Strict left fold over all key-value pairs in the headers map.Example: ,ghci> :set -XOverloadedStrings ghci> import  Data.Monoid ghci> let hdrs = H.y [("Accept", "text/plain"), ("Accept", "text/html")] ghci> let f (cntr, acc) _ val = (cntr+1, val <> ";" <> acc) ghci> H. , f (0, "") hdrs (2,"text/html;text/plain;") Same as  #, but the key parameter is of type ² instead of ³ ²%. The key is case-folded (lowercase).7Right fold over all key-value pairs in the headers map.Example: ,ghci> :set -XOverloadedStrings ghci> import  Data.Monoid ghci> let hdrs = H.y [("Accept", "text/plain"), ("Accept", "text/html")] ghci> let f _ val (cntr, acc) = (cntr+1, val <> ";" <> acc) ghci> H., f (0, "") hdrs (2,"text/plain;text/html;") Same as #, but the key parameter is of type ² instead of ³ ²%. The key is case-folded (lowercase). Convert a $ value to a list of key-value pairs.Example: ighci> :set -XOverloadedStrings ghci> let l = [("Accept", "text/plain"), ("Accept", "text/html")] ghci> H. . H.6 $ l [("accept","text/plain"),("accept","text/html")] Build a & value from a list of key-value pairs.Example: 'ghci> :set -XOverloadedStrings ghci> H.p [("Accept", "text/plain"), ("Accept", "text/html")] H {unH = [("accept","text/plain"),("accept","text/html")]} Like E, but the keys are assumed to be already case-folded (in lowercase).Like #, but does not convert the keys to ³ ²-, so key comparisons will be case-sensitive.   ´µNone;=CQV`IRepresents an HTTP response.­We will need to inspect the content length no matter what, and looking up "content-length" in the headers and parsing the number out of the text will be too expensive.Returns the HTTP status code.Example: ghci> rspStatus b 200 +Returns the HTTP status explanation string.Example: ghci> rspStatusReason b OK 4If true, we are transforming the request body with transformRequestBody+output body is a function that writes to a ¶ stream Noutput body is sendfile(), optional second argument is a byte range to send"?Contains all of the information about an incoming HTTP request.$BThe server name of the request, as it came in from the request's Host: header.Example: 6ghci> :set -XOverloadedStrings ghci> import qualified  Snap.Test as T ghci> import qualified Data.Map° as M ghci> :{ ghci| rq <- T.buildRequest $ do ghci| T.get "/foo/bar" M.empty ghci| T.setHeader "host" "example.com" ghci| :} ghci> rqHostName rq "example.com" %The remote IP address.Example: 6ghci> :set -XOverloadedStrings ghci> import qualified  Snap.Test as T ghci> import qualified Data.MapV as M ghci> rqClientAddr `fmap` T.buildRequest (T.get "/foo/bar" M.empty) "127.0.0.1" &The remote TCP port number.Example: 6ghci> :set -XOverloadedStrings ghci> import qualified  Snap.Test as T ghci> import qualified Data.MapR as M ghci> rqClientPort `fmap` T.buildRequest (T.get "/foo/bar" M.empty) "60000" '&The local IP address for this request.Example: 6ghci> :set -XOverloadedStrings ghci> import qualified  Snap.Test as T ghci> import qualified Data.MapV as M ghci> rqServerAddr `fmap` T.buildRequest (T.get "/foo/bar" M.empty) "127.0.0.1" (§Returns the port number the HTTP server is listening on. This may be useless from the perspective of external requests, e.g. if the server is running behind a proxy.Example: 6ghci> :set -XOverloadedStrings ghci> import qualified  Snap.Test as T ghci> import qualified Data.MapO as M ghci> rqServerPort `fmap` T.buildRequest (T.get "/foo/bar" M.empty) 8080 )fReturns the HTTP server's idea of its local hostname, including port. This is as configured with the Config object at startup.Example: 6ghci> :set -XOverloadedStrings ghci> import qualified  Snap.Test as T ghci> import qualified Data.MapY as M ghci> rqLocalHostname `fmap` T.buildRequest (T.get "/foo/bar" M.empty) "localhost" *Returns True if this is an HTTPS session.Example: 6ghci> :set -XOverloadedStrings ghci> import qualified  Snap.Test as T ghci> import qualified Data.MapN as M ghci> rqIsSecure `fmap` T.buildRequest (T.get "/foo/bar" M.empty) False +Contains all HTTP  associated with this request.Example: 6ghci> :set -XOverloadedStrings ghci> import qualified  Snap.Test as T ghci> import qualified Data.Maph as M ghci> rqHeaders `fmap` T.buildRequest (T.get "/foo/bar" M.empty) H {unH = [("host","localhost")]} ,Actual body of the request.- Returns the Content-Length of the HTTP request body.Example: 6ghci> :set -XOverloadedStrings ghci> import qualified  Snap.Test as T ghci> import qualified Data.MapU as M ghci> rqContentLength `fmap` T.buildRequest (T.get "/foo/bar" M.empty) Nothing . Returns the HTTP request method.Example: 6ghci> :set -XOverloadedStrings ghci> import qualified  Snap.Test as T ghci> import qualified Data.MapJ as M ghci> rqMethod `fmap` T.buildRequest (T.get "/foo/bar" M.empty) GET /,Returns the HTTP version used by the client.Example: 6ghci> :set -XOverloadedStrings ghci> import qualified  Snap.Test as T ghci> import qualified Data.MapM as M ghci> rqVersion `fmap` T.buildRequest (T.get "/foo/bar" M.empty) (1,1) 0JReturns a list of the cookies that came in from the HTTP request headers.Example: 6ghci> :set -XOverloadedStrings ghci> import qualified  Snap.Test as T ghci> import qualified Data.MapJ as M ghci> rqCookies `fmap` T.buildRequest (T.get "/foo/bar" M.empty) [] 1Handlers can be hung on a URI^ "entry point"; this is called the "context path". If a handler is hung on the context path "/foo/", and you request  "/foo/bar", the value of 1 will be "bar".The following identity holds: ÿrqURI r == S.concat [ rqContextPath r , rqPathInfo r , let q = rqQueryString r in if S.null q then "" else S.append "?" q ]Example: 6ghci> :set -XOverloadedStrings ghci> import qualified  Snap.Test as T ghci> import qualified Data.MapR as M ghci> rqPathInfo `fmap` T.buildRequest (T.get "/foo/bar" M.empty) "foo/bar" 2.The "context path" of the request; catenating 2, and 1% should get you back to the original 3 (ignoring query strings). The 2' always begins and ends with a slash ("/"k) character, and represents the path (relative to your component/snaplet) you took to get to your handler.Example: 6ghci> :set -XOverloadedStrings ghci> import qualified  Snap.Test as T ghci> import qualified Data.MapO as M ghci> rqContextPath `fmap` T.buildRequest (T.get "/foo/bar" M.empty) "/" 3 Returns the URI requested by the client.Example: 6ghci> :set -XOverloadedStrings ghci> import qualified  Snap.Test as T ghci> import qualified Data.MapM as M ghci> rqURI `fmap` T.buildRequest (T.get "/foo/bar" M.empty) "foo/bar" 4'Returns the HTTP query string for this ".Example: 6ghci> :set -XOverloadedStrings ghci> import qualified  Snap.Test as T ghci> import qualified Data.Map{ as M ghci> rq <- T.buildRequest (T.get "/foo/bar" (M.fromList [("name", ["value"])])) ghci> rqQueryString rq "name=value" 5(Returns the parameters mapping for this "J. "Parameters" are automatically decoded from the URI's query string and POST* body and entered into this mapping. The 5 value is thus a union of 6 and 7.Example: 6ghci> :set -XOverloadedStrings ghci> import qualified  Snap.Test as T ghci> import qualified Data.Mapã as M ghci> :{ ghci| rq <- T.buildRequest $ do ghci| T.postUrlEncoded "/foo/bar" $ M.fromList [("baz", ["qux"])] ghci| T.setQueryStringRaw "baz=quux" ghci| :} ghci> rqParams rq fromList [("baz",["qux","quux"])] 6:The parameter mapping decoded from the URI's query string.Example: 6ghci> :set -XOverloadedStrings ghci> import qualified  Snap.Test as T ghci> import qualified Data.Mapâ as M ghci> :{ ghci| rq <- T.buildRequest $ do ghci| T.postUrlEncoded "/foo/bar" $ M.fromList [("baz", ["qux"])] ghci| T.setQueryStringRaw "baz=quux" ghci| :} ghci> rqQueryParams rq fromList [("baz",["quux"])] 7|The parameter mapping decoded from the POST body. Note that Snap only auto-decodes POST request bodies when the request's  Content-Type is !application/x-www-form-urlencoded. For multipart/form-data use 3 to decode the POST request and fill this mapping.Example: 6ghci> :set -XOverloadedStrings ghci> import qualified  Snap.Test as T ghci> import qualified Data.Mapà as M ghci> :{ ghci| rq <- T.buildRequest $ do ghci| T.postUrlEncoded "/foo/bar" $ M.fromList [("baz", ["qux"])] ghci| T.setQueryStringRaw "baz=quux" ghci| :} ghci> rqPostParams rq fromList [("baz",["qux"])] 8TA type alias for the HTTP parameters mapping. Each parameter key maps to a list of ²= values; if a parameter is specified multiple times (e.g.: "GET /foo?param=bar1&param=bar2"), looking up "param" in the mapping will give you ["bar1", "bar2"].9'A datatype representing an HTTP cookie.;The name of the cookie.<The cookie's string value.=-The cookie's expiration value, if it has one.>+The cookie's "domain" value, if it has one.?The cookie path.@Tag as secure cookie?A HTTP only?B9Represents a (major, minor) version of the HTTP protocol.C(Enumerates the HTTP method values (see  5http://tools.ietf.org/html/rfc2068.html#section-5.1.1).N5A typeclass for datatypes which contain HTTP headers.OModify the datatype's headers.P6Retrieve the headers from a datatype that has headers.U$Adds a header key-value-pair to the Nj datatype. If a header with the same name already exists, the new value is appended to the headers list.Example: ghci> import qualified Snap.Types.Headers as H ghci> U "Host" "localhost" H.empty( H {unH = [("host","localhost")]} ghci> UB "Host" "127.0.0.1" it H {unH = [("host","localhost,127.0.0.1")]} V"Sets a header key-value-pair in a N` datatype. If a header with the same name already exists, it is overwritten with the new value.Example: ghci> import qualified Snap.Types.Headers as H ghci> V "Host" "localhost" H.emptyi H {unH = [("host","localhost")]} ghci> setHeader "Host" "127.0.0.1" it H {unH = [("host","127.0.0.1")]} WGets a header value out of a N datatype.Example: ghci> import qualified Snap.Types.Headers as H ghci> W "Host" $ V "Host" "localhost" H.empty Just "localhost" XLists all the headers out of a NX datatype. If many headers came in with the same name, they will be catenated together.Example: ghci> import qualified Snap.Types.Headers as H ghci> X $ V "Host" "localhost" H.empty [("host","localhost")] YClears a header value from a N datatype.Example: ghci> import qualified Snap.Types.Headers as H ghci> Y "Host" $ V "Host" "localhost" H.empty H {unH = []} Z?Equate the special case constructors with their corresponding  Method name variant.]Looks up the value(s) for the given named parameter. Parameters initially come from the request's query string and any decoded POST body (if the request's  Content-Type is !application/x-www-form-urlencodedL). Parameter values can be modified within handlers using "rqModifyParams".Example: 6ghci> :set -XOverloadedStrings ghci> import qualified  Snap.Test as T ghci> import qualified Data.Mapµ as M ghci> :{ ghci| rq <- T.buildRequest $ do ghci| T.postUrlEncoded "/foo/bar" $ M.fromList [("baz", ["qux"])] ghci| T.setQueryStringRaw "baz=quux" ghci| :} ghci> ] "baz" rq Just ["qux","quux"] ^TLooks up the value(s) for the given named parameter in the POST parameters mapping.Example: 6ghci> :set -XOverloadedStrings ghci> import qualified  Snap.Test as T ghci> import qualified Data.Mapµ as M ghci> :{ ghci| rq <- T.buildRequest $ do ghci| T.postUrlEncoded "/foo/bar" $ M.fromList [("baz", ["qux"])] ghci| T.setQueryStringRaw "baz=quux" ghci| :} ghci> ^ "baz" rq Just ["qux"] _ULooks up the value(s) for the given named parameter in the query parameters mapping.Example: 6ghci> :set -XOverloadedStrings ghci> import qualified  Snap.Test as T ghci> import qualified Data.Mapµ as M ghci> :{ ghci| rq <- T.buildRequest $ do ghci| T.postUrlEncoded "/foo/bar" $ M.fromList [("baz", ["qux"])] ghci| T.setQueryStringRaw "baz=quux" ghci| :} ghci> _ "baz" rq Just ["quux"] `,Modifies the parameters mapping (which is a Map ByteString ByteString) in a " using the given function.Example: 6ghci> :set -XOverloadedStrings ghci> import qualified  Snap.Test as T ghci> import qualified Data.Mapµ as M ghci> :{ ghci| rq <- T.buildRequest $ do ghci| T.postUrlEncoded "/foo/bar" $ M.fromList [("baz", ["qux"])] ghci| T.setQueryStringRaw "baz=quux" ghci| :} ghci> 5, rq fromList [("baz",["qux","quux"])] ghci> 5 $ `! (M.delete "baz") rq fromList [] aLWrites a key-value pair to the parameters mapping within the given request.Example: 6ghci> :set -XOverloadedStrings ghci> import qualified  Snap.Test as T ghci> import qualified Data.Mapµ as M ghci> :{ ghci| rq <- T.buildRequest $ do ghci| T.postUrlEncoded "/foo/bar" $ M.fromList [("baz", ["qux"])] ghci| T.setQueryStringRaw "baz=quux" ghci| :} ghci> 5, rq fromList [("baz",["qux","quux"])] ghci> 5 $ a2 "baz" ["corge"] rq fromList [("baz", ["corge"])] b An empty .Example: ghci> b HTTP/1.1 200 OK c9Sets an HTTP response body to the given stream procedure.Example: 6ghci> :set -XOverloadedStrings ghci> import qualified System.IO.Streams# as Streams ghci> import qualified Data.ByteString.Builder# as Builder ghci> :{ ghci| let r = cJ ghci| (out -> do ghci| Streams.write (Just $ Builder.·B "Hello, world!") out ghci| return out) ghci| b1 ghci| :} ghci> r HTTP/1.1 200 OK Hello, world! d=Sets the HTTP response status. Note: normally you would use e1 unless you needed a custom response explanation.Example: Sghci> :set -XOverloadedStrings ghci> setResponseStatus 500 "Internal Server Error" b& HTTP/1.1 500 Internal Server Error eSets the HTTP response code.Example: ghci> setResponseCode 404 b HTTP/1.1 404 Not Found fModifies a response body.Example: 6ghci> :set -XOverloadedStrings ghci> import qualified System.IO.Streams# as Streams ghci> import qualified Data.ByteString.Builder# as Builder ghci> :{ ghci| let r = cJ ghci| (out -> do ghci| Streams.write (Just $ Builder.·B "Hello, world!") out ghci| return out) ghci| bI ghci| :} ghci> r HTTP/1.1 200 OK Hello, world! ghci> :{ ghci| let r' = fo ghci| (f out -> do ghci| out' <- f out ghci| Streams.write (Just $ Builder.·v "\nBye, world!") out' ghci| return out') r ghci| :} ghci> r' HTTP/1.1 200 OK Hello, world! Bye, world! g Sets the  Content-Type in the  headers.Example: @ghci> :set -XOverloadedStrings ghci> setContentType "text/html" b+ HTTP/1.1 200 OK content-type: text/html hConvert 9 into ² for output.fTODO: Remove duplication. This function is copied from snap-server/Snap.Internal.Http.Server.Session.iRender cookies from a given  to .fTODO: Remove duplication. This function is copied from snap-server/Snap.Internal.Http.Server.Session.j Adds an HTTP 9 to  headers.Example: 2ghci> :set -XOverloadedStrings ghci> let cookie = 9: "name" "value" Nothing Nothing Nothing False False ghci> k "name" $ j cookie bA Just (Cookie {cookieName = "name", cookieValue = "value", ...}) k Gets an HTTP 9 with the given name from  headers.Example: %ghci> :set -XOverloadedStrings ghci> k "cookie-name" b Nothing lReturns a list of 9 s present in Example: ghci> l b [] mDeletes an HTTP 9 from the Z headers. Please note this does not necessarily erase the cookie from the client browser.Example: 2ghci> :set -XOverloadedStrings ghci> let cookie = 9G "name" "value" Nothing Nothing Nothing False False ghci> let rsp = j cookie b ghci> kR "name" rsp Just (Cookie {cookieName = "name", cookieValue = "value", ...}) ghci> k "name" $ m "name" rsp Nothing nModifies an HTTP 9 with given name in - headers. Nothing will happen if a matching 9 can not be found in .Example: ,ghci> :set -XOverloadedStrings ghci> import  Data.Monoid ghci> let cookie = 9G "name" "value" Nothing Nothing Nothing False False ghci> let rsp = j cookie b ghci> k\ "name" rsp Just (Cookie {cookieName = "name", cookieValue = "value", ...}) ghci> let f ck@(9J { cookieName = name }) = ck { cookieName = name <> "'"} ghci> let rsp' = n "name" f rsp ghci> k> "name'" rsp' Just (Cookie {cookieName = "name'", ...}) ghci> k6 "name" rsp' Just (Cookie {cookieName = "name", ...}) o$A note here: if you want to set the Content-Lengtho for the response, Snap forces you to do it with this function rather than by setting it in the headers; the Content-Length in the headers will be ignored.@The reason for this is that Snap needs to look up the value of Content-Length€ for each request, and looking the string value up in the headers and parsing the number out of the text will be too expensive.vIf you don't set a content length in your response, HTTP keep-alive will be disabled for HTTP/1.0 clients, forcing a Connection: closeO. For HTTP/1.1 clients, Snap will switch to the chunked transfer encoding if Content-Length is not specified.Example: ghci> setContentLength 400 b' HTTP/1.1 200 OK Content-Length: 400 p Removes any Content-Length set in the .Example: ghci> clearContentLength $ o 400 b HTTP/1.1 200 OK q Convert a ¸ into common log entry format.r Convert a ¸ into an HTTP timestamp.Example: ghci> r . ¹& $ 10 "Thu, 01 Jan 1970 00:00:10 GMT" s"Converts an HTTP timestamp into a ¸.Example: %ghci> :set -XOverloadedStrings ghci> s$ "Thu, 01 Jan 1970 00:00:10 GMT" 10 uSee %.vSee &. ]parameter name to look up HTTP request^parameter name to look up HTTP request_parameter name to look up HTTP requestaparameter nameparameter valuesrequestcnew response bodyresponse to modifydHTTP response integer codeHTTP response explanationResponse to be modifiedeHTTP response integer codeResponse to be modifiedj cookie valueresponse to modifyk cookie nameresponse to querylresponse to querym cookie nameresponse to modifyn cookie namemodifier functionresponse to modifyb !"#,76543210/.-*)('&%$+89:A@?>=<;BCLKJIGFEDHMNOPQRSTUVWXYZ[\]^_`abcdefghijklmnopqrstuviTSRQNOPUVWXYCDEFGHIJKLMyxZB9:;<=>?@A8"#$%&'()*+,-./01234567{zw! [\}|]^_`abcdefghijklmnoprqstuv "#$%&'()*+,-./012345679:;<=>?@AC DEFGHIJKLMNOPNone†‰‚Given a , return its body as a ².Example: ghci> ‚  emptyResponse "" ƒGiven a 4, assert that its HTTP status code is 200 (success).Example: 6ghci> :set -XOverloadedStrings ghci> import qualified  Test.HUnit= as T ghci> let test = T.runTestTT . T.TestCase ghci> test $ ƒ q Cases: 1 Tried: 1 Errors: 0 Failures: 0 Counts {cases = 1, tried = 1, errors = 0, failures = 0} ghci> test $ ƒ ( 500 "Internal Server Error" ³) ### Failure: Expected success (200) but got (500) expected: 200 but got: 500 Cases: 1 Tried: 1 Errors: 0 Failures: 1 Counts {cases = 1, tried = 1, errors = 0, failures = 1} „Given a 6, assert that its HTTP status code is 404 (Not Found).Example: %ghci> :set -XOverloadedStrings ghci> „ $  404 "Not Found"  ghci> „ d *** Exception: HUnitFailure "Expected Not Found (404) but got (200)\nexpected: 404\n but got: 200" …Given a m, assert that its HTTP status code is between 300 and 399 (a redirect), and that the Location header of the  points to the specified URI.Example: .ghci> :set -XOverloadedStrings ghci> let r' =  301 "Moved Permanently"  ghci> let r = ' "Location" "www.example.com" r' ghci> … "www.example.com" r ghci> … "www.example.com" K *** Exception: HUnitFailure "Expected redirect but got status code (200)" †Given a H, assert that its HTTP status code is between 300 and 399 (a redirect).Example: %ghci> :set -XOverloadedStrings ghci> † $  301 "Moved Permanently"  ghci> † K *** Exception: HUnitFailure "Expected redirect but got status code (200)" ‡Given a =, assert that its body matches the given regular expression.Example: 6ghci> :set -XOverloadedStrings ghci> import qualified System.IO.Streams# as Streams ghci> import qualified Data.ByteString.Builder# as Builder ghci> :{ ghci| let r = – ghci| (out -> do ghci| Streams.write (Just $ Builder.byteString "Hello, world!") out ghci| return out) ghci|  ghci| :} ghci> ‡ "^Hello" r ghci> ‡] "Bye" r *** Exception: HUnitFailure "Expected body to match regexp \"\"Bye\"\", but didn't" …)The Response should redirect to this URI‡'Regexp that will match the body content‚ƒ„…†‡NoneD¡«“Parser for request headers.¤ Used for ")field-name", and field-name = token, so "Ftoken": comma-separated tokens/field-names, like a header field list.§#Decode an URL-escaped string (see  3http://tools.ietf.org/html/rfc2396.html#section-2.4)Example: ghci> §R "1+attoparsec+%7e%3d+3+*+10%5e-2+meters" Just "1 attoparsec ~= 3 * 10^-2 meters" ¨URL-escape a string (see  3http://tools.ietf.org/html/rfc2396.html#section-2.4)Example: ghci> ¨M "1 attoparsec ~= 3 * 10^-2 meters" "1+attoparsec+%7e%3d+3+*+10%5e-2+meters" ©URL-escape a string (see  3http://tools.ietf.org/html/rfc2396.html#section-2.4 ) into a ¶.Example:  ghci> import Data.ByteString.Builder ghci> º . ©O $ "1 attoparsec ~= 3 * 10^-2 meters" "1+attoparsec+%7e%3d+3+*+10%5e-2+meters" ­Parse a string encoded in !application/x-www-form-urlencoded  Ihttp://en.wikipedia.org/wiki/POST_%28HTTP%29#Use_for_submitting_web_formsformat.Example: ghci> ­F "Name=John+Doe&Name=Jane+Doe&Age=23&Formula=a+%2B+b+%3D%3D+13%25%21"  [(Age ,["23"]),(Formula,["a + b == 13%!"]),(Name,["John Doe","Jane Doe"])] ®Like ¯, but produces a ¶ instead of a ²G. Useful for constructing a large string efficiently in a single step.Example:  ghci> import Data.Map ghci> import  Data.Monoid ghci> import Data.ByteString.Builder ghci> let bldr = ® ( [(Name, ["John Doe"]), (Age, ["23"])]) ghci> º $ · " http://example.com/script?" <> bldr " /http://example.com/script?Age=23&Name=John+Doe" ¯VGiven a collection of key-value pairs with possibly duplicate keys (represented as a »), construct a string in !application/x-www-form-urlencoded format.Example: ghci> ¯ ( [(Name, ["John Doe"]), (Age#, ["23"])]) "Age=23&Name=John+Doe" ,ˆ‰Š‹Œ‘’“”•–—˜™š›œŸ ¡¢£¤¥¦§¨©ª«¬­®¯°±²³,‰Š‹Œ‘’“”•–—˜™š›œŸ ¡¢£¤¥ˆ¦§¨©ª«¬­®¯°±²³Š0None01;<=>?CFKQTVtÀ>´6This exception is thrown if the handler you supply to ÷ fails.¼¼ is the ¼ that user web handlers run in. ¼ gives you: +Stateful access to fetch or modify an HTTP ". 3printRqContextPath :: Snap () printRqContextPath = é . 2 =<< Ü +Stateful access to fetch or modify an HTTP . 7printRspStatusReason :: Snap () printRspStatusReason = é .  =<< Ş  Failure / ½ / ¾ semantics: a ¼9 handler can choose not to handle a given request, using ¿ or its synonym Ñ0, and you can try alternative handlers with the À operator: a :: Snap String a = Ñ< b :: Snap String b = return "foo" c :: Snap String c = a À8 b -- try running a, if it fails then try b Convenience functions (é, ê, ë, ì, ç+) for queueing output to be written to the ), or for streaming to the response using  -http://hackage.haskell.org/package/io-streams io-streams:  example :: (Á ¶ -> IO (Á ¶*)) -> Snap () example streamProc = do é! "I'm a strict bytestring" ê "I'm a lazy bytestring" ë "I'm strict text" ç streamProc Early termination: if you call Ï: a :: Snap () a = do ã $ d! 500 "Internal Server Error" é "500 error" r <- Ş Ï r @then any subsequent processing will be skipped and the supplied  value will be returned from ÷ as-is.Access to the  monad through a à instance: a :: Snap () a = Ä fireTheMissiles PThe ability to set or extend a timeout which will kill the handler thread after N3 seconds of inactivity (the default is 20 seconds): a :: Snap () a =  30 #Throw and catch exceptions using a Å instance: import Control.Exception.Lifted (Æ, Ç, È() foo :: Snap () foo = bar `catch` (e::Æ) -> baz where bar = Ç FooException Log a message to the error log: foo :: Snap () foo = æ "grumble." JYou may notice that most of the type signatures in this module contain a (Ê m) => ... typeclass constraint. ÊB is a typeclass which, in essence, says "you can get back to the ¼ monad from here". Using Ê you can extend the ¼K monad with additional functionality and still have access to most of the ¼ functions without writing Y everywhere. Instances are already provided for most of the common monad transformers (, , É, etc.).ÃUsed internally to implement ö.Æ#Type of external handler passed to ö.ÊÊ is a type class, analogous to à for Â, that makes it easy to wrap ¼ inside monad transformers.ËLift a computation from the ¼ monad.ÌMPass the request body stream to a consuming procedure, returning the result.‰If the consuming procedure you pass in here throws an exception, Snap will attempt to clear the rest of the unread request body (using 0) before rethrowing the exception. If you used õ?, however, Snap will give up and immediately close the socket. To prevent slowloris attacks, the connection will be also terminated if the input socket produces data too slowly (500 bytes per second is the default limit).Example: 6ghci> :set -XOverloadedStrings ghci> import qualified Data.ByteString.Char8 as B8 ghci> import qualified Data.ByteString.Lazy as L ghci> import  Data.Char" (toUpper) ghci> import qualified Data.Map as M ghci> import qualified  Snap.Test as T ghci> import qualified System.IO.Streamsó as Streams ghci> let r = T.put "/foo" "text/plain" "some text" ghci> :{ ghci| let f s = do u <- Streams.map (B8.map toUpper) s ghci| l <- Streams.toList u ghci| return $ L.fromChunks l ghci| :} ghci> T.runHandler r (Ì f >>= êS) HTTP/1.1 200 OK server: Snap/test date: Thu, 07 Aug 2014 20:48:40 GMT SOME TEXT ÍhReturns the request body as a lazy bytestring. /Note that the request is not actually provided lazily!/Example: 6ghci> :set -XOverloadedStrings ghci> import qualified Data.Map as M ghci> import qualified  Snap.TestP as T ghci> let r = T.put "/foo" "text/plain" "some text" ghci> T.runHandler r (Í 2048 >>= êS) HTTP/1.1 200 OK server: Snap/test date: Thu, 07 Aug 2014 20:08:44 GMT some text  Since: 0.6ÎxNormally Snap is careful to ensure that the request body is fully consumed after your web handler runs, but before the ‹ body is streamed out the socket. If you want to transform the request body into some output in O(1) space, you should use this function.nTake care: in order for this to work, the HTTP client must be written with input-to-output streaming in mind.[Note that upon calling this function, response processing finishes early as if you called Ï]. Make sure you set any content types, headers, cookies, etc. before you call this function.Example: 6ghci> :set -XOverloadedStrings ghci> import qualified Data.ByteString.Char8 as B8 ghci> import  Data.Char" (toUpper) ghci> import qualified Data.Map as M ghci> import qualified  Snap.Test as T ghci> import qualified System.IO.Streams as Streams ghci> let r = T.put "/foo" "text/plain" "some text" ghci> let f = Streams.map (B8.map toUpper) ghci> T.runHandler r (Î f >> Í 2048 >>= êS) HTTP/1.1 200 OK server: Snap/test date: Thu, 07 Aug 2014 20:30:15 GMT SOME TEXT ÏShort-circuits a ¼( monad action early, storing the given  value in its state.LIMPORTANT: Be vary careful when using this with things like a DB library's withTransactionƒ function or any other kind of setup/teardown block, as it can prevent the cleanup from being called and result in resource leaks.Example: 6ghci> :set -XOverloadedStrings ghci> import qualified Data.Map as M ghci> import qualified  Snap.Test as T ghci> import Control.Applicative8 ghci> let r = T.get "/" M.empty ghci> T.runHandler r ((Ù $ é "TOP") <|> Ï b) HTTP/1.1 200 OK server: Snap/test date: Thu, 07 Aug 2014 16:58:57 GMT TOP ghci> let r' = T.get "/foo/bar" M.empty ghci> T.runHandler r' ((Ù $ é "TOP") <|> Ï bJ) HTTP/1.1 200 OK server: Snap/test date: Thu, 07 Aug 2014 17:50:50 GMT Ğ4Capture the flow of control in case a handler calls Ï.WARNING: in the event of a call to ÎX it is possible to violate HTTP protocol safety when using this function. If you call Ğ9 it is suggested that you do not modify the body of the  which was passed to the Ï call.Example: 6ghci> :set -XOverloadedStrings ghci> import qualified Data.ByteString.Char8 as B8 ghci> import qualified Data.Map as M ghci> import qualified  Snap.Test as T ghci> import Control.Applicative7 ghci> let r = T.get "/foo/bar" M.empty ghci> let h = (Ù $ é "TOP") <|> Ï b ghci> T.runHandler r (Ğ h >>= éq . B8.pack . show) HTTP/1.1 200 OK server: Snap/test date: Thu, 07 Aug 2014 18:35:42 GMT Left HTTP/1.1 200 OK ÑFails out of a ¼t monad action. This is used to indicate that you choose not to handle the given request within the given handler.Example: 6ghci> :set -XOverloadedStrings ghci> import qualified Data.Map as M ghci> import qualified  Snap.TestB as T ghci> let r = T.get "/foo/bar" M.empty ghci> T.runHandler r ѹ HTTP/1.1 404 Not Found server: Snap/test date: Thu, 07 Aug 2014 13:35:42 GMT <!DOCTYPE html> <html> <head> <title>Not found</title> </head> <body> <code>No handler accepted "/foo/bar" /code </body></html> ÒRuns a ¼J monad action only if the request's HTTP method matches the given method.Example: 6ghci> :set -XOverloadedStrings ghci> import qualified Data.Map as M ghci> import qualified  Snap.TestC as T ghci> let r = T.get "/foo/bar" M.empty ghci> T.runHandler r (Ò D $ ég "OK") HTTP/1.1 200 OK server: Snap/test date: Thu, 07 Aug 2014 13:38:48 GMT OK ghci> T.runHandler r (Ò F $ é" "OK") HTTP/1.1 404 Not Found ... ÓRuns a ¼R monad action only if the request's HTTP method matches one of the given methods.Example: 6ghci> :set -XOverloadedStrings ghci> import qualified Data.Map as M ghci> import qualified  Snap.TestC as T ghci> let r = T.get "/foo/bar" M.empty ghci> T.runHandler r (Ó [D, F] $ ég "OK") HTTP/1.1 200 OK server: Snap/test date: Thu, 07 Aug 2014 13:38:48 GMT OK ghci> T.runHandler r (Ó [F] $ é" "OK") HTTP/1.1 404 Not Found ... ÖRuns a ¼ monad action only when the 19 of the request starts with the given path. For example, dir "foo" handler Will fail if 1 is not "/foo" or "/foo/...", and will add "foo/" to the handler's local 2.Example: 6ghci> :set -XOverloadedStrings ghci> import qualified Data.Map as M ghci> import qualified  Snap.TestC as T ghci> let r = T.get "/foo/bar" M.empty ghci> T.runHandler r (Ö "foo" $ ég "OK") HTTP/1.1 200 OK server: Snap/test date: Thu, 07 Aug 2014 14:52:24 GMT OK ghci> T.runHandler r (Ö "baz" $ é" "OK") HTTP/1.1 404 Not Found ... ×Runs a ¼& monad action only for requests where 1K is exactly equal to the given string. If the path matches, locally sets 2 to the old value of 1, sets 1!="", and runs the given handler.Example: 6ghci> :set -XOverloadedStrings ghci> import qualified Data.Map as M ghci> import qualified  Snap.Test1 as T ghci> T.runHandler (T.get "/foo" M.empty) (× "foo" $ é~ "bar") HTTP/1.1 200 OK server: Snap/test date: Thu, 07 Aug 2014 14:15:42 GMT bar ghci> T.runHandler (T.get "/foo" M.empty) (× "bar" $ é# "baz") HTTP/1.1 404 Not Found ... ØRuns a ¼z monad action only when the first path component is successfully parsed as the argument to the supplied handler function.CNote that the path segment is url-decoded prior to being passed to fromBS$; this is new as of snap-core 0.10.Example: 6ghci> :set -XOverloadedStrings ghci> import qualified Data.Map as M ghci> import qualified  Snap.TestU as T ghci> let r = T.get "/11/foo/bar" M.empty ghci> let f = (\i -> if i == 11 then é "11" else é "???") ghci> T.runHandler r (Ø f) HTTP/1.1 200 OK server: Snap/test date: Thu, 07 Aug 2014 14:27:10 GMT 11 ghci> let r' = T.get "/foo/11/bar" M.empty ghci> T.runHandler r' (Ø f) HTTP/1.1 404 Not Found ... ÙRuns a ¼ monad action only when 1 is empty.Example: 6ghci> :set -XOverloadedStrings ghci> import qualified Data.Map as M ghci> import qualified  Snap.Test< as T ghci> let r = T.get "/" M.empty ghci> T.runHandler r (Ù $ é OK‡) HTTP/1.1 200 OK server: Snap/test date: Thu, 07 Aug 2014 14:56:39 GMT OK ghci> let r' = T.get "/foo" M.empty ghci> T.runHandler r' (Ù $ é" "OK") HTTP/1.1 404 Not Found ... ÚLocal Snap version of get.ÛLocal Snap monad version of modify.Ü Grabs the " object out of the ¼ monad.Example: 6ghci> :set -XOverloadedStrings ghci> import qualified Data.Map as M ghci> import qualified  Snap.TestC as T ghci> let r = T.get "/foo/bar" M.empty ghci> T.runHandler r (é . 3 =<< ÜR) HTTP/1.1 200 OK server: Snap/test date: Sat, 02 Aug 2014 07:51:54 GMT /foo/bar İGrabs something out of the "3 object, using the given projection function. See gets.Example: 6ghci> :set -XOverloadedStrings ghci> import qualified Data.Map as M ghci> import qualified  Snap.TestC as T ghci> let r = T.get "/foo/bar" M.empty ghci> T.runHandler r (é =<< İ 3R) HTTP/1.1 200 OK server: Snap/test date: Sat, 02 Aug 2014 07:51:54 GMT /foo/bar Ş Grabs the  object out of the ¼ monad.Example: 6ghci> :set -XOverloadedStrings ghci> import qualified Data.Map as M ghci> import qualified  Snap.TestC as T ghci> let r = T.get "/foo/bar" M.empty ghci> T.runHandler r (é .  =<< ŞL) HTTP/1.1 200 OK server: Snap/test date: Sat, 02 Aug 2014 15:06:00 GMT OK ßGrabs something out of the 3 object, using the given projection function. See gets.Example: 6ghci> :set -XOverloadedStrings ghci> import qualified Data.Map as M ghci> import qualified  Snap.TestC as T ghci> let r = T.get "/foo/bar" M.empty ghci> T.runHandler r (é =<< ß L) HTTP/1.1 200 OK server: Snap/test date: Wed, 06 Aug 2014 13:35:45 GMT OK à Puts a new  object into the ¼ monad.Example: 6ghci> :set -XOverloadedStrings ghci> import qualified Data.Map as M ghci> import qualified  Snap.Test as T ghci> let rsp = e 404 bB ghci> let req = T.get "/foo/bar" M.empty ghci> T.runHandler req (àU rsp) HTTP/1.1 404 Not Found server: Snap/test date: Wed, 06 Aug 2014 13:59:58 GMT á Puts a new " object into the ¼ monad. Example: 6ghci> :set -XOverloadedStrings ghci> import qualified Data.Map as M ghci> import qualified  Snap.Testi as T ghci> :{ ghci| let hndlr = do rq <- T.buildRequest (T.get "/bar/foo" M.empty) ghci| á! rq ghci| uri' <- İ 3 ghci| é“ uri' ghci| :} ghci> T.runHandler (T.get "/foo/bar" M.empty) hndlr HTTP/1.1 200 OK server: Snap/test date: Wed, 06 Aug 2014 15:13:46 GMT /bar/foo â Modifies the " object stored in a ¼ monad. Example: 6ghci> :set -XOverloadedStrings ghci> import qualified Data.Map as M ghci> import qualified  Snap.Testy as T ghci> let r = T.get "/foo/bar" M.empty ghci> r' <- T.buildRequest $ T.get "/bar/foo" M.empty ghci> T.runHandler r (â (const r') >> İ 3 >>= éR) HTTP/1.1 200 OK server: Snap/test date: Wed, 06 Aug 2014 15:24:25 GMT /bar/foo ã Modifes the  object stored in a ¼ monad. Example: 6ghci> :set -XOverloadedStrings ghci> import qualified Data.Map as M ghci> import qualified  Snap.TestC as T ghci> let r = T.get "/foo/bar" M.empty ghci> T.runHandler r (ã $ eU 404) HTTP/1.1 404 Not Found server: Snap/test date: Wed, 06 Aug 2014 15:27:11 GMT ä#Performs a redirect by setting the LocationH header to the given target URL/path and the status code to 302 in the  object stored in a ¼N monad. Note that the target URL is not validated in any way. Consider using å> instead, which allows you to choose the correct status code.Example: 6ghci> :set -XOverloadedStrings ghci> import qualified Data.Map as M ghci> import qualified  Snap.TestC as T ghci> let r = T.get "/foo/bar" M.empty ghci> T.runHandler r (ä¯ "http://snapframework.com") HTTP/1.1 302 Found content-length: 0 location: http://snapframework.com server: Snap/test date: Thu, 07 Aug 2014 08:52:11 GMT Content-Length: 0 å#Performs a redirect by setting the Locationj header to the given target URL/path and the status code (should be one of 301, 302, 303 or 307) in the  object stored in a ¼> monad. Note that the target URL is not validated in any way.Example: 6ghci> :set -XOverloadedStrings ghci> import qualified Data.Map as M ghci> import qualified  Snap.TestC as T ghci> let r = T.get "/foo/bar" M.empty ghci> T.runHandler r (åÀ "http://snapframework.com" 301) HTTP/1.1 307 Temporary Redirect content-length: 0 location: http://snapframework.com server: Snap/test date: Thu, 07 Aug 2014 08:55:51 GMT Content-Length: 0 æLog an error message in the ¼ monad.Example: ghci> import qualified Data.ByteString.Char8 as B8 ghci> ÷ (æ "fatal error!") (error> . B8.unpack) undefined undefined *** Exception: fatal error! ç9Run the given stream procedure, adding its output to the  stored in the ¼ monad state.Example: 6ghci> :set -XOverloadedStrings ghci> import qualified Data.Map as M ghci> import qualified  Snap.Test as T ghci> import qualified Data.ByteString.Builder as B ghci> import qualified System.IO.StreamsÏ as Streams ghci> let r = T.get "/foo/bar" M.empty ghci> :{ ghci| let f str = do { ghci| Streams.write (Just $ B.byteString "Hello, streams world") str; ghci| return str } ghci| :} ghci> T.runHandler r (ç` f) HTTP/1.1 200 OK server: Snap/test date: Wed, 06 Aug 2014 17:55:47 GMT Hello, streams world èAdds the given ¶ to the body of the  stored in the | ¼ monad state.Example: 6ghci> :set -XOverloadedStrings ghci> import qualified Data.Map as M ghci> import qualified  Snap.Test as T ghci> import qualified Data.ByteString.BuilderC as B ghci> let r = T.get "/foo/bar" M.empty ghci> T.runHandler r (èt $ B.byteString "Hello, world") HTTP/1.1 200 OK server: Snap/test date: Wed, 06 Aug 2014 17:33:33 GMT Hello, world éAdds the given strict ² to the body of the  stored in the ¼ monad state.vWarning: This function is intentionally non-strict. If any pure exceptions are raised by the expression creating the ²B, the exception won't actually be raised within the Snap handler.Example: 6ghci> :set -XOverloadedStrings ghci> import qualified Data.Map as M ghci> import qualified  Snap.TestC as T ghci> let r = T.get "/foo/bar" M.empty ghci> T.runHandler r (é{ "Hello, bytestring world") HTTP/1.1 200 OK server: Snap/test date: Wed, 06 Aug 2014 17:34:27 GMT Hello, bytestring world êAdds the given lazy Ê to the body of the  stored in the ¼ monad state.vWarning: This function is intentionally non-strict. If any pure exceptions are raised by the expression creating the ²B, the exception won't actually be raised within the Snap handler.Example: 6ghci> :set -XOverloadedStrings ghci> import qualified Data.Map as M ghci> import qualified  Snap.TestC as T ghci> let r = T.get "/foo/bar" M.empty ghci> T.runHandler r (ê… "Hello, lazy bytestring world") HTTP/1.1 200 OK server: Snap/test date: Wed, 06 Aug 2014 17:35:15 GMT Hello, lazy bytestring world ëAdds the given strict Ë to the body of the  stored in the ¼ monad state.vWarning: This function is intentionally non-strict. If any pure exceptions are raised by the expression creating the ²B, the exception won't actually be raised within the Snap handler.Example: 6ghci> :set -XOverloadedStrings ghci> import qualified Data.Map as M ghci> import qualified  Snap.TestC as T ghci> let r = T.get "/foo/bar" M.empty ghci> T.runHandler r (ëo "Hello, text world") HTTP/1.1 200 OK server: Snap/test date: Wed, 06 Aug 2014 17:36:38 GMT Hello, text world ìAdds the given lazy Ì to the body of the  stored in the ¼ monad state.vWarning: This function is intentionally non-strict. If any pure exceptions are raised by the expression creating the ²B, the exception won't actually be raised within the Snap handler.Example: 6ghci> :set -XOverloadedStrings ghci> import qualified Data.Map as M ghci> import qualified  Snap.TestC as T ghci> let r = T.get "/foo/bar" M.empty ghci> T.runHandler r (ìy "Hello, lazy text world") HTTP/1.1 200 OK server: Snap/test date: Wed, 06 Aug 2014 17:37:41 GMT Hello, lazy text world í9Sets the output to be the contents of the specified file.Calling í5 will overwrite any output queued to be sent in the :. If the response body is not modified after the call to í, Snap will use the efficient  sendfile()+ system call on platforms that support it.(If the response body is modified (using f ), the file will be read using mmap().Example: 6ghci> :set -XOverloadedStrings ghci> import qualified Data.Map as M ghci> import qualified  Snap.Test as T ghci>  writeFileg "/tmp/snap-file" "Hello, sendFile world" ghci> let r = T.get "/foo/bar" M.empty ghci> T.runHandler r (í– "/tmp/snap-file") HTTP/1.1 200 OK content-length: 21 server: Snap/test date: Wed, 06 Aug 2014 17:45:10 GMT Content-Length: 21 Hello, sendFile world î^Sets the output to be the contents of the specified file, within the given (start,end) range.Calling î5 will overwrite any output queued to be sent in the :. If the response body is not modified after the call to î, Snap will use the efficient  sendfile()+ system call on platforms that support it.(If the response body is modified (using f ), the file will be read using mmap().Example: 6ghci> :set -XOverloadedStrings ghci> import qualified Data.Map as M ghci> import qualified  Snap.Test as T ghci>  writeFilen "/tmp/snap-file" "Hello, sendFilePartial world" ghci> let r = T.get "/foo/bar" M.empty ghci> T.runHandler r (î "/tmp/snap-file" (7, 28)) HTTP/1.1 200 OK content-length: 21 server: Snap/test date: Wed, 06 Aug 2014 17:47:20 GMT Content-Length: 21 sendFilePartial world ïRuns a ¼ action with a locally-modified " state object. The "\ object in the Snap monad state after the call to localRequest will be unchanged. Example: 6ghci> :set -XOverloadedStrings ghci> import qualified Data.Map as M ghci> import qualified  Snap.Testz as T ghci> let r = T.get "/foo/bar" M.empty ghci> r' <- T.buildRequest $ T.get "/bar/foo" M.empty ghci> let printRqURI = İ 3 >>= é >> é* "\n" ghci> T.runHandler r (printRqURI >> ïr (const r') printRqURI) HTTP/1.1 200 OK server: Snap/test date: Wed, 06 Aug 2014 15:34:12 GMT /foo/bar /bar/foo ğ Fetches the "7 from state and hands it to the given action. Example: 6ghci> :set -XOverloadedStrings ghci> import qualified Data.Map as M ghci> import qualified  Snap.Test as T ghci> import Control.Monad.IO.Class6 ghci> let r = T.get "/foo/bar" M.empty ghci> let h = ğ (\rq -> Ä (T.requestToString rq) >>= éˆ) ghci> T.runHandler r h HTTP/1.1 200 OK server: Snap/test date: Wed, 06 Aug 2014 15:44:24 GMT GET /foo/bar HTTP/1.1 host: localhost ñ Fetches the 7 from state and hands it to the given action. Example: 6ghci> :set -XOverloadedStrings ghci> import qualified Data.Map as M ghci> import qualified  Snap.TestC as T ghci> let r = T.get "/foo/bar" M.empty ghci> T.runHandler r (ñ $ é . L) HTTP/1.1 200 OK server: Snap/test date: Wed, 06 Aug 2014 15:48:45 GMT OK ò Modifies the " in the state to set the  rqRemoteAddrm field to the value in the X-Forwarded-For header. If the header is not present, this action has no effect.ÇThis action should be used only when working behind a reverse http proxy that sets the X-Forwarded-For header. This is the only way to ensure the value in the X-Forwarded-For header can be trusted.ÛThis is provided as a filter so actions that require the remote address can get it in a uniform manner. It has specifically limited functionality to ensure that its transformation can be trusted, when used correctly.ó Modifies the " in the state to set the  rqRemoteAddrt field to the value from the header specified. If the header specified is not present, this action has no effect.·This action should be used only when working behind a reverse http proxy that sets the header being looked at. This is the only way to ensure the value in the header can be trusted.ÛThis is provided as a filter so actions that require the remote address can get it in a uniform manner. It has specifically limited functionality to ensure that its transformation can be trusted, when used correctly.ôsThis function brackets a Snap action in resource acquisition and release. This is provided because MonadCatchIO's bracketh function doesn't work properly in the case of a short-circuit return from the action being bracketed.¼In order to prevent confusion regarding the effects of the aquisition and release actions on the Snap state, this function doesn't accept Snap actions for the acquire or release actions.£This function will run the release action in all cases where the acquire action succeeded. This includes the following behaviors from the bracketed Snap action. Normal completion.Short-circuit completion, either from calling Í or ÏAn exception being thrown.Example: 6ghci> :set -XOverloadedStrings ghci> import qualified Data.Map as M ghci> import qualified  Snap.Test as T ghci> let br = ôÊ (putStrLn "before") (const $ putStrLn "after") ghci> T.runHandler (T.get "/" M.empty) (br $ const $ writeBS "OK") before after HTTP/1.1 200 OK server: Snap/test date: Thu, 07 Aug 2014 18:41:50 GMT OK õ4Terminate the HTTP session with the given exception.Example: 6ghci> :set -XOverloadedStrings ghci> import qualified Data.Map as M ghci> import qualified  Snap.Test as T ghci> import qualified Control.Exception¯ as E ghci> let r = T.get "/foo/bar" M.empty ghci> T.runHandler r (terminateConnection $ E.AssertionFailed "Assertion failed!") *** Exception: <terminated: Assertion failed!> öiTerminate the HTTP session and hand control to some external handler, escaping all further HTTP traffic.‚The external handler takes three arguments: a function to modify the thread's timeout, and a read and a write ends to the socket.÷Runs a ¼ monad action.KThis function is mostly intended for library writers; instead of invoking ÷ directly, use  ! or " (for testing).ø'Post-process a finalized HTTP response:fixup content-length header!properly handle 204/304 responses)if request was HEAD, remove response bodyVNote that we do NOT deal with transfer-encoding: chunked or "connection: close" here.ûSee ]9. Looks up a value for the given named parameter in the "D. If more than one value was entered for the given parameter name, û gloms the values together with Î " ".Example: 6ghci> :set -XOverloadedStrings ghci> import qualified Data.Map as M ghci> import qualified  Snap.Test as T ghci> import qualified Data.ByteString.Char8\ as B8 ghci> let r = T.get "/foo/bar" $ M.fromList [("foo", ["bar"])] ghci> T.runHandler r (û "foo" >>= ée . B8.pack . show) HTTP/1.1 200 OK server: Snap/test date: Mon, 11 Aug 2014 12:57:20 GMT Just "bar" üSee ^Y. Looks up a value for the given named parameter in the POST form parameters mapping in "D. If more than one value was entered for the given parameter name, ü" gloms the values together with: Î " ".Example: 6ghci> :set -XOverloadedStrings ghci> import qualified Data.Map as M ghci> import qualified  Snap.Test as T ghci> import qualified Data.ByteString.Char8g as B8 ghci> let r = T.postUrlEncoded "/foo/bar" $ M.fromList [("foo", ["bar"])] ghci> T.runHandler r (ü "foo" >>= ée . B8.pack . show) HTTP/1.1 200 OK server: Snap/test date: Mon, 11 Aug 2014 13:01:04 GMT Just "bar" ıSee _\. Looks up a value for the given named parameter in the query string parameters mapping in "D. If more than one value was entered for the given parameter name, ı" gloms the values together with Î " ".Example: 6ghci> :set -XOverloadedStrings ghci> import qualified Data.Map as M ghci> import qualified  Snap.Test as T ghci> import qualified Data.ByteString.Char8x as B8 ghci> let r = T.postUrlEncoded "/foo/bar" M.empty >> T.setQueryStringRaw "foo=bar&foo=baz" ghci> T.runHandler r (ı "foo" >>= éi . B8.pack . show) HTTP/1.1 200 OK server: Snap/test date: Mon, 11 Aug 2014 13:06:50 GMT Just "bar baz" şSee 5!. Convenience function to return 8 from the " inside of a Ê instance.Example: 6ghci> :set -XOverloadedStrings ghci> import qualified Data.Map as M ghci> import qualified  Snap.Test as T ghci> import qualified Data.ByteString.Char8\ as B8 ghci> let r = T.get "/foo/bar" $ M.fromList [("foo", ["bar"])] ghci> T.runHandler r (ş >>= éu . B8.pack . show) HTTP/1.1 200 OK server: Snap/test date: Mon, 11 Aug 2014 13:02:54 GMT fromList [("foo",["bar"])] ÿSee 5!. Convenience function to return 8 from the " inside of a Ê instance.Example: 6ghci> :set -XOverloadedStrings ghci> import qualified Data.Map as M ghci> import qualified  Snap.Test as T ghci> import qualified Data.ByteString.Char8g as B8 ghci> let r = T.postUrlEncoded "/foo/bar" $ M.fromList [("foo", ["bar"])] ghci> T.runHandler r (ÿ >>= éu . B8.pack . show) HTTP/1.1 200 OK server: Snap/test date: Mon, 11 Aug 2014 13:04:34 GMT fromList [("foo",["bar"])] See 5!. Convenience function to return 8 from the " inside of a Ê instance.Example: 6ghci> :set -XOverloadedStrings ghci> import qualified Data.Map as M ghci> import qualified  Snap.Test as T ghci> import qualified Data.ByteString.Char8x as B8 ghci> let r = T.postUrlEncoded "/foo/bar" M.empty >> T.setQueryStringRaw "foo=bar&foo=baz" ghci> T.runHandler r ( >>= é{ . B8.pack . show) HTTP/1.1 200 OK server: Snap/test date: Mon, 11 Aug 2014 13:10:17 GMT fromList [("foo",["bar","baz"])] Gets the HTTP 9 with the specified name.Example: 6ghci> :set -XOverloadedStrings ghci> import qualified Data.Map as M ghci> import qualified  Snap.Test as T ghci> import qualified Data.ByteString.Char8 as B8 ghci> let cookie = 9Š "name" "value" Nothing Nothing Nothing False False ghci> let r = T.get "/foo/bar" M.empty >> T.addCookies [cookie] ghci> T.runHandler r ( "name" >>= éš . B8.pack . show) HTTP/1.1 200 OK server: Snap/test date: Thu, 07 Aug 2014 12:16:58 GMT Just (Cookie {cookieName = "name", cookieValue = "value", ...}) Gets the HTTP 9Y with the specified name and decodes it. If the decoding fails, the handler calls pass.Example: 6ghci> :set -XOverloadedStrings ghci> import qualified Data.Map as M ghci> import qualified  Snap.Test as T ghci> let cookie = 9Š "name" "value" Nothing Nothing Nothing False False ghci> let r = T.get "/foo/bar" M.empty >> T.addCookies [cookie] ghci> T.runHandler r ( "name" >>= éO) HTTP/1.1 200 OK server: Snap/test date: Thu, 07 Aug 2014 12:20:09 GMT value  Expire given 9 in client's browser.Example: 6ghci> :set -XOverloadedStrings ghci> import qualified Data.Map as M ghci> import qualified  Snap.Test” as T ghci> let r = T.get "/foo/bar" M.empty ghci> let cookie = Cookie "name" "" Nothing (Just "/subsite") Nothing True False ghci> T.runHandler r (ÿD cookie) HTTP/1.1 200 OK set-cookie: name=; path=/subsite; expires=Sat, 24 Dec 1994 06:28:16 GMT; Secure server: Snap/test date: Thu, 07 Aug 2014 12:21:27 GMT ghci> let cookie = Cookie "name" "value" Nothing Nothing Nothing False False ghci> let r2 = T.get "/foo/bar" M.empty >> T.addCookies [cookie] ghci> T.runHandler r ( "name" >>= maybe (return ()) _) HTTP/1.1 200 OK set-cookie: name=; expires=Sat, 24 Dec 1994 06:28:16 GMT server: Snap/test 'Causes the handler thread to be killed n seconds from now.0Causes the handler thread to be killed at least n seconds from now.CModifies the amount of time remaining before the request times out. Returns an Â6 action which you can use to modify the timeout value. Ætimeout modifiersocket read endsocket write endÍnsize of the largest request body we're willing to accept. If a request body longer than this is received, a TooManyBytesReadException is thrown. See takeNoMoreThan.Îthe Ï from the "5 is passed to this function, and then the resulting Ï is fed to the output.Öpath component to matchhandler to run×path to match againsthandler to runç output to add÷Action to run.Error logging action.Timeout action. HTTP request.ûparameter name to look upüparameter name to look upıparameter name to look up® !"#+$%&'()*-./01234567,89:;<=>?@ABCMHDEFGIJKLNPOUVWXYZ[\]^_`abcdefgjklmnopqrst´µ¶·¸¹º»¼½¾¿ÂÀÁÃÄÅÆÇÉÈÊËÌÍÎÏĞÑÒÓÔÕÖרÙÚÛÜİŞßàáâãäåæçèéêëìíîïğñòóôõö÷øùúûüışÿTÊËÇÈÉÆÃÄÅ¿ÀÁ¼½¾¶·¸¹º»ÌÍÎÏĞÑÒÓÔÕÖרÙÚÛÜŞİßáàâãäåæçèéêëìíîïğñòóô´µõö÷øùúûüışÿ´µĞÑÒ¶·¸¹º»¼½¾¿ÀÁÂÃÄÅÇÈÉÊË#None<QV½ ÓXThe internal data type you use to build a routing tree. Matching is done unambiguously.Ô and Õ$ routes can have a "fallback" route:For Ô9, the fallback is routed when there is nothing to captureFor Õ>, the fallback is routed when we can't find a route in its map3Fallback routes are stacked: i.e. for a route like: 5Dir [("foo", Capture "bar" (Action bar) NoRoute)] baz¾visiting the URI foo/ will result in the "bar" capture being empty and triggering its fallback. It's NoRoute, so we go to the nearest parent fallback and try that, which is the baz action.€A web handler which, given a mapping from URL entry points to web handlers, efficiently routes requests to the correct handler.Usage>The URL entry points are given as relative paths, for example: &route [ ("foo/bar/quux", fooBarQuux) ]&If the URI of the incoming request is  /foo/bar/quux or /foo/bar/quux/...anything...% then the request will be routed to  "fooBarQuux", with 2 set to "/foo/bar/quux/" and 1 set to "...anything...".CA path component within an URL entry point beginning with a colon (":") is treated as a variable captureT; the corresponding path component within the request URI will be entered into the 5K parameters mapping with the given name. For instance, if the routes were: )route [ ("foo/:bar/baz", fooBazHandler) ]Then a request for "/foo/saskatchewan/baz" would be routed to  fooBazHandler with a mapping for "bar" => "saskatchewan" in its parameters table.kLonger paths are matched first, and specific routes are matched before captures. That is, if given routes: ([ ("a", h1), ("a/b", h2), ("a/:x", h3) ]a request for "/a/b" will go to h2, "/a/s" for any s will go to h3, and "/a" will go to h1.The following example matches  "/article" to an article index, "/login" to a login, and "/article/..." to an article renderer. _ [ ("article", renderIndex) , ("article/:id", renderArticle) , ("login", $ POST doLogin) ] Note: URL decodingQA short note about URL decoding: path matching and variable capture are done on decoded URLs, but the contents of 2 and 1m will contain the original encoded URL, i.e. what the user entered. For example, in the following scenario: route [ ("a b c d/", foo ) ]A request for "/a+b+c+d" will be sent to foo with 2 set to "a+b+c+d".hThis behaviour changed as of Snap 0.6.1; previous versions had unspecified (and buggy!) semantics here.Example: 6ghci> :set -XOverloadedStrings ghci> import qualified Data.Map as Map ghci> import qualified Data.ByteString.Char8 as B8 ghci> import  Snap.Test& ghci> :{ ghci| let handler = do r <- Ü ghci| % $ "rqContextPath: " <> 2" r <> "\n" ghci| % $ "rqPathInfo: " <> 1" r <> "\n" ghci| %% $ "rqParams: " <> (B8.pack . show $ 5 r) ghci| :} ghci> " (& "/foo/bar" "Map.empty") ( [("foo", handler)]) HTTP/1.1 200 OK server: Snap/test date: Sat, 02 Aug 2014 05:16:59 GMT rqContextPath: /foo/ rqPathInfo: bar rqParams: fromList [] ghci> " (& "/foo/bar" "Map.empty") (k [("foo/:bar", handler)]) [...] rqContextPath: /foo/bar/ rqPathInfo: rqParams: fromList [("bar",["bar"])] The  function is the same as , except it doesn't change the request's context path. This is useful if you want to route to a particular handler but you want that handler to receive the 1 as it is.Example: 6ghci> :set -XOverloadedStrings ghci> import qualified Data.Map as M ghci> import qualified Data.ByteString.Char8 as B8 ghci> import  Snap.Test& ghci> :{ ghci| let handler = do r <- Ü ghci| % $ "rqContextPath: " <> 2" r <> "\n" ghci| % $ "rqPathInfo: " <> 1" r <> "\n" ghci| %% $ "rqParams: " <> (B8.pack . show $ 5 r) ghci| :} ghci> " (& "/foo/bar" M.empty) (‡ [("foo", handler)]) HTTP/1.1 200 OK server: Snap/test date: Sat, 02 Aug 2014 05:17:28 GMT rqContextPath: / rqPathInfo: foo/bar ghci> " (& "/foo/bar" M.empty) (k [("foo/:bar", handler)]) [...] rqContextPath: / rqPathInfo: foo/bar rqParams: fromList [("bar",["bar"])] Ö-action to run before we call the user handler^the "context"; the list of path segments we've already successfully matched, in reverse order0the list of path segments we haven't yet matched Ó×ÔÕØÙÚÛÜÓ×ÔÕØ'NoneU½”None½Í’"76543210/.-*)('&%$+89:;<=>?@ABCMHDEFGIJKLNPOUVWXY]^_`abcdefgjklmnoprsuv§¨©­®¯´µ¼ÃÄÅÆÊËÌÍÎÏĞÑÒÓÖרÙÜİŞßàáâãäåæçèéêëìíîïğñòóôõö÷ûüışÿ§¼÷ÊË´µôÏĞÑÆÃÄÅöõÒÓרÖÙÜİŞßáàâãïğñæÌÍÎ"$%&'()*+-./01234567NOP8CDEFGHIJKLM9:;<=>?@ABUVWXYòó+$%&'()*-./01234567]^_ûüışÿ`auvbedgjklmnopäåcfçèéìëêíîrs­®¯¨©§(None1<CVf€5 A form parameter name-value pair6Upload policy can be set on an "general" basis (using …), but handlers can also make policy decisions on individual files/parts uploaded. For each part uploaded, handlers can decide:'whether to allow the file upload at all.the maximum size of uploaded files, if allowed4File upload policy, if any policy is violated then  is thrown0 controls overall policy decisions relating to multipart/form-data uploads, specifically:Qwhether to treat parts without filenames as form input (reading them into the 5 map)†because form input is read into memory, the maximum size of a form input read in this manner, and the maximum number of form inputsåthe minimum upload rate a client must maintain before we kill the connection; if very low-bitrate uploads were allowed then a Snap server would be vulnerable to a trivial denial-of-service using a "slowloris"-type attackuthe minimum number of seconds which must elapse before we start killing uploads for having too low an upload rate.pthe amount of time we should wait before timing out the connection whenever we receive input from the client.Thrown when an  or  is violated.3Human-readable error message corresponding to the .KThrown when a part is invalid in some way (e.g. the headers are too large). 2Human-readable error message corresponding to the .!<All of the exceptions defined in this package inherit from !, so if you write .foo `catch` \(e :: FileUploadException) -> ...you can catch a , a , etc.""A contains information about a "part" in a request uploaded with !Content-type: multipart/form-data.#EField name associated with this part (i.e., the name specified with <input name="partFieldName" ...).$Name of the uploaded file.%Content type of this part.&#Disposition type of this part. See (.',Remaining headers associated with this part.(2Represents the disposition type specified via the Content-Disposition header field. See  $https://www.ietf.org/rfc/rfc1806.txtRFC 1806.)Content-Disposition: attachment.*Content-Disposition: file.+Content-Disposition: form-data.,Any other value.-EA type alias for a function that will process one of the parts of a multipart/form-data- HTTP request body without usinc accumulator..EA type alias for a function that will process one of the parts of a multipart/form-data$ HTTP request body with accumulator./Contents of form field of type file1Name of a field2Result of storing file3ZReads uploaded files into a temporary directory and calls a user handler to process them.Note: %THE REQUEST MUST BE CORRECTLY ENCODED. If the request's  Content-type is not "multipart/formdata)", this function skips processing using pass.‘Given a temporary directory, global and file-specific upload policies, and a user handler, this function consumes a request body uploaded with !Content-type: multipart/form-dataˆ. Each file is read into the temporary directory, and is then passed to the user handler. After the user handler runs (but before the Response¾ body is streamed to the client), the files are deleted from disk; so if you want to retain or use the uploaded files in the generated response, you need to move or otherwise process them.3The argument passed to the user handler is a tuple: 4(PartInfo, Either PolicyViolationException FilePath)"The first half of this tuple is a "Ÿ, which contains the information the client browser sent about the given upload part (like filename, content-type, etc). The second half of this tuple is an İ stipulating that either: Dthe file was rejected on a policy basis because of the provided  handler3the file was accepted and exists at the given path. ExceptionsFIf the client's upload rate passes below the configured minimum (see @ and BŒ), this function terminates the connection. This setting is there to protect the server against slowloris-style denial of service attacks. If the given ; stipulates that you wish form inputs to be placed in the 5 parameter map (using :U), and a form input exceeds the maximum allowable size, this function will throw a .If an uploaded part contains MIME headers longer than a fixed internal threshold (currently 32KB), this function will throw a .4GProcesses form data and calls provided storage function on file parts.You can use this together with M, L7 or provide your own callback to store uploaded files.cIf you need to process uploaded file mime type or file name, do it in the store callback function. See also 5.DExample using with small files which can safely be stored in memory. ÿU import qualified Data.ByteString.Lazy as Lazy handleSmallFiles :: MonadSnap m => [(ByteString, ByteString, Lazy.ByteString)] handleSmallFiles = handleFormUploads uploadPolicy filePolicy store where uploadPolicy = defaultUploadPolicy filePolicy = setMaximumFileSize (64*1024) $ setMaximumNumberOfFiles 5 defaultUploadPolicy store partInfo stream = do content <- storeAsLazyByteString partInfo stream let fileName = partFileName partInfo fileMime = partContentType partInfo in (fileName, fileMime, content) 5iGiven an upload policy and a function to consume uploaded "parts", consume a request body uploaded with !Content-type: multipart/form-data.If : is Ş, then parts with disposition  form-data… (a form parameter) will be processed and returned as first element of resulting pair. Parts with other disposition will be fed to . handler.If : is ß2, then parts with any disposition will be fed to .› handler and first element of returned pair will be empty. In this case it is important that you limit number of form inputs and sizes of inputs in your .% handler to avoid common DOS attacks.Note: %THE REQUEST MUST BE CORRECTLY ENCODED. If the request's  Content-type is not "multipart/formdata)", this function skips processing using pass.)Most users will opt for the higher-level 30, which writes to temporary files, rather than 6«. This function should be chosen, however, if you need to stream uploaded files directly to your own processing function: e.g. to a database or a remote service via RPC.FIf the client's upload rate passes below the configured minimum (see @ and BŒ), this function terminates the connection. This setting is there to protect the server against slowloris-style denial of service attacks. Exceptions If the given > stipulates that you wish form inputs to be processed (using :‚), and a form input exceeds the maximum allowable size or the form exceeds maximum number of inputs, this function will throw a .If an uploaded part contains MIME headers longer than a fixed internal threshold (currently 32KB), this function will throw a .Since: 1.0.3.06 A variant of 57 accumulating results into a list. Also puts captured &s into rqPostParams and rqParams maps.72Human-readable error message corresponding to the !.8FA reasonable set of defaults for upload policy. The default policy is: maximum form input size128kBmaximum number of form inputs10minimum upload rate1kB/s%seconds before rate limiting kicks in10inactivity timeout 20 seconds9_Does this upload policy stipulate that we want to treat parts without filenames as form input?:ISet the upload policy for treating parts without filenames as form input.;DGet the maximum size of a form input which will be read into our 5 map.<DSet the maximum size of a form input which will be read into our 5 map.=DGet the maximum size of a form input which will be read into our 5 map.>DSet the maximum size of a form input which will be read into our 5 map.?Get the minimum rate (in  bytes/second:) a client must maintain before we kill the connection.@Set the minimum rate (in  bytes/second:) a client must maintain before we kill the connection.A]Get the amount of time which must elapse before we begin enforcing the upload rate minimumB]Set the amount of time which must elapse before we begin enforcing the upload rate minimumC‡Get the "upload timeout". Whenever input is received from the client, the connection timeout is set this many seconds in the future.DSet the upload timeout.E A default  maximum file size1MBmaximum number of files10skip files without nameyesmaximum size of skipped file0F%Maximum size of single uploaded file.G!Maximum number of uploaded files.H!Skip files with empty file names.DIf set, parts without filenames will not be fed to storage function.®HTML5 form data encoding standard states that form input fields of type file, without value set, are encoded same way as if file with empty body, empty file name, and type application/octet-stream was set as value.\You most likely want to use this with zero bytes allowed to avoid storing such fields (see I).+By default files without names are skipped.Since: 1.0.3.0I7Maximum size of file without name which can be skipped. Ignored if H is False.1If skipped file is larger than this setting then ! is thrown."By default maximum file size is 0.Since: 1.0.3.0J"Disallows the file to be uploaded.K2Allows the file to be uploaded, with maximum size n.L.Stores file body in memory as Lazy ByteString.MDStore files in a temporary directory, and clean up on function exit.-Files are safe to move until function exists.PIf asynchronous exception is thrown during cleanup, temporary files may remain. %uploadsHandler = withTemporaryStore "varçtmp" "upload-" $ store -> do (inputs, files) <- handleFormUploads defaultUploadpolicy defaultFileUploadPolicy (const store) saveFiles files à‰Assuming we've already identified the boundary value and split the input up into parts which match and parts which don't, run the given ²E InputStream over each part and grab a list of the resulting values.TODO/FIXME: fix description3temporary directorygeneral upload policyper-part upload policy(user handler (see function description)4general upload policyUpload policy for filesA file storage function5global upload policypart processorseed accumulator6global upload policypart processorMtemporary directoryfile name patternAction taking store functionámaximum size of form inputfile reading codeâmax num fieldsboundary valuepart processorGãäåæçèéêëìíîïğñ !ò"ó#$%'&()*+,-./0123456ô789:;<=>?@ABCDEFGHIJKLM õö÷øùúûüıãäæåçèéîïíëìêğñ şÿ!ò"ó#$%&'()*+,/012)NoneV©|NDA collection of options for serving static files out of a directory.PBFiles to look for when a directory is requested (e.g., index.html)Q=Handler to generate a directory listing if there is no index.R—Map of extensions to pass to dynamic file handlers. This could be used, for example, to implement CGI dispatch, pretty printing of source code, etc.S'MIME type map to look up content types.T†Handler that is called before a file is served. It will only be called when a file is actually found, not for generated index pages.UA type alias for MIME typeV!A type alias for dynamic handlersWGets a path from the " using 1¤ and makes sure it is safe to use for opening files. A path is safe if it is a relative path and has no ".." elements to escape the intended directory structure.Example: 6ghci> :set -XOverloadedStrings ghci> import qualified Data.Map as M ghci> import qualified  Snap.Test as T ghci> import qualified Data.ByteString.Char86 as B8 ghci> T.runHandler (T.get "/foo/bar" M.empty) (W >>= é . B8.pack) HTTP/1.1 200 OK server: Snap/test date: Fri, 08 Aug 2014 16:13:20 GMT foo/bar ghci> T.runHandler (T.get "/foo/../bar" M.empty) (W >>= é' . B8.pack) HTTP/1.1 404 Not Found ... XLThe default set of mime type mappings we use when serving files. Its value: ÿ¥Map.fromList [ ( ".asc" , "text/plain" ), ( ".asf" , "video/x-ms-asf" ), ( ".asx" , "video/x-ms-asf" ), ( ".au" , "audio/basic" ), ( ".avi" , "video/x-msvideo" ), ( ".bmp" , "image/bmp" ), ( ".bz2" , "application/x-bzip" ), ( ".c" , "text/plain" ), ( ".class" , "application/octet-stream" ), ( ".conf" , "text/plain" ), ( ".cpp" , "text/plain" ), ( ".css" , "text/css" ), ( ".cxx" , "text/plain" ), ( ".doc" , "application/msword" ), ( ".docx" , S.append "application/vnd.openxmlformats-officedocument" ".wordprocessingml.document" ), ( ".dotx" , S.append "application/vnd.openxmlformats-officedocument" ".wordprocessingml.template" ), ( ".dtd" , "application/xml-dtd" ), ( ".dvi" , "application/x-dvi" ), ( ".exe" , "application/octet-stream" ), ( ".flv" , "video/x-flv" ), ( ".gif" , "image/gif" ), ( ".gz" , "application/x-gzip" ), ( ".hs" , "text/plain" ), ( ".htm" , "text/html" ), ( ".html" , "text/html" ), ( ".ico" , "image/x-icon" ), ( ".jar" , "application/x-java-archive" ), ( ".jpeg" , "image/jpeg" ), ( ".jpg" , "image/jpeg" ), ( ".js" , "text/javascript" ), ( ".json" , "application/json" ), ( ".log" , "text/plain" ), ( ".m3u" , "audio/x-mpegurl" ), ( ".m3u8" , "application/x-mpegURL" ), ( ".mka" , "audio/x-matroska" ), ( ".mk3d" , "video/x-matroska" ), ( ".mkv" , "video/x-matroska" ), ( ".mov" , "video/quicktime" ), ( ".mp3" , "audio/mpeg" ), ( ".mp4" , "video/mp4" ), ( ".mpeg" , "video/mpeg" ), ( ".mpg" , "video/mpeg" ), ( ".ogg" , "application/ogg" ), ( ".pac" , "application/x-ns-proxy-autoconfig" ), ( ".pdf" , "application/pdf" ), ( ".png" , "image/png" ), ( ".potx" , S.append "application/vnd.openxmlformats-officedocument" ".presentationml.template" ), ( ".ppsx" , S.append "application/vnd.openxmlformats-officedocument" ".presentationml.slideshow" ), ( ".ppt" , "application/vnd.ms-powerpoint" ), ( ".pptx" , S.append "application/vnd.openxmlformats-officedocument" ".presentationml.presentation" ), ( ".ps" , "application/postscript" ), ( ".qt" , "video/quicktime" ), ( ".rtf" , "text/rtf" ), ( ".sig" , "application/pgp-signature" ), ( ".sldx" , S.append "application/vnd.openxmlformats-officedocument" ".presentationml.slide" ), ( ".spl" , "application/futuresplash" ), ( ".svg" , "image/svg+xml" ), ( ".swf" , "application/x-shockwave-flash" ), ( ".tar" , "application/x-tar" ), ( ".tar.bz2" , "application/x-bzip-compressed-tar" ), ( ".tar.gz" , "application/x-tgz" ), ( ".tbz" , "application/x-bzip-compressed-tar" ), ( ".text" , "text/plain" ), ( ".tif" , "image/tiff" ), ( ".tiff" , "image/tiff" ), ( ".tgz" , "application/x-tgz" ), ( ".torrent" , "application/x-bittorrent" ), ( ".ts" , "video/mp2t" ), ( ".txt" , "text/plain" ), ( ".wav" , "audio/x-wav" ), ( ".wax" , "audio/x-ms-wax" ), ( ".webm" , "video/webm" ), ( ".wma" , "audio/x-ms-wma" ), ( ".wmv" , "video/x-ms-wmv" ), ( ".xbm" , "image/x-xbitmap" ), ( ".xlam" , "application/vnd.ms-excel.addin.macroEnabled.12" ), ( ".xls" , "application/vnd.ms-excel" ), ( ".xlsb" , "application/vnd.ms-excel.sheet.binary.macroEnabled.12" ), ( ".xlsx" , S.append "application/vnd.openxmlformats-officedocument." "spreadsheetml.sheet" ), ( ".xltx" , S.append "application/vnd.openxmlformats-officedocument." "spreadsheetml.template" ), ( ".xml" , "text/xml" ), ( ".xpm" , "image/x-xpixmap" ), ( ".xwd" , "image/x-xwindowdump" ), ( ".zip" , "application/zip" ) ]<Style information for the default directory index generator.YAn automatic index generator, which is fairly small and does not rely on any external files (which may not be there depending on external request routing).A U‘ is passed in to display the types of files in the directory listing based on their extension. Preferably, this is the same as the map in the Nÿ The styles parameter allows you to apply styles to the directory listing. The listing itself consists of a table, containing a header row using th elements, and one row per file using td elements, so styles for those pieces may be attached to the appropriate tags.ZfA very simple configuration for directory serving. This configuration uses built-in MIME types from XF, and has no index files, index generator, dynamic file handlers, or T.[mA reasonable default configuration for directory serving. This configuration uses built-in MIME types from X, serves common index files  index.html and  index.htmX, but does not autogenerate directory indexes, nor have any dynamic file handlers. The T will not do anything.\dA more elaborate configuration for file serving. This configuration uses built-in MIME types from X, serves common index files  index.html and  index.htmj, and autogenerates directory indexes with a Snap-like feel. It still has no dynamic file handlers, nor T#, which should be added as needed.$Files recognized as indexes include  index.html,  index.htm,  default.html,  default.htm,  home.html<Example of how the autogenerated directory index looks like: ÿ¤ş]RServes static files from a directory using the default configuration as given in [.^NServes static files from a directory. Configuration options are passed in a NT that captures various choices about desired behavior. The relative path given in 1 is searched for a requested file, and the file is served with the appropriate mime type if it is found. Absolute paths and "..N" are prohibited to prevent files from being served from outside the sandbox._Serves a single file specified by a full or relative path. If the file does not exist, throws an exception (not that it does not8 pass to the next handler). The path restrictions on ]P don't apply to this function since the path is not being supplied by the user.`Same as _*, with control over the MIME mapping used.aPDetermine a given file's MIME type from its filename and the provided MIME map.Y%MIME type mapping for reporting typesStyle info to insert in headerDirectory to generate index for]Directory to serve from^Configuration optionsDirectory to serve from_ path to file` MIME type path to fileNOPQRSTUVWXYZ[\]^_`aNOPQRST*NoneKQVDi"bThe bŒ datatype enumerates the different kinds of HTTP requests you can generate using the testing interface. Most users will prefer to use the }, , €, , and ~ convenience functions.h(Represents a single file upload for the m.jthe file's namekthe file's content-typelthe file contentsm A single "multipart/form-dataQ" form parameter: either a list of regular form values or a set of file uploads.n(a form variable consisting of the given ² values.o&a file upload consisting of the given h values.pA request body of type "multipart/form-dataƒ" consists of a set of named form parameters, each of which can by either a list of regular form values or a set of file uploads.qTRequestBuilder is a monad transformer that allows you to conveniently build a snap " for testing.rRuns a q, producing the desired ".N.B. please} don't use the request you get here in a real Snap application; things will probably break. Don't say you weren't warned :-)Example: 6ghci> :set -XOverloadedStrings ghci> import qualified Data.Map as M ghci> r $ }| "/foo/bar" M.empty GET /foo/bar HTTP/1.1 host: localhost sn="localhost" c=127.0.0.1:60000 s=127.0.0.1:8080 ctx=/ clen=n/a sSets the type of the " being built.Example: 6ghci> :set -XOverloadedStrings ghci> import qualified Data.Map as M ghci> r $ ~ "/foo/bar" M.empty >> st GetRequest GET /foo/bar HTTP/1.1 host: localhost sn="localhost" c=127.0.0.1:60000 s=127.0.0.1:8080 ctx=/ clen=n/a t˜Sets the request's query string to be the raw bytestring provided, without any escaping or other interpretation. Most users should instead choose the u+ function, which takes a parameter mapping.Example: 6ghci> :set -XOverloadedStrings ghci> import qualified Data.Map as M ghci> r $ } "/foo/bar" M.empty >> tÀ "param0=baz&param1=qux" GET /foo/bar?param0=baz&param1=qux HTTP/1.1 host: localhost sn="localhost" c=127.0.0.1:60000 s=127.0.0.1:8080 ctx=/ clen=n/a params: param0: ["baz"], param1: ["qux"] uOEscapes the given parameter mapping and sets it as the request's query string.Example: 6ghci> :set -XOverloadedStrings ghci> import qualified Data.Map as M ghci> r $ } "/foo/bar" M.empty >> uà (M.fromList [("param0", ["baz"]), ("param1", ["qux"])]) GET /foo/bar?param0=baz&param1=qux HTTP/1.1 host: localhost sn="localhost" c=127.0.0.1:60000 s=127.0.0.1:8080 ctx=/ clen=n/a params: param0: ["baz"], param1: ["qux"] vmSets the given header in the request being built, overwriting any header with the same name already present.Example: 6ghci> :set -XOverloadedStrings ghci> import qualified Data.Map as M ghci> :{ ghci| r5 $ do get "/foo/bar" M.empty ghci| v. "Accept" "text/html" ghci| v› "Accept" "text/plain" ghci| :} GET /foo/bar HTTP/1.1 accept: text/plain host: localhost sn="localhost" c=127.0.0.1:60000 s=127.0.0.1:8080 ctx=/ clen=n/a w1Adds the given header to the request being built.Example: 6ghci> :set -XOverloadedStrings ghci> import qualified Data.Map as M ghci> :{ ghci| r $ do }, "/foo/bar" M.empty ghci| w. "Accept" "text/html" ghci| w¥ "Accept" "text/plain" ghci| :} GET /foo/bar HTTP/1.1 accept: text/html,text/plain host: localhost sn="localhost" c=127.0.0.1:60000 s=127.0.0.1:8080 ctx=/ clen=n/a x2Adds the given cookies to the request being built.Example: 6ghci> :set -XOverloadedStrings ghci> import qualified Data.Map as M ghci> import  Snap.Core ghci> let cookie = 9: "name" "value" Nothing Nothing Nothing False False ghci> r $ } "/foo/bar" M.empty >> xÇ [cookie] GET /foo/bar HTTP/1.1 cookie: name=value host: localhost sn="localhost" c=127.0.0.1:60000 s=127.0.0.1:8080 ctx=/ clen=n/a cookies: Cookie {cookieName = "name", cookieValue = "value", ...}  Convert 9 into ² for output.ySets the request's  content-type to the given MIME type.Example: 6ghci> :set -XOverloadedStrings ghci> import qualified Data.Map as M ghci> r $ ' "/foo/bar" "text/html" "some text" >> yŸ "text/plain" PUT /foo/bar HTTP/1.1 content-type: text/plain content-length: 9 host: localhost sn="localhost" c=127.0.0.1:60000 s=127.0.0.1:8080 ctx=/ clen=9 zYControls whether the test request being generated appears to be an https request or not.Example: 6ghci> :set -XOverloadedStrings ghci> import qualified Data.Map as M ghci> r $ ~ "/foo/bar" M.empty >> zx True DELETE /foo/bar HTTP/1.1 host: localhost sn="localhost" c=127.0.0.1:60000 s=127.0.0.1:8080 ctx=/ clen=n/a secure {$Sets the test request's http versionExample: 6ghci> :set -XOverloadedStrings ghci> import qualified Data.Map as M ghci> r $ ~ "/foo/bar" M.empty >> {r (1,0) DELETE /foo/bar HTTP/1.0 host: localhost sn="localhost" c=127.0.0.1:60000 s=127.0.0.1:8080 ctx=/ clen=n/a |>Sets the request's path. The path provided must begin with a "/ " and must notc contain a query string; if you want to provide a query string in your test request, you must use u or t . Note that 2 is never set by any q function.Example: 6ghci> :set -XOverloadedStrings ghci> import qualified Data.Map as M ghci> r $ } "/foo/bar" M.empty >> |t "/bar/foo" GET /bar/foo HTTP/1.1 host: localhost sn="localhost" c=127.0.0.1:60000 s=127.0.0.1:8080 ctx=/ clen=n/a }=Builds an HTTP "GET" request with the given query parameters.Example: 6ghci> :set -XOverloadedStrings ghci> import qualified Data.Map as M ghci> r $ }Õ "/foo/bar" (M.fromList [("param0", ["baz", "quux"])]) GET /foo/bar?param0=baz&param0=quux HTTP/1.1 host: localhost sn="localhost" c=127.0.0.1:60000 s=127.0.0.1:8080 ctx=/ clen=n/a params: param0: ["baz","quux"] ~@Builds an HTTP "DELETE" request with the given query parameters.Example: 6ghci> :set -XOverloadedStrings ghci> import qualified Data.Map as M ghci> r $ ~ "/foo/bar" M.empty DELETE /foo/bar HTTP/1.1 host: localhost sn="localhost" c=127.0.0.1:60000 s=127.0.0.1:8080 ctx=/ clen=n/a wBuilds an HTTP "POST" request with the given form parameters, using the "application/x-www-form-urlencoded" MIME type.Example: 6ghci> :set -XOverloadedStrings ghci> import qualified Data.Map as M ghci> r $ ÿ "/foo/bar" (M.fromList [("param0", ["baz", "quux"])]) POST /foo/bar HTTP/1.1 content-type: application/x-www-form-urlencoded content-length: 22 host: localhost sn="localhost" c=127.0.0.1:60000 s=127.0.0.1:8080 ctx=/ clen=22 params: param0: ["baz","quux"] €iBuilds an HTTP "POST" request with the given form parameters, using the "form-data/multipart" MIME type.Example: %ghci> :set -XOverloadedStrings ghci> r $ €ÿ "/foo/bar" [("param0", FormData ["baz", "quux"])] POST /foo/bar HTTP/1.1 content-type: multipart/form-data; boundary=snap-boundary-572334111ec0c05ad4812481e8585dfa content-length: 406 host: localhost sn="localhost" c=127.0.0.1:60000 s=127.0.0.1:8080 ctx=/ clen=406 Builds an HTTP "PUT" request.Example: %ghci> :set -XOverloadedStrings ghci> r $ ¶ "/foo/bar" "text/plain" "some text" PUT /foo/bar HTTP/1.1 content-type: text/plain content-length: 9 host: localhost sn="localhost" c=127.0.0.1:60000 s=127.0.0.1:8080 ctx=/ clen=9 ‚PBuilds a "raw" HTTP "POST" request, with the given MIME type and body contents.Example: %ghci> :set -XOverloadedStrings ghci> r $ ‚· "/foo/bar" "text/plain" "some text" POST /foo/bar HTTP/1.1 content-type: text/plain content-length: 9 host: localhost sn="localhost" c=127.0.0.1:60000 s=127.0.0.1:8080 ctx=/ clen=9 ƒGiven a web handler in the ¼ monad, and a q? defining a test request, runs the handler, producing an HTTP .ùThis function will produce almost exactly the same output as running the handler in a real server, except that chunked transfer encoding is not applied, and the "Transfer-Encoding" header is not set (this makes it easier to test response output).Example: 6ghci> :set -XOverloadedStrings ghci> import qualified Data.Map as M ghci> import  Snap.Core ghci> ƒ (} "foo/bar" M.empty) (%g "Hello, world!") HTTP/1.1 200 OK server: Snap/test date: Thu, 17 Jul 2014 21:03:23 GMT Hello, world! „&Given a web handler in some arbitrary Ê_ monad, a function specifying how to evaluate it within the context of the test monad, and a q? defining a test request, runs the handler, producing an HTTP .…Given a web handler in the ¼ monad, and a qV defining a test request, runs the handler and returns the monadic value it produces.Throws an exception if the ¼ handler early-terminates with + or ,-.Example: ,ghci> :set -XOverloadedStrings ghci> import  Control.Monad ghci> import qualified Data.Map as M ghci> import  Snap.Core ghci> … (} "foo/bar" M.empty) (%( "Hello, world!" >> return 42) 42 ghci> … (} "foo/bar" M.empty) ,-9 *** Exception: No handler for request: failure was pass †&Given a web handler in some arbitrary Ê_ monad, a function specifying how to evaluate it within the context of the test monad, and a qU defining a test request, runs the handler, returning the monadic value it produces.Throws an exception if the ¼ handler early-terminates with + or ,-.‡Converts the given  to a bytestring.Example:  ghci> import  Snap.Core ghci> ‡  "HTTP/1.1 200 OK\r\n\r\n" ˆConverts the given " to a bytestring.Since: 1.0.0.0Example: 6ghci> :set -XOverloadedStrings ghci> import qualified Data.Map as M ghci> r <- r $ get "/foo/bar" M.empty ghci> ˆ6 r "GET /foo/bar HTTP/1.1\r\nhost: localhost\r\n\r\n" } request pathrequest's form parameters~ request pathrequest's form parameters request pathrequest's form parameters€ request pathmultipart form parameters request pathrequest body MIME content-typerequest body contents‚ request pathrequest body MIME content-typerequest body contentsƒa request builder a web handler„a function defining how the Ê monad should be runa request builder a web handler†a function defining how the Ê monad should be runa request builder a web handler'bcdefghijklmonpqrstuvwxyz{|}~€‚ƒ„…†‡ˆbcdefghijklmnoq NoneE‚-‚ƒ„…†‡bcdefghijklmonpqrstuvwxyz{|}~€‚ƒ„…†‡ˆ-qpmnohijklbcdefgrƒ„…†}€‚~wyvx{ut|szƒ„…†‡‚ˆ‡ Noneeİ‹A newtype over   with a   instance.aSpecify the options to use when building CORS headers for a response. Most of these options are ./M actions to allow you to conditionally determine the setting of each header.8Which origins are allowed to make cross-origin requests.WWhether or not to allow exposing the response when the omit credentials flag is unset.‘‡A list of headers that are exposed to clients. This allows clients to read the values of these headers, if the response includes them.’+A list of request methods that are allowed.“rAn action to determine which of the request headers are allowed. This action is supplied the parsed contents of Access-Control-Request-Headers.”$Used to specify the contents of the Access-Control-Allow-Origin header.•:Allow any origin to access this resource. Corresponds to Access-Control-Allow-Origin: *–"Do not allow cross-origin requests—/Allow cross-origin requests from these origins.˜WA set of origins. RFC 6454 specifies that origins are a scheme, host and port, so the ˜ wrapper around a   ensures that each  % constists of nothing more than this.š(Liberal default options. Specifies that:*All origins may make cross-origin requestsallow-credentials is true.3No extra headers beyond simple headers are exposed.GET, POST, PUT, DELETE and HEAD are all allowed. All request headers are allowed.+All options are determined unconditionally.›·Apply CORS headers to a specific request. This is useful if you only have a single action that needs CORS headers, and you don't want to pay for conditional checks on every request.You should note that ›& needs to be used before you add any Ò@ combinators. For example, the following won't do what you want: 2method POST $ applyCORS defaultOptions $ myHandler'This fails to work as CORS requires an OPTIONSR request in the preflighting stage, but this would get filtered out. Instead, use 2applyCORS defaultOptions $ method POST $ myHandler‰Š‹Œ‘’“”•–—˜™š›œ›‘’“𔕖—˜™œ™‹Œ‰Š‰Š‹Œ‘’“”•–—˜™ NonegNOPQRSTUVWXYZ[\]^_`aWUVNOPQRSTZ[\YXa]^_`NonegŞ5 !"&'%$#()*+,-./0123456789:;<=>?@ABCDEFGHIJKLM<45./012LM36-"#$%&'()*+,#$%'&89:;<=>?@ABCDEFGHIJK!7   None1CV„u£Thrown when the 'Accept-Encoding'# request header has invalid format.¤Runs a Snap+ web handler with compression if available.(If the client has indicated support for gzip or deflate in its Accept-Encoding header, and the  Content-Type0 in the response is one of the following types: application/x-javascript application/json text/css  text/html text/javascript  text/plain text/xml application/x-font-truetype<Then the given handler's output stream will be compressed, Content-Encoding- will be set in the output headers, and the Content-Lengtho will be cleared if it was set. (We can't process the stream in O(1) space if the length is known beforehand.)<The wrapped handler will be run to completion, and then the Response that's contained within the Snap monad state will be passed to Ï to prevent further processing.Example: 6ghci> :set -XOverloadedStrings ghci> import qualified Data.Map as M ghci> import qualified  Snap.Testd as T ghci> let r = T.get "/" M.empty >> T.addHeader "Accept-Encoding" "gzip,deflate" ghci> let h = ã (0 "text/plain") >> %¤ "some text" ghci> T.runHandler r h HTTP/1.1 200 OK content-type: text/plain server: Snap/test date: Fri, 08 Aug 2014 15:40:45 GMT some text ghci> T.runHandler r (¤’ h) HTTP/1.1 200 OK content-type: text/plain vary: Accept-Encoding content-encoding: gzip server: Snap/test date: Fri, 08 Aug 2014 15:40:10 GMT ¥ The same as ¤2, with control over which MIME types to compress.¦WTurn off compression by setting "Content-Encoding: identity" in the response headers. ¤4 is a no-op when a content-encoding is already set.¤the web handler to run¥set of compressible MIME typesthe web handler to run£¤¥¦§¤¥¦£§£ Noneê2What kind of proxy is this? Affects which headers ­) pulls the original remote address from.'Currently only proxy servers that send X-Forwarded-For or  Forwarded-For are supported.«!no proxy, leave the request alone¬Use the  Forwarded-For or X-Forwarded-For header­Rewrite % if we're behind a proxy.Example: 6ghci> :set -XOverloadedStrings ghci> import qualified Data.Map as M ghci> import qualified  Snap.Testb as T ghci> let r = T.get "/foo" M.empty >> T.addHeader "X-Forwarded-For" "1.2.3.4" ghci> let h = 1 % >>= %€) ghci> T.runHandler r h HTTP/1.1 200 OK server: Snap/test date: Fri, 08 Aug 2014 14:32:29 GMT 127.0.0.1 ghci> T.runHandler r (­ ¬S h) HTTP/1.1 200 OK server: Snap/test date: Fri, 08 Aug 2014 14:33:02 GMT 1.2.3.4 ª«¬­ª«¬­ª«¬23456789:;<=>?@ABCDEFFGHIJKLMNOPQRRSTUVWXYZ[\]^_`abcdefghhijklmnopqrstuvwxyzq{|}~€‚ƒ„…†‡ˆ‰Š‹Œ0‘’“”•–—˜™š›œŸ ¡¢£¤¥¦§¨©ª«¬­®¯°±²³´µ¶·¸¹º»¼½¾¿ÀÁÂÃÄÅÆÇÈÉÊËÌÍÎÏĞÑÒÓÔÕÖרÙÚÛÜÜİİŞßàá..âãäåææçèéêëãìíîïğ+ñò$óôõö÷øùúûü1ışÿ%      !"#$%&'()*+,-./01234#5#6(7(8(9(:(;(<(=(>(?(@(A(B(C(D(E(F(G(H(I(J(K(L(M(M(N(O((P(Q(R(S(T(U(V(W(X(Y(Z([(\(](^(_(`(a(b(c(d(e(f(g(h(i)j)j)k)l)m)n)o)p)q)r)s)t)u)v)w)x)y)z){)|*}*~**€**‚*ƒ*ƒ*„*…*†*‡*ˆ*‰*Š*‹*Œ*****‚**0*‘*’*“*&*=*”*•*–*—*"*˜*™*š*›*œ     Ÿ Ÿ   ¡ ¢ £ ¤ ¥ ¦ § ¨ © ª « ¬ ­ ® ¯ ° ± ² ³ ´ µ ¶ · ¸ ¹ º » ¼ ½ ¾ ¿ À Á ÂÃÄÅÆÇÈÉÊÃËÌÃËÍÎÏĞÎÑÒÃÓÔÕÖ×ÎØÙÎØÚÎØÛÎØ5ÎØÜİŞßàáâÎãäÎãåæçèÎéêëìíëìîïğñÃòÅóôõóöõÎØ÷ÃøùİŞúûûü#ı#ş#ÿ#######Îàá àá ( ( ( (8(9(((((:(((((((;(=((@((((((((ş(ş(( (!("("(#($)%)&)')')(**‹)*+,-./01 © ´2'snap-core-1.0.3.2-6c0XIwxbJjXXfnyiXTcpuSnap.Internal.DebugSnap.Types.HeadersSnap.Internal.Http.Types Snap.TestSnap.Internal.ParsingSnap.Internal.Core Snap.CoreSnap.Util.FileUploadsSnap.Util.FileServeSnap.Util.CORSSnap.Util.GZipSnap.Util.ProxyForeign.C.ErrorErrnogetErrnohandleFileUploadsSnap.Internal.Test.Assertions emptyResponsesetResponseStatus setHeadersetResponseBodyData.MapfromListControl.Monad.Trans.ClassliftControl.Monad.Trans.ReaderReaderTControl.Monad.Trans.WriterWriterTSystem.IO.Streams.Combinators skipToEofSnap.Http.Server httpServe runHandlerSnap.Internal.RoutingmethodwriteBSgetSnap.Internal.InstancesSnap.Internal.Util.FileUploadsSnap.Internal.Util.FileServe!Snap.Internal.Test.RequestBuilder finishWith Control.MonadmzeroSnapHandlersetContentType getsRequestdebug debugErrnoHeadersemptynullmemberlookuplookupWithDefaultinsert unsafeInsertsetdeletefoldl' foldedFoldl'foldr foldedFoldrtoListunsafeFromCaseFoldedListunsafeToCaseFoldedList $fShowHeadersResponse rspHeaders rspCookiesrspContentLengthrspBody rspStatusrspStatusReasonrspTransformingRqBody ResponseBodyStreamSendFile StreamProcRequest rqHostName rqClientAddr rqClientPort rqServerAddr rqServerPortrqLocalHostname rqIsSecure rqHeadersrqBodyrqContentLengthrqMethod rqVersion rqCookies rqPathInfo rqContextPathrqURI rqQueryStringrqParams rqQueryParams rqPostParamsParamsCookie cookieName cookieValue cookieExpires cookieDomain cookiePath cookieSecurecookieHttpOnly HttpVersionMethodGETHEADPOSTPUTDELETETRACEOPTIONSCONNECTPATCH HasHeaders updateHeadersheadersc_format_log_timec_format_http_timec_parse_http_time set_c_locale addHeader getHeader listHeaders deleteHeadernormalizeMethod rspBodyMap rspBodyToEnumrqParam rqPostParam rqQueryParamrqModifyParams rqSetParamsetResponseCodemodifyResponseBody cookieToBS renderCookiesaddResponseCookiegetResponseCookiegetResponseCookiesdeleteResponseCookiemodifyResponseCookiesetContentLengthclearContentLength formatLogTimeformatHttpTime parseHttpTimestatusReasonMap rqRemoteAddr rqRemotePort$fHasHeadersHeaders $fOrdMethod $fEqMethod$fHasHeadersRequest $fShowRequest$fHasHeadersResponse$fShowResponse $fShowMethod $fReadMethod $fEqCookie $fShowCookiegetResponseBody assertSuccess assert404assertRedirectToassertRedirectassertBodyContainsDList fullyParse fullyParse'parseNumuntilEOLcrlf toTableListtoTableskipFieldChars isFieldCharpHeaderspWordpWord' pQuotedStringpQuotedString' isRFCTextmatchAllpAvPairspAvPair pParameter pParameter'trimpValueWithParameterspValueWithParameters'pContentTypeWithParameterspTokenisTokenpTokensparseToCompletion pUrlEscaped urlDecode urlEncodeurlEncodeBuilderurlEncodeCleanhexdfinishparseUrlEncodedbuildUrlEncodedprintUrlEncodedpCookies parseCookie unsafeFromHex unsafeFromNatNoHandlerException SnapState _snapRequest _snapResponse _snapLogError_snapModifyTimeoutunSnapZeroPassOnProcessingEarlyTermination EscapeSnapTerminateConnection EscapeHttpEscapeHttpHandler SnapResult SnapValue MonadSnapliftSnaprunRequestBodyreadRequestBodytransformRequestBodycatchFinishWithpassmethodsupdateContextPathpathWithdirpathpathArgifTopsgetsmodify getRequest getResponse getsResponse putResponse putRequest modifyRequestmodifyResponseredirect redirect'logError addToOutput writeBuilderwriteLBS writeText writeLazyTextsendFilesendFilePartial localRequest withRequest withResponseipHeaderFilteripHeaderFilter' bracketSnapterminateConnection escapeHttprunSnap fixupResponseevalSnap getParamFromgetParam getPostParam getQueryParam getParams getPostParamsgetQueryParams getCookie readCookie expireCookie setTimeout extendTimeout modifyTimeoutgetTimeoutModifier$fShowEscapeSnap$fExceptionEscapeSnap$fAlternativeSnap$fApplicativeSnap $fFunctorSnap$fMonadPlusSnap$fMonadBaseIOSnap $fMonadIOSnap$fMonadFailSnap $fMonadSnap$fMonadSnapSnap$fMonadBaseControlIOSnap$fExceptionNoHandlerException$fShowNoHandlerException$fEqNoHandlerExceptionroute routeLocal FormParamPartUploadPolicyFileUploadPolicy UploadPolicyPolicyViolationExceptionpolicyViolationExceptionReasonBadPartExceptionbadPartExceptionReasonFileUploadExceptionPartInfo partFieldName partFileNamepartContentTypepartDisposition partHeadersPartDispositionDispositionAttachmentDispositionFileDispositionFormDataDispositionOther PartProcessorPartFoldFormFile formFileName formFileValuehandleFormUploads foldMultiparthandleMultipartfileUploadExceptionReasondefaultUploadPolicydoProcessFormInputssetProcessFormInputsgetMaximumFormInputSizesetMaximumFormInputSizegetMaximumNumberOfFormInputssetMaximumNumberOfFormInputsgetMinimumUploadRatesetMinimumUploadRategetMinimumUploadSecondssetMinimumUploadSecondsgetUploadTimeoutsetUploadTimeoutdefaultFileUploadPolicysetMaximumFileSizesetMaximumNumberOfFilessetSkipFilesWithoutNamessetMaximumSkippedFileSizedisallowallowWithMaximumSizestoreAsLazyByteStringwithTemporaryStoreDirectoryConfig indexFilesindexGeneratordynamicHandlers mimeTypes preServeHookMimeMap HandlerMap getSafePathdefaultMimeTypesdefaultIndexGeneratorsimpleDirectoryConfigdefaultDirectoryConfigfancyDirectoryConfigserveDirectoryserveDirectoryWith serveFile serveFileAsfileType RequestType GetRequestRequestWithRawBodyMultipartPostRequestUrlEncodedPostRequest DeleteRequestFileData fdFileName fdContentType fdContentsMultipartParamFormDataFilesMultipartParamsRequestBuilder buildRequestsetRequestTypesetQueryStringRawsetQueryString addCookies setSecuresetHttpVersionsetRequestPathpostUrlEncoded postMultipartputpostRaw runHandlerM evalHandler evalHandlerMresponseToStringrequestToStringHashableMethod HashableURI CORSOptionscorsAllowOrigincorsAllowCredentialscorsExposeHeaderscorsAllowedMethodscorsAllowedHeaders OriginList EverywhereNowhereOrigins OriginSetoriginsdefaultOptions applyCORS mkOriginSet$fHashableHashableURI$fShowHashableURI$fShowHashableMethod$fHashableHashableMethod$fEqHashableURI$fEqHashableMethodBadAcceptEncodingExceptionwithCompressionwithCompression' noCompressioncompressibleMimeTypes%$fExceptionBadAcceptEncodingException $fShowBadAcceptEncodingException ProxyTypeNoProxyX_Forwarded_For behindProxy$fReadProxyType$fShowProxyType $fEqProxyType$fOrdProxyTypebytestring-0.10.8.2Data.ByteString.Internal ByteString/case-insensitive-1.2.0.10-1Cqb2PqMAYNmEZ0jMKGvkData.CaseInsensitive.InternalCIHunH Data.ByteString.Builder.InternalBuilder byteStringbaseForeign.C.TypesCTimeGHC.Real fromIntegralData.ByteString.BuildertoLazyByteStringcontainers-0.5.10.2Data.Map.InternalMapGHC.BaseMonad Alternative MonadPlus<|>)io-streams-1.5.0.1-6auM1z4KeQM5ZSLLooUu8lSystem.IO.Streams.Internal OutputStreamghc-prim GHC.TypesIOControl.Monad.IO.ClassMonadIOliftIO,monad-control-1.0.2.3-4Dn1Phj5k8rLcgnEMPHikLControl.Monad.Trans.ControlMonadBaseControl GHC.Exception SomeException+lifted-base-0.2.3.12-9BUfqcMCwBm5ZiNpsDDbGEControl.Exception.LiftedthrowIOcatchtransformers-0.5.2.0Control.Monad.Trans.State.LazyStateTData.ByteString.Lazy.Internal#text-1.2.2.2-EakMpasry3jA6OIwSZhq9MData.Text.InternalTextData.Text.Internal.LazyfailData.ByteString intercalate InputStreamStSnapunStSnapRouteCaptureDirroute'ActionNoRoute routeHeightrouteEarliestNC splitPathpRoute Data.EitherEitherTrueFalse foldPartscaptureVariableOrReadFileinternalFoldMultipartmaxNumberOfFilesmaxFileUploadSizeskipEmptyFileNamemaxEmptyFileNameSize uploadTimeoutminimumUploadRateminimumUploadSecondsmaximumNumberOfFormInputsprocessFormInputsmaximumFormInputSizeWrappedFileUploadExceptiontoPartDispositionMultipartState numFormVars numFormFilescapturedFields accumulatorFileExceptionWithReasonexceptionReason UploadStatenumUploadedFiles uploadedFilessnapIndexStylesdecodeFilePathRangeReqSuffixRangeReq)network-uri-2.6.1.0-RmiO7Es4EVIxdXQfHHaUt Network.URIURI'hashable-1.2.6.1-JDYnvpSucMf1h1i2CUUtVbData.Hashable.ClassHashable3unordered-containers-0.2.8.0-3iSQJVS3Sio885UUC852oj Data.HashSetHashSet