I need to store PNG files as blob/binary in the database and then be able to retrieve and show them.
Here's my model that has a binary field to store the image:
class ImageFile(models.Model):
file = models.BinaryField(editable=True)
I created a widget based on this answer:
class BinaryFileInputWidget(forms.ClearableFileInput):
def is_initial(self, value):
return bool(value)
def format_value(self, value):
if self.is_initial(value):
return f"{len(value)} bytes"
def value_from_datadict(self, data, files, name):
upload = super().value_from_datadict(data, files, name)
if upload:
return upload.read()
And I used it in admin.py
like this:
@admin.register(ImageFile)
class ImageFileAdmin(admin.ModelAdmin):
list_display = ["id"]
formfield_overrides = {
models.BinaryField: {"widget": BinaryFileInputWidget()},
}
Then I encode the file as base64
in the view:
def image_view(request: HttpRequest, id: int):
document_file = ImageFile.objects.filter(id=id).first()
data = ""
if document_file:
data = base64.b64encode(document_file.file).decode(encoding="ascii")
return render(request, "image.html", {"data": data})
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8" />
<meta name="viewport" content="width=device-width, initial-scale=1.0" />
<title>Document</title>
</head>
<body>
<pre>data:image/png;base64,{{ data }}</pre>
<img src="data:image/png;base64,{{ data }}" alt="" />
</body>
</html>
There is no error and the data is shown in the pre
element but the img
element fails to load the image. I tried using some online converters to check if the data is valid but they show errors saying the base64 string is invalid.
It loads properly if I convert the file to base64
in the widget before storing it in the database (return base64.b64encode(upload.read()).decode('ascii')
instead of return upload.read()
), and then convert it again in the view. But I cant use this method because I might need to compress the files and apparently base64
encoded data can't be compressed properly.
Sample output using my profile picture:
<!DOCTYPE html>
<html lang="en">
<head>
<meta http-equiv="Content-Type" content="text/html; charset=UTF-8" />
<meta name="viewport" content="width=device-width, initial-scale=1.0" />
<title>Document</title>
</head>
<body cz-shortcut-listen="true">
<pre>
</pre
>
<img
src=""
alt=""
/>
</body>
</html>
I ended up creating a custom form (instead of widget) that reads the file from a FileField
and then saves it in the model.
forms.py:
class ImageFileForm(forms.ModelForm):
file_upload = forms.FileField()
def save(self, commit=True, *args, **kwargs):
instance = super(ImageFileForm, self).save(commit=False)
file: InMemoryUploadedFile | None = self.cleaned_data.get("file_upload")
if file:
with file.open() as f:
instance.file = f.read()
if commit:
instance.save()
return instance
class Meta:
model = ImageFile
fields = ["file_upload"]
admin.py:
@admin.register(ImageFile)
class ImageFileAdmin(admin.ModelAdmin):
list_display = ["id"]
form = DocumentFileForm