我在Breeze JsonMediaTypeFormatter
的设置中遇到了问题。
我要做的是由WebAPI发送和接收json的日期
总是在UTC工作。
根据this document,可以通过为DateTimeZoneHandling
DateTimeZoneHandling.Utc
设置为JsonSerializerSettings
来实现
然而这不起作用。
调查this source code,我意识到可能影响这种行为的是this other issue所做的黑客攻击。
通过删除所有这些代码,一切正常。
//jsonSerializerSettings.Converters.Add(new IsoDateTimeConverter
//{
// DateTimeFormat = "yyyy-MM-dd\\THH:mm:ss.fffK"
//});
如何在不删除Hack的情况下处理这种情况?
编辑1
我首次尝试设置如下:
var jsonFormatter = Breeze.WebApi.JsonFormatter.Create();
jsonFormatter.SerializerSettings.DateTimeZoneHandling = DateTimeZoneHandling.Utc;
jsonFormatter.SupportedMediaTypes.Add(new MediaTypeHeaderValue("application/json"));
jsonFormatter.SupportedEncodings.Add(new UTF8Encoding(false, true));
GlobalConfiguration.Configuration.Formatters.Insert(
0, jsonFormatter);
但这不起作用,返回的日期不是UTC。
编辑2
首先,我已将Breeze lib更新为0.80.3版本。
在我的App_Start文件夹中,我有这个BreezeWebApiConfig.cs文件:
[assembly: WebActivator.PreApplicationStartMethod(
typeof(Partner.App_Start.BreezeWebApiConfig), "RegisterBreezePreStart")]
namespace Partner.App_Start
{
public static class BreezeWebApiConfig
{
public static void RegisterBreezePreStart()
{
GlobalConfiguration.Configuration.Routes.MapHttpRoute(
name: "BreezeApi",
routeTemplate: "api/{controller}/{action}"
);
var jsonFormatter = Breeze.WebApi.JsonFormatter.Create();
jsonFormatter.SupportedMediaTypes.Add(new MediaTypeHeaderValue("application/json"));
jsonFormatter.SupportedEncodings.Add(new UTF8Encoding(false, true));
GlobalConfiguration.Configuration.Formatters.Insert(
0, jsonFormatter);
// Apply query parameters, expressed as OData URI query strings,
// to results of Web API controller methods that return IQueryable<T>
GlobalConfiguration.Configuration.Filters.Add(
new Breeze.WebApi.ODataActionFilter());
}
}
}
其次,我在一个名为BreezeConfig的文件夹中创建了一个CustomBreezeConfig.cs类(使用Jay下面描述的代码),但是这个新尝试没有用。
此致
Bernardo Pacheco
答案 0 :(得分:2)
从breeze v 0.80.3开始,我们添加了自定义jreeze序列化器设置的功能,这些设置是breeze用于查询和保存的。它涉及添加一个服务器端类,它是新的Breeze.WebApi.BreezeConfig类的子类。这个子类看起来像:
public class CustomBreezeConfig : Breeze.WebApi.BreezeConfig {
/// <summary>
/// Overriden to create a specialized JsonSerializer implementation that uses UTC date time zone handling.
/// </summary>
protected override JsonSerializerSettings CreateJsonSerializerSettings() {
var baseSettings = base.CreateJsonSerializerSettings();
baseSettings.DateTimeZoneHandling = DateTimeZoneHandling.Utc;
return baseSettings;
}
}
现在将自动发现服务器端项目中出现的Breeze.WebApi.BreezeConfig子类的任何实例,并用于自定义breeze的配置。
如果这有帮助(或没有),请告知我们。
答案 1 :(得分:1)
当你说添加DateTimeZoneHandling不起作用时,你是如何尝试设置它的?
您可以尝试在源代码中的“Converters.Add”调用(从上方)上方添加此行(不删除“hack”),并告诉我它是否有效。
jsonSerializerSettings.DateTimeZoneHandling = DateTimeZoneHandling.Utc;
我同意它仍然很笨拙,因为这意味着你必须修改微风源。因此,如果它确实有效,我们将尝试提供一些方法来允许您从格式化程序外部设置它。请告诉我们。
答案 2 :(得分:1)
请尝试使用breeze v 0.80.5以及相应的发行说明。希望'时间应该现在应该正常往返。
答案 3 :(得分:0)
我解决了这个黑客的问题,它仍然闻起来。
app.vm.run.js中的
app.vm.run = (function ($, ko, dataservice, router) {
var currentRunId = ko.observable(),
// run will be an entity
run = ko.observable(),
...
save = function () {
this.run().lastUpdate(makeDatetimeUTC(moment().toDate()));
this.run().runStart(makeDatetimeUTC(this.run().runStart()));
this.run().runEnd(makeDatetimeUTC(this.run().runEnd()));
dataservice.saveChanges();
// the test r === run() succeeds because the local run is a
// ko.observable which is bound to the run in the cache
var r = dataservice.getRunById(currentRunId());
},
...
})($, ko, app.dataservice, app.router);
myScripts.js中的
// Here is a real pain in the neck.
// For some reason, when the entity is saved, it shows up on the server as UTC datetime
// instead of local. Moment parses everything as local by default, so the formatDate function
// used to get a display value needs to be converted to utc before it is returned to the server.
//
// This function takes the value of the dependentObservable in the entity
// and converts it to a string which can be stored back into the entity before sending
// it to the server.
//
// The reason I need to do that is so that it displays properly after the save.
// The date seems to be handled properly by the server.
var makeDatetimeUTC = function(localDatetime) {
var datestring = formatDate(localDatetime);
var utc = moment.utc(datestring);
return formatDate(utc);
};
var formatDate = function(dateToFormat) {
if (dateToFormat === null ||dateToFormat === undefined || dateToFormat.length === 0)
return "";
// intermediate variable is not needed, but is good for debugging
var formattedDate = moment(dateToFormat).format('MM/DD/YYYY hh:mm A');
return formattedDate;
},
formatObservableDate = function(observable) {
if (ko.isObservable(observable))
return observable(formatDate(observable()));
else
throw new Error("formatObservableDate expected a ko.Observable ");
};