2012-01-04 57 views
4

回到零值,我有很多Mongoid可选字段,如:的Rails + Mongoid - 不要在JSON

field :key, type: String    
    field :element, type: String   
    field :rect, type: Array    

如果我回到这个模型的JSON其中只有一个充满我得到null所有其他字段的值。我如何删除这些字段?

我的模型具有嵌套属性,这意味着空值可以在几个级别上。

澄清:

我需要一种方法来移除模型的JSON表示空字段,包括所有嵌套的属性空字段。

代码示例

1.9.3-p0 :005 > u=Muse.new(:key=>'ram').to_json 
=> "{\"_id\":\"4f1ced749c2ee4219d000003\",\"element\":null,\"key\":\"ram\",\"rect\":null}" 
+1

一些示例数据/预期的输入/输出将在这里有所帮助 – Yule 2012-01-04 17:00:26

+0

在模型上调用to_json时,它可以正常工作。 JSON对象没有任何空的字段?你正在使用哪个版本的mongoid?并调用your_doc_obj.to_json它返回null? – RameshVel 2012-01-22 16:16:27

+0

它有空字段。这个例子中只有一个字段是填充的,另一个是空值,我想将它们去掉。我也想删除其他字段(但那些我将剥离:attr_protected) – CamelCamelCamel 2012-01-22 16:24:40

回答

5

默认情况下mongoid有权删除空字段的能力。如果你让一些字段为空,mongoid会在插入时删除它们。

在下面的示例中

,我离开了字段元件& RECT

class User 
    include Mongoid::Document 

    field :key, type: String    
    field :element, type: String   
    field :rect, type: Array  

    embeds_one :home 
end 
>> u=User.new(:key=>'ram').to_json 
=> "{"_id":"4f1c3722b356f82e4a000001","_type":"key":"ram"}" 

和它完美。但是,如果你在该字段中输入零值,那么它将被插入。假设这是你代码中的确切问题。所以,你可以解决此通过转换使用JSON as_json哈希表示并取出零领域

x=u.as_json.reject! {|k,v| v.nil?} 
=> "{"_id":"4f1c3722b356f82e4a000001","_type":"User","key":"ram"}" 

但去到内部的水平,则不能使用as_json。检查下面的代码

>>h=Home.new(:address=>'xxxx',:dummy=>nil) 
    >>u.home = h 
    >>x=u.as_json.reject! {|k,v| v.nil?} 
    =>{"_id"=>BSON::ObjectId('4f1c39b4b356f82e4a000003'), "_type"=>"User","key":"ram","home"=>#<Home _id: 4f1c3c5db356f82e4a000004,address:'xxxx' , dummy: nil >} 

现在你看到嵌入文档屋内场虚拟仍与零。所以我最好的建议是不要把db的零值全部写入。要做到这一点,在您的模型上嵌入before_save回调(嵌入),并删除空字段。

另外我会告诉你如何从嵌套对象中删除零字段。使用它,如果没有其他的办法

我们可以用mongoid模型attributes来获取对象的散列表示包括嵌套级别

x=u.attributes 
=> {"_id"=>BSON::ObjectId4f1c39b4b356f82e4a000003,"key"=>"ram","element"=>nil,"home"=>{"address"=>"xxxx", "_id"=>BSON::ObjectId4f1c3c5db356f82e4a000004,"dummy"=>nil}} 

,你必须发现有任何哈希mongoid内对象,如果有,我们必须使用reject! {|k,v| v.nil?}对哈希太

放在一起的所有

def to_json(obj) 
    obj.reject! {|k,v| v.nil?} 
    obj.find_all {|x| x[1].class==BSON::OrderedHash}.each do |arr| 
      obj[arr[0]] = to_json(arr[1])  
    end 
    obj  
end 

,并呼吁日是与模型的属性

>> to_json u.attributes 
=> {"_id"=>BSON::ObjectId4f1c39b4b356f82e4a000003,"key"=>"ram","home"=>{"address"=>"xxxx", "_id"=>BSON::ObjectId4f1c3c5db356f82e4a000004}} 

那就是所有。希望它有帮助

+0

谢谢!这听起来是合理的,递归哈希thingy是我最终做的,但它很烂,我提出了寻找另一种方式的赏金。我明天会测试它,但是这行看起来不正确:“”{“_id”:“4f1c3722b356f82e4a000001”,“_ type”:“key”,“fname”:“ram”}“” - 什么是fname? – CamelCamelCamel 2012-01-22 19:21:21

+0

@myxospsm,这是一个错字...我混合了我的原始测试数据:( – RameshVel 2012-01-22 20:25:44

+0

对不起,它不起作用。Mongoid自动将空值放入模型中,也许我会使用动态字段。问题) – CamelCamelCamel 2012-01-23 05:26:53

1

据我所知,在Mongoid自己没有这样的功能。 我的建议是去与API发电机中的一个,那么:

http://fabrik42.github.com/acts_as_api/

https://github.com/nesquena/rabl

https://github.com/rails/jbuilder

+0

我已经看过acts_as_api之前,但我不知道它是如何解决null清除问题 – CamelCamelCamel 2012-01-23 04:56:26

+0

你基本上有一些业务规则来编写JSON响应所以,除非你想手工编码(这当然是可能的),你最好使用这些解决方案之一,恕我直言。 – Roman 2012-01-23 19:07:42

2

如果你只是想删除与零场键在向浏览器发送JSON之前查询结果中的值,您可以简单地执行此操作(适用于我的情况):

doc = Model.all.entries.map {|d| d.as_document} 
respond_width(doc) 
0

如果您使用active_model_serializers(请查看this Railscast快速介绍),这很容易。

对于你的使用情况,您可以编写一个muse_serializer.rb,看起来像这样:

class MuseSerializer < ActiveModel::Serializer 
    def attributes 
    muse = super 
    object.attributes.each do |attr| 
     muse[attr[0]] = attr[1] 
    end 
    muse.delete('_id') #in case you want to exclude the _id field 
    muse 
    end 
end 

这将返回存在于当前缪斯对象的所有非空字段。