modern-uri
Modern library for working with URIs
https://github.com/mrkkrp/modern-uri
| LTS Haskell 24.16: | 0.3.6.1@rev:3 | 
| Stackage Nightly 2025-10-25: | 0.3.6.1@rev:3 | 
| Latest on Hackage: | 0.3.6.1@rev:3 | 
modern-uri-0.3.6.1@sha256:1ff00fbd7e695b0695bb325a1fe375cc8c83b55cd7b7c08af083526809a8d76a,3602Module documentation for 0.3.6.1
Modern URI
This is a modern library for working with URIs in Haskell as per RFC 3986:
https://tools.ietf.org/html/rfc3986
Features
The modern-uri package features:
- Correct by construction URIdata type. The correctness is ensured by making sure that every sub-component of theURIrecord cannot be invalid.
- Textual components in the URIdata type are represented asTextrather thanByteString, because they are percent-decoded and so they can contain characters outside of ASCII range (i.e. Unicode). This allows for easier manipulation ofURIs, while encoding and decoding headaches are handled by the parsers and renders for you.
- Absolute and relative URIs differ only by the scheme component: if it’s
Nothing, then the URI is relative, otherwise it’s absolute.
- A Megaparsec parser that can be used as a standalone smart constructor for
the URIdata type (seemkURI) as well as seamlessly integrated into a bigger Megaparsec parser that consumes a strictText(seeparser) or strictByteString(seeparserBs).
- The parser performs some normalization, for example it collapses
consecutive slashes. Some smart constructors such as mkSchemeandmkHostalso perform normalization. So in a sense URIs are also “normalized by construction” to some extent.
- Fast rendering to strict TextandByteStringas well as to their respectiveBuildertypes and toString/ShowS.
- Extensive set of lensy helpers for easier manipulation of the nested data
types (see Text.URI.Lens).
- Quasi-quoters for compile-time construction of the URIdata type and refined text types (seeText.URI.QQ).
Quick start
The modern-uri package serves three main purposes:
- Construction of the URIdata type.
- Inspection and manipulation of the URIdata type (in the sense of changing its parts).
- Rendering of URIs.
Let’s walk through every operation quickly.
Construction of URIs
There are four ways to create a URI value. First off, one could assemble
it manually like so:
λ> :set -XOverloadedStrings
λ> import qualified Text.URI as URI
λ> scheme <- URI.mkScheme "https"
λ> scheme
"https"
λ> host <- URI.mkHost "markkarpov.com"
λ> host
"markkarpov.com"
λ> let uri = URI.URI (Just scheme) (Right (URI.Authority Nothing host Nothing)) Nothing [] Nothing
λ> uri
URI
  { uriScheme = Just "https",
    uriAuthority =
      Right
        ( Authority
            { authUserInfo = Nothing,
              authHost = "markkarpov.com",
              authPort = Nothing
            }
        ),
    uriPath = Nothing,
    uriQuery = [],
    uriFragment = Nothing
  }
In this library we use quite a few refined text values. They only can be
constructed by using smart constructors like mkScheme :: MonadThrow m => Text -> m (RText 'Scheme). For example, if argument to mkScheme is not a
valid scheme, an exception will be thrown. Note that monads such as Maybe
are also instances of the MonadThrow type class, and so the smart
constructors can be used in pure environment as well.
There is a smart constructor that can make an entire URI too, it’s called
(unsurprisingly) mkURI:
λ> uri <- URI.mkURI "https://markkarpov.com"
λ> uri
URI
  { uriScheme = Just "https",
    uriAuthority =
      Right
        ( Authority
            { authUserInfo = Nothing,
              authHost = "markkarpov.com",
              authPort = Nothing
            }
        ),
    uriPath = Nothing,
    uriQuery = [],
    uriFragment = Nothing
  }
If the argument of mkURI is not a valid URI, then an exception will be
thrown. The exception will contain full context and the actual parse error.
If some refined text value or URI is known statically at compile time, we
can use Template Haskell, namely the “quasi quotes” feature. To do so import
the Text.URI.QQ module and enable the QuasiQuotes language extension,
like so:
λ> :set -XQuasiQuotes
λ> import qualified Text.URI.QQ as QQ
λ> let uri = [QQ.uri|https://markkarpov.com|]
λ> uri
URI
  { uriScheme = Just "https",
    uriAuthority =
      Right
        ( Authority
            { authUserInfo = Nothing,
              authHost = "markkarpov.com",
              authPort = Nothing
            }
        ),
    uriPath = Nothing,
    uriQuery = [],
    uriFragment = Nothing
  }
Note how the value returned by the url quasi quote is pure, its
construction cannot fail because when there is an invalid URI inside the
quote it’s a compilation error. The Text.URI.QQ module has quasi-quoters
for scheme, host, and other components.
Finally, the package provides two Megaparsec parsers: parser and
parserBs. The first works on strict Text, while the other one works on
strict ByteStrings. You can use the parsers in a bigger Megaparsec parser
to parse URIs.
Inspection and manipulation
Although one could use record syntax directly, possibly with language
extensions like RecordWildcards, the best way to inspect and edit parts of
URI is with lenses. The lenses can be found in the Text.URI.Lens module.
If you have never used the
lens library, you could
probably start by reading/watching materials suggested in the library
description on Hackage.
Here are some examples, just to show off what you can do:
λ> import Text.URI.Lens
λ> uri <- URI.mkURI "https://example.com/some/path?foo=bar&baz=quux&foo=foo"
λ> uri ^. uriScheme
Just "https"
λ> uri ^? uriAuthority . _Right . authHost
Just "example.com"
λ> uri ^. isPathAbsolute
True
λ> uri ^. uriPath
["some","path"]
λ> k <- URI.mkQueryKey "foo"
λ> uri ^.. uriQuery . queryParam k
["bar","foo"]
-- etc.
Rendering
Rendering turns a URI into a sequence of bytes or characters. Currently
the following options are available:
- renderfor rendering to strict- Text.
- render'for rendering to text- Builder. It’s possible to turn that into lazy- Textby using the- toLazyTextfunction from- Data.Text.Lazy.Builder.
- renderBsfor rendering to strict- ByteString.
- renderBs'for rendering to byte string- Builder. Similarly it’s possible to get a lazy- ByteStringfrom that by using the- toLazyByteStringfunction from- Data.ByteString.Builder.
- renderStrcan be used to render to- String. Sometimes it’s handy. The render uses difference lists internally so it’s not that slow, but in general I’d advise avoiding- Strings.
- renderStr'returns- ShowS, which is just a synonym for- String -> String—a function that prepends the result of rendering to a given- String. This is useful when the- URIyou want to render is a part of a bigger output, just like with the builders mentioned above.
Examples:
λ> uri <- mkURI "https://markkarpov.com/posts.html"
λ> render uri
"https://markkarpov.com/posts.html"
λ> renderBs uri
"https://markkarpov.com/posts.html"
λ> renderStr uri
"https://markkarpov.com/posts.html"
-- etc.
Contribution
Issues, bugs, and questions may be reported in the GitHub issue tracker for this project.
Pull requests are also welcome.
License
Copyright © 2017–present Mark Karpov
Distributed under BSD 3 clause license.
Changes
0.3.6.1
- Host now can contain unreserved characters rather than simply alpha numeric ones, which was against RFC 3986. Issue 73.
0.3.6.0
- Now colons are not escaped in paths, unless the URIin question is a URI-reference, in which case colons in the first path segment are escaped. See RFC 3986, section 3.3. Issue 55.
Modern URI 0.3.5.0
- Added Hashableinstances forURI,Authority,UserInfo,QueryParam,RText.
Modern URI 0.3.4.4
- The mailtoscheme does not escape@in its paths (fixes the regression introduced in 0.3.4.3).
Modern URI 0.3.4.3
- 
Percent encode delimiter characters and @that appear in a path component. PR 47.
- 
Sub-domains that look like IPv4 can now be parsed. Issue 46. 
Modern URI 0.3.4.2
- Improved handling of percent-encoded sequences of bytes that cannot be decoded as UTF-8 text. Now friendly error messages are reported in these cases.
Modern URI 0.3.4.1
- Works with GHC 9.0.1.
Modern URI 0.3.4.0
- URIs with authority component and without path are now rendered without trailing slashes.
Modern URI 0.3.3.1
- Works with bytestring-0.11.
Modern URI 0.3.3.0
- Added mkURIBsfor parsingByteStringas aURI.
Modern URI 0.3.2.0
- 
Quasi-quoters from Text.URI.QQnow can be used in pattern context when theViewPatternsextension is enabled.
- 
Dropped support for GHC 8.2.x. 
Modern URI 0.3.1.0
- 
Dropped support for GHC 8.0 and 7.10. 
- 
Added Template Haskell Liftinstance for theURItype and its sub-components.
Modern URI 0.3.0.1
- Allow superfluous &right after question sign in query parameters.
Modern URI 0.3.0.0
- Uses Megaparsec 7. Visible API changes amount to an adjustment in
definition of the ParseExceptiontype.
Modern URI 0.2.2.0
- 
Removed a potentially overlapping instance Arbitrary (NonEmpty (RText 'PathPiece)).
- 
Fixed a bug that made it impossible to have empty host names. This allows us to parse URIs like file:///etc/hosts.
Modern URI 0.2.1.0
- Added emptyURI—URIvalue representing the empty URI.
Modern URI 0.2.0.0
- 
Changed the type of uriPathfield of theURIrecord from[RText 'PathPiece]toMaybe (Bool, NonEmpty (RText 'PathPiece)). This allows us to store whether there is a trailing slash in the path or not. See the updated documentation for more information.
- 
Added the relativeTofunction.
- 
Added the uriTrailingSlash0-1 traversal inText.URI.Lens.
Modern URI 0.1.2.1
- Allow Megaparsec 6.4.0.
Modern URI 0.1.2.0
- Fixed handling of +in query strings. Now+is parsed as space and serialized as%2bas per RFC 1866 (paragraph 8.2.1). White space in query parameters is serialized as+.
Modern URI 0.1.1.1
- Fixed implementation of Text.URI.Lens.queryParamtraversal.
Modern URI 0.1.1.0
- 
Derived NFDataforParseException.
- 
Adjusted percent-encoding in renders so it’s only used when absolutely necessary. Previously we percent-escaped a bit too much, which, strictly speaking, did not make the renders incorrect, but that didn’t look nice either. 
Modern URI 0.1.0.1
- Updated the readme to include “Quick start” instructions and some examples.
Modern URI 0.1.0.0
- 
Changed the type of uriAuthorityfromMaybe AuthoritytoEither Bool Authority. This allows to know if URI path is absolute or not without duplication of information, i.e. when theAuthoritycomponent is present the path is necessarily absolute, otherwise theBoolvalue tells if it’s absolute (True) or relative (False).
- 
Added isPathAbsoluteinText.URIand the corresponding getter inText.URI.Lens.
Modern URI 0.0.2.0
- 
Added the renderStrandrenderStr'functions for efficient rendering toStringandShowS.
- 
Added the parserBsthat can consume strictByteStringstreams.
Modern URI 0.0.1.0
- Initial release.
