Attualmente mi sto insegnando OData ma mi sono imbattuto in una situazione che non sono stato in grado di risolvere. O sono io a fraintendere le specifiche OData o devo fare qualcosa per farlo funzionare.OData su API Web: come interrogare le proprietà annidate?
Ho creato un piccolo modello di entità Libri e Autori (EF/CF). Roba abbastanza semplice, con una relazione uno-a-molti da autori di libri:
modelBuilder.Entity<Book>().HasRequired(b => b.Author);
modelBuilder.Entity<Author>().HasMany(a => a.Books);
Ora, quando si interroga Autori vorrei essere in grado di espandere la proprietà libri e filtro sui suoi (annidati) proprietà. Ad esempio, se chiedo "che ha scritto i libri di Harry Potter", in questo modo ...
http://myBooksDatabase/Authors?$expand=Books&$filter=contains(Books/Name,'Harry Potter')&$select=Name
... ottengo questo errore di risposta:
{
error: {
code: ""
message: "The query specified in the URI is not valid. The parent value for a property access of a property 'Name' is not a single value. Property access can only be applied to a single value."
innererror: {
message: "The parent value for a property access of a property 'Name' is not a single value. Property access can only be applied to a single value."
type: "Microsoft.OData.Core.ODataException"
stacktrace: " at Microsoft.OData.Core.UriParser.Parsers.EndPathBinder.BindEndPath(EndPathToken endPathToken) at Microsoft.OData.Core.UriParser.Parsers.MetadataBinder.BindEndPath(EndPathToken endPathToken) at Microsoft.OData.Core.UriParser.Parsers.MetadataBinder.Bind(QueryToken token) at Microsoft.OData.Core.UriParser.Parsers.MetadataBinder.BindFunctionParameter(FunctionParameterToken token) at Microsoft.OData.Core.UriParser.Parsers.MetadataBinder.Bind(QueryToken token) at Microsoft.OData.Core.UriParser.Parsers.FunctionCallBinder.<BindFunctionCall>b__8(FunctionParameterToken ar) at System.Linq.Enumerable.WhereSelectEnumerableIterator`2.MoveNext() at System.Collections.Generic.List`1..ctor(IEnumerable`1 collection) at Microsoft.OData.Core.UriParser.Parsers.FunctionCallBinder.BindFunctionCall(FunctionCallToken functionCallToken) at Microsoft.OData.Core.UriParser.Parsers.MetadataBinder.BindFunctionCall(FunctionCallToken functionCallToken) at Microsoft.OData.Core.UriParser.Parsers.MetadataBinder.Bind(QueryToken token) at Microsoft.OData.Core.UriParser.Parsers.FilterBinder.BindFilter(QueryToken filter) at Microsoft.OData.Core.UriParser.ODataQueryOptionParser.ParseFilterImplementation(String filter, ODataUriParserConfiguration configuration, IEdmType elementType, IEdmNavigationSource navigationSource) at Microsoft.OData.Core.UriParser.ODataQueryOptionParser.ParseFilter() at System.Web.OData.Query.FilterQueryOption.get_FilterClause() at System.Web.OData.Query.Validators.FilterQueryValidator.Validate(FilterQueryOption filterQueryOption, ODataValidationSettings settings) at System.Web.OData.Query.FilterQueryOption.Validate(ODataValidationSettings validationSettings) at System.Web.OData.Query.Validators.ODataQueryValidator.Validate(ODataQueryOptions options, ODataValidationSettings validationSettings) at System.Web.OData.Query.ODataQueryOptions.Validate(ODataValidationSettings validationSettings) at System.Web.OData.EnableQueryAttribute.ValidateQuery(HttpRequestMessage request, ODataQueryOptions queryOptions) at System.Web.OData.EnableQueryAttribute.ExecuteQuery(Object response, HttpRequestMessage request, HttpActionDescriptor actionDescriptor) at System.Web.OData.EnableQueryAttribute.OnActionExecuted(HttpActionExecutedContext actionExecutedContext)"
}-
}-
}
mi rendo conto che posso ottenere che da l'interrogazione dei libri entità ...
http://myBooksDatabase/Books?$expand=Author&$filter=contains(Name,'Harry')
... ma il problema ricevo proviene da quando tento riferimento alla proprietà nidificate, non importa come lo faccio. La query sopra funziona, e presenta l'intera entità autore, ma se aggiungo &$select=Author/Name
ottengo la seguente risposta:
{
error: {
code: ""
message: "The query specified in the URI is not valid. Found a path with multiple navigation properties or a bad complex property path in a select clause. Please reword your query such that each level of select or expand only contains either TypeSegments or Properties."
innererror: {
message: "Found a path with multiple navigation properties or a bad complex property path in a select clause. Please reword your query such that each level of select or expand only contains either TypeSegments or Properties."
type: "Microsoft.OData.Core.ODataException"
stacktrace: " at Microsoft.OData.Core.UriParser.Visitors.SelectPropertyVisitor.ProcessTokenAsPath(NonSystemToken tokenIn) at Microsoft.OData.Core.UriParser.Visitors.SelectPropertyVisitor.Visit(NonSystemToken tokenIn) at Microsoft.OData.Core.UriParser.Syntactic.NonSystemToken.Accept(IPathSegmentTokenVisitor visitor) at Microsoft.OData.Core.UriParser.Parsers.SelectBinder.Bind(SelectToken tokenIn) at Microsoft.OData.Core.UriParser.Parsers.SelectExpandBinder.Bind(ExpandToken tokenIn) at Microsoft.OData.Core.UriParser.Parsers.SelectExpandSemanticBinder.Bind(IEdmStructuredType elementType, IEdmNavigationSource navigationSource, ExpandToken expandToken, SelectToken selectToken, ODataUriParserConfiguration configuration) at Microsoft.OData.Core.UriParser.ODataQueryOptionParser.ParseSelectAndExpandImplementation(String select, String expand, ODataUriParserConfiguration configuration, IEdmStructuredType elementType, IEdmNavigationSource navigationSource) at Microsoft.OData.Core.UriParser.ODataQueryOptionParser.ParseSelectAndExpand() at System.Web.OData.Query.Validators.SelectExpandQueryValidator.Validate(SelectExpandQueryOption selectExpandQueryOption, ODataValidationSettings validationSettings) at System.Web.OData.Query.SelectExpandQueryOption.Validate(ODataValidationSettings validationSettings) at System.Web.OData.Query.Validators.ODataQueryValidator.Validate(ODataQueryOptions options, ODataValidationSettings validationSettings) at System.Web.OData.Query.ODataQueryOptions.Validate(ODataValidationSettings validationSettings) at System.Web.OData.EnableQueryAttribute.ValidateQuery(HttpRequestMessage request, ODataQueryOptions queryOptions) at System.Web.OData.EnableQueryAttribute.ExecuteQuery(Object response, HttpRequestMessage request, HttpActionDescriptor actionDescriptor) at System.Web.OData.EnableQueryAttribute.OnActionExecuted(HttpActionExecutedContext actionExecutedContext)"
}-
}-
}
Ecco le mie due controller OData per autori e libri:
namespace My.OData.Controllers
{
public class AuthorsController : ODataController
{
// GET /Author
[EnableQuery]
public IQueryable<Author> Get()
{
return MediaContext.Singleton.Authors;
}
// GET /Authors(<key>)
[EnableQuery]
public SingleResult<Author> Get([FromODataUri] Guid key)
{
var result = MediaContext.Singleton.Authors.Where(b => b.Id == key);
return SingleResult.Create(result);
}
// GET /Authors(<key>)/Books
[EnableQuery]
public IQueryable<Book> GetBooks([FromODataUri] Guid key)
{
return MediaContext.Singleton.Authors.Where(a => a.Id == key).SelectMany(author => author.Books);
}
}
public class BooksController : ODataController
{
// GET /Books
[EnableQuery]
public IQueryable<Book> Get()
{
return MediaContext.Singleton.Books;
}
// GET /Books(<key>)
[EnableQuery]
public SingleResult<Book> Get([FromODataUri] Guid key)
{
var result = MediaContext.Singleton.Books.Where(b => b.Id == key);
return SingleResult.Create(result);
}
// GET /Books(<key>)/Author
[EnableQuery]
public SingleResult<Author> GetAuthor([FromODataUri] Guid key)
{
return SingleResult.Create(MediaContext.Singleton.Books.Where(b => b.Id == key).Select(b => b.Author));
}
}
}
Così, come Ho detto, c'è qualcos'altro che devo aggiungere o configurare per far funzionare le proprietà dei riferimenti nelle entità correlate?
Ok, un po 'di indagine e di lettura delle specifiche del OData v4 mi ha insegnato la sintassi del filtro dovrebbe essere usando la funzione "any", come questa: http: // myBooksDatabase/Authors? $ filter = Books/any (b: contains (b/Name, 'Harry Potter'))). Funziona, ma cosa succede se ho bisogno di * alcune * proprietà dei libri, come il titolo e il codice ISBN? Non riesco ancora a capire come specificare un'istruzione $ select che limiti le proprietà nidificate a quelle di cui ho bisogno. –