This repository has been archived by the owner on Nov 4, 2024. It is now read-only.
-
Notifications
You must be signed in to change notification settings - Fork 48
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Mrazadar/pon 178/unit test checkout (#760)
* test: added stripe service unit test * test: added api service unit test * test: working on improving subscription checkout test * test: checkout test with react-testing-library * test: removed the commented code
- Loading branch information
Showing
5 changed files
with
1,760 additions
and
1,853 deletions.
There are no files selected for viewing
190 changes: 66 additions & 124 deletions
190
src/subscription/checkout/SubscriptionCheckout.test.jsx
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1,155 +1,97 @@ | ||
/* eslint-disable react/jsx-no-constructed-context-values */ | ||
/* eslint-disable global-require */ | ||
import React from 'react'; | ||
import { mount } from 'enzyme'; | ||
import { act } from 'react-dom/test-utils'; | ||
import { createStore, applyMiddleware } from 'redux'; | ||
import thunkMiddleware from 'redux-thunk'; | ||
import { Provider } from 'react-redux'; | ||
import { Factory } from 'rosie'; | ||
import { createSerializer } from 'enzyme-to-json'; | ||
import { IntlProvider, configure as configureI18n } from '@edx/frontend-platform/i18n'; | ||
import { getConfig } from '@edx/frontend-platform'; | ||
import * as analytics from '@edx/frontend-platform/analytics'; | ||
import Cookies from 'universal-cookie'; | ||
import { Elements } from '@stripe/react-stripe-js'; | ||
// import { loadStripe } from '@stripe/stripe-js'; | ||
|
||
import '../__factories__/subscription.factory'; | ||
import '../../payment/__factories__/userAccount.factory'; | ||
import { AppContext } from '@edx/frontend-platform/react'; | ||
import { loadStripe } from '@stripe/stripe-js'; | ||
import { | ||
render, act, screen, store, | ||
} from '../test-utils'; | ||
import * as mocks from '../../payment/checkout/stripeMocks'; | ||
|
||
import { SubscriptionCheckout } from './SubscriptionCheckout'; | ||
import createRootReducer from '../../data/reducers'; | ||
import { fetchSubscriptionDetails, subscriptionDetailsReceived } from '../data/details/actions'; | ||
import { camelCaseObject } from '../../payment/data/utils'; | ||
|
||
expect.addSnapshotSerializer(createSerializer({ mode: 'deep', noKey: true })); | ||
|
||
const config = getConfig(); | ||
const locale = 'en'; | ||
|
||
configureI18n({ | ||
config: { | ||
ENVIRONMENT: process.env.ENVIRONMENT, | ||
LANGUAGE_PREFERENCE_COOKIE_NAME: process.env.LANGUAGE_PREFERENCE_COOKIE_NAME, | ||
}, | ||
loggingService: { | ||
logError: jest.fn(), | ||
logInfo: jest.fn(), | ||
}, | ||
messages: { | ||
uk: {}, | ||
th: {}, | ||
ru: {}, | ||
'pt-br': {}, | ||
pl: {}, | ||
'ko-kr': {}, | ||
id: {}, | ||
he: {}, | ||
ca: {}, | ||
'zh-cn': {}, | ||
fr: {}, | ||
'es-419': {}, | ||
ar: {}, | ||
fa: {}, | ||
'fa-ir': {}, | ||
}, | ||
}); | ||
|
||
jest.mock('universal-cookie', () => { | ||
class MockCookies { | ||
static result = { | ||
[process.env.LANGUAGE_PREFERENCE_COOKIE_NAME]: 'en', | ||
[process.env.CURRENCY_COOKIE_NAME]: { | ||
code: 'MXN', | ||
rate: 19.092733, | ||
}, | ||
}; | ||
|
||
get(cookieName) { | ||
return MockCookies.result[cookieName]; | ||
} | ||
} | ||
return MockCookies; | ||
}); | ||
|
||
jest.mock('@edx/frontend-platform/analytics', () => ({ | ||
sendTrackEvent: jest.fn(), | ||
sendPageEvent: jest.fn(), | ||
})); | ||
|
||
// https://github.com/wwayne/react-tooltip/issues/595#issuecomment-638438372 | ||
jest.mock('react-tooltip/node_modules/uuid', () => ({ | ||
v4: () => '00000000-0000-0000-0000-000000000000', | ||
})); | ||
|
||
jest.mock('./StripeOptions', () => ({ | ||
getStripeOptions: jest.fn().mockReturnValue({}), | ||
getStripeOptions: jest.fn().mockReturnValue({ | ||
mode: 'subscription', | ||
amount: 55, | ||
currency: 'usd', | ||
paymentMethodCreation: 'manual', | ||
}), | ||
})); | ||
|
||
// jest.mock('@stripe/stripe-js', () => ({ | ||
// loadStripe: jest.fn(), | ||
// })); | ||
jest.mock('@stripe/stripe-js', () => ({ | ||
loadStripe: jest.fn(() => ({ | ||
// mock implementation of the stripe object | ||
})), | ||
})); | ||
|
||
/** | ||
* SubscriptionCheckout Test | ||
* https://github.com/stripe-archive/react-stripe-elements/issues/427 | ||
* https://github.com/stripe/react-stripe-js/issues/59 | ||
*/ | ||
describe('<SubscriptionCheckout />', () => { | ||
let store; | ||
let tree; | ||
|
||
let subscriptionDetails; | ||
let mockStripe; | ||
let mockElements; | ||
let mockElement; | ||
let mockStripePromise; | ||
beforeEach(() => { | ||
const authenticatedUser = Factory.build('userAccount'); | ||
store = createStore(createRootReducer(), {}, applyMiddleware(thunkMiddleware)); | ||
// eslint-disable-next-line no-import-assign | ||
analytics.sendTrackingLogEvent = jest.fn(); | ||
Cookies.result[process.env.CURRENCY_COOKIE_NAME] = undefined; | ||
|
||
// Mock the response of loadStripe method | ||
// const mockedStripe = { | ||
// elements: jest.fn(), | ||
// }; | ||
// loadStripe.mockResolvedValue(mockedStripe); | ||
// TODO: make sure to test the form submit events | ||
|
||
const component = ( | ||
<IntlProvider locale="en"> | ||
<AppContext.Provider value={{ authenticatedUser, config, locale }}> | ||
<Provider store={store}> | ||
<SubscriptionCheckout /> | ||
</Provider> | ||
</AppContext.Provider> | ||
</IntlProvider> | ||
); | ||
|
||
tree = mount(component); | ||
}); | ||
|
||
afterEach(() => { | ||
tree.unmount(); | ||
// Arrange | ||
mockStripe = mocks.mockStripe(); | ||
mockElement = mocks.mockElement(); | ||
mockElements = mocks.mockElements(); | ||
mockStripe.elements.mockReturnValue(mockElements); | ||
mockElements.create.mockReturnValue(mockElement); | ||
mockStripePromise = jest.fn(() => Promise.resolve({ | ||
...mockStripe, | ||
})); | ||
subscriptionDetails = camelCaseObject(Factory.build('subscription', {}, { numProducts: 2 })); | ||
loadStripe.mockResolvedValue(mockStripePromise); | ||
}); | ||
|
||
it('should render the loading skeleton for SubscriptionCheckout', () => { | ||
tree.update(); | ||
render(<SubscriptionCheckout />); | ||
expect(screen.queryByText('Last Name (required)')).not.toBeInTheDocument(); // it doesn't exist | ||
expect( | ||
screen.queryByText('You’ll be charged $55.00 USD on April 21, 2025 then every 31 days until you cancel your subscription.'), | ||
).toBeNull(); | ||
}); | ||
|
||
expect(tree.find('CheckoutSkeleton')).toHaveLength(1); | ||
it('should render the <SubscriptionCheckout/> with the subscription details', () => { | ||
const stripePromise = mockStripePromise(); | ||
|
||
expect(tree).toMatchSnapshot(); | ||
}); | ||
loadStripe.mockResolvedValue(stripePromise); | ||
|
||
it('should render the subscription checkout details', () => { | ||
const { container } = render(<SubscriptionCheckout />); | ||
act(() => { | ||
store.dispatch( | ||
subscriptionDetailsReceived( | ||
camelCaseObject(Factory.build('subscription', {}, { numProducts: 1 })), | ||
subscriptionDetails, | ||
), | ||
); | ||
store.dispatch(fetchSubscriptionDetails.fulfill()); | ||
}); | ||
expect(loadStripe).toHaveBeenCalledWith( | ||
process.env.STRIPE_PUBLISHABLE_KEY, | ||
{ | ||
betas: [process.env.STRIPE_DEFERRED_INTENT_BETA_FLAG], | ||
apiVersion: process.env.STRIPE_API_VERSION, | ||
locale: 'en', | ||
}, | ||
); | ||
expect(container).toMatchSnapshot(); | ||
// screen.debug(container.querySelector('#payment-element')); | ||
|
||
tree.update(); | ||
expect(tree).toMatchSnapshot(); | ||
|
||
expect(tree.find('SubscriptionCheckout')).toHaveLength(1); | ||
expect(container.querySelector('#payment-element')).toBeDefined(); | ||
|
||
expect(tree.find(Elements)).toHaveLength(1); | ||
expect(tree.find('StripePaymentForm')).toHaveLength(1); | ||
// verify that Checkout Form fields are present in the DOM | ||
expect(screen.queryByText('Last Name (required)')).toBeDefined(); | ||
// verify that MonthlySubscriptionNotification is present in the DOM | ||
expect( | ||
screen.queryByText('You’ll be charged $55.00 USD on April 21, 2025 then every 31 days until you cancel your subscription.'), | ||
).toBeDefined(); | ||
}); | ||
}); |
Oops, something went wrong.