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 Load/LoadAsync pattern to load OpenApi documents #1446

Closed
wants to merge 6 commits into from

Conversation

MaggieKimani1
Copy link
Contributor

@MaggieKimani1 MaggieKimani1 commented Oct 31, 2023

Fixes #1434

/// <param name="diagnostic"></param>
/// <param name="settings"></param>
/// <returns></returns>
public static OpenApiDocument Load(Stream stream, out OpenApiDiagnostic diagnostic, OpenApiReaderSettings settings = null)
Copy link
Member

Choose a reason for hiding this comment

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

These should be extension methods so that they appear on the OpenApiDocument class.

Copy link
Member

Choose a reason for hiding this comment

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

we don't have an instance of OpenAPIDocument yet, and as far as I know static extension methods are not a thing. Maybe the design needs to be reviewed here?

Copy link
Contributor Author

Choose a reason for hiding this comment

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

These should be extension methods so that they appear on the OpenApiDocument class.

So the format should ideally look like this:

var doc = new OpenApiDocument().Load(stream)

Copy link
Member

Choose a reason for hiding this comment

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

in which case we wouldn't be returning the document from the method? Works for me but I'm not sure the internals of OpenAPIDocument are accessible in an extension method outside of the assembly.

Choose a reason for hiding this comment

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

This would create an odd pattern of

var document = new OpenApiDocument();

document.Load();

Where document would now be null.

Perhaps marking OpenApiDocument as partial could do the trick as well (adding the static load methods in the reader lib)

Copy link
Contributor Author

Choose a reason for hiding this comment

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

in which case we wouldn't be returning the document from the method? Works for me but I'm not sure the internals of OpenAPIDocument are accessible in an extension method outside of the assembly.

We can return a document in order to avoid null references when we return a void as @VisualBean has indicated here.

Copy link
Member

Choose a reason for hiding this comment

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

This pattern is strange though. Why define those methods as extension methods if we're returning a new instance? How is the consumer supposed to know which instance to hold on to?

Copy link
Contributor Author

Choose a reason for hiding this comment

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

Yeah you're right, that's a fair point. This then brings us back to how do we ensure the value of the original object is modified using the extension methods?

Copy link

@VisualBean VisualBean Nov 1, 2023

Choose a reason for hiding this comment

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

Perhaps simply making static methods on the readers will provide an adequate solution without resorting to new classes people will have to use.

The main issue (compared to XmlDocument) is that reading is separate from the models. And extension methods on non-instances aren't a thing (yet).

The main problem Darrel seems to reference is that the readers aren't really 'readers', they simply parse the entirety of a string/stream/whatever and make a document (or fragment), so newing them seems wrong, as they don't keep state (other than options).

Copy link
Member

Choose a reason for hiding this comment

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

the easiest compromise for now is probably to rename the static class to OpenAPIDocumentFactory (or something like that), and revert back to "regular" static method (not extension methods).
But I'd like to get @darrelmiller confirmation since I don't have all the context here for that work.

Copy link

sonarqubecloud bot commented Nov 1, 2023

SonarCloud Quality Gate failed.    Quality Gate failed

Bug A 0 Bugs
Vulnerability A 0 Vulnerabilities
Security Hotspot A 0 Security Hotspots
Code Smell A 0 Code Smells

0.0% 0.0% Coverage
0.0% 0.0% Duplication

warning The version of Java (11.0.21) you have used to run this analysis is deprecated and we will stop accepting it soon. Please update to at least Java 17.
Read more here

idea Catch issues before they fail your Quality Gate with our IDE extension sonarlint SonarLint

@darrelmiller
Copy link
Member

Goals

  • We would like to move the loading code into OpenApi.Net core library so that people who only want to load JSON don't need a dependency on the third party YAML library
  • We would like to avoid developers needing to discover a secondary "factory/reader" class and simply be able to use a static member on the model to load/parse content

Constraints

  • We cannot create a circular dependency between the core library and reader library
  • We must support loading any OpenAPI model object directly, not just a complete document. (fragments)
  • We don't want to duplicate all of the loading/parsing code in every model object's static methods.

Proposal

In the core library we create a static factory for loading/parsing any OpenAPI model object.

T OpenApiModelFactory.LoadAsync(url)
T OpenApiModelFactory.LoadAsync(stream, format)
T OpenApiModelFactory.LoadAsync(textReader, format)
...
T OpenApiModelFactory.Parse(string)

We can add simple static helper methods to the OpenAPIDocument class to avoid the "discovery problem".
OpenApiDocument OpenApiDocument.LoadAsync(stream) -> Calls factory
OpenApiDocument OpenApiDocument.Parse(string) -> Calls factory

We need to be able to add support for other formats in the factory:

OpenApiModelFactory.RegisterReader(IOpenApiReader) <- Making this up. Look at how Kiota does it.

OpenApiJsonReader : IOpenApiReader <- Lives in the core library
OpenApiYamlReader : IOpenApiReader <- Lives in the reader library

OpenApiModelFactory.RegisterReader(new OpenApiJsonReader) <= Do this by default in a static ctor.

// User chooses to enable YAML parsing by adding reference to Microsoft.OpenApiReaders
OpenApiModelFactory.RegisterReader(new OpenApiYamlReader) ;

@baywet
Copy link
Member

baywet commented Nov 7, 2023

@darrelmiller

OpenApiJsonReader : IOpenApiReader <- Lives in the core library

Is this representative of things now or a desired change? Why would we play favorite child with a specific format?

OpenApiDocument OpenApiDocument.LoadAsync(stream) -> Calls factory

Assuming the factory lives in readers, and the LoadAsync method in core, this is not going to be possible due to circular references

User chooses to enable YAML parsing by adding reference to Microsoft.OpenApiReaders

Didn't we say we'd enable auto-registration through static constructors? (I've never done that in the past, I'm not sure this is possible)

@MaggieKimani1
Copy link
Contributor Author

Superseded by #1553

@baywet baywet deleted the mk/use-load-pattern branch November 12, 2024 11:51
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