blob: 8e22acd2c4766b8c91c6cd0c4b01f46d7c644a20 [file] [log] [blame]
Mitja Nikolaus03e412b2018-09-18 17:50:15 +02001"""Utility functions shared by all crashreport stats tests."""
2
Mitja Nikolaus7dc86722018-11-27 14:57:39 +01003from datetime import datetime
Mitja Nikolaus03e412b2018-09-18 17:50:15 +02004
5import pytz
6from django.contrib.auth.models import Group
7from rest_framework import status
8from rest_framework.authtoken.models import Token
9from rest_framework.test import APITestCase, APIClient
10
11from crashreport_stats.models import (
12 Version,
13 VersionDaily,
14 RadioVersion,
15 RadioVersionDaily,
16 StatsMetadata,
17)
18
Mitja Nikolaus7dc86722018-11-27 14:57:39 +010019from crashreports.models import User
20from crashreports.tests.utils import Dummy as CrashreportsDummy
Mitja Nikolaus03e412b2018-09-18 17:50:15 +020021from hiccup.allauth_adapters import FP_STAFF_GROUP_NAME
22
23
Mitja Nikolaus7dc86722018-11-27 14:57:39 +010024class Dummy(CrashreportsDummy):
Mitja Nikolaus03e412b2018-09-18 17:50:15 +020025 """Class for creating dummy instances for testing."""
26
Mitja Nikolaus77dd5652018-12-06 11:27:01 +010027 DEFAULT_VERSION_VALUES = {
Mitja Nikolaus7dc86722018-11-27 14:57:39 +010028 "build_fingerprint": CrashreportsDummy.BUILD_FINGERPRINTS[0],
29 "first_seen_on": CrashreportsDummy.DATES[1],
30 "released_on": CrashreportsDummy.DATES[0],
Mitja Nikolaus03e412b2018-09-18 17:50:15 +020031 "is_beta_release": False,
32 "is_official_release": True,
33 }
34
Mitja Nikolaus77dd5652018-12-06 11:27:01 +010035 DEFAULT_VERSION_DAILY_VALUES = {"date": CrashreportsDummy.DATES[1]}
Mitja Nikolaus03e412b2018-09-18 17:50:15 +020036
Mitja Nikolaus77dd5652018-12-06 11:27:01 +010037 DEFAULT_RADIO_VERSION_VALUES = {
Mitja Nikolaus7dc86722018-11-27 14:57:39 +010038 "radio_version": CrashreportsDummy.RADIO_VERSIONS[0],
39 "first_seen_on": CrashreportsDummy.DATES[1],
40 "released_on": CrashreportsDummy.DATES[0],
Mitja Nikolaus03e412b2018-09-18 17:50:15 +020041 }
42
Mitja Nikolaus77dd5652018-12-06 11:27:01 +010043 DEFAULT_RADIO_VERSION_DAILY_VALUES = {"date": CrashreportsDummy.DATES[1]}
Mitja Nikolaus03e412b2018-09-18 17:50:15 +020044
Mitja Nikolaus77dd5652018-12-06 11:27:01 +010045 DEFAULT_STATSMETADATA_VALUES = {
Mitja Nikolaus03e412b2018-09-18 17:50:15 +020046 "updated_at": datetime(2018, 6, 15, 2, 12, 24, tzinfo=pytz.utc)
47 }
48
Mitja Nikolaus03e412b2018-09-18 17:50:15 +020049 @staticmethod
Mitja Nikolaus77dd5652018-12-06 11:27:01 +010050 def create_version(version_type=Version, **kwargs):
Mitja Nikolaus03e412b2018-09-18 17:50:15 +020051 """Create a dummy version instance.
52
53 The dummy instance is created and saved to the database.
54 Args:
Mitja Nikolaus35a02652018-10-01 15:10:18 +020055 version_type: The class of the version type to be created.
Mitja Nikolaus03e412b2018-09-18 17:50:15 +020056 **kwargs:
57 Optional arguments to extend/overwrite the default values.
58
59 Returns: The created version instance.
60
Mitja Nikolaus35a02652018-10-01 15:10:18 +020061 Raises:
62 ValueError: If version_type is not a valid version class type.
Mitja Nikolaus03e412b2018-09-18 17:50:15 +020063
64 """
Mitja Nikolaus35a02652018-10-01 15:10:18 +020065 if version_type == Version:
66 entity = Version(
Mitja Nikolaus77dd5652018-12-06 11:27:01 +010067 **Dummy._update_copy(Dummy.DEFAULT_VERSION_VALUES, kwargs)
Mitja Nikolaus03e412b2018-09-18 17:50:15 +020068 )
Mitja Nikolaus35a02652018-10-01 15:10:18 +020069 elif version_type == RadioVersion:
70 entity = RadioVersion(
Mitja Nikolaus77dd5652018-12-06 11:27:01 +010071 **Dummy._update_copy(Dummy.DEFAULT_RADIO_VERSION_VALUES, kwargs)
Mitja Nikolaus35a02652018-10-01 15:10:18 +020072 )
73 else:
74 raise ValueError(
75 "No dummy version instance can be created for {}".format(
76 version_type.__name__
77 )
78 )
Mitja Nikolaus03e412b2018-09-18 17:50:15 +020079 entity.save()
80 return entity
81
82 @staticmethod
Mitja Nikolaus77dd5652018-12-06 11:27:01 +010083 def create_daily_version(version, **kwargs):
Mitja Nikolaus03e412b2018-09-18 17:50:15 +020084 """Create a dummy daily version instance.
85
86 The dummy instance is created and saved to the database.
87 Args:
88 **kwargs:
89 Optional arguments to extend/overwrite the default values.
90
91 Returns: The created daily version instance.
92
93 """
94 entity = VersionDaily(
95 version=version,
Mitja Nikolaus77dd5652018-12-06 11:27:01 +010096 **Dummy._update_copy(Dummy.DEFAULT_VERSION_DAILY_VALUES, kwargs)
Mitja Nikolaus03e412b2018-09-18 17:50:15 +020097 )
98 entity.save()
99 return entity
100
101 @staticmethod
Mitja Nikolaus77dd5652018-12-06 11:27:01 +0100102 def create_daily_radio_version(version, **kwargs):
Mitja Nikolaus03e412b2018-09-18 17:50:15 +0200103 """Create a dummy daily radio version instance.
104
105 The dummy instance is created and saved to the database.
106 Args:
107 **kwargs:
108 Optional arguments to extend/overwrite the default values.
109
110 Returns: The created daily radio version instance.
111
112 """
113 entity = RadioVersionDaily(
114 version=version,
Mitja Nikolaus7dc86722018-11-27 14:57:39 +0100115 **Dummy._update_copy(
Mitja Nikolaus77dd5652018-12-06 11:27:01 +0100116 Dummy.DEFAULT_RADIO_VERSION_DAILY_VALUES, kwargs
Mitja Nikolaus03e412b2018-09-18 17:50:15 +0200117 )
118 )
119 entity.save()
120 return entity
121
122 @staticmethod
Mitja Nikolaus77dd5652018-12-06 11:27:01 +0100123 def create_stats_metadata(**kwargs):
Mitja Nikolaus03e412b2018-09-18 17:50:15 +0200124 """Create a dummy stats metadata instance.
125
126 The dummy instance is created and saved to the database.
127 Args:
128 **kwargs:
129 Optional arguments to extend/overwrite the default values.
130
131 Returns: The created stats metadata instance.
132
133 """
134 entity = StatsMetadata(
Mitja Nikolaus77dd5652018-12-06 11:27:01 +0100135 **Dummy._update_copy(Dummy.DEFAULT_STATSMETADATA_VALUES, kwargs)
Mitja Nikolaus03e412b2018-09-18 17:50:15 +0200136 )
137 entity.save()
138 return entity
139
140
141class HiccupStatsAPITestCase(APITestCase):
142 """Abstract class for Hiccup stats REST API test cases to inherit from."""
143
144 @classmethod
145 def setUpTestData(cls): # noqa: N802
Mitja Nikolause0e83772018-11-05 10:00:53 +0100146 """Create client users for accessing the API.
Mitja Nikolaus03e412b2018-09-18 17:50:15 +0200147
148 The APIClient that can be used to make authenticated requests as
Mitja Nikolause0e83772018-11-05 10:00:53 +0100149 Fairphone staff user is stored in self.fp_staff_client. Additionally, a
150 client which is related to a device owner user is stored in
151 self.device_owner_client.
Mitja Nikolaus03e412b2018-09-18 17:50:15 +0200152 """
Mitja Nikolausd6907dd2018-10-17 14:13:06 +0200153 fp_staff_group = Group.objects.get(name=FP_STAFF_GROUP_NAME)
Mitja Nikolaus03e412b2018-09-18 17:50:15 +0200154 fp_staff_user = User.objects.create_user(
155 "fp_staff", "somebody@fairphone.com", "thepassword"
156 )
157 fp_staff_user.groups.add(fp_staff_group)
158 cls.fp_staff_client = APIClient()
159 cls.fp_staff_client.force_login(fp_staff_user)
160
161 cls.device_owner_user = User.objects.create_user(
162 "device_owner", "somebody@somemail.com", "thepassword"
163 )
164 Token.objects.create(user=cls.device_owner_user)
Mitja Nikolaus77dd5652018-12-06 11:27:01 +0100165 cls.device_owner_device = Dummy.create_device(
Mitja Nikolaus03e412b2018-09-18 17:50:15 +0200166 user=cls.device_owner_user
167 )
168 cls.device_owner_client = APIClient()
169 cls.device_owner_client.credentials(
170 HTTP_AUTHORIZATION="Token " + cls.device_owner_user.auth_token.key
171 )
172
Mitja Nikolaus03e412b2018-09-18 17:50:15 +0200173 def _assert_get_as_fp_staff_succeeds(
174 self, url, expected_status=status.HTTP_200_OK
175 ):
176 response = self.fp_staff_client.get(url)
177 self.assertEqual(response.status_code, expected_status)
178
179 def _assert_get_without_authentication_fails(
180 self, url, expected_status=status.HTTP_401_UNAUTHORIZED
181 ):
182 response = self.client.get(url)
183 self.assertEqual(response.status_code, expected_status)
184
185 def _assert_get_as_device_owner_fails(
186 self, url, expected_status=status.HTTP_403_FORBIDDEN
187 ):
188 response = self.device_owner_client.get(url)
189 self.assertEqual(response.status_code, expected_status)