最新消息:Welcome to the puzzle paradise for programmers! Here, a well-designed puzzle awaits you. From code logic puzzles to algorithmic challenges, each level is closely centered on the programmer's expertise and skills. Whether you're a novice programmer or an experienced tech guru, you'll find your own challenges on this site. In the process of solving puzzles, you can not only exercise your thinking skills, but also deepen your understanding and application of programming knowledge. Come to start this puzzle journey full of wisdom and challenges, with many programmers to compete with each other and show your programming wisdom! Translated with DeepL.com (free version)

postgresql - How to use Django BinaryField to store PNG images in postgres? - Stack Overflow

matteradmin2PV0评论

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 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>

Share Improve this question edited Nov 18, 2024 at 13:53 Sadra Saderi asked Nov 18, 2024 at 11:09 Sadra SaderiSadra Saderi 3944 silver badges12 bronze badges 6
  • 2 Storing file-like content in the database is typically not a good idea. Databases are not really designed to store large binary blobs as values. – willeM_ Van Onsem Commented Nov 18, 2024 at 11:53
  • @willeM_VanOnsem I know, usually I don't do things like this but in this project I need to store them in the database. – Sadra Saderi Commented Nov 18, 2024 at 13:15
  • Is it possible to share a (sample) HTML output? – willeM_ Van Onsem Commented Nov 18, 2024 at 13:25
  • @willeM_VanOnsem Yes, I added a sample code snippet to the question. – Sadra Saderi Commented Nov 18, 2024 at 13:55
  • 1 The data looks quite strange: it repeats x00 (and other x a lot), and thus looks as if it was initially a string with \x00\x01, etc. where the backslash was dropped after wrapping it through a str. If this conversion is done, it starts making a bit sense as a binary string, but still not a base64 encoding. – willeM_ Van Onsem Commented Nov 18, 2024 at 15:30
 |  Show 1 more comment

1 Answer 1

Reset to default 1

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
Post a comment

comment list (0)

  1. No comments so far