OpenAPI.NET.OData icon indicating copy to clipboard operation
OpenAPI.NET.OData copied to clipboard

Bound operations not appended to non-contained navigation property paths

Open irvinesunday opened this issue 3 years ago • 6 comments

Related to https://github.com/microsoft/OpenAPI.NET.OData/issues/201

With this bound operation: image

and given this non-contained navigation property: image

and given a schema namespace value of microsoft.graph

We expect to generate paths similar to below: .../followedSites/microsoft.graph.add

This is not the case with the current version because we check the containment of the navigation property here. According to the spec. doc., see ref:

Actions and Functions MAY be bound to an entity type, primitive type, complex type, or a collection...
The namespace- or alias-qualified name of a bound operation may be appended to any URL that identifies 
a resource whose type matches, or is derived from, the type of the binding parameter. 

This may be applied to any navigation property segment as long as its type matches the type of the binding parameter.

irvinesunday avatar Jun 10 '22 14:06 irvinesunday

Thanks for documenting this in details. I'm torn here. Historically we've decided to only bind operations to contained navigation properties, entity sets or singletons. Or in other words to only have operations on the entities canonical path. This is to avoid expanding too many paths and growing the description too much. It makes sense since why would one call an operation on a "deeper path" when a shallower one is already available?

This case is an interesting example, and poor API design IMHO. The followed sites should instead be a collection of "followedSite", contained. FollowedSite is a new entity type that semantically belongs to the user (is contained), has a non-contained reference to site (existing entity), and potentially other properties like "followedDate", "followedReason", etc...

But since the API is not getting fixed any time soon, we probably out to make an opiniated decision here. What would be the consequence of setting contains target true for that property? Can we rely on an annotation/attribute that would signal "bind operations even though this is not contained"? Do we have other cases like this one?

baywet avatar Jun 20 '22 17:06 baywet

after a quick chat with @darrelmiller it looks like there's no such annotation. Can you have a look at the implications of changing contains target for that property? and see if we have other instances of that scenario?

baywet avatar Jun 20 '22 17:06 baywet

after a quick chat with @darrelmiller it looks like there's no such annotation. Can you have a look at the implications of changing contains target for that property? and see if we have other instances of that scenario?

Appending ContainsTarget=true generates 1042 more paths. I'm collating the list of other navigation props. that have a similar issue.

irvinesunday avatar Jun 21 '22 15:06 irvinesunday

The number of similar scenarios are quite significant. Below are just a couple of them:

** Non-containment Navigation properties:**

  • followedSites --> Action Name="add"
  • group --> Action Name="assignLicense"
  • filesFolder --> Action Name="restore"
  • createdOnBehalfOf --> Action Name="restore"/Action Name="follow"/Action Name="copy"
  • manager --> ""
  • owners --> Action Name="getByIds"
  • directReports --> ""
  • memberOf --> ""
  • transitiveMemberOf --> ""
  • registeredOwners --> ""
  • members --> ""
  • driveItem --> Action Name="copy" (binds to mailFolder, driveItem, message entities and root nav. prop)
  • group --> Action Name="assignLicense" (binds to user and group entity)
  • teachers --> Function Name="delta"
  • schools --> Function Name="delta"
  • user
  • acceptedSenders
  • calendar
  • membersWithLicenseErrors
  • parentNotebook
  • parentSection
  • baseTypes
  • sites
  • channel

Entities:

  • directoryObject
    • directReports/{directoryObject-id}
    • memberOf/{directoryObject-id}
    • transitiveMemberOf/{directoryObject-id}
  • driveItem
  • contentType
  • site

Clearly, setting their containment to true will be unfeasible, as the number of paths generated will just be humongous.

irvinesunday avatar Jul 13 '22 12:07 irvinesunday

To provide an update to everyone here: Darrel brought it to the API council and it was decided to introduce a new attribute/annotation to say something like "target properties" which would contain list of non contained properties we want that method to target additionally to the contained properties/entity sets/singleton. Next step is to get @mikePizzo to provide us guidance for that new attribute/annotation, but if you have suggestions, feel free to add them here.

baywet avatar Jul 14 '22 14:07 baywet

From discussions with @darrelmiller, the recommendation was to come up with a custom annotation that will be appended to a given operation. This annotation will list the non-contained navigation properties that should append paths to the given operation. The annotation would look something like:

image

irvinesunday avatar Aug 29 '22 10:08 irvinesunday

This was the proposal from Mike for the annotations to be used. MicrosoftTeams-image (5)

irvinesunday avatar Oct 27 '22 13:10 irvinesunday

After further discussions, the below annotation structure has been arrived at:

<Term Name="RequiresExplicitBinding" Type="Core.Tag" DefaultValue="true" AppliesTo="Action, Function">
  <Annotation Name="Description" Edm.String="This bound action or function is only available on model elements annotated with the ExplicitOperationBinding term."/>
</Term>

<Term Name="ExplicitOperationBindings" Type="Collection(Edm.String)"> 
   <Annotation Name="Description" Edm.String="The qualified names of explicitly bound operations that are supported on the target model element. These bindings are in addition to any bindings to the type of the target model element not annotated with RequiresExplicitBinding"/> 
</Term>

Example usage:

<Action Name="forward" IsBound="true" >
  <AnnotationTerm="Org.OData.Capabilities.V1.RequiresExplicitBinding"/>
  <ParameterName="bindingParameter" Type="self.email"/>
  <ParameterName="recipient" Type="Collection(self.recipent)"/>
  <ReturnTypeType="self.email" />
</Action>

<Annotations Target="self.Container/me/email">
  <Annotation Term="Org.OData.Capabilities.V1.ExplicitOperationBindings">
    <Collection>
      <String>self.forward</String>
    </Collection>
  </Annotation>
</Annotations>

Find further reading of the proposal here

irvinesunday avatar Nov 02 '22 07:11 irvinesunday

This issue is blocked while waiting for the resolution of: https://github.com/OData/odata.net/issues/2543

irvinesunday avatar Feb 28 '23 22:02 irvinesunday

Can we confirm that we are no longer blocked?

darrelmiller avatar Mar 25 '23 19:03 darrelmiller

Can we confirm that we are no longer blocked?

This is now unblocked. The PR to resolve this: https://github.com/microsoft/OpenAPI.NET.OData/pull/378

irvinesunday avatar Apr 28 '23 08:04 irvinesunday

Resolved by: https://github.com/microsoft/OpenAPI.NET.OData/pull/378

irvinesunday avatar May 02 '23 14:05 irvinesunday