tests.py 8.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243
  1. from unittest import skipIf
  2. from django.conf import settings
  3. from django.test import TestCase
  4. from django.urls import reverse
  5. from django.utils import timezone
  6. from djstripe.enums import BillingScheme
  7. from freezegun import freeze_time
  8. from model_bakery import baker
  9. class SubscriptionAPITestCase(TestCase):
  10. @classmethod
  11. def setUpTestData(cls):
  12. cls.user = baker.make("users.user")
  13. cls.organization = baker.make("organizations_ext.Organization")
  14. cls.organization.add_user(cls.user)
  15. cls.url = reverse("api:create_subscription")
  16. def setUp(self):
  17. self.client.force_login(self.user)
  18. def test_detail(self):
  19. customer = baker.make("djstripe.Customer", subscriber=self.organization)
  20. subscription = baker.make(
  21. "djstripe.Subscription",
  22. status="active",
  23. customer=customer,
  24. livemode=False,
  25. created=timezone.make_aware(timezone.datetime(2020, 1, 2)),
  26. )
  27. # Should get most recent
  28. baker.make(
  29. "djstripe.Subscription",
  30. customer=customer,
  31. livemode=False,
  32. created=timezone.make_aware(timezone.datetime(2020, 1, 1)),
  33. )
  34. # should not get canceled subscriptions
  35. baker.make(
  36. "djstripe.Subscription",
  37. status="canceled",
  38. customer=customer,
  39. livemode=False,
  40. created=timezone.make_aware(timezone.datetime(2020, 1, 3)),
  41. )
  42. baker.make("djstripe.Subscription")
  43. url = reverse("api:get_subscription", args=[self.organization.slug])
  44. res = self.client.get(url)
  45. self.assertContains(res, subscription.id)
  46. def test_events_count(self):
  47. """
  48. Event count should be accurate and work when there are multiple subscriptions for a given customer
  49. """
  50. customer = baker.make("djstripe.Customer", subscriber=self.organization)
  51. # Ensure we don't filter on any unrelated subscription
  52. baker.make("djstripe.Subscription", status="active")
  53. baker.make(
  54. "djstripe.Subscription",
  55. customer=customer,
  56. livemode=False,
  57. status="cancelled",
  58. current_period_start=timezone.make_aware(timezone.datetime(2019, 1, 2)),
  59. current_period_end=timezone.make_aware(timezone.datetime(2019, 2, 2)),
  60. )
  61. baker.make(
  62. "djstripe.Subscription",
  63. customer=customer,
  64. livemode=False,
  65. status="active",
  66. current_period_start=timezone.make_aware(timezone.datetime(2020, 1, 2)),
  67. current_period_end=timezone.make_aware(timezone.datetime(2020, 2, 2)),
  68. )
  69. baker.make(
  70. "djstripe.Subscription",
  71. customer=customer,
  72. livemode=False,
  73. status="cancelled",
  74. current_period_start=timezone.make_aware(timezone.datetime(2019, 1, 2)),
  75. current_period_end=timezone.make_aware(timezone.datetime(2019, 2, 2)),
  76. )
  77. url = reverse(
  78. "api:get_subscription_events_count", args=[self.organization.slug]
  79. )
  80. with freeze_time(timezone.datetime(2020, 3, 1)):
  81. baker.make(
  82. "issue_events.IssueEvent",
  83. issue__project__organization=self.organization,
  84. )
  85. with freeze_time(timezone.datetime(2020, 1, 5)):
  86. baker.make("issue_events.IssueEvent")
  87. baker.make(
  88. "issue_events.IssueEvent",
  89. issue__project__organization=self.organization,
  90. )
  91. baker.make(
  92. "projects.IssueEventProjectHourlyStatistic",
  93. project__organization=self.organization,
  94. count=1,
  95. )
  96. baker.make(
  97. "performance.TransactionEvent",
  98. group__project__organization=self.organization,
  99. )
  100. baker.make(
  101. "projects.TransactionEventProjectHourlyStatistic",
  102. project__organization=self.organization,
  103. count=1,
  104. )
  105. baker.make(
  106. "sourcecode.DebugSymbolBundle",
  107. file__blob__size=1000000,
  108. organization=self.organization,
  109. release__organization=self.organization,
  110. _quantity=2,
  111. )
  112. res = self.client.get(url)
  113. self.assertEqual(
  114. res.json(),
  115. {
  116. "eventCount": 1,
  117. "fileSizeMB": 2,
  118. "transactionEventCount": 1,
  119. "uptimeCheckEventCount": 0,
  120. },
  121. )
  122. def test_events_count_without_customer(self):
  123. """
  124. Due to async nature of Stripe integration, a customer may not exist
  125. """
  126. baker.make("djstripe.Subscription", livemode=False)
  127. url = reverse(
  128. "api:get_subscription_events_count", args=[self.organization.slug]
  129. )
  130. res = self.client.get(url)
  131. self.assertEqual(sum(res.json().values()), 0)
  132. @skipIf(
  133. settings.STRIPE_TEST_PUBLIC_KEY == "fake", "requires real Stripe test API key"
  134. )
  135. def test_create_free(self):
  136. """
  137. Users should not be able to create a free subscription if they have another non-canceled subscription
  138. """
  139. price = baker.make(
  140. "djstripe.Price",
  141. unit_amount=0,
  142. id="price_1KO6e1J4NuO0bv3IEXhpWpzt",
  143. billing_scheme=BillingScheme.per_unit,
  144. )
  145. baker.make("djstripe.Product", id="prod_L4F8CtH20Oad6S", default_price=price)
  146. data = {"price": price.id, "organization": self.organization.id}
  147. res = self.client.post(self.url, data, content_type="application/json")
  148. self.assertEqual(res.json()["price"], price.id)
  149. # Second attempt should fail
  150. res = self.client.post(self.url, data)
  151. self.assertEqual(res.status_code, 400)
  152. def test_create_invalid_org(self):
  153. """Only owners may create subscriptions"""
  154. user = baker.make("users.user") # Non owner member
  155. plan = baker.make("djstripe.Plan", amount=0)
  156. self.organization.add_user(user)
  157. self.client.force_login(user)
  158. data = {"plan": plan.id, "organization": self.organization.id}
  159. res = self.client.post(self.url, data)
  160. self.assertEqual(res.status_code, 400)
  161. class ProductAPITestCase(TestCase):
  162. def test_product_list(self):
  163. price = baker.make(
  164. "djstripe.Price",
  165. unit_amount=0,
  166. billing_scheme=BillingScheme.per_unit,
  167. active=True,
  168. product__active=True,
  169. product__livemode=False,
  170. product__metadata={"events": 10, "is_public": "true"},
  171. )
  172. inactive_price = baker.make(
  173. "djstripe.Price",
  174. unit_amount=0,
  175. billing_scheme=BillingScheme.per_unit,
  176. active=False,
  177. product__active=False,
  178. product__livemode=False,
  179. product__metadata={"events": 10, "is_public": "true"},
  180. )
  181. hidden_price = baker.make(
  182. "djstripe.Price",
  183. unit_amount=0,
  184. billing_scheme=BillingScheme.per_unit,
  185. active=True,
  186. product__active=True,
  187. product__livemode=False,
  188. product__metadata={"events": 10, "is_public": "false"},
  189. )
  190. user = baker.make("users.user")
  191. self.client.force_login(user)
  192. res = self.client.get(reverse("api:list_products"))
  193. self.assertContains(res, price.id)
  194. self.assertNotContains(res, inactive_price.id)
  195. self.assertNotContains(res, hidden_price.id)
  196. # Price ID must be from a real price actually set up on Stripe Test account
  197. class StripeAPITestCase(TestCase):
  198. @skipIf(
  199. settings.STRIPE_TEST_PUBLIC_KEY == "fake", "requires real Stripe test API key"
  200. )
  201. def test_create_checkout(self):
  202. price = baker.make(
  203. "djstripe.Price",
  204. id="price_1MZhMWJ4NuO0bv3IGMoDoFFI",
  205. )
  206. user = baker.make("users.user")
  207. organization = baker.make("organizations_ext.Organization")
  208. organization.add_user(user)
  209. url = reverse(
  210. "api:create_stripe_subscription_checkout", args=[organization.slug]
  211. )
  212. self.client.force_login(user)
  213. data = {"price": price.id}
  214. res = self.client.post(url, data, content_type="application/json")
  215. self.assertEqual(res.status_code, 200)
  216. @skipIf(
  217. settings.STRIPE_TEST_PUBLIC_KEY == "fake", "requires real Stripe test API key"
  218. )
  219. def test_manage_billing(self):
  220. user = baker.make("users.user")
  221. organization = baker.make("organizations_ext.Organization")
  222. url = reverse("api:stripe_billing_portal", args=[organization.slug])
  223. organization.add_user(user)
  224. self.client.force_login(user)
  225. data = {"organization": organization.id}
  226. res = self.client.post(url, data)
  227. self.assertEqual(res.status_code, 200)