Skip to content
Projects
Groups
Snippets
Help
This project
Loading...
Sign in / Register
Toggle navigation
D
django-rest-framework
Overview
Overview
Details
Activity
Cycle Analytics
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Charts
Issues
0
Issues
0
List
Board
Labels
Milestones
Merge Requests
0
Merge Requests
0
CI / CD
CI / CD
Pipelines
Jobs
Schedules
Charts
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Charts
Create a new issue
Jobs
Commits
Issue Boards
Open sidebar
edx
django-rest-framework
Commits
9fdb2280
Commit
9fdb2280
authored
Sep 18, 2014
by
Tom Christie
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
First pass on ManyRelation
parent
3bc628ed
Show whitespace changes
Inline
Side-by-side
Showing
3 changed files
with
89 additions
and
9 deletions
+89
-9
rest_framework/relations.py
+41
-1
rest_framework/utils/representation.py
+2
-0
tests/test_model_serializer.py
+46
-8
No files found.
rest_framework/relations.py
View file @
9fdb2280
...
@@ -10,7 +10,6 @@ from django.utils.translation import ugettext_lazy as _
...
@@ -10,7 +10,6 @@ from django.utils.translation import ugettext_lazy as _
class
RelatedField
(
Field
):
class
RelatedField
(
Field
):
def
__init__
(
self
,
**
kwargs
):
def
__init__
(
self
,
**
kwargs
):
self
.
queryset
=
kwargs
.
pop
(
'queryset'
,
None
)
self
.
queryset
=
kwargs
.
pop
(
'queryset'
,
None
)
self
.
many
=
kwargs
.
pop
(
'many'
,
False
)
assert
self
.
queryset
is
not
None
or
kwargs
.
get
(
'read_only'
,
None
),
(
assert
self
.
queryset
is
not
None
or
kwargs
.
get
(
'read_only'
,
None
),
(
'Relational field must provide a `queryset` argument, '
'Relational field must provide a `queryset` argument, '
'or set read_only=`True`.'
'or set read_only=`True`.'
...
@@ -21,6 +20,13 @@ class RelatedField(Field):
...
@@ -21,6 +20,13 @@ class RelatedField(Field):
)
)
super
(
RelatedField
,
self
)
.
__init__
(
**
kwargs
)
super
(
RelatedField
,
self
)
.
__init__
(
**
kwargs
)
def
__new__
(
cls
,
*
args
,
**
kwargs
):
# We override this method in order to automagically create
# `ManyRelation` classes instead when `many=True` is set.
if
kwargs
.
pop
(
'many'
,
False
):
return
ManyRelation
(
child_relation
=
cls
(
*
args
,
**
kwargs
))
return
super
(
RelatedField
,
cls
)
.
__new__
(
cls
,
*
args
,
**
kwargs
)
def
get_queryset
(
self
):
def
get_queryset
(
self
):
queryset
=
self
.
queryset
queryset
=
self
.
queryset
if
isinstance
(
queryset
,
QuerySet
):
if
isinstance
(
queryset
,
QuerySet
):
...
@@ -216,3 +222,37 @@ class SlugRelatedField(RelatedField):
...
@@ -216,3 +222,37 @@ class SlugRelatedField(RelatedField):
def
to_representation
(
self
,
obj
):
def
to_representation
(
self
,
obj
):
return
getattr
(
obj
,
self
.
slug_field
)
return
getattr
(
obj
,
self
.
slug_field
)
class
ManyRelation
(
Field
):
"""
Relationships with `many=True` transparently get coerced into instead being
a ManyRelation with a child relationship.
The `ManyRelation` class is responsible for handling iterating through
the values and passing each one to the child relationship.
You shouldn't need to be using this class directly yourself.
"""
def
__init__
(
self
,
child_relation
=
None
,
*
args
,
**
kwargs
):
self
.
child_relation
=
child_relation
assert
child_relation
is
not
None
,
'`child_relation` is a required argument.'
super
(
ManyRelation
,
self
)
.
__init__
(
*
args
,
**
kwargs
)
def
bind
(
self
,
field_name
,
parent
,
root
):
# ManyRelation needs to provide the current context to the child relation.
super
(
ManyRelation
,
self
)
.
bind
(
field_name
,
parent
,
root
)
self
.
child_relation
.
bind
(
field_name
,
parent
,
root
)
def
to_internal_value
(
self
,
data
):
return
[
self
.
child_relation
.
to_internal_value
(
item
)
for
item
in
data
]
def
to_representation
(
self
,
obj
):
return
[
self
.
child_relation
.
to_representation
(
value
)
for
value
in
obj
.
all
()
]
rest_framework/utils/representation.py
View file @
9fdb2280
...
@@ -73,6 +73,8 @@ def serializer_repr(serializer, indent, force_many=None):
...
@@ -73,6 +73,8 @@ def serializer_repr(serializer, indent, force_many=None):
ret
+=
serializer_repr
(
field
,
indent
+
1
)
ret
+=
serializer_repr
(
field
,
indent
+
1
)
elif
hasattr
(
field
,
'child'
):
elif
hasattr
(
field
,
'child'
):
ret
+=
list_repr
(
field
,
indent
+
1
)
ret
+=
list_repr
(
field
,
indent
+
1
)
elif
hasattr
(
field
,
'child_relation'
):
ret
+=
field_repr
(
field
.
child_relation
,
force_many
=
field
.
child_relation
)
else
:
else
:
ret
+=
field_repr
(
field
)
ret
+=
field_repr
(
field
)
return
ret
return
ret
...
...
tests/test_model_serializer.py
View file @
9fdb2280
...
@@ -199,7 +199,7 @@ class RelationalModel(models.Model):
...
@@ -199,7 +199,7 @@ class RelationalModel(models.Model):
class
TestRelationalFieldMappings
(
TestCase
):
class
TestRelationalFieldMappings
(
TestCase
):
def
test_
flat_relational_field
s
(
self
):
def
test_
pk_relation
s
(
self
):
class
TestSerializer
(
serializers
.
ModelSerializer
):
class
TestSerializer
(
serializers
.
ModelSerializer
):
class
Meta
:
class
Meta
:
model
=
RelationalModel
model
=
RelationalModel
...
@@ -214,7 +214,7 @@ class TestRelationalFieldMappings(TestCase):
...
@@ -214,7 +214,7 @@ class TestRelationalFieldMappings(TestCase):
"""
)
"""
)
self
.
assertEqual
(
repr
(
TestSerializer
()),
expected
)
self
.
assertEqual
(
repr
(
TestSerializer
()),
expected
)
def
test_nested_relation
al_field
s
(
self
):
def
test_nested_relations
(
self
):
class
TestSerializer
(
serializers
.
ModelSerializer
):
class
TestSerializer
(
serializers
.
ModelSerializer
):
class
Meta
:
class
Meta
:
model
=
RelationalModel
model
=
RelationalModel
...
@@ -238,7 +238,7 @@ class TestRelationalFieldMappings(TestCase):
...
@@ -238,7 +238,7 @@ class TestRelationalFieldMappings(TestCase):
"""
)
"""
)
self
.
assertEqual
(
repr
(
TestSerializer
()),
expected
)
self
.
assertEqual
(
repr
(
TestSerializer
()),
expected
)
def
test_
flat_hyperlinked_field
s
(
self
):
def
test_
hyperlinked_relation
s
(
self
):
class
TestSerializer
(
serializers
.
HyperlinkedModelSerializer
):
class
TestSerializer
(
serializers
.
HyperlinkedModelSerializer
):
class
Meta
:
class
Meta
:
model
=
RelationalModel
model
=
RelationalModel
...
@@ -253,7 +253,7 @@ class TestRelationalFieldMappings(TestCase):
...
@@ -253,7 +253,7 @@ class TestRelationalFieldMappings(TestCase):
"""
)
"""
)
self
.
assertEqual
(
repr
(
TestSerializer
()),
expected
)
self
.
assertEqual
(
repr
(
TestSerializer
()),
expected
)
def
test_nested_hyperlinked_
field
s
(
self
):
def
test_nested_hyperlinked_
relation
s
(
self
):
class
TestSerializer
(
serializers
.
HyperlinkedModelSerializer
):
class
TestSerializer
(
serializers
.
HyperlinkedModelSerializer
):
class
Meta
:
class
Meta
:
model
=
RelationalModel
model
=
RelationalModel
...
@@ -277,7 +277,7 @@ class TestRelationalFieldMappings(TestCase):
...
@@ -277,7 +277,7 @@ class TestRelationalFieldMappings(TestCase):
"""
)
"""
)
self
.
assertEqual
(
repr
(
TestSerializer
()),
expected
)
self
.
assertEqual
(
repr
(
TestSerializer
()),
expected
)
def
test_
flat
_reverse_foreign_key
(
self
):
def
test_
pk
_reverse_foreign_key
(
self
):
class
TestSerializer
(
serializers
.
ModelSerializer
):
class
TestSerializer
(
serializers
.
ModelSerializer
):
class
Meta
:
class
Meta
:
model
=
ForeignKeyTargetModel
model
=
ForeignKeyTargetModel
...
@@ -291,7 +291,7 @@ class TestRelationalFieldMappings(TestCase):
...
@@ -291,7 +291,7 @@ class TestRelationalFieldMappings(TestCase):
"""
)
"""
)
self
.
assertEqual
(
repr
(
TestSerializer
()),
expected
)
self
.
assertEqual
(
repr
(
TestSerializer
()),
expected
)
def
test_
flat
_reverse_one_to_one
(
self
):
def
test_
pk
_reverse_one_to_one
(
self
):
class
TestSerializer
(
serializers
.
ModelSerializer
):
class
TestSerializer
(
serializers
.
ModelSerializer
):
class
Meta
:
class
Meta
:
model
=
OneToOneTargetModel
model
=
OneToOneTargetModel
...
@@ -305,7 +305,7 @@ class TestRelationalFieldMappings(TestCase):
...
@@ -305,7 +305,7 @@ class TestRelationalFieldMappings(TestCase):
"""
)
"""
)
self
.
assertEqual
(
repr
(
TestSerializer
()),
expected
)
self
.
assertEqual
(
repr
(
TestSerializer
()),
expected
)
def
test_
flat
_reverse_many_to_many
(
self
):
def
test_
pk
_reverse_many_to_many
(
self
):
class
TestSerializer
(
serializers
.
ModelSerializer
):
class
TestSerializer
(
serializers
.
ModelSerializer
):
class
Meta
:
class
Meta
:
model
=
ManyToManyTargetModel
model
=
ManyToManyTargetModel
...
@@ -319,7 +319,7 @@ class TestRelationalFieldMappings(TestCase):
...
@@ -319,7 +319,7 @@ class TestRelationalFieldMappings(TestCase):
"""
)
"""
)
self
.
assertEqual
(
repr
(
TestSerializer
()),
expected
)
self
.
assertEqual
(
repr
(
TestSerializer
()),
expected
)
def
test_
flat
_reverse_through
(
self
):
def
test_
pk
_reverse_through
(
self
):
class
TestSerializer
(
serializers
.
ModelSerializer
):
class
TestSerializer
(
serializers
.
ModelSerializer
):
class
Meta
:
class
Meta
:
model
=
ThroughTargetModel
model
=
ThroughTargetModel
...
@@ -332,3 +332,41 @@ class TestRelationalFieldMappings(TestCase):
...
@@ -332,3 +332,41 @@ class TestRelationalFieldMappings(TestCase):
reverse_through = PrimaryKeyRelatedField(many=True, read_only=True)
reverse_through = PrimaryKeyRelatedField(many=True, read_only=True)
"""
)
"""
)
self
.
assertEqual
(
repr
(
TestSerializer
()),
expected
)
self
.
assertEqual
(
repr
(
TestSerializer
()),
expected
)
class
TestIntegration
(
TestCase
):
def
setUp
(
self
):
self
.
foreign_key_target
=
ForeignKeyTargetModel
.
objects
.
create
(
name
=
'foreign_key'
)
self
.
one_to_one_target
=
OneToOneTargetModel
.
objects
.
create
(
name
=
'one_to_one'
)
self
.
many_to_many_targets
=
[
ManyToManyTargetModel
.
objects
.
create
(
name
=
'many_to_many (
%
d)'
%
idx
)
for
idx
in
range
(
3
)
]
self
.
instance
=
RelationalModel
.
objects
.
create
(
foreign_key
=
self
.
foreign_key_target
,
one_to_one
=
self
.
one_to_one_target
,
)
self
.
instance
.
many_to_many
=
self
.
many_to_many_targets
self
.
instance
.
save
()
class
TestSerializer
(
serializers
.
ModelSerializer
):
class
Meta
:
model
=
RelationalModel
self
.
serializer_cls
=
TestSerializer
def
test_pk_relationship_representations
(
self
):
serializer
=
self
.
serializer_cls
(
self
.
instance
)
expected
=
{
'id'
:
self
.
instance
.
pk
,
'foreign_key'
:
self
.
foreign_key_target
.
pk
,
'one_to_one'
:
self
.
one_to_one_target
.
pk
,
'many_to_many'
:
[
item
.
pk
for
item
in
self
.
many_to_many_targets
],
'through'
:
[]
}
self
.
assertEqual
(
serializer
.
data
,
expected
)
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment