91超碰碰碰碰久久久久久综合_超碰av人澡人澡人澡人澡人掠_国产黄大片在线观看画质优化_txt小说免费全本

溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

使用Django如何實現自定義YamlField

發布時間:2020-11-12 15:03:40 來源:億速云 閱讀:240 作者:Leah 欄目:開發技術

本篇文章給大家分享的是有關使用Django如何實現自定義YamlField,小編覺得挺實用的,因此分享給大家學習,希望大家閱讀完這篇文章后可以有所收獲,話不多說,跟著小編一起來看看吧。

需求

在使用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)

以上就是使用Django如何實現自定義YamlField,小編相信有部分知識點可能是我們日常工作會見到或用到的。希望你能通過這篇文章學到更多知識。更多詳情敬請關注億速云行業資訊頻道。

向AI問一下細節

免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。

AI

阿合奇县| 郑州市| 高要市| 常熟市| 襄樊市| 宾川县| 青龙| 当雄县| 陆河县| 黑河市| 庐江县| 青岛市| 本溪市| 石门县| 浪卡子县| 渝北区| 武川县| 万安县| 新宁县| 白朗县| 库伦旗| 文成县| 英山县| 化州市| 江津市| 清丰县| 邹城市| 镇远县| 凤山县| 浦北县| 岳阳县| 芦溪县| 东乌| 定安县| 五莲县| 浦东新区| 青浦区| 江北区| 泗水县| 噶尔县| 徐汇区|