On Mar 11, 2010, at 5:22 AM, James Strachan wrote:
>
> This seems great for 'root resources' or top level resources; though
> it gets harder when you build deeply nested resources, where (say) a
> Shipment resource might be a child of a Widget resource - with the
> Shipment resource only knowing how its linked within a Widget. e.g.
>
> class Widget {
> @Path("shipment/{shipmentId}")
> Shipment getShipment(@PathParam("shipmentId") String shipmentId) {
> return new Shipment(this, shipmentId)
> }
> }
>
> I wonder if the @Link annotation could also be configured with the
> 'owner'? e.g.
>
> class Shipment {
> @Link("shipment/{shipmentId}", ownedBy="widget")
> URI link;
>
> String shipmentId;
> Widget widget;
> }
>
> Though this is not very DRY as there's a need to replicate the child
> URI template in both @Link and @Path.
>
> I wonder if there's a way to just specify the URI template once on
> Widget which can then be used both when navigating from Widget ->
> Shipment and when trying to reverse engineer the URI from a Shipment?
>
I've been thinking about this too. One issue I see is that subresources can be made available via more than one root resource so it might not always be ideal to fix on a particular path (though I suppose a canonical URI for accessing a resource isn't really a bad thing). When you are happy to fix the path then we could offer something like (using your example above):
@Link(resource=Widget.class, locator={"getShipment"})
where the locator property provides the sub-resource locator method. Given this info the link processor can assemble the URI from the root resource and locator path templates. Things get trickier if there are multiple locators used within a single request, annotations don't offer much in the way of syntax to represent a list of class+method pairs other than parallel arrays.
When you want to reflect the actual URI used to access the subresource then I was thinking that we could offer access to UriInfo via EL like:
@Link("${UriInfo.requestUri}")
> e.g. if the @Link could just refer to the owner, then you could find
> the getShipment() method on Widget and find its URI template from the
> @Path annotation? e.g.
>
> class Shipment {
> @Link(ownedBy="widget")
> URI link;
>
> String shipmentId;
> Widget widget;
> }
>
> Then the URI template is only specified once - in the usual place -
> and it can be deduced by looking for the method in the owner with a
> @Path annotation which returns an object of the type Shipment? (For
> the @Path URI template on Widget.getShipment() to be usable with @Link
> we'd need to use consistent naming of the 'shipmentId' parameter).
>
> You could argue that the owner could be deduced too (by analysing all
> of the available resources and figuring out which resource is capable
> of making a Shipment) - but you could possibly have a resource which
> is available via multiple URIs maybe - so being explicit in this case
> is not such a bad thing IMHO.
>
For the URI of the resource or its subresources then I think the ${UriInfo.requestUri} approach will get us most of the way. For linking back up the call stack then UriInfo also has the getMatchedURIs method so you can, e.g., pull out the URI of each resource in the call stack.
Marc.