Django自定义YamlField实现过程解析


Posted in Python onNovember 11, 2020

需求

在使用django admin时希望后台的Textarea多行文本框可以按yaml格式编写,数据库保存为Text文本类型,字段和接口中读取出来自动变为字典或列表格式。

试过pip install django-yamlfied,修改支持新版django之后

接口中返回的字段是字符串形式,不符合预期。

之前写过一版。

import yaml
from django.db import models

class YamlField(models.TextField):
  def to_python(self, value): # 将数据库内容转为python对象时调用
    if not value:
      value = {}
    if isinstance(value, (list, dict)):
      return value
    return yaml.safe_load(value)

  def get_prep_value(self, value): # create时插入数据, 转为字符串存储
    return value if value is None else yaml.dump(value, default_flow_style=False)

  def from_db_value(self, value, expression, connection): # 从数据库读取字段是调用
    return self.to_python(value)

问题是输入框输入

- a
- b
- c

保存后就会变成字典的字符串形式

['a','b','c']

无法原样保存,反复研究后,参考django-jsonfield写了一版。

原理是,改为继承models.Field类,(继承models.TextField类,则formfield和value_to_string不生效)

数据库依旧将数据库中的yaml文本转为dict/list,在django admin中通过自定义widget显示为yaml字符串格式。

为了保存时,验证表单中yaml字符串格式是否正确,还需要自定义一个form。完整代码如下。

import django
from django.db import models
from django import forms
from django.core.exceptions import ValidationError
import yaml


class YamlWidget(forms.Textarea):
  def render(self, name, value, attrs=None, renderer=None):
    if value is None:
      value = ""
    if not isinstance(value, str):
      value = yaml.safe_dump(value, default_flow_style=False)
    if django.VERSION < (2, 0):
      return super().render(name, value, attrs)
    return super().render(name, value, attrs, renderer)


class YamlFormField(forms.CharField):
  empty_values = [None, '']

  def __init__(self, *args, **kwargs):
    if 'widget' not in kwargs:
      kwargs['widget'] = YamlWidget
    super().__init__(*args, **kwargs)

  def to_python(self, value):
    if isinstance(value, str) and value:
      try:
        return yaml.safe_load(value)
      except Exception as exc:
        raise forms.ValidationError('Yaml decode error: %s' % (exc.args[0],))
    else:
      return value

  def validate(self, value):
    if value in self.empty_values and self.required:
      raise forms.ValidationError(self.error_messages['required'], code='required')


class YamlField(models.Field):
  description = "Yaml object"

  def get_internal_type(self):
    return 'TextField'

  def formfield(self, **kwargs):
    defaults = {
      'form_class': YamlFormField,
      'widget': YamlWidget
    }
    defaults.update(**kwargs)
    return super().formfield(**defaults)

  def to_python(self, value: str): # 将数据库内容转为python对象时调用
    if value is None:
      if not self.null and self.blank:
        return ""
      return None
    if isinstance(value, (list, dict)):
      return value
    value = yaml.safe_load(value)
    return value

  def validate(self, value, model_instance): # 验证从接受到字典格式
    if not self.null and value is None:
      raise ValidationError(self.error_messages['null'])
    try:
      self.get_prep_value(value)
    except ValueError:
      raise ValidationError(self.error_messages['invalid'] % value)

  def get_prep_value(self, value: (list, dict)): # 保存时插入数据, 转为字符串存储
    if value is None:
      return None
    value = yaml.safe_dump(value, default_flow_style=False)
    return value

  def from_db_value(self, value: str, expression, connection, *args, **kwargs): # 从数据库读取字段是调用
    return self.to_python(value)

  def value_to_string(self, obj): # Rest Framework调用时
    return self.value_from_object(obj)

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持三水点靠木。

Python 相关文章推荐
Python基于有道实现英汉字典功能
Jul 25 Python
Python中异常重试的解决方案详解
May 05 Python
python扫描proxy并获取可用代理ip的实例
Aug 07 Python
Python网络编程 Python套接字编程
Sep 13 Python
浅谈Django自定义模板标签template_tags的用处
Dec 20 Python
python3.X 抓取火车票信息【修正版】
Jun 19 Python
python运行时强制刷新缓冲区的方法
Jan 14 Python
python实现诗歌游戏(类继承)
Feb 26 Python
Python实用工具FuckIt.py介绍
Jul 02 Python
Python实现AES加密,解密的两种方法
Oct 03 Python
python实现数据结构中双向循环链表操作的示例
Oct 09 Python
baselines示例程序train_cartpole.py的ImportError
May 20 Python
Python监听剪切板实现方法代码实例
Nov 11 #Python
如何通过python计算圆周率PI
Nov 11 #Python
python中turtle库的简单使用教程
Nov 11 #Python
python 怎样进行内存管理
Nov 10 #Python
python tqdm实现进度条的示例代码
Nov 10 #Python
python 解决Windows平台上路径有空格的问题
Nov 10 #Python
Python在后台自动解压各种压缩文件的实现方法
Nov 10 #Python
You might like
php对gzip文件或者字符串解压实例参考
2008/07/25 PHP
dhtmlxTree目录树增加右键菜单以及拖拽排序的实现方法
2013/04/26 PHP
php+ajax实现的点击浏览量加1
2015/04/16 PHP
PHP与SQL语句常用大全
2016/12/10 PHP
php实现简单的守护进程创建、开启与关闭操作
2019/08/13 PHP
JavaScript 入门基础知识 想学习js的朋友可以参考下
2009/12/26 Javascript
JavaScript加强之自定义event事件
2013/09/21 Javascript
JS获取Table中td值的方法
2015/03/19 Javascript
js实现温度计时间样式代码分享
2015/08/21 Javascript
浅谈$('div a') 与$('div&gt;a')的区别
2016/07/18 Javascript
JavaScript中的splice方法用法详解
2016/07/20 Javascript
基于Vue实现tab栏切换内容不断实时刷新数据功能
2017/04/13 Javascript
Vue.2.0.5实现Class 与 Style 绑定的实例
2017/06/20 Javascript
解决vue 项目引入字体图标报错、不显示等问题
2018/09/01 Javascript
vuex2中使用mapGetters/mapActions报错的解决方法
2018/10/20 Javascript
vue实现的下拉框功能示例
2019/01/29 Javascript
基于Proxy的小程序状态管理实现
2019/06/14 Javascript
[56:21]LGD vs IG 2018国际邀请赛小组赛BO2 第二场 8.18
2018/08/19 DOTA
[01:01:18]VP vs NIP 2019国际邀请赛小组赛 BO2 第二场 8.15
2019/08/17 DOTA
Python作用域用法实例详解
2016/03/15 Python
基于python 处理中文路径的终极解决方法
2018/04/12 Python
Python实现的批量修改文件后缀名操作示例
2018/12/07 Python
Python实现合并两个有序链表的方法示例
2019/01/31 Python
python2.7实现复制大量文件及文件夹资料
2019/08/31 Python
python FTP批量下载/删除/上传实例
2019/12/22 Python
Python中使用socks5设置全局代理的方法示例
2020/04/15 Python
可视化pytorch 模型中不同BN层的running mean曲线实例
2020/06/24 Python
html5的自定义data-*属性与jquery的data()方法的使用
2014/07/02 HTML / CSS
德国团购网站:Groupon德国
2018/03/13 全球购物
Notino法国:购买香水和化妆品
2019/04/15 全球购物
暑期社会实践感言
2014/02/25 职场文书
小学少先队工作总结2015
2015/05/26 职场文书
2016年社会管理综治宣传月活动总结
2016/03/16 职场文书
Python WSGI 规范简介
2021/04/11 Python
Redis源码阅读:Redis字符串SDS详解
2021/07/15 Redis
Oracle表空间与权限的深入讲解
2021/11/17 Oracle