Ara*_*ier 5 python database django postgresql django-rest-framework
我试图在 PostgreSQL 数据库中存储大量的交通数据,其中客户端一次上传多条记录(数万条)。我只想要每个站点、每条路线、每次行程的一个到达时间,并且该站点、路线和行程的唯一标识符是我的表的主键(以及不同表中的外键)。我正在尝试在序列化器中使用 Django 的 update_or_create 来创建该到达时间的条目,或者使用最新数据更新到达时间(如果已存在)。不幸的是,在我的序列化器上调用 is_valid() 时,它识别出重复记录违反了主键的唯一性约束(这是有道理的),并给出了以下错误消息:
'real_id': [ErrorDetail(string='actual with this real id already exists.', code='unique')]
Run Code Online (Sandbox Code Playgroud)
我想覆盖此行为,因为如果主键不唯一,它只会更新条目。我已经尝试过循环并删除验证器,如下所示:
def run_validators(self, value):
for validator in self.validators:
self.validators.remove(validator)
super(ActualSerializer, self).run_validators(value)
Run Code Online (Sandbox Code Playgroud)
我还尝试使用extra_kwargs
序列化器元类的字段删除所有验证器,如下所示:
extra_kwargs = {
'name': {'validators': []}
}
Run Code Online (Sandbox Code Playgroud)
我真的不想在我的视图中进行插入,因为理想情况下序列化器正在解析值并验证其他约束。
但这些解决方案都没有改变任何事情。我找不到任何其他可以解决我的问题的东西(但如果有人找到我错过的答案,那将是天赐之物)。我认为我的问题与这个问题有点相似,但那个问题没有一个好的答案。
作为参考,我的序列化器文件:
from .models import Future, Actual
from rest_framework import serializers
class ActualSerializer(serializers.ModelSerializer):
def create(self, validated_data):
actual, created = Actual.objects.update_or_create(
real_id=validated_data['real_id'],
defaults={'arrival': validated_data.get('arrival', None)}
)
return actual
class Meta:
model = Actual
fields = ['real_id', 'arrival']
class FutureSerializer(serializers.ModelSerializer):
class Meta:
model = Future
fields = ['row_id', 'stop_id', 'route_id', 'record_time', 'predicted_arrival', 'delay', 'real_id']
Run Code Online (Sandbox Code Playgroud)
我的模型:
from django.db import models
import json
class Actual(models.Model):
real_id = models.CharField(max_length=100, primary_key=True)
arrival = models.BigIntegerField(null=True)
def __str__(self):
return json.dumps([self.real_id, self.arrival])
class Future(models.Model):
row_id = models.BigAutoField(primary_key=True)
stop_id = models.CharField(max_length=20)
route_id = models.CharField(max_length=10)
record_time = models.BigIntegerField()
predicted_arrival = models.IntegerField(null=True)
delay = models.IntegerField()
real_id = models.ForeignKey(Actual, on_delete=models.CASCADE)
def __str__(self):
return json.dumps([self.row_id,
self.stop_id,
self.route_id,
self.record_time,
self.predicted_arrival,
self.delay,
self.real_id])
Run Code Online (Sandbox Code Playgroud)
最后,我的观点:
class ArrivalData(APIView):
queryset = Future.objects.all()
permission_classes = [OperationAllowed]
def post(self, request: Request, format=None) -> Response:
parsed_data = json.loads(request.data['data'])
actual_arrivals = []
for datum in parsed_data:
saved_time = None
if datum['predicted_arrival'] is not None and -200 < datum['predicted_arrival'] - datum['record_time'] < 0:
saved_time = datum['predicted_arrival']
actual_arrivals.append({
'real_id': datum['real_id'],
'arrival': saved_time,
})
actual_serializer = ActualSerializer(data=actual_arrivals, many=True)
if actual_serializer.is_valid(raise_exception=False):
actual_serializer.save()
future_serializer = FutureSerializer(data=parsed_data, many=True)
if future_serializer.is_valid():
future_serializer.save()
return Response("Data saved")
return Response(str(actual_serializer.errors))
Run Code Online (Sandbox Code Playgroud)
因此,事实证明,唯一约束是字段级验证器,这就是为什么尝试在类级别删除它不起作用。
我在序列化器类中显式声明了该字段,没有任何验证器real_id = serializers.CharField(validators=[])
,这解决了问题。
如果其他人也遇到此问题,此页面最终对我有所帮助。
归档时间: |
|
查看次数: |
4929 次 |
最近记录: |