-
-
Notifications
You must be signed in to change notification settings - Fork 533
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
Object is not transformed into the URL variable correctly #917
Comments
@altarrok There is a fix for that in the current development branch (complex query params support). This should fix your issue. Once i have a few more things fixed i will push a new version to NPM |
Thanks a lot @ferdikoomen !! Let me know if we can contribute in some way, thanks for keeping the library updated |
@altarrok New version is pushed to NPM 0.13.0 Could you check if that resolves your issue? |
@altarrok New version is pushed 0.15.0, that should include all options (nested props, arrays, etc.) |
@ferdikoomen unfortunately the fix doesn't help our case (from my original issue): The fix only works for lists, not objects, objects are still translated as |
@altarrok do you by change have a link to the spec or a copy of the spec that i can check? Im mostly interested in the definition of that parameter. That would make it easier to reproduce the scenario |
@ferdikoomen this is the openapi spec for pageable we are using:
I believe in request.ts::getQueryString line#45, the value is being checked to be an array, but there is no other check that if the value is a string or not, but it is being casted to a string downstream. My current best solution would be to check if the type of the value is a string, if not, check if it is an array, if not accept it as an object and call recursively using the objects key value pairs. That way, an object containing other objects, strings, or arrays can be flattened to a query string. Let me know what you think |
@altarrok I assume you then have something like this in the operation parameters right?:
I'm testing this now with a client, will come back with results in a bit |
@altarrok When im using the latest version (0.18.1), then i see the following behaviour When im sending the following parameter:
Then this is converted to:
If we decode this, then it looks like:
Our backend (NodeJS Express) just parses this and parses this as:
That seems exactly right... Are you using the latest version of the generator? |
@ferdikoomen we were using npx. Specifying the version now, we got the same behavior as you do. However this behavior: If you disagree with this, I respect your opinion, and please let me know so that I can fork and fix the issue. Thanks! |
@altarrok Do you have the full spec for me (if you cannot send it via github, then please send me an email: info@madebyferdi.com). Then i'll have a look to see why this generated differently. |
@ferdikoomen The generation is identical, however I think that nested query objects should be flattened to a key:value pair list, instead of having the nested structure stringified into parentkey[childkey]:value list. The swagger behavior is matching what I believe the behavior should be. I can not share the full spec as it would breach company policies, sorry :( |
Can you share the operation spec itself? For the tests i am using the following: Is there a big difference between this and what you are using?
|
No it is practically same:
|
@altarrok That is weird, since the specs basically describes your server is expecting a property in the query called |
@ferdikoomen Yeah, I will do some testing around it, let's see what is the actual behavior of swagger. May I ask how is the current behavior useful? If you could give some cases where this is used. |
@ferdikoomen Immediately I tested an object as a request parameter, and the swagger generated using the same behavior, and the request is a bad request (as the response confirms). So this tells me that Swagger and Spring are using the same behavior, where if there is a request parameter specified, it should be a simple type of a variable like a string or a UUID. If the controller input parameter is not specified as a request parameter (in Spring), then it accepts the fields of that variable type as request parameters. For example:
Then the request parameters would be the fields of CompositeKey: fooId and barId If controller method parameter DOES specify it as a request parameter such as:
Then the request parameter is myKey The weird thing is that, swagger always uses the first behavior no matter if it is specified or not. That tells me that having objects in the request parameter is not good since it is not supported by swagger. |
Yeah there is indeed some logic in the Java based generator that im trying to figure out:
It seems that in some cases (not sure when) they 'explode' the query param and indeed generate a url that looks like
In general a beter solution would indeed be to specify separate query params for each of the props inside this object. |
@ferdikoomen Also, Tomcat throws an error when URL has "[" or "]":
|
Is that Java based generator publicly available? Maybe I can also take a look and help |
You can download and run it like this:
|
So I have been looking into different libraries and examples, it seems like this is really a grey area, so I believe it is your decision as how this library should behave in case we have objects passed as request parameters That being said, there is a library that has a configuration available for stringifying the request parameters, if no config provided fallback is to cast the object to a String. |
Ill need to have a look here: https://swagger.io/docs/specification/describing-parameters/#query-parameters And just pick a direction 😃 anyway it seems some work is needed here. Will keep the ticket open. |
Hey @ferdikoomen, just wanted to see if there is an update on this issue? |
I wanted to share a doc I found: https://spec.openapis.org/oas/v3.1.0 Note that the default "style" is "form" when parameter is in "query":
And the default "explode" is "true" when "style" is "form":
And when "explode" is "true", the parameters are seperated, look at the table in: |
Also, we found an easy solution on spring-doc side, so if anyone is down in this rabbit hole (I doubt), hit me up and I can explain the solution |
@altarrok Sure, send me a message (info@madebyferdi.com) and let's chat. Would be nice to implement this |
I have the same problem and I fix it by referencing the param as to a schema: components:
schemas:
Sort:
type: array
items:
type: string
parameters:
sort:
name: sort
in: query
description: 'Sorted request'
example: ['pessoa.nome:asc']
schema:
# This fixes the parameter not being an array
$ref: '#/components/schemas/Sort' |
@altarrok I have the exact same problem. What did you change on the spring-doc site? |
@Arnagos I believe we used
for the endpoints on controllers more info: @PageableAsQueryParam |
Hi @ferdikoomen. Do you have any updates on this behaviour? I'm also having this issue and I disagree that this a grey area. OpenAPI allows you to specify object parameters that should explicitly be exploded in this way. See "Query Parameters" here: https://swagger.io/docs/specification/serialization/. What I'm trying to achieve here is an endpoint that allows for dynamic query parameters that cannot be neatly documented, so I'm generating the API with object query parameters with style=form and explode=true in order to indicate that the endpoint accepts arbitrary query parameters. Is there currently any way to get this behaviour with this library? |
We are using SpringDoc to generate the openapi specs of our backend, and we are using a feature of SpringDoc where commonly used Pageable object parameter is mapped automatically to openapi specs. The result of the mapping looks like this:
As you can see the Pageable parameter is defined as object. The generated request URL looks like this:
http://localhost:8080/hackathon/search?page=0&size=1&sort=location
So we can see that swagger-ui generates the correct request URL for the given Pageable object.
However -problem begins here-, when the generated client using openapi-typescript-codegen is used to make a backend call, the request URL looks like this:
http://localhost:8080/hackathon/search?pageable=%5Bobject%20Object%5D
Tracking the code, I can see that in request.ts::getQueryString arrays are accounted for but the objects aren't, and therefore the Pageable object is casted to a string, resulting in the given bug.
Is there a possible fix on our side (although I doubt that)?
A possible fix on the library side (I haven't tested) may be to call the getQueryString method recursively for each entry in the object, and concatenating the result (qs).
The text was updated successfully, but these errors were encountered: