Fixing attachments with long names
parent
099fae2c42
commit
26d10ca7a3
|
@ -0,0 +1,42 @@
|
||||||
|
# Copyright (C) 2014-2016 Andrey Antukh <niwi@niwi.be>
|
||||||
|
# Copyright (C) 2014-2016 Jesús Espino <jespinog@gmail.com>
|
||||||
|
# Copyright (C) 2014-2016 David Barragán <bameda@dbarragan.com>
|
||||||
|
# Copyright (C) 2014-2016 Alejandro Alonso <alejandro.alonso@kaleidos.net>
|
||||||
|
# This program is free software: you can redistribute it and/or modify
|
||||||
|
# it under the terms of the GNU Affero General Public License as
|
||||||
|
# published by the Free Software Foundation, either version 3 of the
|
||||||
|
# License, or (at your option) any later version.
|
||||||
|
#
|
||||||
|
# This program is distributed in the hope that it will be useful,
|
||||||
|
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
# GNU Affero General Public License for more details.
|
||||||
|
#
|
||||||
|
# You should have received a copy of the GNU Affero General Public License
|
||||||
|
# along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
|
import hashlib
|
||||||
|
|
||||||
|
from os import path, urandom
|
||||||
|
from unidecode import unidecode
|
||||||
|
|
||||||
|
from django.template.defaultfilters import slugify
|
||||||
|
from django.utils import timezone
|
||||||
|
from django.utils.encoding import force_bytes
|
||||||
|
|
||||||
|
from taiga.base.utils.iterators import split_by_n
|
||||||
|
|
||||||
|
def get_file_path(instance, filename, base_path):
|
||||||
|
basename = path.basename(filename).lower()
|
||||||
|
base, ext = path.splitext(basename)
|
||||||
|
base = slugify(unidecode(base))[0:100]
|
||||||
|
basename = "".join([base, ext])
|
||||||
|
|
||||||
|
hs = hashlib.sha256()
|
||||||
|
hs.update(force_bytes(timezone.now().isoformat()))
|
||||||
|
hs.update(urandom(1024))
|
||||||
|
|
||||||
|
p1, p2, p3, p4, *p5 = split_by_n(hs.hexdigest(), 1)
|
||||||
|
hash_part = path.join(p1, p2, p3, p4, "".join(p5))
|
||||||
|
|
||||||
|
return path.join(base_path, hash_part, basename)
|
|
@ -16,35 +16,20 @@
|
||||||
# along with this program. If not, see <http://www.gnu.org/licenses/>.
|
# along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
import hashlib
|
import hashlib
|
||||||
import os
|
|
||||||
import os.path as path
|
|
||||||
|
|
||||||
from unidecode import unidecode
|
|
||||||
|
|
||||||
from django.db import models
|
from django.db import models
|
||||||
from django.conf import settings
|
from django.conf import settings
|
||||||
from django.contrib.contenttypes.models import ContentType
|
from django.contrib.contenttypes.models import ContentType
|
||||||
from django.contrib.contenttypes.fields import GenericForeignKey
|
from django.contrib.contenttypes.fields import GenericForeignKey
|
||||||
from django.utils import timezone
|
from django.utils import timezone
|
||||||
from django.utils.encoding import force_bytes
|
|
||||||
from django.utils.translation import ugettext_lazy as _
|
from django.utils.translation import ugettext_lazy as _
|
||||||
from django.utils.text import get_valid_filename
|
from django.utils.text import get_valid_filename
|
||||||
|
|
||||||
from taiga.base.utils.iterators import split_by_n
|
from taiga.base.utils.files import get_file_path
|
||||||
|
|
||||||
|
|
||||||
def get_attachment_file_path(instance, filename):
|
def get_attachment_file_path(instance, filename):
|
||||||
basename = path.basename(filename)
|
return get_file_path(instance, filename, "attachments")
|
||||||
basename = get_valid_filename(basename)
|
|
||||||
|
|
||||||
hs = hashlib.sha256()
|
|
||||||
hs.update(force_bytes(timezone.now().isoformat()))
|
|
||||||
hs.update(os.urandom(1024))
|
|
||||||
|
|
||||||
p1, p2, p3, p4, *p5 = split_by_n(hs.hexdigest(), 1)
|
|
||||||
hash_part = path.join(p1, p2, p3, p4, "".join(p5))
|
|
||||||
|
|
||||||
return path.join("attachments", hash_part, basename)
|
|
||||||
|
|
||||||
|
|
||||||
class Attachment(models.Model):
|
class Attachment(models.Model):
|
||||||
|
|
|
@ -15,6 +15,6 @@ class Migration(migrations.Migration):
|
||||||
migrations.AddField(
|
migrations.AddField(
|
||||||
model_name='project',
|
model_name='project',
|
||||||
name='logo',
|
name='logo',
|
||||||
field=models.FileField(null=True, blank=True, upload_to=taiga.projects.models.get_user_file_path, verbose_name='logo', max_length=500),
|
field=models.FileField(null=True, blank=True, upload_to=taiga.projects.models.get_project_logo_file_path, verbose_name='logo', max_length=500),
|
||||||
),
|
),
|
||||||
]
|
]
|
||||||
|
|
|
@ -15,14 +15,9 @@
|
||||||
# You should have received a copy of the GNU Affero General Public License
|
# You should have received a copy of the GNU Affero General Public License
|
||||||
# along with this program. If not, see <http://www.gnu.org/licenses/>.
|
# along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
import hashlib
|
|
||||||
import os
|
|
||||||
import os.path as path
|
|
||||||
import itertools
|
import itertools
|
||||||
import uuid
|
import uuid
|
||||||
|
|
||||||
from unidecode import unidecode
|
|
||||||
|
|
||||||
from django.conf import settings
|
from django.conf import settings
|
||||||
from django.core.exceptions import ValidationError
|
from django.core.exceptions import ValidationError
|
||||||
from django.db import models
|
from django.db import models
|
||||||
|
@ -32,8 +27,6 @@ from django.conf import settings
|
||||||
from django.dispatch import receiver
|
from django.dispatch import receiver
|
||||||
from django.contrib.auth import get_user_model
|
from django.contrib.auth import get_user_model
|
||||||
from django.utils.translation import ugettext_lazy as _
|
from django.utils.translation import ugettext_lazy as _
|
||||||
from django.template.defaultfilters import slugify
|
|
||||||
from django.utils.encoding import force_bytes
|
|
||||||
from django.utils import timezone
|
from django.utils import timezone
|
||||||
|
|
||||||
from django_pgjson.fields import JsonField
|
from django_pgjson.fields import JsonField
|
||||||
|
@ -41,7 +34,7 @@ from djorm_pgarray.fields import TextArrayField
|
||||||
|
|
||||||
from taiga.base.tags import TaggedMixin
|
from taiga.base.tags import TaggedMixin
|
||||||
from taiga.base.utils.dicts import dict_sum
|
from taiga.base.utils.dicts import dict_sum
|
||||||
from taiga.base.utils.iterators import split_by_n
|
from taiga.base.utils.files import get_file_path
|
||||||
from taiga.base.utils.sequence import arithmetic_progression
|
from taiga.base.utils.sequence import arithmetic_progression
|
||||||
from taiga.base.utils.slug import slugify_uniquely
|
from taiga.base.utils.slug import slugify_uniquely
|
||||||
from taiga.base.utils.slug import slugify_uniquely_for_queryset
|
from taiga.base.utils.slug import slugify_uniquely_for_queryset
|
||||||
|
@ -62,20 +55,8 @@ from . import choices
|
||||||
from dateutil.relativedelta import relativedelta
|
from dateutil.relativedelta import relativedelta
|
||||||
|
|
||||||
|
|
||||||
def get_user_file_path(instance, filename):
|
def get_project_logo_file_path(instance, filename):
|
||||||
basename = path.basename(filename).lower()
|
return get_file_path(instance, filename, "project")
|
||||||
base, ext = path.splitext(basename)
|
|
||||||
base = slugify(unidecode(base))
|
|
||||||
basename = "".join([base, ext])
|
|
||||||
|
|
||||||
hs = hashlib.sha256()
|
|
||||||
hs.update(force_bytes(timezone.now().isoformat()))
|
|
||||||
hs.update(os.urandom(1024))
|
|
||||||
|
|
||||||
p1, p2, p3, p4, *p5 = split_by_n(hs.hexdigest(), 1)
|
|
||||||
hash_part = path.join(p1, p2, p3, p4, "".join(p5))
|
|
||||||
|
|
||||||
return path.join("project", hash_part, basename)
|
|
||||||
|
|
||||||
|
|
||||||
class Membership(models.Model):
|
class Membership(models.Model):
|
||||||
|
@ -167,7 +148,7 @@ class Project(ProjectDefaults, TaggedMixin, models.Model):
|
||||||
description = models.TextField(null=False, blank=False,
|
description = models.TextField(null=False, blank=False,
|
||||||
verbose_name=_("description"))
|
verbose_name=_("description"))
|
||||||
|
|
||||||
logo = models.FileField(upload_to=get_user_file_path,
|
logo = models.FileField(upload_to=get_project_logo_file_path,
|
||||||
max_length=500, null=True, blank=True,
|
max_length=500, null=True, blank=True,
|
||||||
verbose_name=_("logo"))
|
verbose_name=_("logo"))
|
||||||
|
|
||||||
|
|
|
@ -15,15 +15,10 @@
|
||||||
# You should have received a copy of the GNU Affero General Public License
|
# You should have received a copy of the GNU Affero General Public License
|
||||||
# along with this program. If not, see <http://www.gnu.org/licenses/>.
|
# along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
import hashlib
|
|
||||||
import os
|
|
||||||
import os.path as path
|
|
||||||
import random
|
import random
|
||||||
import re
|
import re
|
||||||
import uuid
|
import uuid
|
||||||
|
|
||||||
from unidecode import unidecode
|
|
||||||
|
|
||||||
from django.apps import apps
|
from django.apps import apps
|
||||||
from django.conf import settings
|
from django.conf import settings
|
||||||
from django.contrib.contenttypes.models import ContentType
|
from django.contrib.contenttypes.models import ContentType
|
||||||
|
@ -33,15 +28,13 @@ from django.utils.translation import ugettext_lazy as _
|
||||||
from django.contrib.auth.models import UserManager, AbstractBaseUser
|
from django.contrib.auth.models import UserManager, AbstractBaseUser
|
||||||
from django.core import validators
|
from django.core import validators
|
||||||
from django.utils import timezone
|
from django.utils import timezone
|
||||||
from django.utils.encoding import force_bytes
|
|
||||||
from django.template.defaultfilters import slugify
|
|
||||||
|
|
||||||
from django_pgjson.fields import JsonField
|
from django_pgjson.fields import JsonField
|
||||||
from djorm_pgarray.fields import TextArrayField
|
from djorm_pgarray.fields import TextArrayField
|
||||||
|
|
||||||
from taiga.auth.tokens import get_token_for_user
|
from taiga.auth.tokens import get_token_for_user
|
||||||
from taiga.base.utils.slug import slugify_uniquely
|
from taiga.base.utils.slug import slugify_uniquely
|
||||||
from taiga.base.utils.iterators import split_by_n
|
from taiga.base.utils.files import get_file_path
|
||||||
from taiga.permissions.permissions import MEMBERS_PERMISSIONS
|
from taiga.permissions.permissions import MEMBERS_PERMISSIONS
|
||||||
from taiga.projects.choices import BLOCKED_BY_OWNER_LEAVING
|
from taiga.projects.choices import BLOCKED_BY_OWNER_LEAVING
|
||||||
from taiga.projects.notifications.choices import NotifyLevel
|
from taiga.projects.notifications.choices import NotifyLevel
|
||||||
|
@ -54,19 +47,7 @@ def generate_random_hex_color():
|
||||||
|
|
||||||
|
|
||||||
def get_user_file_path(instance, filename):
|
def get_user_file_path(instance, filename):
|
||||||
basename = path.basename(filename).lower()
|
return get_file_path(instance, filename, "user")
|
||||||
base, ext = path.splitext(basename)
|
|
||||||
base = slugify(unidecode(base))[0:100]
|
|
||||||
basename = "".join([base, ext])
|
|
||||||
|
|
||||||
hs = hashlib.sha256()
|
|
||||||
hs.update(force_bytes(timezone.now().isoformat()))
|
|
||||||
hs.update(os.urandom(1024))
|
|
||||||
|
|
||||||
p1, p2, p3, p4, *p5 = split_by_n(hs.hexdigest(), 1)
|
|
||||||
hash_part = path.join(p1, p2, p3, p4, "".join(p5))
|
|
||||||
|
|
||||||
return path.join("user", hash_part, basename)
|
|
||||||
|
|
||||||
|
|
||||||
class PermissionsMixin(models.Model):
|
class PermissionsMixin(models.Model):
|
||||||
|
|
|
@ -45,3 +45,19 @@ def test_create_attachment_on_wrong_project(client):
|
||||||
client.login(issue1.owner)
|
client.login(issue1.owner)
|
||||||
response = client.post(url, data)
|
response = client.post(url, data)
|
||||||
assert response.status_code == 400
|
assert response.status_code == 400
|
||||||
|
|
||||||
|
|
||||||
|
def test_create_attachment_with_long_file_name(client):
|
||||||
|
issue1 = f.create_issue()
|
||||||
|
f.MembershipFactory(project=issue1.project, user=issue1.owner, is_owner=True)
|
||||||
|
|
||||||
|
url = reverse("issue-attachments-list")
|
||||||
|
|
||||||
|
data = {"description": "test",
|
||||||
|
"object_id": issue1.pk,
|
||||||
|
"project": issue1.project.id,
|
||||||
|
"attached_file": SimpleUploadedFile(500*"x"+".txt", b"test")}
|
||||||
|
|
||||||
|
client.login(issue1.owner)
|
||||||
|
response = client.post(url, data)
|
||||||
|
assert response.data["attached_file"].endswith("/"+100*"x"+".txt")
|
||||||
|
|
|
@ -629,6 +629,24 @@ def test_update_project_logo(client):
|
||||||
assert not any(list(map(os.path.exists, original_photo_paths)))
|
assert not any(list(map(os.path.exists, original_photo_paths)))
|
||||||
|
|
||||||
|
|
||||||
|
@pytest.mark.django_db(transaction=True)
|
||||||
|
def test_update_project_logo_with_long_file_name(client):
|
||||||
|
user = f.UserFactory.create(is_superuser=True)
|
||||||
|
project = f.create_project()
|
||||||
|
url = reverse("projects-change-logo", args=(project.id,))
|
||||||
|
|
||||||
|
with NamedTemporaryFile(delete=False) as logo:
|
||||||
|
logo.name=500*"x"+".bmp"
|
||||||
|
logo.write(DUMMY_BMP_DATA)
|
||||||
|
logo.seek(0)
|
||||||
|
|
||||||
|
client.login(user)
|
||||||
|
post_data = {'logo': logo}
|
||||||
|
response = client.post(url, post_data)
|
||||||
|
|
||||||
|
assert response.status_code == 200
|
||||||
|
|
||||||
|
|
||||||
@pytest.mark.django_db(transaction=True)
|
@pytest.mark.django_db(transaction=True)
|
||||||
def test_remove_project_logo(client):
|
def test_remove_project_logo(client):
|
||||||
user = f.UserFactory.create(is_superuser=True)
|
user = f.UserFactory.create(is_superuser=True)
|
||||||
|
|
Loading…
Reference in New Issue