-
-
Notifications
You must be signed in to change notification settings - Fork 17
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
Forward references are not always correctly identified. #23
Comments
Ran into the same issue while generating the schema types. I think this should be easily fixed by refactoring the "self_referential" (which are actually forward references ) checks to be happening in the annotation recursion and keeping track of reference of not yet defined types so that they will have forward references. Also I need to refactor and document that part a lot better :D This btw would be something that would be nice to catch while testing. I was thinking of having tests that actually |
I've had a bit of a play with the testing stuff. I can validate that the generated code is valid by writing to a temporary file and running something like subprocess.run([sys.executable, file_name]) This still doesn't catch the issue I described because it doesn't try to build each model. For that I was thinking that we could write a test/series of tests that write the generated files to disk, and for each BaseModel in the file try and build them. (Maybe using hypothesis to automate this) Do you think we should include the .py files generated from each schema in git? |
@strue36 I would love hypothesis like testing. It is something that i have been meaning to test out for quite some time now. If you have some tests I would love to add them. Also great that subprocess is working. :) |
On the topic of forward references:
This however would break another idea i had of using a process pool executor for running the plugins in parallel. Parellelization could then only work between projects. I have not stress tested turms with massive schemas, so not really sure if this would be necessary... Whats your take on that? |
Accidentally closed this |
There was a similar issue in the ObjectsPlugin. turms/turms/plugins/objects.py Lines 247 to 260 in 87f1a23
EDITSame goes for the InputsPlugin. Lines 143 to 152 in 87f1a23
|
@j-riebe Agreed there was a lot of duplicated code there, I discovered however that i might a quite dramatic conceptual error here, the forward reference update needs to be happening on the parent not on the invoked child, i am currently refactoring this part of the mess :D will keep you posted! |
Fortunately, this error didn't hit me yet 😅 |
Its this kind of error where you wonder how things ever worked... 😄 Works well on linux and windows now, but funnily macos gives me the weirdest error right now, that Also there is one breaking change given a test you wrote: type Foo {
forward: Ref
}
interface Ref {
id: ID
} Will yield non working code. The big querstio: is that actually something that we should consider? As a interface is an abstract type anyways and needs to resolve to a type (correct me if i am wrong). So i am wondering if we should return an error when such a thing happens or actually generate the code? in the scenario that: type Foo {
forward: Ref
}
interface Ref {
id: ID
}
type X implements Ref {
x: ID
} Everything works as expected... |
In terms of GraphQL-SDL it should be perfectly fine, to use interfaces without implementations. What exactly won't be working in the test case? from pydantic import BaseModel
from typing import Optional
class RefBase(BaseModel):
id: Optional[str]
class Foo(BaseModel):
forward: Optional['Ref']
RefBase.update_forward_refs() Another question: What is the Union of the implementations about? Why don't we get rid of the Unions? |
If 'stand alone' interfaces are generally useful the non-union approach should definitely be possible. It makes me wonder however what actually is are the purposes of generating a pydantic schema generation. What is your use case? My usecase for code generation is so focused on fragments and operations on the client side, that I am a bit oblivious to what schema generation would be used for. |
Well, my use-case is server-side static typing. I'm trying to use the pydantic schema together with another code-generation approach for creating ariadne bindables (python's schema first GQL library). The pydantic schema is extremly useful for static-typing. |
Ah ariadne :) This was actually something that I thought would make a great plugin and started working on it myself. I was also looking at generating sqlalchemy types from graphql (+ some schema directives). Would you like to join forces on that? Another use case mentioned in some private requests to me was using the pydantic types to validate their response on the server side and serialize to json then. Something that I think is probably an antipattern, as pydantic validation is unnecessarily long. I think it would be great to have a list of common usecases and antipatterns in the documentation! |
If you speak about sqlalchemy, have a look at SQLModel. Speaking about anti-patterns it might be unwise to generate your DB Models based on the Graph, as this couples your API Design with DB-concerns. |
Regarding the ariadne generator, I already thought about rebuilding it as a turms plugin. |
#26 should have solved the issue and provide both solutions ( "always_resolve_interfaces" would be a strict mode, where this would raise a Generation Error) |
move to #28 |
I came across an issue with a generated schema that was missing a call to update_forward_refs()
It has an InputType that references another InputType that is later in the same file. The InputsPlugin looks like it tries to check for this by marking any InputType that references another InputType as self_referential.
My schema contains a nested non nullable InputType. This is is of type GraphQLNonNull and we don't check the value.type.of_type which shows that this is actually wrapping a GraphQLInputObjectType.
The reference is correctly written as a string. I'm not 100% sure where the code that determines which type hints should be strings lives.
When looking into this, I also noticed that there is a (presumably) related issue in the test_arkitekt_operations test.
This generates the following class at line 305
But ArgPort is not defined until line 544
The text was updated successfully, but these errors were encountered: