- ๐ซ๐ทFrance andypost
The blocker is closed as outdated so IS needs update
- ๐ณ๐ฑNetherlands bbrala Netherlands
Think this is the issue, we can use the symfony property for that I think.
#3252872: Use CacheableSupportsMethodInterface for performance improvement in normalizers โ
- Merge request !5008Issue #3031367: Introduce "schematic" normalizers โ (Closed) created by bradjones1
- last update
over 1 year ago 30,396 pass, 1 fail - last update
over 1 year ago Custom Commands Failed - last update
over 1 year ago Custom Commands Failed - last update
over 1 year ago 30,434 pass - Status changed to Needs review
over 1 year ago 1:27am 16 October 2023 - ๐บ๐ธUnited States bradjones1 Digital Nomad Life
Changing the component to serialization because this is mostly not specific to JSON:API.
I am marking NR because I'm deep enough into this now that I would really like some maintainer/committer review to ensure this isn't totally off-base. I'm personally pretty proud of how this is implemented with the interface and the mostly-automatic integration of the base normalizer with the new Attribute.
- last update
over 1 year ago 30,434 pass - last update
over 1 year ago Custom Commands Failed - last update
over 1 year ago 30,438 pass - last update
over 1 year ago Custom Commands Failed - last update
over 1 year ago Build Successful - last update
over 1 year ago Composer error. Unable to continue. - last update
over 1 year ago Build Successful - ๐บ๐ธUnited States bradjones1 Digital Nomad Life
So, yeah, this now requires a change to
symfony/serializer
which is a bummer, and I'm not quite sure what the work-around would be. I've chosen to just include the patch for now, in hopes that as we work on this that either 1) Symfony will make a minor change to two methods from private to protected, or 2) someone smarter than I comes up with a different alternative.The issue is that we are adding a new method to normalizers to express information about their normalization, but the selection of said normalizer is hidden behind the serializer's
::normalize()
method, where it finds the correct normalizer from all those registered. We need to be able to essentially statically-analyze the selected normalizer. We need to peer into::getNormalizer()
, which is currently private.There is a dirty hack to invoke private methods with reflection, but I doubt that would pass a core quality gate and makes me feel very dirty.
Still leaving NR because I would love feedback on this and the overall approach. Not letting this slow down development, but it is a sticking point.
- last update
over 1 year ago Build Successful - last update
over 1 year ago Build Successful - ๐บ๐ธUnited States bradjones1 Digital Nomad Life
This should now be to the point of producing a spec-compliant schema for a resource object. There's plenty more work to be done, but I'm pretty happy with how relatively easy it was to implement the earlier generic work on schematic normalizers to JSON:API in basically an afternoon.
Re: our need for the Symfony Serializer to change two methods from private to protected, I've received some initial feedback from two Symfony maintainers and both asked good questions for follow-up.
The concrete bad news is that Symfony 6.4 and 7.0 are in feature-freeze, so the earliest this change would be accepted is 7.1.
Parallels were drawn between what we're doing and api-platform, which is an official-ish reference implementation of Symfony for doing fancy API things. They are generating JSON Schema using object reflection, and I laid out how this isn't much of a parallel to the Drupal entity and field APIs. We'll see if anyone's convinced by my reasoning enough to change the method visibility.
So optimistically, this would be blocked on Symfony 7. If this is a non-starter, there are two other options:
- Most radical: Ditch the Symfony serializer/normalizer because we barely use it now and abuse it when we do. This would be a major lift/BC break, but it would also alleviate a lot of headaches encountered in the D8+ lifecycle.
- Fork the Symfony Serializer and implement our own which implements
SerializerInterface
. This increases maintenance overhead and we are forked further from Symfony core, which we're trying to avoid. But there is precedence, e.g. we have our ownYamlLoader
. - (Ab)use
NormalizerInterface::normalize()
's context parameter and allow normalizers to return a value object containing a schema. There is some precedence for returning a value object as JSON:API module needs cacheability metadata and so it passes aroundCacheableNormalization
s. This is a bit of an escape hatch if we get cornered by other options. The downside is that is makes ::normalize() even more polymorphic and magical and also implies the value you pass is "real data," instead of potentially being "just" a supported class or interface name, which is all we might have during schema generation. Then again, that parameter is already mixed and so we could just define by convention that if the special context flag is passed, the value is to be regarded as the same as to::supportsNormalization()
. We could also use this approach to start and then deprecate it in favor of a more explicit method if we can ever make it publicly callable.
- last update
over 1 year ago Build Successful - last update
over 1 year ago Build Successful - last update
over 1 year ago Build Successful - ๐ณ๐ฑNetherlands bbrala Netherlands
First off, im so happy to see you working on this issue. :)
I've been tring to wrap my head around this and am having a hard time hehe.
One of the things we might be able to do is perhaps use the serializer to keep track? We already overwrite the contructor for that and pass that up to the parent. Perhaps there is a way to add the required data to the normalizers there using our own interface. This also feels very tacky in a way, but could be a place where we can track that information and allow it to be pulled in the normalizer itself perhaps.
This is purely based on looking at code, have not tried and see if that would work and how that would look.
The suggestion to use another format for the json_schema in the github issue and go from there is also interesting, wouldn't that work? it might not be the fastest of things to serialize again and go through all the paces to get the schema and would add another whole normalization pipeline, but it could be cached pretty well I guess. This does seem to combine pretty well with the 'describeby' member that was added in jsonapi 1.1 though, which is a link. It could then be a link like '/jsonapi/node/article?format=json_schema'.
Basically, i have no real awnser here right now. I'm not convinced we will see Symfony change the visibility right now. Hopefully my thoughts are helpfull in a way.
- Status changed to Needs work
over 1 year ago 6:11pm 18 October 2023 - ๐บ๐ธUnited States bradjones1 Digital Nomad Life
Thanks @bbrala for the feedback. Sorry I couldn't make it to Lille this year to work on this in person! We got a lot done last time we pair programmed in Portland.
One of the things we might be able to do is perhaps use the serializer to keep track? We already overwrite the contructor for that and pass that up to the parent. Perhaps there is a way to add the required data to the normalizers there using our own interface. This also feels very tacky in a way, but could be a place where we can track that information and allow it to be pulled in the normalizer itself perhaps.
This is purely based on looking at code, have not tried and see if that would work and how that would look.
I think I follow what you're saying, and in theory yes we could do a lot of shenanigans within the @internal portion of JSON:API module. However there are two main gotchas to this approach: 1) despite these methods not being "technically" extend-able, modules like Extras violate that with imposters and so any normalizers from modules like that, while violating the internal rule, would need to also change. And 2) this would only address JSON:API's normalizers, but not those in the core typed data API, which provides all the schema for properties.
The suggestion to use another format for the json_schema in the github issue and go from there is also interesting, wouldn't that work?
I ran into your feedback while coming here to say that yes, I think this suggestion for treating JSON Schema as another "format" is going to be the path forward.
...it might not be the fastest of things to serialize again and go through all the paces to get the schema and would add another whole normalization pipeline, but it could be cached pretty well I guess.
Unless we're talking about different things here, and I don't think we are, I don't see a performance hit. The approach with ::getNormalizer() would depend on the same normalizer resolving process that was optimized recently with the help of upstream changes, and this is just another path for resolving a normalizer but for a different $format. Also as it stands I don't believe that performance should be a huge issue here, as the resulting schemas could be cached and won't vary much.
This does seem to combine pretty well with the 'describeby' member that was added in jsonapi 1.1 though, which is a link. It could then be a link like '/jsonapi/node/article?format=json_schema'.
That's an angle I hadn't explored yet, however I think it's really powerful and would open the door to a very low-code way for us to bring the functionality that currently lives in jsonapi_schema module โ into core. I almost wonder if that shouldn't be the near-term goal vs. the OpenAPI integration, though OAI is basically just formatting the same data in a slightly different way, so these are not in competition.
Basically, i have no real awnser here right now. I'm not convinced we will see Symfony change the visibility right now. Hopefully my thoughts are helpfull in a way.
Yeah, after the third maintainer basically said "no," I agree, and honestly this is how the process should work. I iterated on the initial idea (an interface for the normalizer), it turns out that won't work and is blocked on upstream cooperation anyway, but we land on a solution that might even be "better."
I am toying with different ways of implementing this, however, that would still perhaps leverage an interface and/or trait to do most of the heavy lifting for the normalizers. I have some ideas that I'll toy around with in the MR as I refactor out of this solution and into the alternative for
$format
.Marking NW as I only really had it in NR to solicit this kind of help.
- last update
over 1 year ago Custom Commands Failed - last update
over 1 year ago Custom Commands Failed - last update
about 1 year ago 30,384 pass, 8 fail - last update
about 1 year ago 30,333 pass, 17 fail - last update
about 1 year ago Custom Commands Failed - last update
about 1 year ago 30,370 pass, 12 fail 55:21 52:12 Running- last update
about 1 year ago 30,451 pass, 1 fail - last update
about 1 year ago 30,452 pass - Status changed to Needs review
about 1 year ago 3:17pm 23 October 2023 - ๐บ๐ธUnited States bradjones1 Digital Nomad Life
Putting this back to NR as I would still love feedback from maintainers and committers.
The MR now contains a refactor of the initial approach, which I think overall is an improvement. TL;DR, you now request schema from the normalization system by specifying the
$format
asjson_schema
.The only real nit I can see with this approach is that normalizers are resolved based on the data to be normalized and the format, and so there is theoretically a different universe of normalizers selected for schema vs. a specific format. That's pretty much unavoidable with this approach. In theory we could get 100% the same normalizers by specifying the format same as the eventual normalization and hinting in $context that we actually want the schema. (This is in spirit what we wanted by having access to ::getNormalizer().)
I don't love this but don't hate it,
in theory
. The practical reason we can't take this approach is that normalizers which don't care about the $format (which is actually the majority of them) need to be aware of that context, and might return a "normal" normalization instead of schema. We have to address this as it is with this change toNormalizerBase::checkFormat()
to special-case the json_schema format as an exception to the "if I don't specify any formats, I serve them all" default.Another question would be if the JSON:API implementation should wrap schema in
CacheableNormalization
s or not. - ๐บ๐ธUnited States bradjones1 Digital Nomad Life
One thought about type-safety on the returned value (since we are using
::normalize()
for schema as well as normalizations) could be to require it to specify the JSON Schema meta-schema it implements in$schema
, which would be a quick check for the calling code to say "this is for sure a schema." I'm not sure if that's necessary, but one idea to throw out there if people are worried about non-core normalizers somehow getting tricked into returning a normalization when we really want a schema. - last update
about 1 year ago Custom Commands Failed - last update
about 1 year ago Custom Commands Failed - last update
about 1 year ago 30,419 pass, 4 fail - last update
about 1 year ago 30,462 pass - last update
about 1 year ago 30,464 pass - last update
about 1 year ago 30,464 pass - ๐บ๐ธUnited States bradjones1 Digital Nomad Life
Making title less technical and more accurate as to the current goals.
- Status changed to Needs work
about 1 year ago 3:02pm 13 January 2024 The Needs Review Queue Bot โ tested this issue. It no longer applies to Drupal core. Therefore, this issue status is now "Needs work".
This does not mean that the patch needs to be re-rolled or the MR rebased. Read the Issue Summary, the issue tags and the latest discussion here to determine what needs to be done.
Consult the Drupal Contributor Guide โ to find step-by-step guides for working with issues.
- Status changed to Needs review
about 1 year ago 11:55pm 13 January 2024 - ๐บ๐ธUnited States bradjones1 Digital Nomad Life
Config schema generation is very different and a path forward for them is not very clear yet. The recent work on validation for config entities will help unlock this in the future, however.
- Status changed to Needs work
about 1 year ago 11:40am 20 January 2024 The Needs Review Queue Bot โ tested this issue. It no longer applies to Drupal core. Therefore, this issue status is now "Needs work".
This does not mean that the patch necessarily needs to be re-rolled or the MR rebased. Read the Issue Summary, the issue tags and the latest discussion here to determine what needs to be done.
Consult the Drupal Contributor Guide โ to find step-by-step guides for working with issues.
- ๐ญ๐บHungary Gรกbor Hojtsy Hungary
Is there a high level summary for this issue? I can't tell from the huge changeset what exactly is being proposed here. What's the before/after? "Content entities were serialized in BLOBs before (in which cases?) but now they are serialized in JSON?" What's the benefit? Which APIs are affected, etc? Especially of an issue of this magnitude I think it would be important to outline these. It should help with reviews as well :)
- ๐ง๐ชBelgium wim leers Ghent ๐ง๐ช๐ช๐บ
+1 to what @Gรกbor Hojtsy said. Change records would be really helpful to understand this functionality too. ๐
- Status changed to Needs review
11 months ago 6:00pm 29 February 2024 - ๐บ๐ธUnited States bradjones1 Digital Nomad Life
Draft CR added. IS updated. MR rebased. Back to NR.
- Status changed to Needs work
11 months ago 1:32am 4 March 2024 The Needs Review Queue Bot โ tested this issue. It no longer applies to Drupal core. Therefore, this issue status is now "Needs work".
This does not mean that the patch necessarily needs to be re-rolled or the MR rebased. Read the Issue Summary, the issue tags and the latest discussion here to determine what needs to be done.
Consult the Drupal Contributor Guide โ to find step-by-step guides for working with issues.
- Status changed to Needs review
11 months ago 9:03pm 7 March 2024 - ๐บ๐ธUnited States bradjones1 Digital Nomad Life
Conflict was from the conversion of typed data plugins from annotations to attributes ๐ฏ
- ๐บ๐ธUnited States bradjones1 Digital Nomad Life
Related: โจ Generate JSON Schema for config entity types Active
- Status changed to Needs work
10 months ago 10:23am 6 April 2024 The Needs Review Queue Bot โ tested this issue. It no longer applies to Drupal core. Therefore, this issue status is now "Needs work".
This does not mean that the patch necessarily needs to be re-rolled or the MR rebased. Read the Issue Summary, the issue tags and the latest discussion here to determine what needs to be done.
Consult the Drupal Contributor Guide โ to find step-by-step guides for working with issues.
- Status changed to Needs review
9 months ago 3:11am 4 May 2024 - ๐ณ๐ฑNetherlands bbrala Netherlands
I went through the MR and have some comments. All and all it is a good implementation of what we talked about (quite) a while back.
Change record is available.
Issue summary seems up to date, although format is a little more freeform, would prefer to move to default setup. So keeping that tag. - Status changed to Needs work
9 months ago 1:29pm 10 May 2024 - ๐บ๐ธUnited States bradjones1 Digital Nomad Life
Related note, so close to having a proper JSON:API 1.0 and 1.1 schema available on jsonapi.org but it's blocked on infra: https://github.com/json-api/json-api/issues/1749
That doesn't mean we have to block this issue on that, and now that it's in a const it's easy enough to update.
- Status changed to Needs review
5 months ago 8:30pm 18 August 2024 - Status changed to RTBC
5 months ago 9:06am 21 August 2024 - ๐ณ๐ฑNetherlands bbrala Netherlands
I've gone through the changes and all previous comments. Think we got through everything and the feedback has been adressed. Remove NISU since the wording has changed referencing the other issue.
I'm gonna go out on a limb and push to RTBC. <3
- Status changed to Needs work
4 months ago 11:59am 11 September 2024 - ๐ฌ๐งUnited Kingdom alexpott ๐ช๐บ๐
I've added a couple of small comments to the MR that could be addressed.
- Status changed to Needs review
4 months ago 12:09am 13 September 2024 - ๐บ๐ธUnited States bradjones1 Digital Nomad Life
I have addressed your comments.
- ๐บ๐ธUnited States bradjones1 Digital Nomad Life
A bit of a fat-fingered rebase but I am hoping that this is very close to RTBC still.
- Status changed to RTBC
4 months ago 10:05am 21 September 2024 - ๐ณ๐ฑNetherlands bbrala Netherlands
The comments by alex have been adressed. I agree with the argument for the array argument. Back to RTBC
- ๐ฆ๐บAustralia larowlan ๐ฆ๐บ๐.au GMT+10
bbrala โ credited larowlan โ .
The Needs Review Queue Bot โ tested this issue. It fails the Drupal core commit checks. Therefore, this issue status is now "Needs work".
This does not mean that the patch necessarily needs to be re-rolled or the MR rebased. Read the Issue Summary, the issue tags and the latest discussion here to determine what needs to be done.
Consult the Drupal Contributor Guide โ to find step-by-step guides for working with issues.
- ๐บ๐ธUnited States bradjones1 Digital Nomad Life
Putting this back to RTBC as it only needed a phpstan-related update relating to improved analysis as this awaits commit.
- ๐บ๐ธUnited States bradjones1 Digital Nomad Life
Updating tags. Marking this as a contrib blocker because new versions of OpenAPI and OpenAPI JSON:API depend on this.
- ๐ฎ๐ฑIsrael jsacksick
I think the patch needs a reroll as it doesn't apply to 11.0.6.
- ๐บ๐ธUnited States bradjones1 Digital Nomad Life
I've rebased against 11.x, if it doesn't apply to a specific tag it would be because of a change in HEAD.
The Needs Review Queue Bot โ tested this issue. It no longer applies to Drupal core. Therefore, this issue status is now "Needs work".
This does not mean that the patch necessarily needs to be re-rolled or the MR rebased. Read the Issue Summary, the issue tags and the latest discussion here to determine what needs to be done.
Consult the Drupal Contributor Guide โ to find step-by-step guides for working with issues.
- ๐บ๐ธUnited States bradjones1 Digital Nomad Life
This is likely due to โจ Allow specifying `meta` data on JSON:API objects Needs work going in which is a good thing. Needs a rebase then should be back to RTBC.
- ๐ณ๐ฑNetherlands bbrala Netherlands
Rebasing took a little effort because we need to deprecate some extra arguments. I also had to undo contructor promotion since the merged json:api metadata issue changed the contructor parameter.
I think the changes are small enough no to warrent a new review, all is green still.
- ๐ฆ๐บAustralia larowlan ๐ฆ๐บ๐.au GMT+10
Some questions on the MR, will keep an eye out for when this is addressed, as getting this into 11.2 early is a priority
- ๐ณ๐ฑNetherlands bbrala Netherlands
All things have been adressed.
Exciting times :)
-
larowlan โ
committed 8e29ee2b on 11.x
Issue #3031367 by bradjones1, bbrala, gabesullice, wim leers, larowlan,...
-
larowlan โ
committed 8e29ee2b on 11.x
- ๐ฆ๐บAustralia larowlan ๐ฆ๐บ๐.au GMT+10
Committed to 11.x - thanks
We've got ~6 months to find any issues downstream from here - likely started with JSON API extras which is maintained by several of the people in this issue.
Great work all
- ๐บ๐ธUnited States bradjones1 Digital Nomad Life
Is this eligible to go into D10 LTS?
- ๐ฆ๐บAustralia larowlan ๐ฆ๐บ๐.au GMT+10
Under https://www.drupal.org/about/core/policies/core-change-policies/allowed-... โ - I don't think so.
Published the change record.
- ๐ง๐ชBelgium wim leers Ghent ๐ง๐ช๐ช๐บ
๐คฏ What a christmas present! ๐คฉ
Questions:
- The change record doesn't mention it, but the commit does change the JSON:API version:
JsonApiSpec::SUPPORTED_SPECIFICATION_VERSION = '1.1'
is now a fact. So #3305324 is partially done? Completely? See #3305324-18: [Meta] JSON:API 1.1 spec compliance/support โ . It seems that meta indicates there's a lot more to be done to be 1.1-compliant, so it's a bit surprising to see that constant being changed. I think either an updated change record that clarifies it, or an additional change record would be appropriate? - Next up: #3426508, and I posted some pointers at #3426508-4: Generate JSON Schema for config entity types โ .
- This seems to statically define JSON schemas, which is not good enough: many field types have settings that impact what the correct JSON schema would be. IOW: the JSON schemas have to be dynamically defined for them to be precise. For example: min/max for integers and numbers, a limited set of valid values (
enum
in JSON schema), many strings have a particular format (https://json-schema.org/understanding-json-schema/reference/string#built...), and so on. Are there plans/discussions for how to support those more precise JSON schema descriptions? Perhaps this needs a follow-up?
Either way: HUUUGE leap forward! ๐คฉ๐
- The change record doesn't mention it, but the commit does change the JSON:API version:
- ๐บ๐ธUnited States bradjones1 Digital Nomad Life
Thanks, Wim!
I will reply to your specific points soon, but why is this now marked as to-be-ported? The reply above indicates this can't go into 10.x, so I think there's nothing to backport?
(To be fair, my main project in production is "still" D10 and this applies relatively cleanly, but I also understand if it can't go officially into D10 because of policy.) I would love for this to be supported in D10 as that would personally benefit me and make the new majors of OpenAPI/OpenAPI JSON:API D10 compatible but appreciate if that's just too much.
- ๐ฆ๐บAustralia larowlan ๐ฆ๐บ๐.au GMT+10
I think NW to add a follow up for at least the third of Wim's points is an appropriate status - tagging as such.
Getting this early into 11.2 allows us to smooth out any pain points so it would be great to see what the current approach is missing for XB and refine this before 11.2 comes out
- ๐บ๐ธUnited States bradjones1 Digital Nomad Life
Re: ##8 - thanks Wim for your very kind words.
The change record doesn't mention it, but the commit does change the JSON:API version: JsonApiSpec::SUPPORTED_SPECIFICATION_VERSION = '1.1' is now a fact. So ๐ฑ [Meta] JSON:API 1.1 spec compliance/support Active is partially done? Completely? See #3305324-18: [Meta] JSON:API 1.1 spec compliance/support โ . It seems that meta indicates there's a lot more to be done to be 1.1-compliant, so it's a bit surprising to see that constant being changed. I think either an updated change record that clarifies it, or an additional change record would be appropriate?
I bumped the version in this MR for a few reasons. One, our testing and validation features (e.g., that validates responses when assertions are enabled) depend on a schema for JSON:API itself, which is not exactly stable but the upstream maintenance of JSON:API overall is glacially slow. The schema file we had for 1.0 was draft and not that valid and so the updated 1.1 draft is much closer to truly representing the spec's requirements.
After reviewing the linked issue I would say that we very nearly support 1.1 at a basic level, though we "should" also finish ๐ Spec Compliance: JSON API's profile/extention (Fancy Filters, Drupal sorting, Drupal pagination, relationship arity) needs to be explicitly communicated Active to convey our profile.
TL;DR, it's time to bump it because we were far ahead of 1.0 with implementing features that would later go into 1.1. We don't strictly validate, but it's much closer to the current spec than we've ever been.
This seems to statically define JSON schemas, which is not good enough: many field types have settings that impact what the correct JSON schema would be. IOW: the JSON schemas have to be dynamically defined for them to be precise. For example: min/max for integers and numbers, a limited set of valid values (enum in JSON schema), many strings have a particular format (https://json-schema.org/understanding-json-schema/reference/string#built...), and so on. Are there plans/discussions for how to support those more precise JSON schema descriptions? Perhaps this needs a follow-up?
Schemas _may_ be defined statically, but they need not be. In addition, JSON Schema generation has no caching layer (the assumption being that the caller is responsible for this, e.g. OpenAPI module, and invalidation is based on entity definitions or whatever) so the entire schema for the normalization you are introspecting is generated on-demand. It is true that included in the merged MR was a trait and attributes that help provide static schemas... but that's not a requirement. This allowed us to add a lot of default schemas for primitive types and make it easy to define them this way, but you can also return any valid schema generated any way you like. So I think a follow-up for option module to provide a
ListStringItemNormalizer
with its own dynamic logic would be amazing. But it requires no change to the underlying API. Perhaps we could introduce some additional traits or whatever to assist with code re-use, but yeah.