[1.7.x] Fixed #19905 -- Fixed leakage of file descriptors in form wizard.

Backport of c4c2c99669 from master.
This commit is contained in:
Florian Apolloner 2014-05-25 19:37:31 +02:00
parent 08a2b3b9cf
commit e3792bb95f
5 changed files with 119 additions and 44 deletions

View File

@ -1,5 +1,7 @@
from __future__ import unicode_literals from __future__ import unicode_literals
import copy
from django.core.urlresolvers import reverse from django.core.urlresolvers import reverse
from django.http import QueryDict from django.http import QueryDict
from django.test import TestCase from django.test import TestCase
@ -11,14 +13,31 @@ from django.contrib.formtools.wizard.views import (NamedUrlSessionWizardView,
NamedUrlCookieWizardView) NamedUrlCookieWizardView)
from django.contrib.formtools.tests.wizard.test_forms import get_request, Step1, Step2 from django.contrib.formtools.tests.wizard.test_forms import get_request, Step1, Step2
from .forms import temp_storage
UPLOADED_FILE_NAME = 'tests.py'
class NamedWizardTests(object): class NamedWizardTests(object):
urls = 'django.contrib.formtools.tests.wizard.namedwizardtests.urls' urls = 'django.contrib.formtools.tests.wizard.namedwizardtests.urls'
def setUp(self): def setUp(self):
self.testuser, created = User.objects.get_or_create(username='testuser1') self.testuser, created = User.objects.get_or_create(username='testuser1')
# Get new step data, since we modify it during the tests.
self.wizard_step_data = copy.deepcopy(self.wizard_step_data)
self.wizard_step_data[0]['form1-user'] = self.testuser.pk self.wizard_step_data[0]['form1-user'] = self.testuser.pk
def tearDown(self):
# Ensure that there are no files in the storage which could lead to false
# results in the next tests. Deleting the whole storage dir is not really
# an option since the storage is defined on the module level and can't be
# easily reinitialized. (FIXME: The tests here should use the view classes
# directly instead of the test client, then the storage issues would go
# away too.)
for file in temp_storage.listdir('')[1]:
temp_storage.delete(file)
def test_initial_call(self): def test_initial_call(self):
response = self.client.get(reverse('%s_start' % self.wizard_urlname)) response = self.client.get(reverse('%s_start' % self.wizard_urlname))
self.assertEqual(response.status_code, 302) self.assertEqual(response.status_code, 302)
@ -123,17 +142,21 @@ class NamedWizardTests(object):
self.assertEqual(response.context['wizard']['steps'].current, 'form2') self.assertEqual(response.context['wizard']['steps'].current, 'form2')
post_data = self.wizard_step_data[1] post_data = self.wizard_step_data[1]
post_data['form2-file1'].close() with open(__file__, 'rb') as post_file:
post_data['form2-file1'] = open(__file__, 'rb') post_data['form2-file1'] = post_file
response = self.client.post( response = self.client.post(
reverse(self.wizard_urlname, reverse(self.wizard_urlname,
kwargs={'step': response.context['wizard']['steps'].current}), kwargs={'step': response.context['wizard']['steps'].current}),
post_data) post_data)
response = self.client.get(response.url) response = self.client.get(response.url)
self.assertEqual(response.status_code, 200) self.assertEqual(response.status_code, 200)
self.assertEqual(response.context['wizard']['steps'].current, 'form3') self.assertEqual(response.context['wizard']['steps'].current, 'form3')
# Check that the file got uploaded properly.
with open(__file__, 'rb') as f, temp_storage.open(UPLOADED_FILE_NAME) as f2:
self.assertEqual(f.read(), f2.read())
response = self.client.post( response = self.client.post(
reverse(self.wizard_urlname, reverse(self.wizard_urlname,
kwargs={'step': response.context['wizard']['steps'].current}), kwargs={'step': response.context['wizard']['steps'].current}),
@ -150,10 +173,10 @@ class NamedWizardTests(object):
response = self.client.get(response.url) response = self.client.get(response.url)
self.assertEqual(response.status_code, 200) self.assertEqual(response.status_code, 200)
# After the wizard is done no files should exist anymore.
self.assertFalse(temp_storage.exists(UPLOADED_FILE_NAME))
all_data = response.context['form_list'] all_data = response.context['form_list']
with open(__file__, 'rb') as f:
self.assertEqual(all_data[1]['file1'].read(), f.read())
all_data[1]['file1'].close()
del all_data[1]['file1'] del all_data[1]['file1']
self.assertEqual(all_data, [ self.assertEqual(all_data, [
{'name': 'Pony', 'thirsty': True, 'user': self.testuser}, {'name': 'Pony', 'thirsty': True, 'user': self.testuser},
@ -174,22 +197,22 @@ class NamedWizardTests(object):
self.assertEqual(response.status_code, 200) self.assertEqual(response.status_code, 200)
post_data = self.wizard_step_data[1] post_data = self.wizard_step_data[1]
post_data['form2-file1'] = open(__file__, 'rb') with open(__file__, 'rb') as post_file:
response = self.client.post( post_data['form2-file1'] = post_file
reverse(self.wizard_urlname, response = self.client.post(
kwargs={'step': response.context['wizard']['steps'].current}), reverse(self.wizard_urlname,
post_data) kwargs={'step': response.context['wizard']['steps'].current}),
post_data)
response = self.client.get(response.url) response = self.client.get(response.url)
self.assertEqual(response.status_code, 200) self.assertEqual(response.status_code, 200)
self.assertTrue(temp_storage.exists(UPLOADED_FILE_NAME))
step2_url = reverse(self.wizard_urlname, kwargs={'step': 'form2'}) step2_url = reverse(self.wizard_urlname, kwargs={'step': 'form2'})
response = self.client.get(step2_url) response = self.client.get(step2_url)
self.assertEqual(response.status_code, 200) self.assertEqual(response.status_code, 200)
self.assertEqual(response.context['wizard']['steps'].current, 'form2') self.assertEqual(response.context['wizard']['steps'].current, 'form2')
with open(__file__, 'rb') as f: with open(__file__, 'rb') as f, temp_storage.open(UPLOADED_FILE_NAME) as f2:
self.assertEqual( self.assertEqual(f.read(), f2.read())
response.context['wizard']['form'].files['form2-file1'].read(),
f.read())
response = self.client.post( response = self.client.post(
reverse(self.wizard_urlname, reverse(self.wizard_urlname,
@ -206,9 +229,9 @@ class NamedWizardTests(object):
self.assertEqual(response.status_code, 200) self.assertEqual(response.status_code, 200)
all_data = response.context['all_cleaned_data'] all_data = response.context['all_cleaned_data']
with open(__file__, 'rb') as f: self.assertEqual(all_data['file1'].name, UPLOADED_FILE_NAME)
self.assertEqual(all_data['file1'].read(), f.read()) self.assertTrue(all_data['file1'].closed)
all_data['file1'].close() self.assertFalse(temp_storage.exists(UPLOADED_FILE_NAME))
del all_data['file1'] del all_data['file1']
self.assertEqual( self.assertEqual(
all_data, all_data,
@ -237,12 +260,12 @@ class NamedWizardTests(object):
self.assertEqual(response.status_code, 200) self.assertEqual(response.status_code, 200)
post_data = self.wizard_step_data[1] post_data = self.wizard_step_data[1]
post_data['form2-file1'].close() with open(__file__, 'rb') as post_file:
post_data['form2-file1'] = open(__file__, 'rb') post_data['form2-file1'] = post_file
response = self.client.post( response = self.client.post(
reverse(self.wizard_urlname, reverse(self.wizard_urlname,
kwargs={'step': response.context['wizard']['steps'].current}), kwargs={'step': response.context['wizard']['steps'].current}),
post_data) post_data)
response = self.client.get(response.url) response = self.client.get(response.url)
self.assertEqual(response.status_code, 200) self.assertEqual(response.status_code, 200)

View File

@ -85,3 +85,19 @@ class TestStorage(object):
storage.extra_data['test'] = True storage.extra_data['test'] = True
self.assertTrue('test' in storage.extra_data) self.assertTrue('test' in storage.extra_data)
def test_reset_deletes_tmp_files(self):
request = get_request()
storage = self.get_storage()('wizard1', request, temp_storage)
step = 'start'
file_ = SimpleUploadedFile('file.txt', b'content')
storage.set_step_files(step, {'file': file_})
with storage.get_step_files(step)['file'] as file:
tmp_name = file.name
self.assertTrue(storage.file_storage.exists(tmp_name))
storage.reset()
self.assertFalse(storage.file_storage.exists(tmp_name))

View File

@ -1,5 +1,6 @@
from __future__ import unicode_literals from __future__ import unicode_literals
import copy
import os import os
from django import forms from django import forms
@ -12,6 +13,11 @@ from django.contrib.formtools.wizard.views import CookieWizardView
from django.utils._os import upath from django.utils._os import upath
from django.contrib.formtools.tests.models import Poet, Poem from django.contrib.formtools.tests.models import Poet, Poem
from .forms import temp_storage
UPLOADED_FILE_NAME = 'tests.py'
class UserForm(forms.ModelForm): class UserForm(forms.ModelForm):
class Meta: class Meta:
@ -28,8 +34,20 @@ class WizardTests(object):
def setUp(self): def setUp(self):
self.testuser, created = User.objects.get_or_create(username='testuser1') self.testuser, created = User.objects.get_or_create(username='testuser1')
# Get new step data, since we modify it during the tests.
self.wizard_step_data = copy.deepcopy(self.wizard_step_data)
self.wizard_step_data[0]['form1-user'] = self.testuser.pk self.wizard_step_data[0]['form1-user'] = self.testuser.pk
def tearDown(self):
# Ensure that there are no files in the storage which could lead to false
# results in the next tests. Deleting the whole storage dir is not really
# an option since the storage is defined on the module level and can't be
# easily reinitialized. (FIXME: The tests here should use the view classes
# directly instead of the test client, then the storage issues would go
# away too.)
for file in temp_storage.listdir('')[1]:
temp_storage.delete(file)
def test_initial_call(self): def test_initial_call(self):
response = self.client.get(self.wizard_url) response = self.client.get(self.wizard_url)
wizard = response.context['wizard'] wizard = response.context['wizard']
@ -98,11 +116,16 @@ class WizardTests(object):
self.assertEqual(response.context['wizard']['steps'].current, 'form2') self.assertEqual(response.context['wizard']['steps'].current, 'form2')
post_data = self.wizard_step_data[1] post_data = self.wizard_step_data[1]
post_data['form2-file1'] = open(upath(__file__), 'rb') with open(upath(__file__), 'rb') as post_file:
response = self.client.post(self.wizard_url, post_data) post_data['form2-file1'] = post_file
response = self.client.post(self.wizard_url, post_data)
self.assertEqual(response.status_code, 200) self.assertEqual(response.status_code, 200)
self.assertEqual(response.context['wizard']['steps'].current, 'form3') self.assertEqual(response.context['wizard']['steps'].current, 'form3')
# Check that the file got uploaded properly.
with open(upath(__file__), 'rb') as f, temp_storage.open(UPLOADED_FILE_NAME) as f2:
self.assertEqual(f.read(), f2.read())
response = self.client.post(self.wizard_url, self.wizard_step_data[2]) response = self.client.post(self.wizard_url, self.wizard_step_data[2])
self.assertEqual(response.status_code, 200) self.assertEqual(response.status_code, 200)
self.assertEqual(response.context['wizard']['steps'].current, 'form4') self.assertEqual(response.context['wizard']['steps'].current, 'form4')
@ -110,10 +133,10 @@ class WizardTests(object):
response = self.client.post(self.wizard_url, self.wizard_step_data[3]) response = self.client.post(self.wizard_url, self.wizard_step_data[3])
self.assertEqual(response.status_code, 200) self.assertEqual(response.status_code, 200)
# After the wizard is done no files should exist anymore.
self.assertFalse(temp_storage.exists(UPLOADED_FILE_NAME))
all_data = response.context['form_list'] all_data = response.context['form_list']
with open(upath(__file__), 'rb') as f:
self.assertEqual(all_data[1]['file1'].read(), f.read())
all_data[1]['file1'].close()
del all_data[1]['file1'] del all_data[1]['file1']
self.assertEqual(all_data, [ self.assertEqual(all_data, [
{'name': 'Pony', 'thirsty': True, 'user': self.testuser}, {'name': 'Pony', 'thirsty': True, 'user': self.testuser},
@ -134,6 +157,7 @@ class WizardTests(object):
post_data['form2-file1'] = post_file post_data['form2-file1'] = post_file
response = self.client.post(self.wizard_url, post_data) response = self.client.post(self.wizard_url, post_data)
self.assertEqual(response.status_code, 200) self.assertEqual(response.status_code, 200)
self.assertTrue(temp_storage.exists(UPLOADED_FILE_NAME))
response = self.client.post(self.wizard_url, self.wizard_step_data[2]) response = self.client.post(self.wizard_url, self.wizard_step_data[2])
self.assertEqual(response.status_code, 200) self.assertEqual(response.status_code, 200)
@ -142,9 +166,9 @@ class WizardTests(object):
self.assertEqual(response.status_code, 200) self.assertEqual(response.status_code, 200)
all_data = response.context['all_cleaned_data'] all_data = response.context['all_cleaned_data']
with open(upath(__file__), 'rb') as f: self.assertEqual(all_data['file1'].name, UPLOADED_FILE_NAME)
self.assertEqual(all_data['file1'].read(), f.read()) self.assertTrue(all_data['file1'].closed)
all_data['file1'].close() self.assertFalse(temp_storage.exists(UPLOADED_FILE_NAME))
del all_data['file1'] del all_data['file1']
self.assertEqual(all_data, { self.assertEqual(all_data, {
'name': 'Pony', 'thirsty': True, 'user': self.testuser, 'name': 'Pony', 'thirsty': True, 'user': self.testuser,
@ -161,9 +185,9 @@ class WizardTests(object):
self.assertEqual(response.status_code, 200) self.assertEqual(response.status_code, 200)
post_data = self.wizard_step_data[1] post_data = self.wizard_step_data[1]
post_data['form2-file1'].close() with open(upath(__file__), 'rb') as post_file:
post_data['form2-file1'] = open(upath(__file__), 'rb') post_data['form2-file1'] = post_file
response = self.client.post(self.wizard_url, post_data) response = self.client.post(self.wizard_url, post_data)
self.assertEqual(response.status_code, 200) self.assertEqual(response.status_code, 200)
response = self.client.post(self.wizard_url, self.wizard_step_data[2]) response = self.client.post(self.wizard_url, self.wizard_step_data[2])
@ -189,9 +213,9 @@ class WizardTests(object):
self.assertEqual(response.context['wizard']['steps'].current, 'form2') self.assertEqual(response.context['wizard']['steps'].current, 'form2')
post_data = self.wizard_step_data[1] post_data = self.wizard_step_data[1]
post_data['form2-file1'].close() with open(upath(__file__), 'rb') as post_file:
post_data['form2-file1'] = open(upath(__file__), 'rb') post_data['form2-file1'] = post_file
response = self.client.post(self.wizard_url, post_data) response = self.client.post(self.wizard_url, post_data)
self.assertEqual(response.status_code, 200) self.assertEqual(response.status_code, 200)
self.assertEqual(response.context['wizard']['steps'].current, 'form3') self.assertEqual(response.context['wizard']['steps'].current, 'form3')

View File

@ -16,6 +16,7 @@ class BaseStorage(object):
self.prefix = 'wizard_%s' % prefix self.prefix = 'wizard_%s' % prefix
self.request = request self.request = request
self.file_storage = file_storage self.file_storage = file_storage
self._files = {}
def init_data(self): def init_data(self):
self.data = { self.data = {
@ -77,8 +78,10 @@ class BaseStorage(object):
for field, field_dict in six.iteritems(wizard_files): for field, field_dict in six.iteritems(wizard_files):
field_dict = field_dict.copy() field_dict = field_dict.copy()
tmp_name = field_dict.pop('tmp_name') tmp_name = field_dict.pop('tmp_name')
files[field] = UploadedFile( if (step, field) not in self._files:
file=self.file_storage.open(tmp_name), **field_dict) self._files[(step, field)] = UploadedFile(
file=self.file_storage.open(tmp_name), **field_dict)
files[field] = self._files[(step, field)]
return files or None return files or None
def set_step_files(self, step, files): def set_step_files(self, step, files):
@ -106,4 +109,12 @@ class BaseStorage(object):
return self.get_step_files(self.current_step) return self.get_step_files(self.current_step)
def update_response(self, response): def update_response(self, response):
pass def post_render_callback(response):
for file in self._files.values():
if not file.closed:
file.close()
if hasattr(response, 'render'):
response.add_post_render_callback(post_render_callback)
else:
post_render_callback(response)

View File

@ -27,6 +27,7 @@ class CookieStorage(storage.BaseStorage):
return json.loads(data, cls=json.JSONDecoder) return json.loads(data, cls=json.JSONDecoder)
def update_response(self, response): def update_response(self, response):
super(CookieStorage, self).update_response(response)
if self.data: if self.data:
response.set_signed_cookie(self.prefix, self.encoder.encode(self.data)) response.set_signed_cookie(self.prefix, self.encoder.encode(self.data))
else: else: