Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Use full header type in response header instances #1697

Merged

Conversation

worm2fed
Copy link
Contributor

@worm2fed worm2fed commented Jul 19, 2023

This is required to be able provide Description into headers, for example like this:

type PaginationTotalCountHeader =
  Header'
    '[ Description "Indicates to the client total count of items in collection"
     , Optional
     , Strict
     ]
    "Total-Count"
    Int

This PR is related to another one: biocad/servant-openapi3#48 and related to issue biocad/servant-openapi3#45

@worm2fed worm2fed marked this pull request as ready for review July 19, 2023 12:57
@tchoutri
Copy link
Contributor

@worm2fed thanks a lot for this PR. Does this unlock nice things beyond the openapi feature? If so we should probably have a changelog entry for it. :)

@worm2fed
Copy link
Contributor Author

worm2fed commented Jul 19, 2023

@tchoutri thank you)
I hope yes, it allows to bring more information into generated openapi3 spec.

As I can see there a few things have to be done:

  • add entry to changelog
  • update documentation to reflect that this is possible to provide descriptions to Headers via modifiers
  • review more instances where limited; I found that it's not possible to use functions with AddHeader, for example
    addHeader :: [AddHeader](https://hackage.haskell.org/package/servant-0.20/docs/Servant-API.html#t:AddHeader) h v orig new => v -> orig -> new

Maybe it will be better to define helper type

-- | Helper type to define 'Header' with 'Description'.
type HeaderWithDescription description = 
 Header' '[Description description, Optional, Strict]

@worm2fed worm2fed force-pushed the worm2fed/use-full-header-type branch 2 times, most recently from 0d237e5 to 7dfae41 Compare July 19, 2023 22:08
@worm2fed
Copy link
Contributor Author

Seems like problem with order of mods in AddHeader. When I add it to the end I got another error

>     • Expected one of:
>           '[WithStatus 200 (Headers '[Header "H1" Int] String),
>             WithStatus 404 String]
>       But got:
>           WithStatus
>             200 (Headers '[Servant.API.Header.Header' mods0 "H1" Int] [Char])
>     • In the first argument of ‘(.)’, namely ‘respond’
>       In the first argument of ‘($)’, namely
>         ‘respond . WithStatus @200 . addHeader @"H1" (5 :: Int)’
>       In the expression:
>         respond . WithStatus @200 . addHeader @"H1" (5 :: Int)
>           $ ("foo" :: String)
>     |
> 749 | uverbResponseHeadersServer True = respond . WithStatus @200 . addHeader @"H1" (5 :: Int) $ ("foo" :: String)

Actually, I think it's correct to do that - add mods to AddHeader. Before it was type alias with mods as well, but with specific ones.
The reason I've added mods to AddHeader is because of strange error in these two instances:

-- In this instance, we add a Header on top of something that is already decorated with some headers
instance {-# OVERLAPPING #-} ( KnownSymbol h, ToHttpApiData v )
         => AddHeader h v (Headers (fst ': rest) a) (Headers (Header' mods h v  ': fst ': rest) a) where
    addOptionalHeader hdr (Headers resp heads) = Headers resp (HCons hdr heads)

-- In this instance, 'a' parameter is decorated with a Header.
instance {-# OVERLAPPABLE #-} ( KnownSymbol h, ToHttpApiData v , new ~ Headers '[Header' mods h v] a)
         => AddHeader h v a new where
    addOptionalHeader hdr resp = Headers resp (HCons hdr HNil)

Here is this error:

>     • Illegal instance declaration for
>         ‘AddHeader
>            h
>            v
>            (Headers (fst : rest) a)
>            (Headers (Header' mods h v : fst : rest) a)’
>         The liberal coverage condition fails in class ‘AddHeader’
>           for functional dependency: ‘h v orig -> new’
>         Reason: lhs types ‘h’, ‘v’, ‘Headers (fst : rest) a’
>           do not jointly determine rhs type ‘Headers
>                                                (Header' mods h v : fst : rest) a’
>         Un-determined variable: mods
>     • In the instance declaration for
>         ‘AddHeader h v (Headers (fst : rest) a) (Headers (Header' mods h v
>                                                           : fst : rest) a)’
>     |
> 166 | instance {-# OVERLAPPING #-} ( KnownSymbol h, ToHttpApiData v )
>     |                              ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^...
> 
>     • Illegal instance declaration for ‘AddHeader h v a new’
>         The liberal coverage condition fails in class ‘AddHeader’
>           for functional dependency: ‘h v orig -> new’
>         Reason: lhs types ‘h’, ‘v’, ‘a’
>           do not jointly determine rhs type ‘new’
>         Un-determined variable: new
>     • In the instance declaration for ‘AddHeader h v a new’
>     |
> 171 | instance {-# OVERLAPPABLE #-} ( KnownSymbol h, ToHttpApiData v , new ~ Headers '[Header' mods h v] a)
>     |        

Honestly, I do not understand what is it

@worm2fed
Copy link
Contributor Author

worm2fed commented Jul 20, 2023

@tchoutri maybe you can explain what is going on?

My current understanding is that

-- We need all these fundeps to save type inference
class AddHeader h v orig new (mods :: [*])
    | h v orig mods -> new, new -> h, new -> v, new -> orig, new -> mods where
  addOptionalHeader :: ResponseHeader h v -> orig -> new  -- ^ N.B.: The same header can't be added multiple times

in this class all these dependencies required to validate that there is only one exact header. And by using mods we break this rule, cause Header' '[Optional, Strict] "MyHeader" Int is not same as Header' '[Description "Just header", Optional, Strict] "MyHeader" Int

And in sense of logic it is same because description is just "documentation label", but current implementation of providing description to headers via modifiers limits to implement this.
Any way, it can raise another question, how would we have two "same" headers but with different description.

So, to perform such validation we need to add mods to AddHeader.
Add this leads to error in tests.

HLS gives me a bit more information:

• Expected one of:
      '[WithStatus 200 (Headers '[Header "H1" Int] String),
        WithStatus 404 String]
  But got:
      WithStatus
        200 (Headers '[Servant.API.Header.Header' mods0 "H1" Int] [Char])
• In the first argument of ‘(.)’, namely ‘respond’
  In the first argument of ‘($)’, namely
    ‘respond . WithStatus @200 . addHeader @"H1" (5 :: Int)’
  In the expression:
    respond . WithStatus @200 . addHeader @"H1" (5 :: Int)
      $ ("foo" :: String)typecheck(-Wdeferred-type-errors)
• Overlapping instances for Servant.API.UVerb.Union.UElem
                              (WithStatus
                                 200 (Headers '[Servant.API.Header.Header' mods0 "H1" Int] [Char]))
                              '[WithStatus 200 (Headers '[Header "H1" Int] String),
                                WithStatus 404 String]
    arising from a use of ‘respond’
  Matching instances:
    instance [overlapping] forall a (x :: a) (xs :: [a]) (x' :: a).
                           Servant.API.UVerb.Union.UElem x xs =>
                           Servant.API.UVerb.Union.UElem x (x' : xs)
      -- Defined in ‘Servant.API.UVerb.Union’
    instance [overlapping] forall a (x :: a) (xs :: [a]).
                           Servant.API.UVerb.Union.UElem x (x : xs)
      -- Defined in ‘Servant.API.UVerb.Union’
  (The choice depends on the instantiation of ‘mods0’
   To pick the first instance above, use IncoherentInstances
   when compiling the other instance declarations)
• In the first argument of ‘(.)’, namely ‘respond’
  In the first argument of ‘($)’, namely
    ‘respond . WithStatus @200 . addHeader @"H1" (5 :: Int)’
  In the expression:
    respond . WithStatus @200 . addHeader @"H1" (5 :: Int)
      $ ("foo" :: String)typecheck(-Wdeferred-type-errors)

@worm2fed
Copy link
Contributor Author

Oh, I see in test
addHeader @"H1" (5 :: Int) $ ("foo" :: String) must become addHeader @"H1" @_ @_ @_ @('[Optional, Strict]) (5 :: Int) $ ("foo" :: String)

@blackheaven
Copy link
Contributor

I think you may consider mimicking Summary:

Describe "Indicates to the client total count of items in collection"
  (Header "Total-Count" Int)

@worm2fed
Copy link
Contributor Author

@blackheaven I do no see how it will work with servant-openapi3 this way, only by providing an instance there to extract that description.

I will take a look

@worm2fed
Copy link
Contributor Author

so, smith like this:

data Describe (sym :: Symbol) (a :: *)

instance (KnownSymbol sym, ToParamSchema a, KnownSymbol description) => ToResponseHeader (Describe description (Header sym a)) where
  toResponseHeader _ = (hname, header)
    where
      hname = Text.pack (symbolVal (Proxy :: Proxy sym))
      header = mempty
        & description ?~ Text.pack (symbolVal (Proxy :: Proxy description))
        & schema ?~ Inline (toParamSchema (Proxy :: Proxy a))

which can be used not only for response headers, but for any

@blackheaven is it what you are suggesting?

@blackheaven
Copy link
Contributor

Exactly, thanks.

@worm2fed
Copy link
Contributor Author

worm2fed commented Jul 21, 2023

@blackheaven I think it will be good to include Describe into servant package but it will be slower.
Or, we can keep all in servant-openapi3, wdyt?

@blackheaven
Copy link
Contributor

I would say, in a perfect world, it should be in servant-openapi3, but

  1. There is already Summary
  2. It not a big patch
  3. It contribute to a better API readability

@worm2fed worm2fed force-pushed the worm2fed/use-full-header-type branch from 939b8ef to a32345e Compare July 21, 2023 23:48
@worm2fed
Copy link
Contributor Author

worm2fed commented Jul 21, 2023

@blackheaven it is a little more difficult than it might seem at first

I've added test and it's failing, addHeader cannot add Header wrapped in Describe.
So it's required to add AddHeader instance / change current. I left a commented version there, but it's not working.
Do you know how to solve this problem without introducing new param to AddHeader?

servant-server/test/Servant/ServerSpec.hs:137:24: error:
    • Couldn't match type: Describe "desc" (Header "D" Int)
                     with: Servant.API.Header.Header'
                             '[Servant.API.Modifiers.Optional, Strict] h0 v0
        arising from a use of ‘addHeader’
    • In the first argument of ‘return’, namely ‘(addHeader 5 alice)’
      In the first argument of ‘(:<|>)’, namely
        ‘return (addHeader 5 alice)’
      In the second argument of ‘(:<|>)’, namely
        ‘return (addHeader 5 alice)
           :<|>             
             (return alice :<|> return "B")
               :<|> return (S.source ["bytestring"])’
    |                       
137 |           :<|> return (addHeader 5 alice)

This showed, that it will be impossible to add this just to servant-opeanapi3, because servant changes required as well

@blackheaven
Copy link
Contributor

I tried also, I didn't find a solution, it doesn't seems a good idea in the end, sorry for that :/

@worm2fed
Copy link
Contributor Author

I will leave diff here, maybe someone else will want to try
but I will revert this PR to initial version

haskell-servant_servant_pull_1697.diff.txt

Use `Header'` instead of `Header` in response, so it's possible to provide
`Description`, for example:

```
type PaginationTotalCountHeader =
  Header'
    '[ Description "Indicates to the client total count of items in collection"
     , Optional
     , Strict
     ]
    "Total-Count"
    Int
```

Note: if you want to add header with description you should use `addHeader'`
or `noHeader'` which accepts `Header'` with all modifiers.
@worm2fed worm2fed force-pushed the worm2fed/use-full-header-type branch from a32345e to 89680f8 Compare July 22, 2023 19:37
@worm2fed
Copy link
Contributor Author

worm2fed commented Jul 25, 2023

@tchoutri should I add servant-auth targets to stack.yaml? currently they are only in .cabal

@tchoutri
Copy link
Contributor

@worm2fed good question, I must admit I haven't used stack in some years now. If you add them and it works, I'll approve it. :)

@worm2fed
Copy link
Contributor Author

mm.. seems like MonadPlus and guard not in base for ghc 9.6.2

worm2fed added 2 commits July 26, 2023 10:17
It's in `cabal.project`, but not in `stack.yaml`.
Actually, there is more packages in `cabal.project` - `doc/cookbook/*`,
but stack is not happy with their cabal files and asks to rename, so
I do not add them.
@ysangkok
Copy link
Contributor

They are in base, you just don't get them automatically through mtl. See changelog

@worm2fed
Copy link
Contributor Author

@ysangkok thanks for explanation

@worm2fed
Copy link
Contributor Author

worm2fed commented Aug 3, 2023

@tchoutri can you please review this PR?

Copy link
Contributor

@tchoutri tchoutri left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

LGTM, thanks a lot!

@tchoutri tchoutri merged commit 72f5d5c into haskell-servant:master Aug 4, 2023
@worm2fed
Copy link
Contributor Author

worm2fed commented Aug 4, 2023

@tchoutri I think we had to squash fixups before merge..

Any way, thanks)

@tchoutri
Copy link
Contributor

tchoutri commented Aug 4, 2023

@worm2fed I'm pretty sure I did! Please do check though :) 72f5d5c

@worm2fed
Copy link
Contributor Author

worm2fed commented Aug 4, 2023

@tchoutri oh, I see, sorry
Was it merge-with squash?

@tchoutri
Copy link
Contributor

tchoutri commented Aug 4, 2023

Yep

ohri-anurag pushed a commit to bellroy/servant that referenced this pull request Aug 21, 2023
* Use `Header'` in response headers

Use `Header'` instead of `Header` in response, so it's possible to provide
`Description`, for example:

```
type PaginationTotalCountHeader =
  Header'
    '[ Description "Indicates to the client total count of items in collection"
     , Optional
     , Strict
     ]
    "Total-Count"
    Int
```

Note: if you want to add header with description you should use `addHeader'`
or `noHeader'` which accepts `Header'` with all modifiers.
@maksbotan maksbotan mentioned this pull request Oct 1, 2023
@maksbotan
Copy link
Contributor

Hi @worm2fed!

As far as I understand, your change to AddHeader (adding mods type argument) is not backwards-compatible and therefore would require a new servant version, e.g. 0.21.

Can we instead add AddHeader' with new argument and make AddHeader an alias to it? So that we can release this in 0.20 series without breaking public API.

@tchoutri what do you think?

@tchoutri
Copy link
Contributor

tchoutri commented Oct 1, 2023

A compat alias could be nice yes!

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

5 participants