-
-
Notifications
You must be signed in to change notification settings - Fork 280
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
Hyper-schema: Link subject/anchor #140
Comments
I don't understand the purpose of this at all. Could you expand with a practical use case? |
Suppose I have a JSON document that specifies metadata about a blog post, or another image, or something. In that case, I don't want to specify a link from the current document, I want to specify a link about some third resource. So I have a document like: [ {
post: 34
next: 35
}, {
post: 35,
next: 36
} ] and I want to generate the following links:
With the HTTP Link header, you'd specify it like this:
|
@awwright @Relequestual @jdesrosiers this just clicked for me and I understand why it is essential. I endorse the The simple example is that without this feature, collections with the "collection" and "item" relations simply don't work. The key to understanding this is the following bit from the definition of
So if a link is attached to the sub-schema, then the context URI of the link (which is what is meant by "from") is the resource to which the sub-schema is attached, not the complete instance document (which corresponds to the complete schema document). Consider this minimal collection schema:
With this representation, retrieved from
The context URI of the collection's "self" link as applied to this instance is However, the context URIs of the "item" link as applied to each of the two collection members are But the "item" link is defined as relating a collection (the context) to one of its members (the target). However, those context URIs do not identify the collection instance. They each identify one array element. So we need a way to adjust the context URI to identify the collection, which would be Why not just use the whole instance as context?In other situations this turns out to be critically important. If you have a collection of books, and in addition to the "item" link for each element of the collection array, you also define an "author" link, the context URI should be for the book, which is the individual array element. Similarly, each element in the collection array could have a "self" link, and once again the context should be the array item. The "self" link for the collection is defined at the top level. You can also get more complicated with nested collection representations. So an "item" link from an inner collection element should have the inner collection as its context, not the collection described by the full schema document. OK now what?
For instance, you may define a schema with an items link in I don't think that giving the re-used schema an "$id" would help, because it doesn't uniquely identify a position in the instance. The main schema might even re-use the re-usable sub-schema multiple times in the same representation (no idea why, but it's legal), and that would definitely be ambiguous. Perhaps we support two keywords:
(there is no way to do Relative JSON Pointers as URI fragments, which I can explain if anyone doesn't follow) |
I think The issues with "collection"/"item" might be why draft-04 went with "instances"/"full" instead. @handrews I was with you up until "OK now what?". Can you elaborate on the purpose of |
:-) Yeah Both This means that if a link is deeply nested in an array item, and needs to set a less-deeply-nested object that is still within the same array item as the context, then there is no URI reference that can express that, as they all start from the root of the instance document. Plus, if you re-use a schema that needs to go up a level within the re-used unit, you can't know how the subschema is being re-used so you can't write a URI reference that will always work. Here's a silly example which is not even attempting to be a coherent design of anything, just illustrate what happens: The "1" is a Relative JSON Pointer, indicating the next enclosing object/array up (like ".." in paths). {
"definitions": {
"foo": {
"properties": {
"outer": {
"properties": {
"inner": {
"links": [{"anchorPointer": "1"}]
}
}
}
}
},
"bar": {"properties": {"x": {"$ref": "#/definitions/foo"}}},
"baz": {"items": { "$ref": "#/definitions/foo" }}
}
} So, if you just had a JSON instance that matched foo on its own: {"outer": {"inner": null} Then But with a bar instance: {"x": {"outer": {"inner": null}}} you would need to replace it with Finally, with a baz instance: [{"outer": {"inner": null}}, {"outer": {"inner": null}}] you can't write this one properly with |
As far as "instances" and "full": All of that dates back to at least 2009 (in Draft 00). The collection and item link relations were registered in 2012. My guess is that either no one realized the duplication, or no one ever got around to resolving it until now. Although "instances" is not a direct analogue- the context of "instances" was the schema (same for "create"), which was confusing and (I'm fairly sure) unnecessary if we define the specific usage of "collection" and "item" for |
@handrews If I understand the "anchor" proposal correctly, based on your first example at #140 (comment), one should add an "anchor" within inner {
"title": "Generic collection with id references",
"type": "array",
"items": {
"type": "object",
"properties": {
"id": {"type": "integer", "minimum": 1}
},
"links": [
{
"rel": "item",
"href": "/stuff/element/{id}",
"anchor": "#"
}
]
},
"links": [
{
"rel": "self",
"href": "/stuff"
}
]
} is that correct? |
@dlax yes, that is correct! I am going to write up a PR for the As for I think I'd like to try writing that up as part of the |
Hmm... now I'm realizing my point about naming subschemas isn't as useful as I thought. Writing up a PR, I was quickly reminded that the context URI is an instance URI, so naming subschemas is irrelevant. Per the
There's another implication here: Using JSON Pointer-style URI fragments for "anchor" at all is dubious, as
|
@handrews Can't we say that the |
@dlax I did consider that and worried that it was too convoluted. Although it makes perfect sense to me (however, experience tells me that this is not an effective metric for whether it makes sense to anyone else ;-) More seriously, One reason that I did not post Also, the value could be a URI Reference, but it would only make sense when referencing another part of the schema that is currently being used to validate the instance. I would not want to allow |
I find this quite symmetrical with how Hyper-Schema's LDO are currently defining the context which is already deviating from RFC5988 because the link is not on the instance. Accordingly, I don't think it'd be more convoluted to assume that anything related to link's context (here, the Anyways, I'm trying to figure out limitations of this "simple" approach to get convinced that something more complex is needed (e.g. #351). For relative references to subschemas within the same schema (the case of |
@dlax we do not yet use any keywords that are also used in RFC 5988 in completely different ways. The most different is "href" being a template. But really an LDO only becomes an actual link when the template is resolved, and at that point "href", "rel", "mediaType", and "title" have the same semantics as in RFC 5988. Since JSON Schema does not control the exact media type of the instance, it is not always possible to calculate a context URI that your definition would produce. There may be no way to associate a URI fragment with the position in the instance that the designated schema matches. If we were able to do that, it would map down into RFC 5988 semantics and I might have been persuadable. I still don't see how referencing another schema produces unambiguous instance locations. A given sub-schema may validate many points within the instance. It may not always be clear which is intended or desirable. |
@dlax I didn't really express this well earlier. We may be able to come up with rules that makes this unambiguous, in which case I would really like this idea (independent of the keyword we use for it). This reminds me a bit of the debate on how powerful relative pointers need to be (e.g. do you need to be able to move laterally in a parent array, which the current Relative JSON Pointer spec does not support). I should try to find that discussion, as I think we managed to prove that some of the use cases really had no application. A similar approach could work here. How would you feel about splitting this off into a new issue, and letting the current |
@handrews That's fine by me. |
JSON Schema should be able to define the subject (anchor, or source) of a link relation.
Suppose I define a link right here (HTML):
We can express the information links convey in a format called N-Triples (a subset of the more general grammar Turtle). Since links by default have a subject of the current document, it would look like:
But it might be the case we want to use JSON Schema to let documents make links about other documents to different other documents. A subject that's not
<https://github.com/json-schema-org/json-schema-spec/issues/140>
Well in that case we need to allow JSON Schema to define what that "subject"/"source"/"anchor" is.
The text was updated successfully, but these errors were encountered: