-
Notifications
You must be signed in to change notification settings - Fork 134
/
Copy pathservices.py
87 lines (65 loc) · 3.21 KB
/
services.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
from typing import Any, Dict, List, Tuple
from django.db import models
from django.utils import timezone
from styleguide_example.common.types import DjangoModelType
def model_update(
*, instance: DjangoModelType, fields: List[str], data: Dict[str, Any], auto_updated_at=True
) -> Tuple[DjangoModelType, bool]:
"""
Generic update service meant to be reused in local update services.
For example:
def user_update(*, user: User, data) -> User:
fields = ['first_name', 'last_name']
user, has_updated = model_update(instance=user, fields=fields, data=data)
// Do other actions with the user here
return user
Return value: Tuple with the following elements:
1. The instance we updated.
2. A boolean value representing whether we performed an update or not.
Some important notes:
- Only keys present in `fields` will be taken from `data`.
- If something is present in `fields` but not present in `data`, we simply skip.
- There's a strict assertion that all values in `fields` are actual fields in `instance`.
- `fields` can support m2m fields, which are handled after the update on `instance`.
- If `auto_updated_at` is True, we'll try bumping `updated_at` with the current timestmap.
"""
has_updated = False
m2m_data = {}
update_fields = []
model_fields = {field.name: field for field in instance._meta.get_fields()}
for field in fields:
# Skip if a field is not present in the actual data
if field not in data:
continue
# If field is not an actual model field, raise an error
model_field = model_fields.get(field)
assert model_field is not None, f"{field} is not part of {instance.__class__.__name__} fields."
# If we have m2m field, handle differently
if isinstance(model_field, models.ManyToManyField):
m2m_data[field] = data[field]
continue
if getattr(instance, field) != data[field]:
has_updated = True
update_fields.append(field)
setattr(instance, field, data[field])
# Perform an update only if any of the fields were actually changed
if has_updated:
if auto_updated_at:
# We want to take care of the `updated_at` field,
# Only if the models has that field
# And if no value for updated_at has been provided
if "updated_at" in model_fields and "updated_at" not in update_fields:
update_fields.append("updated_at")
instance.updated_at = timezone.now() # type: ignore
instance.full_clean()
# Update only the fields that are meant to be updated.
# Django docs reference:
# https://docs.djangoproject.com/en/dev/ref/models/instances/#specifying-which-fields-to-save
instance.save(update_fields=update_fields)
for field_name, value in m2m_data.items():
related_manager = getattr(instance, field_name)
related_manager.set(value)
# Still not sure about this.
# What if we only update m2m relations & nothing on the model? Is this still considered as updated?
has_updated = True
return instance, has_updated