4
4
5
5
""" # Complete reference of resource definition
6
6
7
+ This section explains how to define a resource, its fields and its links. See
8
+ `BaseResource.jsonapi_dict()` for a documentation on exporting objects into
9
+ JSON:API format.
10
+
7
11
## Fields
8
12
9
13
### Attributes
@@ -135,9 +139,10 @@ class Meta:
135
139
```
136
140
137
141
A concrete resource definition without `"id"` member will raise an `AttributeError`.
138
- When a resource subclasses another resource, all fields are copied to the sub-resource.
142
+ When a resource subclasses another resource, all fields are copied to the sub-resource,
143
+ but the `Meta` inner class is not inherited, so a resource is concrete by default.
139
144
140
- ## Special class attributes
145
+ ## Accessing configuration and meta information about resources
141
146
142
147
These resource classes are used in the following examples.
143
148
@@ -157,7 +162,7 @@ class Meta:
157
162
"related__self": lambda x: f"/b/{x}/relationships/related"}
158
163
```
159
164
160
- ### Accessing resource metadata
165
+ ### Basic information about fields
161
166
162
167
Some metadata about a resource can be accessed through top level functions applied on
163
168
a resource class:
@@ -176,7 +181,25 @@ class Meta:
176
181
and refer to the following sections to know more about special metadata class
177
182
attributes.
178
183
179
- ### Configuration attributes defined in the `Meta` inner class
184
+ ### Atomic and relationships fields
185
+
186
+ When a field is a instance of `BaseResource`, it is considered as a
187
+ relationship field. The other fields are considered as "atomic": the `id` used
188
+ for identification, and the attributes that are exported in the `"attributes"`
189
+ object.
190
+
191
+ Some special attributes provide the sets of atomic and relationships fields names.
192
+
193
+ ```python
194
+ BResource.__fields_types__
195
+ # {'id': int, 'name': str, 'related': __main__.AResource}
196
+ BResource.__atomic_fields_set__
197
+ # {'id', 'name'}
198
+ BResource.__relationships_fields_set__
199
+ # {'related'}
200
+ ```
201
+
202
+ ### Configuration special attributes
180
203
181
204
Summary of attributes which can be defined in the `Meta` inner class:
182
205
@@ -193,24 +216,33 @@ class Meta:
193
216
During runtime, these metadata can be accessed with special attributes directly
194
217
on the resource classes. For example, the value of `is_abstract` is available
195
218
on the `__is_abstract__` attribute. The `Meta` inner class is not accessible
196
- during runtime.
219
+ during runtime. See the `BaseResourceMeta` metaclass for more information about
220
+ these attributes initialization.
197
221
198
- ### Atomic and relationships fields
222
+ ## Creating a resource class dynamically
199
223
200
- When a field is a instance of `BaseResource`, it is considered as a
201
- relationship field. The other fields are considered as "atomic": the `id` used
202
- for identification, and the attributes that are exported in the `"attributes"`
203
- object.
224
+ This module provides a `create_resource()` functions for creating resources
225
+ classes at runtime. We can recreate `BResource`:
204
226
205
- Some special attributes provide the sets of atomic and relationships fields names.
227
+ ```python
228
+ BResource = create_resource(
229
+ "BResource",
230
+ {"links_factories":
231
+ {"self": lambda x: f"/b/{x}",
232
+ "related__self": lambda x: f"/b/{x}/relationships/related"}},
233
+ id=int,
234
+ name=str,
235
+ related=AResource)
206
236
207
- ```pycon
208
- >>> BResource.__fields_types__
209
- {'id': int, 'name': str, 'related': __main__.AResource}
210
- >>> BResource.__atomic_fields_set__
211
- {'id', 'name'}
212
- >>> BResource.__relationships_fields_set__
213
- {'related'}
237
+ attributes_names(BResource)
238
+ # {'name'}
239
+ relationships_names(BResource)
240
+ # {'related'}
241
+ fields_types(BResource)
242
+ # {'id': int, 'name': str, 'related': __main__.AResource}
243
+ BResource.__links_factories__
244
+ # {'self': <function __main__.<lambda>(x)>,
245
+ # 'related__self': <function __main__.<lambda>(x)>}
214
246
```
215
247
"""
216
248
@@ -235,10 +267,13 @@ class Meta:
235
267
236
268
__all__ = ("BaseResource" , "create_resource" , "BaseResourceMeta" )
237
269
238
- IdType = TypeVar ("IdType" )
239
-
240
270
241
271
def _validate_link_name (klass , name ):
272
+ """Check if the link name is consistent with the resource class.
273
+
274
+ If the link name is a relationship-qualified name, check if the
275
+ relationship exists. Else raise a `ValueError`.
276
+ """
242
277
split_name = name .split ("__" )
243
278
if len (split_name ) > 1 :
244
279
relationship_name = split_name [0 ]
@@ -353,18 +388,18 @@ class BaseResource(metaclass=BaseResourceMeta):
353
388
__resource_name__ : str
354
389
__is_abstract__ : bool
355
390
__identifier_meta_attributes__ : Set [str ]
356
- __links_factories__ : Dict [str , Callable [[ IdType ] , str ]]
391
+ __links_factories__ : Dict [str , Callable [... , str ]]
357
392
_identifier_fields : Set [str ]
358
393
_forbidden_fields : Set [str ]
359
394
360
395
# must be provided by subclasses
361
- id : IdType
396
+ id : Any
362
397
363
398
class Meta :
364
399
is_abstract : bool = True
365
400
resource_name : str
366
401
identifier_meta_attributes : Set [str ]
367
- links_factories : Dict [str , Callable [[ IdType ] , str ]]
402
+ links_factories : Dict [str , Callable [... , str ]]
368
403
369
404
def __init__ (self , ** kwargs ):
370
405
"""Automatically set all passed arguments.
@@ -514,7 +549,7 @@ def dump(
514
549
)
515
550
516
551
@classmethod
517
- def register_link_factory (cls , name : str , factory : Callable [[ IdType ] , str ]):
552
+ def register_link_factory (cls , name : str , factory : Callable [... , str ]):
518
553
"""Add a link factory to the resource.
519
554
520
555
When the resources are dump, these factories are used to produce their
@@ -678,26 +713,6 @@ def _formatted_relationships(self, relationships: Dict) -> Dict:
678
713
)
679
714
return relationships_dict
680
715
681
- # def _relationship_dict(
682
- # self,
683
- # related_object: "BaseResource",
684
- # data_is_required: bool,
685
- # relationship_links: Set[str],
686
- # relationship_name
687
- # ):
688
- # """Make a single relationship object.
689
- #
690
- # Return a relationship object containing:
691
- # - data if needed
692
- # - links if needed
693
- # """
694
- # rel_data = {}
695
- # if data_is_required:
696
- # rel_data["data"] = related_object._identifier_dict
697
- # if relationship_links:
698
- # rel_data["links"] = self._make_links(relationship_links, relationship=relationship_name)
699
- # return rel_data
700
-
701
716
def _make_links (self ,
702
717
links : Mapping [str , Union [str , Dict [str , Any ]]],
703
718
relationship : Optional [str ] = None ):
@@ -734,8 +749,8 @@ def __getattr__(self, name):
734
749
735
750
def create_resource (
736
751
name : str ,
737
- meta_conf : Dict [str , Any ],
738
- bases : Tuple [type ] = (),
752
+ meta_conf : Optional [ Dict [str , Any ]] = None ,
753
+ bases : Tuple [type ] = (BaseResource , ),
739
754
metaklass : type = BaseResourceMeta ,
740
755
/ ,
741
756
** fields_types
@@ -744,12 +759,18 @@ def create_resource(
744
759
745
760
###### Parameters ######
746
761
762
+ Positional:
763
+
747
764
* `name`: The resource class name.
748
765
* `meta_conf`: A dictionary containg configuration attributes (of the
749
766
`Meta` inner class).
750
- * `bases`: A tuple containing parent classes.
767
+ * `bases`: A tuple containing parent classes. It must contain include
768
+ `BaseResource`.
751
769
* `metaklass`: The metaclass used to create the resource class (must
752
770
be a subclass of `BaseResourceMeta`).
771
+
772
+ Keywords:
773
+
753
774
* `**fields_types`: The types of the fields as keyword arguments.
754
775
755
776
###### Returned value ######
@@ -759,12 +780,17 @@ def create_resource(
759
780
###### Errors raised ######
760
781
761
782
A `TypeError` is raised if `metaklass` is not a subclass of `BaseResourceMeta`.
783
+ A `ValueError` is raised if `BaseResource` is not in `bases` argument.
762
784
"""
763
785
if not issubclass (metaklass , BaseResourceMeta ):
764
786
raise TypeError (
765
787
"Only a submetaclass of BaseResourceMeta can create a new "
766
788
f"resource class. ('{ metaklass } ' provided.)" )
789
+ if BaseResource not in bases :
790
+ raise ValueError (
791
+ "'BaseResource' class must be a parent class of any resource "
792
+ "class." )
767
793
768
- meta_inncer_class = type ("Meta" , (), meta_conf )
794
+ meta_inncer_class = type ("Meta" , (), meta_conf or {} )
769
795
namespace = {"__annotations__" : fields_types , "Meta" : meta_inncer_class }
770
796
return metaklass (name , bases , namespace )
0 commit comments