projects can be public, registered, private

This commit is contained in:
Élie Bouttier 2014-08-22 23:14:32 -07:00
parent c0e1dcdde0
commit 6ad5e46776
6 changed files with 68 additions and 60 deletions

File diff suppressed because one or more lines are too long

View file

@ -8,9 +8,9 @@ from issue.models import *
AddProjectForm = modelform_factory(Project,
fields=['display_name', 'name', 'description', 'public'])
fields=['display_name', 'name', 'description', 'access'])
EditProjectForm = modelform_factory(Project,
fields=['display_name', 'description', 'public'])
fields=['display_name', 'description', 'access'])
LabelForm = modelform_factory(Label,
fields=['name', 'color', 'inverted'])
TeamForm = modelform_factory(Team,

View file

@ -29,8 +29,9 @@ class ProjectMiddleware:
if request.user.is_authenticated() and request.user.is_staff:
projects = Project.objects.all()
else:
query = Q(public=True)
query = Q(access=Project.ACCESS_PUBLIC)
if request.user.is_authenticated():
query |= Q(access=Project.ACCESS_REGISTERED)
# access granted through a team
teams = request.user.teams.values_list('name')
query |= Q(permissions__grantee_type=PermModel.GRANTEE_TEAM,

View file

@ -0,0 +1,24 @@
# -*- coding: utf-8 -*-
from __future__ import unicode_literals
from django.db import models, migrations
class Migration(migrations.Migration):
dependencies = [
('issue', '0001_initial'),
]
operations = [
migrations.AddField(
model_name='project',
name='access',
field=models.IntegerField(choices=[(1, 'Public'), (2, 'Connected users'), (3, 'Private')], default=1),
preserve_default=True,
),
migrations.RemoveField(
model_name='project',
name='public',
),
]

View file

@ -29,6 +29,15 @@ class User(AbstractUser):
@python_2_unicode_compatible
class Project(models.Model):
ACCESS_PUBLIC = 1
ACCESS_REGISTERED = 2
ACCESS_PRIVATE = 3
ACCESS_TYPE = (
(ACCESS_PUBLIC, 'Public'),
(ACCESS_REGISTERED, 'Registration required'),
(ACCESS_PRIVATE, 'Private'),
)
url_name_validator = RegexValidator(regex='^[a-z0-9_-]+$',
message="Please enter only lowercase characters, number, "
"underscores or hyphens.")
@ -43,9 +52,7 @@ class Project(models.Model):
description = models.TextField(blank=True, default="",
verbose_name="Description")
public = models.BooleanField(default=True,
verbose_name="Do unregistered users have read access "
"to this project?")
access = models.IntegerField(choices=ACCESS_TYPE, default=ACCESS_PUBLIC)
subscribers = models.ManyToManyField(User, blank=True, null=True,
related_name='subscribed_projects')

View file

@ -72,32 +72,6 @@ class TestPermissions(TestCase):
self.assertFalse(user.has_perm('modify_issue', project))
self.assertTrue(user.has_perm('delete_issue', project))
def test_unregistered_project_list(self):
response = self.client.get('/')
projects = response.context['projects']
project = Project.objects.get(name='project-1')
self.assertEqual(len(projects), 1)
self.assertEqual(projects[0], project)
def test_ungranted_project_list(self):
self.client.login(username='user1', password='user1')
response = self.client.get('/')
projects = response.context['projects']
project = Project.objects.get(name='project-1')
self.assertEqual(len(projects), 1)
self.assertEqual(projects[0], project)
self.client.logout()
def test_granted_project_list(self):
self.client.login(username='user2', password='user2')
response = self.client.get('/')
projects = response.context['projects']
project1 = Project.objects.get(name='project-1')
project2 = Project.objects.get(name='project-2')
self.assertEqual(len(projects), 2)
self.assertTrue(project1 in projects)
self.assertTrue(project2 in projects)
class TestNoProject(TestCase):
@ -180,7 +154,8 @@ class TestProjectsViews(TestCase):
lambda x: x)
def test_home_as_user1(self):
expected = Project.objects.filter(name='project-1')
expected = Project.objects \
.filter(Q(name='project-1') | Q(name='project-3'))
self.client.login(username='user1', password='user1')
url = reverse('list-project')
self.assertEqual(url, '/')
@ -202,7 +177,8 @@ class TestProjectsViews(TestCase):
self.assertNotContains(response, 'New project')
def test_home_as_user3(self):
expected = Project.objects.filter(name='project-1')
expected = Project.objects \
.filter(Q(name='project-1') | Q(name='project-3'))
self.client.login(username='user3', password='user3')
url = reverse('list-project')
self.assertEqual(url, '/')
@ -225,41 +201,41 @@ class TestProjectsViews(TestCase):
def test_add_project_granted(self):
self.client.login(username='user3', password='user3')
expected_url = reverse('list-project-permission', args=['project-3'])
expected_url = reverse('list-project-permission', args=['project-4'])
url = reverse('add-project')
response = self.client.post(url, {
'name': 'project-3',
'display_name': 'Project 3',
'description': 'This is the third project.',
'name': 'project-4',
'display_name': 'Project 4',
'description': 'This is the fourth project.',
'access': Project.ACCESS_PUBLIC,
})
self.assertRedirects(response, expected_url)
self.assertQuerysetEqual(Project.objects.all(),
['project-1', 'project-2', 'project-3'],
lambda x: x.name, ordered=False)
self.assertQuerysetEqual(Project.objects.all(), ['project-%s' % x
for x in (1, 2, 3, 4)], lambda x: x.name, ordered=False)
def test_add_project_forbidden(self):
self.client.login(username='user1', password='user1')
url = reverse('add-project')
response = self.client.post(url, {
'name': 'project-3',
'display_name': 'Project 3',
'description': 'This is the third project.',
'name': 'project-4',
'display_name': 'Project 4',
'description': 'This is the foorth project.',
})
self.assertEqual(response.status_code, 403)
self.assertQuerysetEqual(Project.objects.all(),
['project-1', 'project-2'], lambda x: x.name, ordered=False)
self.assertQuerysetEqual(Project.objects.all(), ['project-%s' % x
for x in (1, 2, 3)], lambda x: x.name, ordered=False)
def test_add_project_forbidden_ano(self):
expected_url = reverse('login') + '?next=' + reverse('add-project')
url = reverse('add-project')
response = self.client.post(url, {
'name': 'project-3',
'display_name': 'Project 3',
'description': 'This is the third project.',
'name': 'project-4',
'display_name': 'Project 4',
'description': 'This is the foorth project.',
})
self.assertRedirects(response, expected_url)
self.assertQuerysetEqual(Project.objects.all(),
['project-1', 'project-2'], lambda x: x.name, ordered=False)
self.assertQuerysetEqual(Project.objects.all(), ['project-%s' % x
for x in (1, 2, 3)], lambda x: x.name, ordered=False)
def test_delete_project_get(self):
self.client.login(username='user1', password='user1')
@ -267,8 +243,8 @@ class TestProjectsViews(TestCase):
url = reverse('delete-project', args=['project-1'])
response = self.client.get(url)
self.assertEqual(response.status_code, 405)
self.assertQuerysetEqual(Project.objects.all(),
['project-1', 'project-2'], lambda x: x.name, ordered=False)
self.assertQuerysetEqual(Project.objects.all(), ['project-%s' % x
for x in (1, 2, 3)], lambda x: x.name, ordered=False)
def test_delete_project_granted(self):
self.client.login(username='user1', password='user1')
@ -276,16 +252,16 @@ class TestProjectsViews(TestCase):
url = reverse('delete-project', args=['project-1'])
response = self.client.post(url)
self.assertRedirects(response, expected_url)
self.assertQuerysetEqual(Project.objects.all(),
['project-2'], lambda x: x.name, ordered=False)
self.assertQuerysetEqual(Project.objects.all(), ['project-%s' % x
for x in (2, 3)], lambda x: x.name, ordered=False)
def test_delete_project_forbidden(self):
self.client.login(username='user2', password='user2')
url = reverse('delete-project', args=['project-1'])
response = self.client.post(url)
self.assertEqual(response.status_code, 403)
self.assertQuerysetEqual(Project.objects.all(),
['project-1', 'project-2'], lambda x: x.name, ordered=False)
self.assertQuerysetEqual(Project.objects.all(), ['project-%s' % x
for x in (1, 2, 3)], lambda x: x.name, ordered=False)
def test_delete_project_forbidden_ano(self):
expected_url = reverse('login') + '?next=' \
@ -293,8 +269,8 @@ class TestProjectsViews(TestCase):
url = reverse('delete-project', args=['project-1'])
response = self.client.post(url)
self.assertRedirects(response, expected_url)
self.assertQuerysetEqual(Project.objects.all(),
['project-1', 'project-2'], lambda x: x.name, ordered=False)
self.assertQuerysetEqual(Project.objects.all(), ['project-%s' % x
for x in (1, 2, 3)], lambda x: x.name, ordered=False)
class TestIssuesViews(TestCase):