【问题标题】:Azure Mobile Service TableController not returning inner objectsAzure 移动服务 TableController 不返回内部对象
【发布时间】:2015-09-04 09:54:44
【问题描述】:

我正在使用表存储创建一个基本的(我的第一个)Azure 移动服务来控制一个简单的事件应用程序。我的 DataObjects 由 2 种对象类型组成:CoordinatorEvent,我希望协调器成为一个单独的表来存储我不希望它在事件中非规范化的特定信息,但事件也有一个内部对象位置来存储事件位置的详细信息,但我想非规范化存储,因为我不想将这些详细信息与事件分开维护。

这是我目前拥有的对象: 数据对象:

public class Coordinator : EntityData {
    public string Name { get; set; }
    public int Points { get; set; }
    public bool IsActive { get; set; }
}
public class Event: EntityData {
        public Coordinator Coordinator { get; set; }
        public DateTime EventDate { get; set; }
        public int Attendees { get; set; }
        public Location Location { get; set; }
}
public class Location {
        public string Name { get; set; }
        public string Address1 { get; set; }
        public string Address2 { get; set; }
        public string City { get; set; }
        public string County { get; set; }
        public string PostCode { get; set; }
}

控制器作为基本的TableController由VS的脚手架生成,我所做的唯一更改是在事件控制器上公开MobileServiceContext以启用Post方法在保存时找到现有的协调器,因为客户端只会发布协调员 ID:

public class EventController : TableController<Event> {
        private MobileServiceContext context;
        protected override void Initialize(HttpControllerContext controllerContext) {
            base.Initialize(controllerContext);
            context = new MobileServiceContext();
            DomainManager = new EntityDomainManager<Event>(context, Request, Services);
        }
        protected override void Dispose(bool disposing) {
            context?.Dispose();
            base.Dispose(disposing);
        }

        public IQueryable<Event> GetAllEvent() {
            return Query();
        }

        public async Task<IHttpActionResult> PostEvent(Event item) {
            var coordinator = context.Coordinators.Find(item.Coordinator.Id);
            item.Coordinator = coordinator;
            Event current = await InsertAsync(item);
            return CreatedAtRoute("Tables", new { id = current.Id }, current);
        }

}

从客户端发布数据按预期工作,我有一个包含正确数据的协调器表:

ID  Name    Points  IsActive    Version CreatedAt   UpdatedAt   Deleted
cdf96b0f93644f1e945bd16d63aa96e0    John Smith  10  True    0x00000000000007D2  04/09/2015 09:15:02 +00:00  04/09/2015 09:15:02 +00:00  False
f216406eb9ad4cdcb7b8866fdf126727    Rebecca Jones   10  True    0x00000000000007D4  04/09/2015 09:15:30 +00:00  04/09/2015 09:15:30 +00:00  False

还有一个与第一个 Coordinator 关联的事件:

Id  EventDate   Attendees   Location_Name   Location_Address1   Location_Address2   Location_City   Location_County Location_PostCode   Version CreatedAt   UpdatedAt   Deleted Coordinator_Id
961abbf839bf4e3481ff43a214372b7f    04/11/2015 09:00:00 0   O2 Arena    Peninsula Square        London      SE10 0DX    0x00000000000007D6  04/09/2015 09:18:11 +00:00  04/09/2015 09:18:11 +00:00  False   cdf96b0f93644f1e945bd16d63aa96e0

此时一切看起来都不错,我的 2 个问题与事件的获取有关,其中 CoordinatorLocation 对象均未返回,而我的 EventController 获取的 Json 就是这样:

[{"id":"961abbf839bf4e3481ff43a214372b7f","attendees":0,"eventDate":"2015-11-04T09:00:00Z"}]

所以我的两个问题是:

1) Location 对象由服务器上的EventController 正确加载,如果我在返回之前中断,我可以看到属性正确加载,对我来说这看起来像是 Json 序列化问题,但我已经尝试更改序列化程序的配置(在WebApiConfig上)没有太大影响,我尝试的最后一个选项是MaxDepth,但仍然没有返回Location对象。

2) Coordinator 对象我根本没有加载到服务器上,甚至没有 Id(正确存储在表中)所以我不能强制加载整个对象,当然它不会返回到客户。

关于我在这里做错了什么有什么想法吗?

提前致谢

克莱顿·洛瓦托

【问题讨论】:

    标签: c# azure azure-mobile-services azure-table-storage


    【解决方案1】:

    这是 TableController 的默认行为。 为了以一种时尚的方式实现您的目标,您应该在控制器中实现 OData $expand

    本文为解决方案提供了一个很好的演练
    Retrieving data from 1:n relationship using .NET backend Azure Mobile Services

    作为进一步的扩展,我实现了一个自定义属性,您可以在控制器方法中使用它来指定客户端可以请求扩展哪些属性。您可能不想总是返回所有子关系(扩展对象)

    [AttributeUsage(AttributeTargets.Method, AllowMultiple = true)]
    public class ExpandablePropertyAttribute : ActionFilterAttribute
    {
        #region [ Constants ]
        private const string ODATA_EXPAND = "$expand=";
        #endregion
    
        #region [ Fields ]
        private string _propertyName;
        private bool _alwaysExpand;
        #endregion
    
        #region [ Ctor ]
        public ExpandablePropertyAttribute(string propertyName, bool alwaysExpand = false)
        {
            this._propertyName = propertyName;
            this._alwaysExpand = alwaysExpand;
        }
        #endregion
    
        #region [ Public Methods - OnActionExecuting ]
        public override void OnActionExecuting(HttpActionContext actionContext)
        {
            base.OnActionExecuting(actionContext);
            var uriBuilder = new UriBuilder(actionContext.Request.RequestUri);
            var queryParams = uriBuilder.Query.TrimStart('?').Split(new char[1] { '&' }, StringSplitOptions.RemoveEmptyEntries).ToList();
            int expandIndex = -1;
    
            for (var i = 0; i < queryParams.Count; i++)
            {
                if (queryParams[i].StartsWith(ODATA_EXPAND, StringComparison.Ordinal))
                {
                    expandIndex = i;
                    break;
                }
            }
    
            if (expandIndex >= 0 || this._alwaysExpand)
            {
                if (expandIndex < 0)
                {
                    queryParams.Add(string.Concat(ODATA_EXPAND, this._propertyName));
                }
                else
                {
                    queryParams[expandIndex] = queryParams[expandIndex] + "," + this._propertyName;
                }
    
                uriBuilder.Query = string.Join("&", queryParams);
                actionContext.Request.RequestUri = uriBuilder.Uri;
            }
    
        }
        #endregion
    }
    

    我以这种方式在我的控制器中使用:

        [ExpandableProperty("Documents", false)]
        public IQueryable<ClientActivity> GetAllClientActivities()
        {
            return Query(); 
        }
    

    【讨论】:

    • 嗨 Massimo,非常感谢,这很有意义,现在我可以将 Coordinator 项目发送给客户,不幸的是这不适用于我的“位置”对象。我仍然对序列化感到非常困惑,因为 $expand 内部的“协调器”对象被序列化了,但“位置”不是......
    • 奇怪,不知道为什么...我看到你的类只使用简单的类型,所以这两个实体应该以相同的方式工作(被序列化)。我在答案中添加了更多代码,以使用自定义属性以更高级的方式对其进行管理。希望有帮助
    • 您好 Massimo,非常感谢您的帮助,它确实帮助我确定了我做错了什么,并且 Coordinator 对象现在使用您的代码正确返回。 Location 对象仍未返回,但看起来这是 SDK 上的 OData 实现的错误。 (social.msdn.microsoft.com/Forums/azure/en-US/…) 在 MSDN 上有同样的问题。我现在将尝试找到解决方法,因为我不想创建一个新实体来存储位置
    • 大家好!这篇文章对我帮助很大,谢谢。顺便问一下:如果我正在查询一个对象,它有一个嵌套对象,并且嵌套对象也有一个嵌套对象,这是怎么回事?
    • 通过这种方式,您可能很容易得到一个提取整个数据库的方法。那时你可能要三思而后行。它也与 RESTful 原则有些冲突,并且 OData 没有为此提供规范。
    【解决方案2】:

    我已经设法将复杂的属性序列化并发送回客户端。也许解决方案不是最干净的,但它适用于我的情况。希望其他人也会觉得它有用。

    首先,创建一个继承自EnableQueryAttribute 的类并重写GetModel 方法,如下所示:

    public class CustomEnableQueryAttribute : EnableQueryAttribute
    {
        public override IEdmModel GetModel(Type elementClrType, HttpRequestMessage request, HttpActionDescriptor actionDescriptor)
        {
            var modelBuilder = new ODataConventionModelBuilder();
            modelBuilder.EntitySet<Event>("Events");
    
            return modelBuilder.GetEdmModel();
        }
    }
    

    然后,在控制器的 Initialize 方法中,添加以下行:

        protected override void Initialize(HttpControllerContext controllerContext)
        {
            base.Initialize(controllerContext);
            context = new MobileServiceContext();
            DomainManager = new EntityDomainManager<Event>(context, Request, Services);
    
            //Add these lines
            var service = Configuration.Services.GetServices(typeof(IFilterProvider)).ToList();
            service.Remove(service.FirstOrDefault(f => f.GetType() == typeof(TableFilterProvider)));
            service.Add(new TableFilterProvider(new CustomEnableQueryAttribute()));
            Configuration.Services.ReplaceRange(typeof(IFilterProvider), service.ToList().AsEnumerable());
    
            Request.Properties.Add("MS_IsQueryableAction", true);
        }
    

    【讨论】:

      猜你喜欢
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      相关资源
      最近更新 更多