import React, {StrictMode} from 'react';
import {render, act, waitFor} from '@testing-library/react';
import {renderHook} from '@testing-library/react-hooks';

import {CheckoutElementsProvider} from './CheckoutElementsProvider';
import {
  useCheckout,
  useCheckoutElements,
  useCheckoutForm,
} from './CheckoutContext';
import {Elements} from '../../components/Elements';
import {useStripe} from '../../components/useStripe';
import * as mocks from '../../../test/mocks';
import makeDeferred from '../../../test/makeDeferred';

describe('CheckoutElementsProvider', () => {
  let mockStripe: any;
  let mockStripePromise: any;
  let mockCheckoutSdk: any;
  let consoleError: any;
  let consoleWarn: any;
  let mockCheckoutActions: any;

  beforeEach(() => {
    mockCheckoutSdk = mocks.mockCheckoutSdk();
    mockCheckoutActions = mocks.mockCheckoutActions();
    mockCheckoutSdk.loadActions.mockResolvedValue({
      type: 'success',
      actions: mockCheckoutActions,
    });

    mockStripe = mocks.mockStripe();
    mockStripe.initCheckoutElementsSdk.mockReturnValue(mockCheckoutSdk);
    mockStripePromise = Promise.resolve(mockStripe);

    jest.spyOn(console, 'error');
    jest.spyOn(console, 'warn');
    consoleError = console.error;
    consoleWarn = console.warn;
  });

  afterEach(() => {
    jest.restoreAllMocks();
  });

  const fakeClientSecret = 'cs_123';

  const wrapper = ({stripe, clientSecret, children}: any) => (
    <CheckoutElementsProvider
      stripe={stripe === undefined ? mockStripe : stripe}
      options={{
        clientSecret: clientSecret || fakeClientSecret,
      }}
    >
      {children}
    </CheckoutElementsProvider>
  );

  describe('interaction with useStripe()', () => {
    it('works with a Stripe instance', async () => {
      const {result, waitForNextUpdate} = renderHook(() => useStripe(), {
        wrapper,
        initialProps: {stripe: mockStripe},
      });

      expect(result.current).toBe(mockStripe);

      await waitForNextUpdate();

      expect(result.current).toBe(mockStripe);
    });

    it('works when updating null to a Stripe instance', async () => {
      const {result, rerender, waitForNextUpdate} = renderHook(
        () => useStripe(),
        {
          wrapper,
          initialProps: {stripe: null},
        }
      );

      expect(result.current).toBe(null);

      rerender({stripe: mockStripe});
      await waitForNextUpdate();

      expect(result.current).toBe(mockStripe);
    });

    it('works with a Promise', async () => {
      const deferred = makeDeferred();
      const {result} = renderHook(() => useStripe(), {
        wrapper,
        initialProps: {stripe: deferred.promise},
      });

      expect(result.current).toBe(null);

      await act(() => deferred.resolve(mockStripe));

      expect(result.current).toBe(mockStripe);
    });
  });

  describe('interaction with useCheckout()', () => {
    it('works when loadActions resolves', async () => {
      const stripe: any = mocks.mockStripe();
      const deferred = makeDeferred();
      const mockSdk = mocks.mockCheckoutSdk();
      const testMockCheckoutActions = mocks.mockCheckoutActions();
      const testMockSession = mocks.mockCheckoutSession();

      mockSdk.loadActions.mockReturnValue(deferred.promise);
      stripe.initCheckoutElementsSdk.mockReturnValue(mockSdk);

      const {result} = renderHook(() => useCheckout(), {
        wrapper,
        initialProps: {stripe},
      });

      expect(result.current).toEqual({type: 'loading'});
      expect(stripe.initCheckoutElementsSdk).toHaveBeenCalledTimes(1);

      await act(() =>
        deferred.resolve({
          type: 'success',
          actions: testMockCheckoutActions,
        })
      );

      const {on: _on, loadActions: _loadActions, ...elementsMethods} = mockSdk;
      const {
        getSession: _getSession,
        ...otherCheckoutActions
      } = testMockCheckoutActions;

      const expectedCheckout = {
        ...elementsMethods,
        ...otherCheckoutActions,
        ...testMockSession,
      };

      expect(result.current).toEqual({
        type: 'success',
        checkout: expectedCheckout,
      });
      expect(stripe.initCheckoutElementsSdk).toHaveBeenCalledTimes(1);
    });

    it('works when loadActions rejects', async () => {
      const stripe: any = mocks.mockStripe();
      const deferred = makeDeferred();
      const mockSdk = mocks.mockCheckoutSdk();
      mockSdk.loadActions.mockReturnValue(deferred.promise);
      stripe.initCheckoutElementsSdk.mockReturnValue(mockSdk);

      const {result} = renderHook(() => useCheckout(), {
        wrapper,
        initialProps: {stripe},
      });

      expect(result.current).toEqual({type: 'loading'});
      expect(stripe.initCheckoutElementsSdk).toHaveBeenCalledTimes(1);

      await act(() =>
        deferred.reject(new Error('initCheckoutElementsSdk error'))
      );

      expect(result.current).toEqual({
        type: 'error',
        error: new Error('initCheckoutElementsSdk error'),
      });
      expect(stripe.initCheckoutElementsSdk).toHaveBeenCalledTimes(1);
    });

    it('does not set context if Promise resolves after CheckoutElementsProvider is unmounted', async () => {
      const result = render(
        <CheckoutElementsProvider
          stripe={mockStripePromise}
          options={{clientSecret: Promise.resolve(fakeClientSecret)}}
        >
          {null}
        </CheckoutElementsProvider>
      );

      result.unmount();
      await act(() => mockStripePromise);

      expect(consoleError).not.toHaveBeenCalled();
    });

    // Regression guard for the public Elements SDK surface documented at
    // https://docs.stripe.com/sdks/stripejs-react. Any accidental narrowing
    // of StripeCheckoutValue that drops these methods would break every
    // <CheckoutElementsProvider> consumer calling them from useCheckout().
    it('exposes every action method returned by loadActions on useCheckout().checkout', async () => {
      const testMockCheckoutActions = mocks.mockCheckoutActions();
      mockCheckoutSdk.loadActions.mockResolvedValue({
        type: 'success',
        actions: testMockCheckoutActions,
      });

      const {result, waitForNextUpdate} = renderHook(() => useCheckout(), {
        wrapper,
        initialProps: {stripe: mockStripe},
      });

      await waitForNextUpdate();

      if (result.current.type !== 'success') {
        throw new Error(
          `expected success, got ${JSON.stringify(result.current)}`
        );
      }
      const {checkout} = result.current;

      // Every action key from the SDK (except getSession, whose fields are
      // spread onto checkout as session data) must be a function. Derived
      // from the mock so this stays honest if the shared mock is updated.
      const {
        getSession: _getSession,
        ...expectedActionKeys
      } = testMockCheckoutActions;
      Object.keys(expectedActionKeys).forEach((key) => {
        expect(typeof (checkout as any)[key]).toBe('function');
      });
    });

    // Mirrors the regression guard above for useCheckoutElements(), which
    // returns the same Elements-shaped surface but with a type that makes
    // the contract explicit for Elements consumers.
    it('useCheckoutElements() exposes every action method returned by loadActions', async () => {
      const testMockCheckoutActions = mocks.mockCheckoutActions();
      mockCheckoutSdk.loadActions.mockResolvedValue({
        type: 'success',
        actions: testMockCheckoutActions,
      });

      const {result, waitForNextUpdate} = renderHook(
        () => useCheckoutElements(),
        {
          wrapper,
          initialProps: {stripe: mockStripe},
        }
      );

      await waitForNextUpdate();

      if (result.current.type !== 'success') {
        throw new Error(
          `expected success, got ${JSON.stringify(result.current)}`
        );
      }
      const {checkout} = result.current;

      const {
        getSession: _getSession,
        ...expectedActionKeys
      } = testMockCheckoutActions;
      Object.keys(expectedActionKeys).forEach((key) => {
        expect(typeof (checkout as any)[key]).toBe('function');
      });
    });

    // useCheckoutForm() under the wrong provider must fail loudly at first
    // render instead of silently returning undefined-shaped actions. The
    // error must name the expected provider so integrators can self-serve.
    it('useCheckoutForm() throws when used under <CheckoutElementsProvider>', async () => {
      consoleError.mockImplementation(() => {});
      const testMockCheckoutActions = mocks.mockCheckoutActions();
      mockCheckoutSdk.loadActions.mockResolvedValue({
        type: 'success',
        actions: testMockCheckoutActions,
      });

      const {result, waitForNextUpdate} = renderHook(() => useCheckoutForm(), {
        wrapper,
        initialProps: {stripe: mockStripe},
      });

      await waitForNextUpdate();

      expect(result.error).toBeDefined();
      expect(result.error?.message).toMatch(
        /useCheckoutForm\(\) must be used inside <CheckoutFormProvider>/
      );
    });
  });

  describe('stripe prop', () => {
    it('validates stripe prop type', async () => {
      consoleError.mockImplementation(() => {});

      const renderWithProp = (stripeProp: unknown) => () => {
        render(
          <CheckoutElementsProvider
            stripe={stripeProp as any}
            options={{clientSecret: fakeClientSecret}}
          >
            <div />
          </CheckoutElementsProvider>
        );
      };

      expect(renderWithProp(undefined)).toThrow(
        'Invalid prop `stripe` supplied to `CheckoutElementsProvider`.'
      );
      expect(renderWithProp(false)).toThrow(
        'Invalid prop `stripe` supplied to `CheckoutElementsProvider`.'
      );
      expect(renderWithProp('foo')).toThrow(
        'Invalid prop `stripe` supplied to `CheckoutElementsProvider`.'
      );
      expect(renderWithProp({foo: 'bar'})).toThrow(
        'Invalid prop `stripe` supplied to `CheckoutElementsProvider`.'
      );
    });

    it('when stripe prop changes from null to a Stripe instance', async () => {
      const stripe: any = mocks.mockStripe();
      const deferred = makeDeferred();
      const mockSdk = mocks.mockCheckoutSdk();
      const testMockCheckoutActions = mocks.mockCheckoutActions();
      const testMockSession = mocks.mockCheckoutSession();

      mockSdk.loadActions.mockReturnValue(deferred.promise);
      stripe.initCheckoutElementsSdk.mockReturnValue(mockSdk);

      const {result, rerender} = renderHook(() => useCheckout(), {
        wrapper,
        initialProps: {stripe: null},
      });

      expect(result.current).toEqual({type: 'loading'});
      expect(stripe.initCheckoutElementsSdk).toHaveBeenCalledTimes(0);

      rerender({stripe});

      expect(result.current).toEqual({type: 'loading'});
      expect(stripe.initCheckoutElementsSdk).toHaveBeenCalledTimes(1);

      await act(() =>
        deferred.resolve({
          type: 'success',
          actions: testMockCheckoutActions,
        })
      );

      const {on: _on, loadActions: _loadActions, ...elementsMethods} = mockSdk;
      const {
        getSession: _getSession,
        ...otherCheckoutActions
      } = testMockCheckoutActions;

      const expectedCheckout = {
        ...elementsMethods,
        ...otherCheckoutActions,
        ...testMockSession,
      };

      expect(result.current).toEqual({
        type: 'success',
        checkout: expectedCheckout,
      });
      expect(stripe.initCheckoutElementsSdk).toHaveBeenCalledTimes(1);
    });

    it('when the stripe prop is a Promise', async () => {
      const stripe: any = mocks.mockStripe();
      const stripeDeferred = makeDeferred();
      const deferred = makeDeferred();
      const mockSdk = mocks.mockCheckoutSdk();
      const testMockCheckoutActions = mocks.mockCheckoutActions();
      const testMockSession = mocks.mockCheckoutSession();

      mockSdk.loadActions.mockReturnValue(deferred.promise);
      stripe.initCheckoutElementsSdk.mockReturnValue(mockSdk);

      const {result} = renderHook(() => useCheckout(), {
        wrapper,
        initialProps: {stripe: stripeDeferred.promise},
      });

      expect(result.current).toEqual({type: 'loading'});
      expect(stripe.initCheckoutElementsSdk).toHaveBeenCalledTimes(0);

      await act(() => stripeDeferred.resolve(stripe));

      expect(result.current).toEqual({type: 'loading'});
      expect(stripe.initCheckoutElementsSdk).toHaveBeenCalledTimes(1);

      await act(() =>
        deferred.resolve({
          type: 'success',
          actions: testMockCheckoutActions,
        })
      );

      const {on: _on, loadActions: _loadActions, ...elementsMethods} = mockSdk;
      const {
        getSession: _getSession,
        ...otherCheckoutActions
      } = testMockCheckoutActions;

      const expectedCheckout = {
        ...elementsMethods,
        ...otherCheckoutActions,
        ...testMockSession,
      };

      expect(result.current).toEqual({
        type: 'success',
        checkout: expectedCheckout,
      });
      expect(stripe.initCheckoutElementsSdk).toHaveBeenCalledTimes(1);
    });

    it('when the stripe prop changes from null to a Promise', async () => {
      const stripe: any = mocks.mockStripe();
      const stripeDeferred = makeDeferred();
      const deferred = makeDeferred();
      const mockSdk = mocks.mockCheckoutSdk();
      const testMockCheckoutActions = mocks.mockCheckoutActions();
      const testMockSession = mocks.mockCheckoutSession();

      mockSdk.loadActions.mockReturnValue(deferred.promise);
      stripe.initCheckoutElementsSdk.mockReturnValue(mockSdk);

      const {result, rerender} = renderHook(() => useCheckout(), {
        wrapper,
        initialProps: {stripe: null},
      });

      expect(result.current).toEqual({type: 'loading'});
      expect(stripe.initCheckoutElementsSdk).toHaveBeenCalledTimes(0);

      rerender({stripe: stripeDeferred.promise as any});

      expect(result.current).toEqual({type: 'loading'});
      expect(stripe.initCheckoutElementsSdk).toHaveBeenCalledTimes(0);

      await act(() => stripeDeferred.resolve(stripe));

      expect(result.current).toEqual({type: 'loading'});
      expect(stripe.initCheckoutElementsSdk).toHaveBeenCalledTimes(1);

      await act(() =>
        deferred.resolve({
          type: 'success',
          actions: testMockCheckoutActions,
        })
      );

      const {on: _on, loadActions: _loadActions, ...elementsMethods} = mockSdk;
      const {
        getSession: _getSession,
        ...otherCheckoutActions
      } = testMockCheckoutActions;

      const expectedCheckout = {
        ...elementsMethods,
        ...otherCheckoutActions,
        ...testMockSession,
      };

      expect(result.current).toEqual({
        type: 'success',
        checkout: expectedCheckout,
      });
      expect(stripe.initCheckoutElementsSdk).toHaveBeenCalledTimes(1);
    });

    it('when the stripe prop is a Promise(null)', async () => {
      const stripeDeferred = makeDeferred<any>();

      const {result} = renderHook(() => useCheckout(), {
        wrapper,
        initialProps: {stripe: stripeDeferred.promise},
      });

      expect(result.current).toEqual({type: 'loading'});

      await act(() => stripeDeferred.resolve(null));

      expect(result.current).toEqual({type: 'loading'});
    });

    it('does not allow changes to an already set Stripe object', async () => {
      consoleWarn.mockImplementation(() => {});
      let result: any;
      act(() => {
        result = render(
          <CheckoutElementsProvider
            stripe={mockStripe}
            options={{clientSecret: fakeClientSecret}}
          />
        );
      });

      const mockStripe2: any = mocks.mockStripe();
      act(() => {
        result.rerender(
          <CheckoutElementsProvider
            stripe={mockStripe2}
            options={{clientSecret: fakeClientSecret}}
          />
        );
      });

      await waitFor(() => {
        expect(mockStripe.initCheckoutElementsSdk).toHaveBeenCalledTimes(1);
        expect(mockStripe2.initCheckoutElementsSdk).toHaveBeenCalledTimes(0);
        expect(consoleWarn).toHaveBeenCalledWith(
          'Unsupported prop change on CheckoutElementsProvider: You cannot change the `stripe` prop after setting it.'
        );
      });
    });
  });

  it('only calls initCheckoutElementsSdk once and allows changes to elementsOptions appearance after setting the Stripe object', async () => {
    const result = render(
      <CheckoutElementsProvider
        stripe={mockStripe}
        options={{
          clientSecret: fakeClientSecret,
          elementsOptions: {
            appearance: {theme: 'stripe'},
          },
        }}
      />
    );

    await waitFor(() => {
      expect(mockStripe.initCheckoutElementsSdk).toHaveBeenCalledWith({
        clientSecret: fakeClientSecret,
        elementsOptions: {
          appearance: {theme: 'stripe'},
        },
      });
    });

    act(() => {
      result.rerender(
        <CheckoutElementsProvider
          stripe={mockStripe}
          options={{
            clientSecret: fakeClientSecret,
            elementsOptions: {appearance: {theme: 'night'}},
          }}
        />
      );
    });

    await waitFor(() => {
      expect(mockStripe.initCheckoutElementsSdk).toHaveBeenCalledTimes(1);
      expect(mockCheckoutSdk.changeAppearance).toHaveBeenCalledTimes(1);
      expect(mockCheckoutSdk.changeAppearance).toHaveBeenCalledWith({
        theme: 'night',
      });
    });
  });

  test('it does not call loadFonts a 2nd time if they do not change', async () => {
    let result: any;
    act(() => {
      result = render(
        <CheckoutElementsProvider
          stripe={mockStripe}
          options={{
            clientSecret: fakeClientSecret,
            elementsOptions: {
              fonts: [
                {
                  cssSrc: 'https://example.com/font.css',
                },
              ],
            },
          }}
        />
      );
    });

    await waitFor(() =>
      expect(mockStripe.initCheckoutElementsSdk).toHaveBeenCalledWith({
        clientSecret: fakeClientSecret,
        elementsOptions: {
          fonts: [
            {
              cssSrc: 'https://example.com/font.css',
            },
          ],
        },
      })
    );

    act(() => {
      result.rerender(
        <CheckoutElementsProvider
          stripe={mockStripe}
          options={{
            clientSecret: fakeClientSecret,
            elementsOptions: {
              fonts: [
                {
                  cssSrc: 'https://example.com/font.css',
                },
              ],
            },
          }}
        />
      );
    });

    act(() => {
      result.rerender(
        <CheckoutElementsProvider
          stripe={mockStripe}
          options={{
            clientSecret: fakeClientSecret,
            elementsOptions: {
              fonts: [
                {
                  cssSrc: 'https://example.com/font.css',
                },
              ],
            },
          }}
        />
      );
    });

    await waitFor(() => {
      expect(mockStripe.initCheckoutElementsSdk).toHaveBeenCalledTimes(1);
      expect(mockCheckoutSdk.loadFonts).toHaveBeenCalledTimes(0);
    });
  });

  test('allows changes to elementsOptions fonts', async () => {
    let result: any;
    act(() => {
      result = render(
        <CheckoutElementsProvider
          stripe={mockStripe}
          options={{
            clientSecret: fakeClientSecret,
            elementsOptions: {},
          }}
        />
      );
    });

    await waitFor(() =>
      expect(mockStripe.initCheckoutElementsSdk).toHaveBeenCalledWith({
        clientSecret: fakeClientSecret,
        elementsOptions: {},
      })
    );

    act(() => {
      result.rerender(
        <CheckoutElementsProvider
          stripe={mockStripe}
          options={{
            clientSecret: fakeClientSecret,
            elementsOptions: {
              fonts: [
                {
                  cssSrc: 'https://example.com/font.css',
                },
              ],
            },
          }}
        />
      );
    });

    await waitFor(() => {
      expect(mockStripe.initCheckoutElementsSdk).toHaveBeenCalledTimes(1);
      expect(mockCheckoutSdk.loadFonts).toHaveBeenCalledTimes(1);
      expect(mockCheckoutSdk.loadFonts).toHaveBeenCalledWith([
        {
          cssSrc: 'https://example.com/font.css',
        },
      ]);
    });
  });

  it('allows options changes before setting the Stripe object', async () => {
    const result = render(
      <CheckoutElementsProvider
        stripe={null}
        options={{
          clientSecret: fakeClientSecret,
          elementsOptions: {
            appearance: {theme: 'stripe'},
          },
        }}
      />
    );

    await waitFor(() =>
      expect(mockStripe.initCheckoutElementsSdk).toHaveBeenCalledTimes(0)
    );

    act(() => {
      result.rerender(
        <CheckoutElementsProvider
          stripe={mockStripe}
          options={{
            clientSecret: fakeClientSecret,
            elementsOptions: {appearance: {theme: 'stripe'}},
          }}
        />
      );
    });

    await waitFor(() => {
      expect(console.warn).not.toHaveBeenCalled();
      expect(mockStripe.initCheckoutElementsSdk).toHaveBeenCalledTimes(1);
      expect(mockStripe.initCheckoutElementsSdk).toHaveBeenCalledWith({
        clientSecret: fakeClientSecret,
        elementsOptions: {
          appearance: {theme: 'stripe'},
        },
      });
    });
  });

  describe('React.StrictMode', () => {
    test('initCheckoutElementsSdk once in StrictMode', async () => {
      const TestComponent = () => {
        useCheckout();
        return <div />;
      };

      act(() => {
        render(
          <StrictMode>
            <CheckoutElementsProvider
              stripe={mockStripe}
              options={{clientSecret: fakeClientSecret}}
            >
              <TestComponent />
            </CheckoutElementsProvider>
          </StrictMode>
        );
      });

      await waitFor(() =>
        expect(mockStripe.initCheckoutElementsSdk).toHaveBeenCalledTimes(1)
      );
    });

    test('initCheckoutElementsSdk once with stripePromise in StrictMode', async () => {
      const TestComponent = () => {
        useCheckout();
        return <div />;
      };

      act(() => {
        render(
          <StrictMode>
            <CheckoutElementsProvider
              stripe={mockStripePromise}
              options={{clientSecret: fakeClientSecret}}
            >
              <TestComponent />
            </CheckoutElementsProvider>
          </StrictMode>
        );
      });

      await waitFor(() =>
        expect(mockStripe.initCheckoutElementsSdk).toHaveBeenCalledTimes(1)
      );
    });

    test('allows changes to options via changeAppearance after setting the Stripe object in StrictMode', async () => {
      let result: any;
      act(() => {
        result = render(
          <StrictMode>
            <CheckoutElementsProvider
              stripe={mockStripe}
              options={{
                clientSecret: fakeClientSecret,
                elementsOptions: {
                  appearance: {theme: 'stripe'},
                },
              }}
            />
          </StrictMode>
        );
      });

      await waitFor(() => {
        expect(mockStripe.initCheckoutElementsSdk).toHaveBeenCalledTimes(1);
        expect(mockStripe.initCheckoutElementsSdk).toHaveBeenCalledWith({
          clientSecret: fakeClientSecret,
          elementsOptions: {
            appearance: {theme: 'stripe'},
          },
        });
      });

      act(() => {
        result.rerender(
          <StrictMode>
            <CheckoutElementsProvider
              stripe={mockStripe}
              options={{
                clientSecret: fakeClientSecret,
                elementsOptions: {appearance: {theme: 'night'}},
              }}
            />
          </StrictMode>
        );
      });

      await waitFor(() => {
        expect(mockCheckoutSdk.changeAppearance).toHaveBeenCalledTimes(1);
        expect(mockCheckoutSdk.changeAppearance).toHaveBeenCalledWith({
          theme: 'night',
        });
      });
    });
  });

  describe('providers <> hooks', () => {
    it('throws when trying to call useCheckout outside of checkout context', () => {
      const {result} = renderHook(() => useCheckout());

      expect(result.error && result.error.message).toBe(
        'Could not find checkout context; You need to wrap the part of your app that calls useCheckout() in a <CheckoutElementsProvider> or <CheckoutFormProvider> provider.'
      );
    });

    it('throws when trying to call useStripe outside of Elements context', () => {
      const {result} = renderHook(() => useStripe());

      expect(result.error && result.error.message).toBe(
        'Could not find Elements context; You need to wrap the part of your app that calls useStripe() in an <Elements> provider.'
      );
    });

    it('throws when trying to call useStripe in Elements -> CheckoutElementsProvider nested context', async () => {
      const wrapper = ({children}: any) => (
        <Elements stripe={mockStripe}>
          <CheckoutElementsProvider
            stripe={mockStripe}
            options={{clientSecret: fakeClientSecret}}
          >
            {children}
          </CheckoutElementsProvider>
        </Elements>
      );

      const {result} = renderHook(() => useStripe(), {
        wrapper,
      });

      expect(result.error && result.error.message).toBe(
        'You cannot wrap the part of your app that calls useStripe() in both a checkout provider and <Elements> provider.'
      );
    });

    it('throws when trying to call useStripe in CheckoutElementsProvider -> Elements nested context', async () => {
      const wrapper = ({children}: any) => (
        <CheckoutElementsProvider
          stripe={mockStripe}
          options={{clientSecret: fakeClientSecret}}
        >
          <Elements stripe={mockStripe}>{children}</Elements>
        </CheckoutElementsProvider>
      );

      const {result} = renderHook(() => useStripe(), {
        wrapper,
      });
      expect(result.error && result.error.message).toBe(
        'You cannot wrap the part of your app that calls useStripe() in both a checkout provider and <Elements> provider.'
      );
    });
  });
});
