@@ -300,7 +300,7 @@ def get_text_score(self):
300
300
301
301
return self ._data ['_text_score' ]
302
302
303
- def to_mongo (self , use_db_field = True , fields = None ):
303
+ def to_mongo (self , use_db_field = True , fields = None , populate = False ):
304
304
"""
305
305
Return as SON data ready for use with MongoDB.
306
306
"""
@@ -310,11 +310,21 @@ def to_mongo(self, use_db_field=True, fields=None):
310
310
data = SON ()
311
311
data ["_id" ] = None
312
312
data ['_cls' ] = self ._class_name
313
-
313
+
314
+ Document = _import_class ("Document" )
315
+ EmbeddedDocumentField = _import_class ("EmbeddedDocumentField" )
316
+ ReferenceField = _import_class ("ReferenceField" )
317
+ ListField = _import_class ("ListField" )
318
+
314
319
# only root fields ['test1.a', 'test2'] => ['test1', 'test2']
315
320
root_fields = set ([f .split ('.' )[0 ] for f in fields ])
316
321
322
+ if populate :
323
+ populate_list = [ _field .split ('.' ) for _field in populate .split (',' )]
324
+ populate_domain = dict ((_field [0 ],_field [1 :]) for _field in populate_list )
325
+
317
326
for field_name in self :
327
+
318
328
if root_fields and field_name not in root_fields :
319
329
continue
320
330
@@ -323,22 +333,49 @@ def to_mongo(self, use_db_field=True, fields=None):
323
333
324
334
if field is None and self ._dynamic :
325
335
field = self ._dynamic_fields .get (field_name )
326
-
336
+
327
337
if value is not None :
328
- f_inputs = field .to_mongo .__code__ .co_varnames
329
- ex_vars = {}
330
- if fields and 'fields' in f_inputs :
331
- key = '%s.' % field_name
332
- embedded_fields = [
333
- i .replace (key , '' ) for i in fields
334
- if i .startswith (key )]
338
+ if populate and field_name in populate_domain .keys ():
339
+ if isinstance (field , ListField ):
340
+ _obj = []
341
+ for ref in value :
342
+ if isinstance (ref , DBRef ):
343
+ _ref_model = field .field .document_type
344
+
345
+ if isinstance (ref .id , ObjectId ):
346
+ ref_id = ref .id
347
+ _obj .append (_ref_model .objects .get (id = ref_id ).to_mongo (populate = '.' .join (populate_domain [field_name ])))
348
+ elif isinstance (ref , Document ):
349
+ _obj .append (ref .to_mongo (populate = '.' .join (populate_domain [field_name ])))
350
+ else :
351
+ _obj .append (ref )
352
+
353
+ value = _obj
354
+
355
+ if isinstance (field , ReferenceField ) :
356
+ _ref_model = field .document_type
357
+ try :
358
+ _obj = _ref_model .objects .get (id = value .id ).to_mongo (populate = '.' .join (populate_domain [field_name ]))
359
+ except :
360
+ _obj = None
361
+
362
+ value = _obj
363
+
364
+ else :
365
+ f_inputs = field .to_mongo .__code__ .co_varnames
366
+ ex_vars = {}
367
+ if fields and 'fields' in f_inputs :
368
+ key = '%s.' % field_name
369
+ embedded_fields = [
370
+ i .replace (key , '' ) for i in fields
371
+ if i .startswith (key )]
335
372
336
- ex_vars ['fields' ] = embedded_fields
373
+ ex_vars ['fields' ] = embedded_fields
337
374
338
- if 'use_db_field' in f_inputs :
339
- ex_vars ['use_db_field' ] = use_db_field
340
-
341
- value = field .to_mongo (value , ** ex_vars )
375
+ if 'use_db_field' in f_inputs :
376
+ ex_vars ['use_db_field' ] = use_db_field
377
+
378
+ value = field .to_mongo (value , ** ex_vars )
342
379
343
380
# Handle self generating fields
344
381
if value is None and field ._auto_gen :
@@ -350,9 +387,9 @@ def to_mongo(self, use_db_field=True, fields=None):
350
387
data [field .db_field ] = value
351
388
else :
352
389
data [field .name ] = value
390
+
353
391
354
392
# If "_id" has not been set, then try and set it
355
- Document = _import_class ("Document" )
356
393
if isinstance (self , Document ):
357
394
if data ["_id" ] is None :
358
395
data ["_id" ] = self ._data .get ("id" , None )
@@ -365,7 +402,8 @@ def to_mongo(self, use_db_field=True, fields=None):
365
402
not self ._meta .get ('allow_inheritance' , ALLOW_INHERITANCE )):
366
403
data .pop ('_cls' )
367
404
368
- return data
405
+ return data
406
+
369
407
370
408
def validate (self , clean = True ):
371
409
"""Ensure that all fields' values are valid and that required fields
@@ -416,9 +454,11 @@ def to_json(self, *args, **kwargs):
416
454
"""Converts a document to JSON.
417
455
:param use_db_field: Set to True by default but enables the output of the json structure with the field names
418
456
and not the mongodb store db_names in case of set to False
457
+ :param populate:
419
458
"""
459
+ populate = kwargs .pop ('populate' , False )
420
460
use_db_field = kwargs .pop ('use_db_field' , True )
421
- return json_util .dumps (self .to_mongo (use_db_field ), * args , ** kwargs )
461
+ return json_util .dumps (self .to_mongo (use_db_field , populate = populate ), * args , ** kwargs )
422
462
423
463
@classmethod
424
464
def from_json (cls , json_data , created = False ):
0 commit comments