-
Notifications
You must be signed in to change notification settings - Fork 19
/
Copy pathtest_openapi_validation_errors.py
556 lines (424 loc) · 18.5 KB
/
test_openapi_validation_errors.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
import pytest
from django.contrib.auth.models import Group, User
from django.views.generic import UpdateView
from drf_spectacular.utils import extend_schema
from rest_framework import serializers
from rest_framework.decorators import action, api_view
from rest_framework.generics import DestroyAPIView, GenericAPIView, UpdateAPIView
from rest_framework.response import Response
from rest_framework.versioning import URLPathVersioning
from rest_framework.viewsets import ModelViewSet
from drf_standardized_errors.openapi_validation_errors import extend_validation_errors
from .utils import generate_versioned_view_schema, generate_view_schema, get_error_codes
class UserSerializer(serializers.ModelSerializer):
class Meta:
fields = ["first_name"]
model = User
@pytest.fixture
def viewset_with_extra_errors():
@extend_validation_errors(
["extra_error"], field_name="first_name", actions=["create"]
)
class ValidationViewSet(ModelViewSet):
serializer_class = UserSerializer
queryset = User.objects.all()
return ValidationViewSet
def test_extra_validation_errors_to_viewset(viewset_with_extra_errors):
"""simple test for using @extend_validation_errors with ViewSets"""
route = "validate/"
view = viewset_with_extra_errors.as_view({"post": "create"})
schema = generate_view_schema(route, view)
error_codes = get_error_codes(schema, "ValidateCreateFirstNameErrorComponent")
assert "extra_error" in error_codes
@pytest.fixture
def view_with_extra_errors():
@extend_validation_errors(["extra_error"], field_name="first_name", methods=["put"])
class ValidationView(UpdateAPIView):
serializer_class = UserSerializer
queryset = User.objects.all()
return ValidationView
def test_extra_validation_errors_to_view(view_with_extra_errors):
"""simple test for using @extend_validation_errors with APIViews"""
route = "validate/"
view = view_with_extra_errors.as_view()
schema = generate_view_schema(route, view)
error_codes = get_error_codes(schema, "ValidateUpdateFirstNameErrorComponent")
assert "extra_error" in error_codes
error_codes = get_error_codes(
schema, "ValidatePartialUpdateFirstNameErrorComponent"
)
assert "extra_error" not in error_codes
@pytest.fixture
def function_based_view_with_extra_errors():
@extend_validation_errors(
["extra_error"], field_name="first_name", methods=["post"]
)
@extend_schema(request=UserSerializer, responses={201: None})
@api_view(http_method_names=["post"])
def validate(request):
serializer = UserSerializer(data=request.data)
serializer.is_valid(raise_exception=True)
return Response(status=201)
return validate
def test_extra_validation_errors_to_function_based_api_view(
function_based_view_with_extra_errors,
):
route = "validate/"
view = function_based_view_with_extra_errors
schema = generate_view_schema(route, view)
error_codes = get_error_codes(schema, "ValidateCreateFirstNameErrorComponent")
assert "extra_error" in error_codes
@pytest.fixture
def validation_viewset():
class ValidationViewSet(ModelViewSet):
serializer_class = UserSerializer
queryset = User.objects.all()
return ValidationViewSet
def test_methods_case_sensitivity(validation_viewset):
"""make sure it doesn't matter if we pass 'post' or 'POST' or 'PosT'"""
extend_validation_errors(
["another_code"], field_name="first_name", methods=["PosT"]
)(validation_viewset)
route = "validate/"
view = validation_viewset.as_view({"post": "create"})
schema = generate_view_schema(route, view)
error_codes = get_error_codes(schema, "ValidateCreateFirstNameErrorComponent")
assert "another_code" in error_codes
@pytest.fixture
def function_based_view():
def get_users(request):
serializer = UserSerializer(instance=User.objects.all())
return Response(serializer.data)
return get_users
def test_decorating_non_api_view_functions(function_based_view, capsys):
extend_validation_errors(["new_code"])(function_based_view)
stderr = capsys.readouterr().err
assert "`@extend_validation_errors` can only be applied to APIViews" in stderr
@pytest.fixture
def django_class_based_view():
class UserView(UpdateView):
model = User
fields = ["first_name"]
return UserView
def test_decorating_non_api_view_classes(django_class_based_view, capsys):
extend_validation_errors(["new_code"])(django_class_based_view)
stderr = capsys.readouterr().err
assert "`@extend_validation_errors` can only be applied to APIViews" in stderr
def test_not_passing_error_codes(validation_viewset, capsys):
extend_validation_errors([])(validation_viewset)
stderr = capsys.readouterr().err
assert "No error codes are passed to the `@extend_validation_errors`" in stderr
def test_passing_field_name_as_none(validation_viewset):
extend_validation_errors(["some_code"], methods=["post"])(validation_viewset)
route = "validate/"
view = validation_viewset.as_view({"post": "create"})
schema = generate_view_schema(route, view)
error_codes = get_error_codes(schema, "ValidateCreateErrorComponent")
assert "some_code" in error_codes
def test_passing_incorrect_action(validation_viewset, capsys):
extend_validation_errors(["some_code"], actions=["no_action"])(validation_viewset)
stderr = capsys.readouterr().err
assert "not in the list of actions defined on the viewset" in stderr
@pytest.fixture
def function_based_api_view():
@extend_schema(request=UserSerializer, responses={201: None})
@api_view(http_method_names=["post"])
def validate(request):
serializer = UserSerializer(data=request.data)
serializer.is_valid(raise_exception=True)
return Response(status=201)
return validate
def test_passing_action_for_api_view(function_based_api_view, capsys):
extend_validation_errors(["some_error"], actions=["some_action"])(
function_based_api_view
)
stderr = capsys.readouterr().err
warning_msg = (
"The 'actions' argument of 'extend_validation_errors' should "
"only be set when decorating viewsets."
)
assert warning_msg in stderr
@pytest.fixture
def validation_view():
class ValidationView(UpdateAPIView):
serializer_class = UserSerializer
queryset = User.objects.all()
return ValidationView
def test_passing_incorrect_method(validation_view, capsys):
extend_validation_errors(["some_code"], methods=["get"])(validation_view)
stderr = capsys.readouterr().err
assert "not in the list of allowed http methods" in stderr
def test_passing_multiple_actions(validation_viewset):
extend_validation_errors(
["some_error"], field_name="first_name", actions=["create", "partial_update"]
)(validation_viewset)
route = "validate/"
view = validation_viewset.as_view(
{"post": "create", "put": "update", "patch": "partial_update"}
)
schema = generate_view_schema(route, view)
error_codes = get_error_codes(schema, "ValidateCreateFirstNameErrorComponent")
assert "some_error" in error_codes
error_codes = get_error_codes(
schema, "ValidatePartialUpdateFirstNameErrorComponent"
)
assert "some_error" in error_codes
error_codes = get_error_codes(schema, "ValidateUpdateFirstNameErrorComponent")
assert "some_error" not in error_codes
def test_passing_actions_as_none(validation_viewset):
extend_validation_errors(["some_error"], field_name="first_name")(
validation_viewset
)
route = "validate/"
view = validation_viewset.as_view(
{"post": "create", "put": "update", "patch": "partial_update"}
)
schema = generate_view_schema(route, view)
error_codes = get_error_codes(schema, "ValidateCreateFirstNameErrorComponent")
assert "some_error" in error_codes
error_codes = get_error_codes(
schema, "ValidatePartialUpdateFirstNameErrorComponent"
)
assert "some_error" in error_codes
error_codes = get_error_codes(schema, "ValidateUpdateFirstNameErrorComponent")
assert "some_error" in error_codes
def test_passing_multiple_methods(validation_viewset):
extend_validation_errors(
["some_error"], field_name="first_name", methods=["post", "put"]
)(validation_viewset)
route = "validate/"
view = validation_viewset.as_view(
{"post": "create", "put": "update", "patch": "partial_update"}
)
schema = generate_view_schema(route, view)
error_codes = get_error_codes(schema, "ValidateCreateFirstNameErrorComponent")
assert "some_error" in error_codes
error_codes = get_error_codes(schema, "ValidateUpdateFirstNameErrorComponent")
assert "some_error" in error_codes
error_codes = get_error_codes(
schema, "ValidatePartialUpdateFirstNameErrorComponent"
)
assert "some_error" not in error_codes
def test_passing_methods_as_none(validation_viewset):
extend_validation_errors(["some_error"], field_name="first_name")(
validation_viewset
)
route = "validate/"
view = validation_viewset.as_view(
{"post": "create", "put": "update", "patch": "partial_update"}
)
schema = generate_view_schema(route, view)
error_codes = get_error_codes(schema, "ValidateCreateFirstNameErrorComponent")
assert "some_error" in error_codes
error_codes = get_error_codes(schema, "ValidateUpdateFirstNameErrorComponent")
assert "some_error" in error_codes
error_codes = get_error_codes(
schema, "ValidatePartialUpdateFirstNameErrorComponent"
)
assert "some_error" in error_codes
@pytest.fixture
def versioned_view():
class ValidationView(UpdateAPIView):
serializer_class = UserSerializer
queryset = User.objects.all()
versioning_class = URLPathVersioning
return ValidationView
def test_passing_multiple_versions(versioned_view):
extend_validation_errors(
["some_error"], field_name="first_name", versions=["v1", "v2"]
)(versioned_view)
view = versioned_view.as_view()
versioned_schema = generate_versioned_view_schema(view, "v1")
error_codes = get_error_codes(
versioned_schema, "V1ValidateUpdateFirstNameErrorComponent"
)
assert "some_error" in error_codes
versioned_schema = generate_versioned_view_schema(view, "v2")
error_codes = get_error_codes(
versioned_schema, "V2ValidateUpdateFirstNameErrorComponent"
)
assert "some_error" in error_codes
versioned_schema = generate_versioned_view_schema(view, "v3")
error_codes = get_error_codes(
versioned_schema, "V3ValidateUpdateFirstNameErrorComponent"
)
assert "some_error" not in error_codes
def test_passing_versions_as_none(versioned_view):
extend_validation_errors(["some_error"], field_name="first_name")(versioned_view)
view = versioned_view.as_view()
versioned_schema = generate_versioned_view_schema(view, "v1")
error_codes = get_error_codes(
versioned_schema, "V1ValidateUpdateFirstNameErrorComponent"
)
assert "some_error" in error_codes
versioned_schema = generate_versioned_view_schema(view, "v2")
error_codes = get_error_codes(
versioned_schema, "V2ValidateUpdateFirstNameErrorComponent"
)
assert "some_error" in error_codes
versioned_schema = generate_versioned_view_schema(view, "v3")
error_codes = get_error_codes(
versioned_schema, "V3ValidateUpdateFirstNameErrorComponent"
)
assert "some_error" in error_codes
def test_applying_decorator_multiple_times(validation_view):
"""all error codes should be added to corresponding fields"""
extend_first_name_errors = extend_validation_errors(
["short_name"], field_name="first_name"
)
extend_non_field_errors = extend_validation_errors(
["some_error"], field_name="non_field_errors"
)
extend_non_field_errors(extend_first_name_errors(validation_view))
route = "validate/"
view = validation_view.as_view()
schema = generate_view_schema(route, view)
error_codes = get_error_codes(schema, "ValidateUpdateFirstNameErrorComponent")
assert "short_name" in error_codes
error_codes = get_error_codes(schema, "ValidateUpdateNonFieldErrorsErrorComponent")
assert "some_error" in error_codes
def test_applying_decorator_multiple_times_same_field(validation_viewset):
"""only second_error should appear in the resulting schema"""
add_first_error = extend_validation_errors(["first_error"], field_name="first_name")
add_second_error = extend_validation_errors(
["second_error"], field_name="first_name"
)
add_second_error(add_first_error(validation_viewset))
route = "validate/"
view = validation_viewset.as_view({"post": "create"})
schema = generate_view_schema(route, view)
error_codes = get_error_codes(schema, "ValidateCreateFirstNameErrorComponent")
assert "second_error" in error_codes
@pytest.fixture
def child_viewset():
@extend_validation_errors(["parent_error"], field_name="first_name")
class ParentViewSet(ModelViewSet):
serializer_class = UserSerializer
queryset = User.objects.all()
class ChildViewSet(ParentViewSet):
pass
return ChildViewSet
def test_inherited_validation_errors(child_viewset):
"""
errors defined on a parent are found on the child and parent errors are
not affected
"""
extend_validation_errors(["child_error"], field_name="non_field_errors")(
child_viewset
)
route = "validate/"
view = child_viewset.as_view({"post": "create"})
schema = generate_view_schema(route, view)
error_codes = get_error_codes(schema, "ValidateCreateFirstNameErrorComponent")
assert "parent_error" in error_codes
error_codes = get_error_codes(schema, "ValidateCreateNonFieldErrorsErrorComponent")
assert "child_error" in error_codes
def test_overriding_inherited_validation_errors(child_viewset):
extend_validation_errors(["child_error"], field_name="first_name")(child_viewset)
route = "validate/"
view = child_viewset.as_view({"post": "create"})
schema = generate_view_schema(route, view)
error_codes = get_error_codes(schema, "ValidateCreateFirstNameErrorComponent")
assert "child_error" in error_codes
assert "parent_error" not in error_codes
@pytest.fixture
def delete_view():
class ValidationView(DestroyAPIView):
serializer_class = UserSerializer
queryset = User.objects.all()
return ValidationView
def test_extra_validation_errors_for_unexpected_method(delete_view):
"""
Test that it is possible to add validation errors even for delete even though
validation errors are auto-generated only for post,put,patch or get on a list action
"""
extend_validation_errors(
["some_error"], field_name="first_name", methods=["delete"]
)(delete_view)
route = "validate/"
view = delete_view.as_view()
schema = generate_view_schema(route, view)
error_codes = get_error_codes(schema, "ValidateDestroyFirstNameErrorComponent")
assert "some_error" in error_codes
@pytest.fixture
def viewset_with_custom_action():
class CustomActionViewSet(ModelViewSet):
serializer_class = UserSerializer
queryset = User.objects.all()
@action(methods=["get"], detail=False)
def fetch_superusers(self, request, *args, **kwargs):
serializer = UserSerializer(instance=User.objects.filter(is_superuser=True))
return Response(serializer.data)
return CustomActionViewSet
def test_extra_validation_errors_for_unexpected_action(viewset_with_custom_action):
"""
Test that it is possible to add validation errors even for get on custom action
even though validation errors are auto-generated only for post,put,patch or get
on a list action
"""
extend_validation_errors(["some_error"], field_name="first_name", methods=["get"])(
viewset_with_custom_action
)
route = "superusers/"
view = viewset_with_custom_action.as_view({"get": "fetch_superusers"})
schema = generate_view_schema(route, view)
error_codes = get_error_codes(schema, "SuperusersRetrieveFirstNameErrorComponent")
assert "some_error" in error_codes
@pytest.fixture
def viewset_with_nested_serializer():
class GroupSerializer(serializers.ModelSerializer):
class Meta:
fields = ["name"]
model = Group
class UserSerializer(serializers.ModelSerializer):
groups = GroupSerializer(many=True)
class Meta:
fields = ["first_name", "groups"]
model = User
class NestedViewSet(ModelViewSet):
serializer_class = UserSerializer
queryset = User.objects.all()
return NestedViewSet
def test_extra_validation_errors_for_nested_list_serializer_field(
viewset_with_nested_serializer,
):
extend_validation_errors(["some_error"], field_name="groups.INDEX.name")(
viewset_with_nested_serializer
)
route = "validate/"
view = viewset_with_nested_serializer.as_view({"post": "create"})
schema = generate_view_schema(route, view)
error_codes = get_error_codes(schema, "ValidateCreateGroupsINDEXNameErrorComponent")
assert "some_error" in error_codes
def test_pattern_for_list_serializer_field(viewset_with_nested_serializer):
route = "validate/"
view = viewset_with_nested_serializer.as_view({"post": "create"})
schema = generate_view_schema(route, view)
attr = schema["components"]["schemas"][
"ValidateCreateGroupsINDEXNameErrorComponent"
]["properties"]["attr"]
assert attr["pattern"] == r"groups\.\d+\.name"
@pytest.fixture
def list_dict_fields_view():
class SomeSerializer(serializers.Serializer):
field1 = serializers.DictField(child=serializers.IntegerField())
field2 = serializers.ListField(child=serializers.IntegerField())
class SomeView(GenericAPIView):
serializer_class = SomeSerializer
def post(self, request, *args, **kwargs):
serializer = self.get_serializer(data=request.data)
serializer.is_valid(raise_exception=True)
return Response(serializer.data)
return SomeView
def test_pattern_for_list_dict_fields(list_dict_fields_view):
route = "validate/"
view = list_dict_fields_view.as_view()
schema = generate_view_schema(route, view)
dict_attr = schema["components"]["schemas"][
"ValidateCreateField1KEYErrorComponent"
]["properties"]["attr"]
assert dict_attr["pattern"] == r"field1\..+"
list_attr = schema["components"]["schemas"][
"ValidateCreateField2INDEXErrorComponent"
]["properties"]["attr"]
assert list_attr["pattern"] == r"field2\.\d+"