Skip to content

Commit ff2fd01

Browse files
committed
Fixed issue with uploaded file handling in fundraising tests
The tests currently fail on docker. Using the TemporaryMediaRootMixin originally developped for the views tests should fix that.
1 parent 8e5a19f commit ff2fd01

File tree

3 files changed

+52
-58
lines changed

3 files changed

+52
-58
lines changed

fundraising/tests/test_models.py

+5-20
Original file line numberDiff line numberDiff line change
@@ -1,14 +1,12 @@
1-
import os
21
from datetime import date
32

4-
from django.conf import settings
53
from django.test import TestCase
6-
from PIL import Image
74

85
from ..models import DjangoHero, Donation, InKindDonor
6+
from .utils import ImageFileFactory, TemporaryMediaRootMixin
97

108

11-
class TestDjangoHero(TestCase):
9+
class TestDjangoHero(TemporaryMediaRootMixin, TestCase):
1210
def setUp(self):
1311
kwargs = {
1412
"approved": True,
@@ -26,26 +24,13 @@ def setUp(self):
2624
self.today = date.today()
2725

2826
def test_thumbnail(self):
29-
try:
30-
os.makedirs(os.path.join(settings.MEDIA_ROOT, "fundraising/logos/"))
31-
except OSError: # directory may already exist
32-
pass
33-
image_path = os.path.join(
34-
settings.MEDIA_ROOT, "fundraising/logos/test_logo.jpg"
35-
)
36-
image = Image.new("L", (500, 500))
37-
image.save(image_path)
38-
self.h1.logo = image_path
27+
self.h1.logo = ImageFileFactory(name="logo.jpeg")
3928
self.h1.save()
4029
thumbnail = self.h1.thumbnail
4130
self.assertEqual(thumbnail.x, 170)
4231
self.assertEqual(thumbnail.y, 170)
43-
os.remove(image_path)
44-
self.assertTrue(
45-
os.path.exists(
46-
thumbnail.url.replace(settings.MEDIA_URL, f"{settings.MEDIA_ROOT}/")
47-
)
48-
)
32+
self.h1.logo.delete()
33+
self.assertTrue(thumbnail.exists())
4934

5035
def test_thumbnail_no_logo(self):
5136
self.assertIsNone(self.h2.thumbnail)

fundraising/tests/test_views.py

+6-38
Original file line numberDiff line numberDiff line change
@@ -1,55 +1,21 @@
11
import json
2-
import shutil
3-
import tempfile
42
from datetime import date, datetime
5-
from io import BytesIO
63
from operator import attrgetter
74
from unittest.mock import patch
85

96
import stripe
107
from django.conf import settings
118
from django.core import mail
12-
from django.core.files.images import ImageFile
139
from django.template.defaultfilters import date as date_filter
1410
from django.test import TestCase
1511
from django.urls import reverse
1612
from django_hosts.resolvers import reverse as django_hosts_reverse
1713
from django_recaptcha.client import RecaptchaResponse
18-
from PIL import Image
1914

2015
from members.models import CorporateMember, Invoice
2116

2217
from ..models import DjangoHero, Donation
23-
24-
25-
def _make_image(name, width=1, height=1, color=(12, 75, 51)):
26-
img = Image.new("RGB", (width, height), color=color)
27-
out = BytesIO()
28-
img.save(out, format=name.split(".")[-1])
29-
return ImageFile(out, name=name)
30-
31-
32-
class TemporaryMediaRootMixin:
33-
"""
34-
A TestCase mixin that overrides settings.MEDIA_ROOT for every test on the
35-
class to point to a temporary directory that is destroyed when the tests
36-
finished.
37-
The content of the directory persists between different tests on the class.
38-
"""
39-
40-
@classmethod
41-
def setUpClass(cls):
42-
super().setUpClass()
43-
cls.tmpdir = tempfile.mkdtemp(prefix="djangoprojectcom_")
44-
45-
@classmethod
46-
def tearDownClass(cls):
47-
shutil.rmtree(cls.tmpdir, ignore_errors=True)
48-
super().tearDownClass()
49-
50-
def run(self, result=None):
51-
with self.settings(MEDIA_ROOT=self.tmpdir):
52-
return super().run(result)
18+
from .utils import ImageFileFactory, TemporaryMediaRootMixin
5319

5420

5521
class TestIndex(TestCase):
@@ -77,7 +43,9 @@ def test_corporate_member_without_logo(self):
7743

7844
def test_corporate_member_with_logo(self):
7945
member = CorporateMember.objects.create(
80-
display_name="Test Member", membership_level=1, logo=_make_image("logo.png")
46+
display_name="Test Member",
47+
membership_level=1,
48+
logo=ImageFileFactory("logo.png"),
8149
)
8250
Invoice.objects.create(amount=100, expiration_date=date.today(), member=member)
8351
response = self.client.get(self.index_url)
@@ -96,7 +64,7 @@ def test_corporate_member_with_logo(self):
9664
)
9765

9866
def test_corporate_member_with_non_square_logo(self):
99-
logo = _make_image("wide.png", width=10)
67+
logo = ImageFileFactory("wide.png", width=10)
10068
member = CorporateMember.objects.create(
10169
display_name="Test Member", membership_level=1, logo=logo
10270
)
@@ -130,7 +98,7 @@ def test_anonymous_donor_with_logo(self):
13098
is_visible=True,
13199
approved=True,
132100
hero_type="individual",
133-
logo=_make_image("anonymous.png"),
101+
logo=ImageFileFactory("anonymous.png"),
134102
)
135103
donation = hero.donation_set.create(subscription_amount="5")
136104
donation.payment_set.create(amount="5")

fundraising/tests/utils.py

+41
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,41 @@
1+
import shutil
2+
import tempfile
3+
from io import BytesIO
4+
5+
from django.core.files.images import ImageFile
6+
from PIL import Image
7+
8+
9+
def ImageFileFactory(name, width=1, height=1, color=(12, 75, 51)):
10+
"""
11+
Return an ImageFile instance with the given name.
12+
The image will be of the given size, and of solid color. The format will
13+
be inferred from the filename.
14+
"""
15+
img = Image.new("RGB", (width, height), color=color)
16+
out = BytesIO()
17+
img.save(out, format=name.split(".")[-1])
18+
return ImageFile(out, name=name)
19+
20+
21+
class TemporaryMediaRootMixin:
22+
"""
23+
A TestCase mixin that overrides settings.MEDIA_ROOT for every test on the
24+
class to point to a temporary directory that is destroyed when the tests
25+
finished.
26+
The content of the directory persists between different tests on the class.
27+
"""
28+
29+
@classmethod
30+
def setUpClass(cls):
31+
super().setUpClass()
32+
cls.tmpdir = tempfile.mkdtemp(prefix="djangoprojectcom_")
33+
34+
@classmethod
35+
def tearDownClass(cls):
36+
shutil.rmtree(cls.tmpdir, ignore_errors=True)
37+
super().tearDownClass()
38+
39+
def run(self, result=None):
40+
with self.settings(MEDIA_ROOT=self.tmpdir):
41+
return super().run(result)

0 commit comments

Comments
 (0)