diff --git a/.github/workflows/migrations-mysql8.yml b/.github/workflows/migrations-mysql8.yml index c03b8ef9956..2394c14eb9a 100644 --- a/.github/workflows/migrations-mysql8.yml +++ b/.github/workflows/migrations-mysql8.yml @@ -48,6 +48,7 @@ jobs: mysql -V - name: Install Python dependencies run: | + pip install pip==21.2.1 --upgrade # set pip version to be compatible with deprecated pip version syntax pip install -r requirements/pip_tools.txt pip install -r requirements/production.txt pip uninstall -y mysqlclient diff --git a/ecommerce/coupons/tests/test_views.py b/ecommerce/coupons/tests/test_views.py index 24a81b5252e..2ec4ba1c369 100644 --- a/ecommerce/coupons/tests/test_views.py +++ b/ecommerce/coupons/tests/test_views.py @@ -342,7 +342,7 @@ def tearDown(self): super().tearDown() responses.reset() - def redeem_url_with_params(self, code=COUPON_CODE, consent_token=None): + def redeem_url_with_params(self, code=COUPON_CODE, consent_token=None, other_qs_params=None): """ Constructs the coupon redemption URL with the proper string query parameters. """ params = { 'code': code, @@ -350,6 +350,8 @@ def redeem_url_with_params(self, code=COUPON_CODE, consent_token=None): } if consent_token is not None: params['consent_token'] = consent_token + if other_qs_params is not None: + params.update(other_qs_params) return format_url(base=self.redeem_url, params=params) def create_coupon_and_get_code( @@ -387,9 +389,12 @@ def create_coupon_and_get_code( self.assertEqual(Voucher.objects.filter(code=coupon_code).count(), 1) return coupon_code, coupon - def redeem_coupon(self, code=COUPON_CODE, consent_token=None): + def redeem_coupon(self, code=COUPON_CODE, consent_token=None, other_qs_params=None): self.request.user = self.user - return self.client.get(self.redeem_url_with_params(code=code, consent_token=consent_token)) + return self.client.get(self.redeem_url_with_params( + code=code, + consent_token=consent_token, + other_qs_params=other_qs_params)) def assert_redirects_to_receipt_page(self, code=COUPON_CODE, consent_token=None): response = self.redeem_coupon(code=code, consent_token=consent_token) @@ -406,9 +411,10 @@ def assert_redirects_to_receipt_page(self, code=COUPON_CODE, consent_token=None) self.assertRedirects(response, expected_url, status_code=302, fetch_redirect_response=False) - def assert_redemption_page_redirects(self, expected_url, target=200, code=COUPON_CODE, consent_token=None): + def assert_redemption_page_redirects(self, expected_url, target=200, code=COUPON_CODE, + consent_token=None, other_qs_params=None): """ Verify redirect from redeem page to expected page. """ - response = self.redeem_coupon(code=code, consent_token=consent_token) + response = self.redeem_coupon(code=code, consent_token=consent_token, other_qs_params=other_qs_params) self.assertRedirects( response, expected_url, status_code=302, target_status_code=target, fetch_redirect_response=False ) @@ -495,6 +501,19 @@ def test_basket_redirect_discount_code(self): self.create_coupon(catalog=self.catalog, code=COUPON_CODE, benefit_value=5) self.assert_redemption_page_redirects(self.get_coupon_redeem_success_expected_redirect_url()) + @responses.activate + def test_basket_redirect_discount_code_with_paypal(self): + """ Verify the view redirects to the basket view when a discount code is provided with a paypal qs param. """ + self.mock_course_api_response(course=self.course) + self.mock_account_api(self.request, self.user.username, data={'is_active': True}) + self.mock_access_token_response() + + self.create_coupon(catalog=self.catalog, code=COUPON_CODE, benefit_value=5) + self.assert_redemption_page_redirects( + self.get_coupon_redeem_success_expected_redirect_url() + '&paypal_redirect=1', + other_qs_params={'paypal_redirect': '1'} + ) + @override_flag(REDIRECT_WITH_WAFFLE_TESTING_QUERYSTRING, active=True) @responses.activate def test_basket_redirect_discount_code_with_waffle(self): diff --git a/ecommerce/coupons/views.py b/ecommerce/coupons/views.py index f8a78968890..f195c66ef22 100644 --- a/ecommerce/coupons/views.py +++ b/ecommerce/coupons/views.py @@ -303,7 +303,16 @@ def get(self, request): # pylint: disable=too-many-statements # and should not display the payment form before making that determination. # TODO: It would be cleaner if the user could be redirected to their final destination up front. redirect_url = get_payment_microfrontend_or_basket_url(self.request) + "?coupon_redeem_redirect=1" + + # Check for the paypal_redirect=1 parameter from the ecommerce checkout and add it to the + # redirect URL if present + paypal_redirect = request.GET.get('paypal_redirect') + + if paypal_redirect: + redirect_url += "&paypal_redirect=1" + redirect_url = add_stripe_flag_to_url(redirect_url, self.request) + return HttpResponseRedirect(redirect_url) diff --git a/ecommerce/extensions/basket/tests/test_views.py b/ecommerce/extensions/basket/tests/test_views.py index 52a56f7306e..610dc4e5f76 100644 --- a/ecommerce/extensions/basket/tests/test_views.py +++ b/ecommerce/extensions/basket/tests/test_views.py @@ -119,6 +119,16 @@ def test_basket_with_utm_params(self): expected_url = self.get_full_url(reverse('basket:summary')) + '?utm_source=test' self.assertEqual(response.url, expected_url) + def test_basket_with_paypal_param(self): + """ Verify the basket includes paypal param after redirect. """ + products = ProductFactory.create_batch(3, stockrecords__partner=self.partner) + response = self._get_response( + [product.stockrecords.first().partner_sku for product in products], + paypal_redirect='1', + ) + expected_url = self.get_full_url(reverse('basket:summary')) + '?paypal_redirect=1' + self.assertEqual(response.url, expected_url) + @responses.activate def test_redirect_to_basket_summary(self): """ diff --git a/ecommerce/extensions/basket/views.py b/ecommerce/extensions/basket/views.py index 241655e789d..1bd33ab52fe 100644 --- a/ecommerce/extensions/basket/views.py +++ b/ecommerce/extensions/basket/views.py @@ -492,6 +492,15 @@ def _redirect_response_to_basket_or_payment(self, request, invalid_code=None): redirect_url = get_payment_microfrontend_or_basket_url(request) redirect_url = add_utm_params_to_url(redirect_url, list(self.request.GET.items())) redirect_url = add_invalid_code_message_to_url(redirect_url, invalid_code) + + # Check for the paypal_redirect=1 parameter from the ecommerce checkout and add it to the + # redirect URL if present + paypal_redirect = request.GET.get('paypal_redirect') + + if paypal_redirect: + redirect_url += '&' if '?' in redirect_url else '?' + redirect_url += 'paypal_redirect=1' + redirect_url = add_stripe_flag_to_url(redirect_url, request) return HttpResponseRedirect(redirect_url, status=303) diff --git a/requirements/pip_tools.in b/requirements/pip_tools.in index f2177985a94..bee11765df0 100644 --- a/requirements/pip_tools.in +++ b/requirements/pip_tools.in @@ -1,4 +1,4 @@ -c constraints.txt # Dependencies to run compile tools -pip-tools # Contains pip-compile, used to generate pip requirements files +pip-tools==6.0 # Contains pip-compile, used to generate pip requirements files diff --git a/requirements/pip_tools.txt b/requirements/pip_tools.txt index b6150065453..669c1f956a0 100644 --- a/requirements/pip_tools.txt +++ b/requirements/pip_tools.txt @@ -10,7 +10,7 @@ click==8.1.3 # via pip-tools packaging==23.1 # via build -pip-tools==6.13.0 +pip-tools==6.0 # via -r requirements/pip_tools.in pyproject-hooks==1.0.0 # via build