Escrevendo campos personalizados de modelos.¶
Introdução¶
A documentação referência de modelo explica como usar as classes de campo padrão do Django –CharField
, DateField
, etc. Para muitos propósitos, essas classes são tudo o que você irá precisar. Às vezes, porém, a versão do Django não irá atender às suas necessidades específicas, ou você irá querer usar um campo que é totalmente diferente daqueles fornecidos com Django.
Os tipos de campos embutidos no Django não cobrem todas os possíveis tipos de colunas do banco de dados – somente os tipos comuns, tais como VARCHAR``e `ÌNTEGER
. Para tipos de colunas mais obscuros, tal como polígonos geográficos ou ainda tipos criados pelo usuário como `PostgreSQL custom Types`__, é possível definir sua própria subclasse de Field
do Django.
Por outro lado, pode haver um tipo complexo de objeto no Python que pode de alguma maneira ser serializado para que se encaixe em um tipo de coluna padrão no banco de dados. Este é um outro caso onde uma subclasse de Field
pode ajudar a utilizar o objeto com o “models”.
Nosso objeto de exemplo¶
Criar campos personalizados requer um pouco de atenção. Para que seja mais fácil de acompanhar, usaremos um exemplo consistente através desta documentação: um objeto Python que represente a coleção de cartas de uma mão de Bridge. Não se preocupe, não é necessário saber como jogar Bridge para entender o exemplo. É necessário saber que 52 cartas são distribuídas igualmente para 4 jogadores, os quais são tradicionalmente chamados north (norte), east (leste), south (sul) and west (oeste). Nossa classe se parece com isso:
class Hand:
"""A hand of cards (bridge style)"""
def __init__(self, north, east, south, west):
# Input parameters are lists of cards ('Ah', '9s', etc.)
self.north = north
self.east = east
self.south = south
self.west = west
# ... (other possibly useful methods omitted) ...
Esta é somente uma classe Python comum, não tem nada específico do Django. Gostaríamos de ser capazes de fazer coisas como essa no nosso modelo (assumimos que o atributo hand
do modelo é uma instância de Hand
):
example = MyModel.objects.get(pk=1)
print(example.hand.north)
new_hand = Hand(north, east, south, west)
example.hand = new_hand
example.save()
Atribuimos ou lemos o valor do atributo hand
no nosso modelo como em qualquer outra classe Python. Este truque é para dizer ao Django como manipular a gravação e a leitura como um objeto.
Para utilizar a classe Hand
em nossos modelos, nós não precisamos alterar essa classe em nada. Isso é ideal, porque quer dizer que podemos facilmente escrever modelos de suporte para classes existentes cujo o código fonte não pode ser alterado.
Nota
Você pode estar querendo tirar vantagem dos tipos customizados de colunas de banco de dados e lidar com o dado como um tipo de dado Python padrão nos seus modelos; Este caso é similar ao nosso exemplo Hand
e iremos explicitar qualquer diferença enquanto continuamos.
A teoria por trás.¶
Armazenamento no Banco de Dados.¶
A maneira mais simples de pensar em um campo de modelo, é que ele fornece uma maneira de converter um objeto Python normal –string, boolean, datetime
, ou alguma coisa mais complexa como o Hand
– para ou de um formato que seja útil quando interagir com o banco de dados (e serialização, mas, como veremos mais tarde isso parece bastante natural uma vez que você tenha o lado do banco de dados sob controle).
Campos em um modelo devem ser de alguma maneira convertidos para que se encaixem em um tipo de coluna de banco de dados já existente. Bancos de dados diferentes provêem diferentes tipos de colunas, mas a regra ainda é a mesma: aqueles são os únicos dados que se tem para trabalhar. Qualquer coisa que se queira guardar em um banco de dados deve se encaixar em um daqueles tipos.
Normalmente, ou você está escrevendo um campo Django para lidar com um tipo de coluna de banco de dados em particular, ou existe uma maneira bastante simples de converter seu dado para, por exemplo, uma string.
Para o nosso exemplo Hand`, podemos converter o dado carta para uma string de 104 caracteres concatenando todos as cartas juntas em uma ordem pré-determinada --por exemplo, primeiro todos os *north*, depois os *east*, *south* e *west*. Então os objetos ``Hand
podem ser salvos em colunas do tipo texto ou caracter em um banco de dados.
O que uma classe campo faz?¶
Todos os campos do Django (e quando dizemos campo neste documento, sempre queremos dizer campos do modelo e não campos de forms) são subclasses de django.db.models.Field
. A maioria das informações que o Django registra sobre um campo é comum a todos os campos –name, help text, uniqueness e por diante. Armazenar toda essa informação é papel de Field
. Iremos entrar em detalhes precisos do que o Field
pode fazer mais tarde; por agora, basta dizer que tudo descende de Field
e então customizamos partes chaves do comportamento da classe.
É importante dizer que a classe de campo do Django não é o que é armazenado no seus atributos do modelo. Os atributos dos modelos contém objetos Python normais. As classes de campo que são definidas em um modelo são na verdade armazenadas dentro da classe Meta
quando a classe é criada (os detalhes precisos de como isso é feito são importantes aqui). Isso é porque as classes de campos não são necessárias enquanto se está modificando atributos. Na verdade, eles fornecem o maquinário para conversão entre o valor do atributo e o que é armazenado no banco de dados ou enviado para o serializador.
Tenha isso em mente ao criar seus próprios campos. A subclasse Field
do Django que você escreve provê o maquinário para conversão entre suas instancias Python e os valores do banco de dados/serializador de várias maneiras (existe uma diferença entre armazenar um valor e usar um valor para lookups por exemplo). Se isso lhe soa um pouco confuso, não se preocupe – se tornará claro no exemplo abaixo. Mas lembre que muitas vezes acabará por criar duas classes quando quiser um campo personalizado:
- A primeira classe é o objeto Python que os usuários irão manipular. Irão assinalar isso a um atributo de model, lerão para mostrar, coisas assim. Esta é a classe
Hand
em nosso exemplo. - A segunda classe é a subclasse
Field
. Essa é a classe que sabe como converter sua classe entre sua forma de armazenamento permanente e a forma do Python e vice-versa.
Escrevendo uma subclasse de campo¶
Quando estiver panejando sua subclasse de Field
, primeiro de uma com qual classe já existente Field
seu campo se parece mais. É possível fazer uma subclasse de campo Django já existente e economizar algum trabalho ? Caso não, você deve construir uma subclasse de Field
, da qual tudo descende.
Iniciar seu novo campo é uma questão de separar quaisquer argumentos que são específicos para seu caso dos argumentos comuns e passar este último para o método __init__()
da Field
(ou sua classe pai).
No nosso exemplo, iremos chamar nosso campo de HandField
. (é uma boa idéia chamar sua subclasse Field
<AgumaCoisa>Field
, então ela fica facilmente identificável como uma subclasse de Field
) . Nosso campo não se comporta como nenhum outro campo existente, então iremos herdar diretamente de Field
:
from django.db import models
class HandField(models.Field):
description = "A hand of cards (bridge style)"
def __init__(self, *args, **kwargs):
kwargs['max_length'] = 104
super().__init__(*args, **kwargs)
Nosso HandField
aceita a maioria das opções dos campos padrão (veja a lista abaixo), mas asseguramos que ele tenha um comprimento fixo, já que só precisa de 52 valores de carta mais seus naipes; 104 caracteres no total.
Nota
Muitos dos modelos do Django aceitam opções que estes nunca usam pra nada. Por exemplo, é possível passar ambos os editable
e auto_now
para um django.db.models.DateField
e o parâmetro editable
será ignorado (auto_now
sendo definido implica em editable=False
). Nenhum erro é gerado neste caso.
Este comportamento simplifica as classes de campos, porque não precisam checar opções que não são necessárias. Apenas passam todas as opções para a classe mãe e estes não são usados mais tarde. Você decide se quer que seus campos sejam mais rigorosos sobre as opções que eles selecionam, ou usar o mais simples, um comportamento mais permissivo dos campos existentes.
O método Field.__init__()
recebe os seguintes parâmetros:
verbose_name
name
primary_key
max_length
unique
blank
null
db_index
rel
: Usado para campos relacionados (comoForeignKey
). Somente para uso avançado.default
editable
serialize
: seFalse
, o campo não será “serializado” quando o modelo for passado para os serializadores. Padrão éTrue
.unique_for_date
unique_for_month
unique_for_year
choices
help_text
db_column
db_tablespace
: Somente para a criação de índice, se o “backend” suportar tablespaces. Normalmente é possível ignorar essa opção.auto_created
:True
se o campo foi criado automaticamente, como para oOneToOneField
usado pela herança de modelo. Somente para uso avançado.
Todas as opções sem uma explicação na lista acima tem o mesmo significado que os campos normais do Django. veja a field documentation para exemplos e detalhes.
A descontrução do Campo¶
O contraponto de escrever o seu próprio método __init__()
é escrever o método deconstruct()
. Ele é usando durante migrações de modelo para dizer ao Django como pegar a instância de seu novo campo e reduzir ela para a forma serializada - em particular, quais argumentos passar para o __init__()
para recriá-lo.
Se não foram adicionadas opções extras ao campo que foi herdado, então não há necessidade de reescrever o método deconstruct()
. Por outro lado, se você mudar os argumentos passados no __init__()
(como em HandField
), será preciso complementar os argumentos sendo passados.
O contrato da deconstruct()``é simples; ele retorna uma tupla de quatro item: o atributo "name" do campo, o path completo de importação da classe do campo, os argumentos posicionais (como lista), e os argumentos nomeados (como dicionários). Note que é diferente do método ``deconstruct()
para classes customizadas o qual retorna uma tupla de 3 elementos
Como autor do campo customizado, não é necessário se preocupar sobre os dois primeiros valores; a classe Field
básica tem todo o código para o atributo “name” e “import path” funcionar. Entretanto, devemos nos preocupar com o argumentos posicionais e os nomeados, já que estes são as alterações que fizemos
Por exemplo, na nossa classe “HandField” sempre somos forçados a definir max_length em ” __init __ () “. O método “deconstruct()” da classe base “Field” ierá tentar retornar os argumentos da palavra-chave; assim, podemos lançar a partir dos argumentos palavras-chaves para facilitar a leitura
from django.db import models
class HandField(models.Field):
def __init__(self, *args, **kwargs):
kwargs['max_length'] = 104
super().__init__(*args, **kwargs)
def deconstruct(self):
name, path, args, kwargs = super().deconstruct()
del kwargs["max_length"]
return name, path, args, kwargs
Se você adicionar um novo argumento chave, você vai precisar escrever código em deconstruct()
que põe o valor dentro de kwargs
você mesmo. Você deveria também omitir o valor de kwargs
quando ele não é necessário para reconstruir o estado do campo, como quando se usa o valor padrão:
from django.db import models
class CommaSepField(models.Field):
"Implements comma-separated storage of lists"
def __init__(self, separator=",", *args, **kwargs):
self.separator = separator
super().__init__(*args, **kwargs)
def deconstruct(self):
name, path, args, kwargs = super().deconstruct()
# Only include kwarg if it's not the default
if self.separator != ",":
kwargs['separator'] = self.separator
return name, path, args, kwargs
Exemplos mais complexos estão além do escopo deste documento, mas lembre - para qualquer configuração da sua instância de Field, o deconstruct()
tem que retornar argumentos que se possa passar para o __init__
para reconstruir aquele estado.
Preste atenção extra se foram definidos novos valores padrão para argumentos na superclasse do Field
, tem que ter certeza de que estes são sempre incluídos, ao invés de desaparecer se recebem o valor antigo.
E mais, tente evitar retornar valores como argumentos posicionais; onde for possível, retorne valores como argumentos “nomeados” para compatibilidade máxima no futuro. Claro, se os nomes são trocados mais frequentemente que as posições na lista do construtor, talvez seja preferível o posicional, mas tenha em mente que as pessoas estarão reconstruindo seu campo a partir da verão serializada por algum tempo (talvez anos), dependendo de quanto tempo suas migrações sobrevivam.
É possível ver os resultados de desconstrução olhando as migrações que incluem o campo, e é possível testar a desconstrução com “unit tests” somente desconstruindo e reconstruindo o campo:
name, path, args, kwargs = my_field_instance.deconstruct()
new_instance = MyField(*args, **kwargs)
self.assertEqual(my_field_instance.some_attribute, new_instance.some_attribute)
Alterando a classe básica de um campo personalizado¶
Você não pode alterar a classe base de um campo personalizado porque o Django não irá detectar a mudança, e não irá fazer uma migração para ele. Por exemplo, se você começar com:
class CustomCharField(models.CharField):
...
e decidir que quer usar um TextFeld
no lugar, você não pode mudar a classe base como estea aqui:
class CustomCharField(models.TextField):
...
Ao invés, você deve criar uma nova classe de campo personalizado e atualizar seu modelo para referenciá-la.
class CustomCharField(models.CharField):
...
class CustomTextField(models.TextField):
...
Como discutido em removendo campos, você deve manter a classe CustomCharField
original enquanto tiver migrações que façam referência a ela.
Documentando o seu campo customizado¶
Como sempre, é devido documentar o tipo do campo, então os usuários irão saber o que é. Além disso prover uma “docstring” para ele, que é útil para desenvolvedores, é possível permitir que usuários da aplicação admin vejam uma pequena descrição do tipo do campo através do django.contrib.admindocs . Para tal simplesmente forneça um texto descritivo no atributo description
da classe do seu campo personalizado. NO evento acima, a descrição mostrada pela aplicação admindocs
para o HandField
será ‘A hand of cards (bridge style)’.
No display do django.contrib.admindocs
, a descrição do campo é interpolada com field.__dict__
o qual habilita a descrição incorporar argumentos do campo. Por exemplo, a descrição para CharField
é:
description = _("String (up to %(max_length)s)")
Métodos úteis¶
Uma vez criada a subclasse de Field
, você deve considerar sobrescrever um poucos métodos, dependendo do comportamento do seu campo. A lista de métodos abaixo está mais ou menos em uma ordem decrescente de importância, então comece do topo.
Tipos de banco de dados personalizados¶
Vamos dizer que você tenha criado um tipo personalizado para o PostgreSQL chamado mytype
. Você pode herdar de Field
e implementar o método db_type()
, como a:
from django.db import models
class MytypeField(models.Field):
def db_type(self, connection):
return 'mytype'
Uma vez que tenha MyTypeField
, pode usá-lo em qualquer modelo, tal como qualquer outro tipo Field
:
class Person(models.Model):
name = models.CharField(max_length=80)
something_else = MytypeField()
Se o seu objetivo é criar uma aplicação agnóstica quanto ao banco de dados, deve levar em conta as diferenças nos tipos de colunas no banco de dados. Por exemplo, a tipo date/time para o PostgreSQL é chamado``timestamp`` enquanto que para o MySQL é chamado datetime
. O jeito mais simples de lidar com isso é no método db_type()
verificar o atributo connection.settings_dict['ENGINE']
.
Por exemplo:
class MyDateField(models.Field):
def db_type(self, connection):
if connection.settings_dict['ENGINE'] == 'django.db.backends.mysql':
return 'datetime'
else:
return 'timestamp'
Os métodos db_type()
e rel_db_type()
são chamados pelo Django quando o framework constrói comandos CREATE TABLE
para a sua aplicação – isto é, quando você cria as tabelas pela primeira vez. Os métodos também são chamados quando constróem cláusulas WHERE
que incluem um campo de modelo – isto é, quando você retorna dados usando métodos QuerySet como get()
, filter()
, e exclude()
e tem um campo de modelo como argumento. Eles não são chamados em nenhum outro momento, portanto podem executar códigos complexos, como a verificação connection.settings_dict
no exemplo acima.
Alguns tipos de coluna do banco de dados aceitam parâmetros, como CHAR(25)
, onde o parâmetro 25
representa o tamanho máximo da coluna. Em casos como este, é mais flexível se o parâmetro for especificado no modelo ao invés de ser hard-coded no método db_type()
. Por exemplo, não faria muito sentido ter CharMaxlength25Field
, mostrado aqui:
# This is a silly example of hard-coded parameters.
class CharMaxlength25Field(models.Field):
def db_type(self, connection):
return 'char(25)'
# In the model:
class MyModel(models.Model):
# ...
my_field = CharMaxlength25Field()
O melhor jeito de fazer isso seria fazer o parâmetro ser especificado em tempo de execução – isto é, quando a classe é instanciada. Para fazer isso, basta implementar Field.__init__()
, como em:
# This is a much more flexible example.
class BetterCharField(models.Field):
def __init__(self, max_length, *args, **kwargs):
self.max_length = max_length
super().__init__(*args, **kwargs)
def db_type(self, connection):
return 'char(%s)' % self.max_length
# In the model:
class MyModel(models.Model):
# ...
my_field = BetterCharField(25)
Finalmente, se sua coluna requer que defina um SQL realmente complexo, retorne None
do db_type()
. Isso irá fazer que o código de criação de SQL do Django ignore este campo. Então você é o responsável por criar a coluna na tabela correta de alguma outra maneira, claro, isso é uma maneira de dizer ao Django para sair do caminho.
O método rel_db_type()
é chamado por campos como o ForeignKey
e o OneToOneField
que apontam para outros campos para determinar o tipo da sua coluna de banco de dados. Por exemplo, se tiver um UnsignedAutoField
, você também precisa de chaves estrangeiras que apontam para aquele campo para usar o mesmo tipo de dado:
# MySQL unsigned integer (range 0 to 4294967295).
class UnsignedAutoField(models.AutoField):
def db_type(self, connection):
return 'integer UNSIGNED AUTO_INCREMENT'
def rel_db_type(self, connection):
return 'integer UNSIGNED'
Convertendo valores para objetos Python¶
Se sua classe customizada de Field
lida com estruturas de dados mais complexas que strings, datas, inteiros, or floats, então talvez precise sobrescrever o from_db_value()
e to_python()
.
Se presente para a subclasse do campo, from_db_value()
será chamado em todas as circunstâncias quando dados são lidos do banco de dados, incluindo em agregacões e chamadas do values()
.
o to_python()
é chamado pela deserialização e durante o método clean()
usado nos forms.
Como regra geral, ``to_python()``deve lidar tranquilamente com qualquer dos seguintes argumentos:
- Uma instância do tipo correto (ex., ``Hand``no nosso exemplo corrente).
- Uma string
None``(se o campo permitir ``null=True
)
Na nossa classe HandField
, estamos armazendo os dados como campo VARCHAR no banco de dados, então temos que ser capazes de processar strings e None
no from_db_value()
. No to_python()
, precisamos também lidar com instâncias de Hand
:
import re
from django.core.exceptions import ValidationError
from django.db import models
from django.utils.translation import gettext_lazy as _
def parse_hand(hand_string):
"""Takes a string of cards and splits into a full hand."""
p1 = re.compile('.{26}')
p2 = re.compile('..')
args = [p2.findall(x) for x in p1.findall(hand_string)]
if len(args) != 4:
raise ValidationError(_("Invalid input for a Hand instance"))
return Hand(*args)
class HandField(models.Field):
# ...
def from_db_value(self, value, expression, connection):
if value is None:
return value
return parse_hand(value)
def to_python(self, value):
if isinstance(value, Hand):
return value
if value is None:
return value
return parse_hand(value)
Perceba que sempre retornamos uma instância destes métodos. Este é o tipo do objeto Python que queremos armazenar no atributo do modelo.
Para to_python()
, se qualquer coisa der errado durante a conversão do valor, deve-se gerar a exceção ValidationError
.
Convertendo objetos Python para valores em queries¶
Uma vez que usar um banco de dados requer conversão em ambos os sentidos, se sobrescrever o to_python()
é necessário sobrescrever também get_prep_value()
para converter os objetos Python de novo em valores de query.
Por exemplo:
class HandField(models.Field):
# ...
def get_prep_value(self, value):
return ''.join([''.join(l) for l in (value.north,
value.east, value.south, value.west)])
Aviso
Caso seu campo personalizado use os tipos CHAR
, VARCHAR
or TEXT
parao MySQL, assegure que o get_prep_value()
sempre retorne um tipo string. O MySQL responde a pesquisa de maneira variável e inesperada quando uma query é executada com estes tipos e o valor fornecido é um inteiro, o que pode resultar em queries qeu contenham objetos inesperados nos seus resultados. Este problema não pode ocorrer se sempre retornar um tipo string do get_prep_value()
.
Convertendo valores de queries para valores de banco de dados¶
Alguns tipos de dados (por exemplo datas) precisam estar em um formato específico antes deles poderem ser usados por um “backend” de banco de dados. O get_db_prep_value()
é um método onde estas conversões devem ser feitas. A conexão específica que será usada para a query é passada pelo parâmetro connection
. Isso lhe possibilita usar a lógica de conversão espeçifica para um backend se isso for necessário.
Por exemplo, o Django usa o seguinte método para isso BinaryField
:
def get_db_prep_value(self, value, connection, prepared=False):
value = super().get_db_prep_value(value, connection, prepared)
if value is not None:
return connection.Database.Binary(value)
return value
No caso do seu campo personalizado precisar de uma conversão especial quando for salvo e isso não é o mesmo que a conversão usada para parâmetros de query, você pode sobrescrever get_db_prep_save()
.
Processando valores antes de salvar¶
Caso queira processar o valor logo antes de salvar, pode-se usar pre_save()
. Por exemplo, o DateTimeField
do Django usa este método para definir o atributo corretamente no caso de auto_now
ou auto_now_add
.
Caso sobrescreva este método, é necessário retornar o valor do atributo no fim. Deve-se também atualizar o atributo do modelo caso tenham sido feito mudanças no valor, de modo que que o código que mantém as referências ao modelo sempre veja o valor correto.
Especificando o campo de um formulário para o campo de um modelo.¶
Para personalizar o campo de formulário usado pelo ModelForm
, você pode sobrescrever formfield()
.
A classe do campo de formulário pode ser especificada via o form_class
e os argumentos de choices_form_class
; o último é especificado se o campo tiver opções (choices) especificadas, do contrário o primeiro. Se estes argumentos não são fornecidos, será usada a CharField
ou TypedChoiceField
.
Todos os dicionários kwargs
é passado diretamente para o método __init__()` do campo do formulário. Normalmente, tudo o que você precisa fazer é definir um bom padrão para o argumento do form_class
(e talvez choices_form_class
) e delegar futuras manipulações para a classe pai. Isso talvez requeira que seja escrito um campo de form personalizado (e mesmo o widget de formulário). Veja o forms documentation para informações sobre isso.
Continuando nosso exemplo em andamento, podemos escrever o método formfield()
como:
class HandField(models.Field):
# ...
def formfield(self, **kwargs):
# This is a fairly standard way to set up some defaults
# while letting the caller override them.
defaults = {'form_class': MyFormField}
defaults.update(kwargs)
return super().formfield(**defaults)
Isso assume que importamos uma classe de campo MyModelField
(o qual tem seu próprio widget padrão). Este documento não cobre os detalhes de escrever um campo de formulário personalizado.
Emulando tipos de campos internos.¶
Se você criou um método db_type()
, não precisa se preocupar sobre get_internal_type()
– isso não será usado muito. As vezes, porém, seu armazenamento no banco de dados é similar em tipo a outro campo, então você pode usar a lógica deste outro campo para criar a coluna correta.
Por exemplo:
class HandField(models.Field):
# ...
def get_internal_type(self):
return 'CharField'
Não importa qual backend de banco de dados que esteja sendo usado, isso quer dizer que o migrate
e outros comandos SQL criam o tipo de coluna correto para armazenar uma string.
Se o get_internal_type()
retorna uma string que não é conhecida para o Django para o backend de banco de dados que está sendo usado – que é, ele não aparece no django.db.backends.<db_name>.base.DatabaseWrapper.data_types
– a string ainda será usada pelo serializador, mas o método padrão db_type()
retornará None
. Veja a documentação do db_type()
pelas razões pelas quais este talvez seja útil.
Convertendo dado do campo para serialização¶
Para personalizar como os valores são serializados por um serializador, você pode sobrescrever :meth:’~Field.value_to_string’. Utilizando :meth:’~Field.value_from_object’ é a melhor forma para obter o valor dos campos antes da serialização. Por exemplo, visto que “HandField” utiliza strings para armazenar dados, nós podemos reutilizar algum código de conversão existente:
class HandField(models.Field):
# ...
def value_to_string(self, obj):
value = self.value_from_object(obj)
return self.get_prep_value(value)
Alguns avisos gerais¶
Escrever um um campo personalizado pode ser um um pouco complicado, particularmente se estiver fazendo conversões complexas entre tipos Python e o banco de dados e formatos de serialização. Aqui algumas dicas par fazer as coisas mais tranquilas.
- Olhe para os campos existentes no Django (em
django/db/models/fields/__init__.py
) para inspação. Tente encontrar um campo que seja similar ao o que você quer extender um pouco, ao invés de criar um inteiramente novo desde o início. - Coloque o método
__str__()
na classe que você está utilizando para envelopar o campo. Em vários lugares o comportamento padrão do código do campo é chamarstr()
. (Em nossos exemplos neste documento,value
seria uma instância deHand
e não deHandField
). Então se o método__str__()
da sua classe converter o objeto para o formato de texto, você poderá poupar bastante trabalho.
Escrevendo uma subclasse FileField
¶
Além dos métodos acima, campos que lidam com arquivos tem alguns outros requerimentos especiais que devem ser levados em conta. A maioria dos mecanismos fornecidos pelo ``FileField`, como controlar o armazenamento e leitura na base de dados, podem permanecer inalterados, deixando as subclasses lidarem com o desafio de dar suporte a um tipo de arquivo particular.
O Django fornece uma classe File
, a qual é usada como um proxy para conteúdos e operações de arquivos . Isso pode ser herdado para customizar como o arquivo é acessado, e quais métodos estão disponíveis. Isso está em django.db.models.fields.files
, e seu comportamento padrão é explicado no arquivo documentação.
Uma vez que uma subclasse de File
é criada, a nova subclasse FileField
deve ser informada para usá-lo. Para isso, simplesmente passe a nova subclasse de File
para o atributo especial attr_class
da subclasse de FileField
.
Algumas sugestões¶
Além dos detalhes acima, tem algumas orientações as quais podem melhorar muito a eficiência e a legibilidade do código do campo.
- O código fonte para o
ImageField
do Django (emdjango/db/models/fields/files.py
) é um bom exemplo de como herdarFileField
para dar suporte ao um tipo de arquivo em particular, já que ele incorpora todas as técnicas descritas acima. - Faça o cache dos atributos do arquivo sempre que possível. Uma vez que é possível que arquivos sejam armazenados em sistemas de armazenamento remotos, recuperá-los talvez requeira tempo extra, ou mesmo dinheiro, que não é sempre necessário. Uma vez que o arquivo é recuperado para obter alguns dados sobre seu conteúdo, faça o cache de dados o quanto possível para reduzir o número de vezes que o arquivo deve ser recuperado em chamadas subsequentes para aquela informação.