-
Notifications
You must be signed in to change notification settings - Fork 48
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
rebased aiopenapi3 #69
base: master
Are you sure you want to change the base?
Conversation
describe the base types using dataclasses, use the data class annotations for parsing all unit tests pass
- using Optional - default value - have a dataclass constructor
- the classmethod property ObjectBase.required_fields requires python 3.9
- local data loading for split specifications (open5gs) edit - not including all the api specs …
- required as __root__ can not have additional fields (extensions) in Reference
- test_model supporting basic discrimination
Reference uses __{g,s}etattr__ to forward to target
for discriminated objects create all types it is required to iterate the paths and pre-cache the types for the body & return values parameter types & v20 discriminated types are left
taking care of discriminated Schemas.
- arbitrary_types_allowed = False
Who will review 7kloc across 100 files with a bunch of unrelated changes... |
Hi, actually it ain't that bad, its a bunch of files with pydantic models for OpenAPI 2/3./3.1 description documents, some logic for request/response processing, the dynamic pydantic class generation being the worst of it. But I agree, I doubt this will be reviewed/merged - I already forked it https://github.com/commonism/aiopenapi3 . You are involved with OpenAPI client development as well - https://github.com/python-lapidary/lapidary and lapidary even shares some common requirements with aiopenapi3 - pydantic/httpx, basic difference is lapidary generates pydantic classes - aiopenapi3 is dynamic. From my experience using pydantic will give you a hard time with additionalProperties (defaulting to True even), as it will allow the data to overwrite the Models methods/properties/attributes, requiring a default of Extra.ignore. Good luck with lapidary. |
I think it's a good idea, your project outgrew this one.
I was actually thinking of making it dynamic, but I wanted generated model and at least method stubs for operations, for code completion. The problem is, that the schema that I want it to work with is 1.2MB and it takes too long to load.
I think the problem is that many programmers interprets openapi/jsonschema as a recipe for types in their languages (that's my case in the beginning, but I see it in other projects). It' not, it's a validation description, and incidentally, so is pydantic BaseModel (that's at least the intention). That's why I think is a good match, tho not 1-to-1. I think with generated validators it should take me a long a way. I think not supporting additionalProperties=True is a good decision, as it's impossible to decode such data to a concrete class.
I found anyOf trivial to implement - simply a typing.Union. Works both ways - encoding and decoding.
Unique param names are trivial too, I generate names in Hungarian notation - adding (p/q/c/h)_ prefix, and putting the original name in alias. I might change it to suffix to make it work better with code completion (so that developer starts typing the param name and not the prefix). Feel free to check out the client generated with Lapidary https://github.com/oeklo/gsmtasks-client (the errata got bigger since the last push) Thank you for your thoughts. |
And don't forget read- and writeOnly properties :) |
https://api.gsmtasks.com/docs/schema/ - 4s loading with aiopenapi3, pickle it for re-use.
A different use of additionalProperties - dynamic map. type: object
additionalProperties:
type: string
which is rather helpful.
One could hide the unknowns in
I've had my time with SOAP before, the aiopenapi3 plugin interface is inspired by suds.
The keywords used to combine schemas are:
So far I optimized for the case where parameter names do not collide.
Talking about additionalProperties: https://github.com/oeklo/gsmtasks-client/blob/main/gen/gsmtasks/components/schemas/account.py#L24: import pydantic
class AccountTaskExpiryDurationFromCompleteAfter(pydantic.BaseModel):
class Config(pydantic.BaseConfig):
extra = pydantic.Extra.allow
b = AccountTaskExpiryDurationFromCompleteAfter(dict='yes')
b.dict()
#Traceback (most recent call last):
# File "<stdin>", line 1, in <module>
#TypeError: 'str' object is not callable Authentication - I don't see authentication besides ApiToken to work properly. Aside from having different authentication types, authentication is modular, and you can have multiple SecuritySchemes defined per PathItem. api.authenticate(tokenAuth="BearToken", userAuth=("name","password"), adminAuth=("admin","betterpassword")) , and lookup which scheme to use per request/on the PathItem. Talking about IDE integration and completion, I start to see value in code generation, even if it was just stub files for a dynamic client. Stubs are a good idea. Nice meeting you. |
I find this discussion very useful, so I'm moving it to discussions in Lapidary - python-lapidary/lapidary#19
https://github.com/python-lapidary/lapidary/discussions/20
Or you could do
Funny you should say that, I'm thinking of using aiopenapi3 as the runtime library :)
Likewise |
Rebased aiopenapi3 for merging.
Can't re-open #62