Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

[SDK-1694] Camel case props #22

Merged
merged 7 commits into from
May 29, 2020
Merged
Show file tree
Hide file tree
Changes from 5 commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
83 changes: 50 additions & 33 deletions EXAMPLES.md
Original file line number Diff line number Diff line change
@@ -1,9 +1,9 @@
# Examples

1. [Protecting a route in a `react-router-dom` app](#1-protecting-a-route-in-a--react-router-dom--app)
1. [Protecting a route in a `react-router-dom` app](#1-protecting-a-route-in-a-react-router-dom-app)
2. [Protecting a route in a Gatsby app](#2-protecting-a-route-in-a-gatsby-app)
3. [Protecting a route in a Next.js app (in SPA mode)](#3-protecting-a-route-in-a-nextjs-app--in-spa-mode-)
4. [Create a `useApi` hook for accessing protected APIs with an access token.](#4-create-a--useapi--hook-for-accessing-protected-apis-with-an-access-token)
3. [Protecting a route in a Next.js app (in SPA mode)](#3-protecting-a-route-in-a-nextjs-app-in-spa-mode)
4. [Create a `useApi` hook for accessing protected APIs with an access token.](#4-create-a-useapi-hook-for-accessing-protected-apis-with-an-access-token)

## 1. Protecting a route in a `react-router-dom` app

Expand Down Expand Up @@ -33,8 +33,8 @@ export default function App() {
return (
<Auth0Provider
domain="YOUR_AUTH0_DOMAIN"
client_id="YOUR_AUTH0_CLIENT_ID"
redirect_uri={window.location.origin}
clientId="YOUR_AUTH0_CLIENT_ID"
redirectUri={window.location.origin}
onRedirectCallback={onRedirectCallback}
>
{/* Don't forget to add the history to your router */}
Expand Down Expand Up @@ -70,8 +70,8 @@ export const wrapRootElement = ({ element }) => {
return (
<Auth0Provider
domain="YOUR_AUTH0_DOMAIN"
client_id="YOUR_AUTH0_CLIENT_ID"
redirect_uri={window.location.origin}
clientId="YOUR_AUTH0_CLIENT_ID"
redirectUri={window.location.origin}
onRedirectCallback={onRedirectCallback}
>
{element}
Expand Down Expand Up @@ -125,8 +125,8 @@ class MyApp extends App {
return (
<Auth0Provider
domain="YOUR_AUTH0_DOMAIN"
client_id="YOUR_AUTH0_CLIENT_ID"
redirect_uri={typeof window !== 'undefined' && window.location.origin}
clientId="YOUR_AUTH0_CLIENT_ID"
redirectUri={typeof window !== 'undefined' && window.location.origin}
onRedirectCallback={onRedirectCallback}
>
<Component {...pageProps} />
Expand Down Expand Up @@ -164,24 +164,21 @@ export default withAuthenticationRequired(Profile);
```js
// use-api.js
import { useEffect, useState } from 'react';
import { useAuth0 } from '@auth0/auth0-react';

export const useApi = (url, options = {}) => {
const { isAuthenticated } = useAuth0();
const [error, setError] = useState(null);
const [loading, setLoading] = useState(true);
const [response, setResponse] = useState(true);
const { getAccessToken } = useAuth0();
const [state, setState] = useState({
error: null,
loading: true,
data: null,
});
const [refreshIndex, setRefreshIndex] = useState(0);

useEffect(() => {
if (!isAuthenticated) {
setLoading(false);
setError(new Error('The user is not signed in'));
return;
}
(async () => {
try {
const { audience, scope, ...fetchOptions } = options;
const accessToken = await getAccessTokenSilently({ audience, scope });
const accessToken = await getAccessToken({ audience, scope });
stevehobbsdev marked this conversation as resolved.
Show resolved Hide resolved
const res = await fetch(url, {
...fetchOptions,
headers: {
Expand All @@ -190,19 +187,25 @@ export const useApi = (url, options = {}) => {
Authorization: `Bearer ${accessToken}`,
},
});
setResponse(await res.json());
setState({
...state,
data: await res.json(),
error: null,
loading: false,
});
} catch (error) {
setError(error);
} finally {
setLoading(false);
setState({
...state,
error,
loading: false,
});
}
})();
}, []);
}, [refreshIndex]);

return {
loading,
error,
response,
...state,
refresh: () => setRefreshIndex(refreshIndex + 1),
};
};
```
Expand All @@ -214,17 +217,31 @@ Then use it for accessing protected APIs from your components:
import { useApi } from './use-api';

export const Profile = () => {
const { loading, error, response: users } = useApi(
const opts = {
audience: 'https://api.example.com/',
scope: 'read:users',
};
const { login, getTokenWithPopup } = useAuth0();
const { loading, error, refresh, data: users } = useApi(
'https://api.example.com/users',
{
audience: 'https://api.example.com/',
scope: 'read:users',
}
opts
);
const getTokenAndTryAgain = async () => {
await getTokenWithPopup(opts);
refresh();
};
if (loading) {
return <div>Loading...</div>;
}
if (error) {
if (error.error === 'login_required') {
return <button onClick={() => login(opts)}>Login</button>;
}
if (error.error === 'consent_required') {
return (
<button onClick={getTokenAndTryAgain}>Consent to reading users</button>
);
}
return <div>Oops {error.message}</div>;
}
return (
Expand Down
4 changes: 2 additions & 2 deletions README.md
Original file line number Diff line number Diff line change
Expand Up @@ -50,8 +50,8 @@ import App from './App';
ReactDOM.render(
<Auth0Provider
domain="YOUR_AUTH0_DOMAIN"
client_id="YOUR_AUTH0_CLIENT_ID"
redirect_uri={window.location.origin}
clientId="YOUR_AUTH0_CLIENT_ID"
redirectUri={window.location.origin}
>
<App />
</Auth0Provider>,
Expand Down
41 changes: 38 additions & 3 deletions __tests__/auth-provider.test.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -20,14 +20,23 @@ describe('Auth0Provider', () => {

it('should configure an instance of the Auth0Client', async () => {
const opts = {
client_id: 'foo',
clientId: 'foo',
domain: 'bar',
redirectUri: 'baz',
maxAge: 'qux',
extra_param: '__test_extra_param__',
};
const wrapper = createWrapper(opts);
const { waitForNextUpdate } = renderHook(() => useContext(Auth0Context), {
wrapper,
});
expect(Auth0Client).toHaveBeenCalledWith(opts);
expect(Auth0Client).toHaveBeenCalledWith({
client_id: 'foo',
domain: 'bar',
redirect_uri: 'baz',
max_age: 'qux',
extra_param: '__test_extra_param__',
});
await waitForNextUpdate();
});

Expand Down Expand Up @@ -198,7 +207,7 @@ describe('Auth0Provider', () => {
await waitForNextUpdate();
expect(result.current.loginWithRedirect).toBeInstanceOf(Function);
await result.current.loginWithRedirect({
redirect_uri: '__redirect_uri__',
redirectUri: '__redirect_uri__',
});
expect(clientMock.loginWithRedirect).toHaveBeenCalledWith({
redirect_uri: '__redirect_uri__',
Expand Down Expand Up @@ -233,6 +242,19 @@ describe('Auth0Provider', () => {
expect(token).toBe('token');
});

it('should normalize errors from getAccessTokenSilently method', async () => {
clientMock.getTokenSilently.mockRejectedValue(new ProgressEvent('error'));
const wrapper = createWrapper();
const { waitForNextUpdate, result } = renderHook(
() => useContext(Auth0Context),
{ wrapper }
);
await waitForNextUpdate();
expect(result.current.getAccessTokenSilently).rejects.toThrowError(
'Get access token failed'
);
});

it('should provide a getAccessTokenWithPopup method', async () => {
clientMock.getTokenWithPopup.mockResolvedValue('token');
const wrapper = createWrapper();
Expand All @@ -247,6 +269,19 @@ describe('Auth0Provider', () => {
expect(token).toBe('token');
});

it('should normalize errors from getAccessTokenWithPopup method', async () => {
clientMock.getTokenWithPopup.mockRejectedValue(new ProgressEvent('error'));
const wrapper = createWrapper();
const { waitForNextUpdate, result } = renderHook(
() => useContext(Auth0Context),
{ wrapper }
);
await waitForNextUpdate();
expect(result.current.getAccessTokenWithPopup).rejects.toThrowError(
'Get access token failed'
);
});

it('should provide a getIdTokenClaims method', async () => {
clientMock.getIdTokenClaims.mockResolvedValue({
claim: '__test_claim__',
Expand Down
4 changes: 2 additions & 2 deletions __tests__/helpers.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -3,13 +3,13 @@ import React, { PropsWithChildren } from 'react';
import Auth0Provider from '../src/auth0-provider';

export const createWrapper = ({
client_id = '__test_client_id__',
clientId = '__test_client_id__',
domain = '__test_domain__',
...opts
}: Partial<Auth0ClientOptions> = {}) => ({
children,
}: PropsWithChildren<{}>): JSX.Element => (
<Auth0Provider domain={domain} client_id={client_id} {...opts}>
<Auth0Provider domain={domain} clientId={clientId} {...opts}>
{children}
</Auth0Provider>
);
24 changes: 19 additions & 5 deletions __tests__/utils.test.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -2,6 +2,8 @@ import {
defaultOnRedirectCallback,
hasAuthParams,
loginError,
OAuthError,
tokenError,
} from '../src/utils';

describe('utils hasAuthParams', () => {
Expand Down Expand Up @@ -52,17 +54,20 @@ describe('utils defaultOnRedirectCallback', () => {
});
});

describe('utils loginError', () => {
describe('utils error', () => {
it('should return the original error', async () => {
const error = new Error('__test_error__');
expect(loginError(error)).toBe(error);
});

it('should convert an OAuth error to a JS error', async () => {
const error = { error_description: '__test_error__' };
it('should convert OAuth error data to an OAuth JS error', async () => {
const error = {
error: '__test_error__',
error_description: '__test_error_description__',
};
expect(() => {
throw loginError(error);
}).toThrowError('__test_error__');
throw tokenError(error);
}).toThrow(OAuthError);
});

it('should convert a ProgressEvent error to a JS error', async () => {
Expand All @@ -71,4 +76,13 @@ describe('utils loginError', () => {
throw loginError(error);
}).toThrowError('Login failed');
});

it('should produce an OAuth JS error with error properties', async () => {
const error = new OAuthError(
'__test_error__',
'__test_error_description__'
);
expect(error.error).toBe('__test_error__');
expect(error.error_description).toBe('__test_error_description__');
});
});
8 changes: 4 additions & 4 deletions __tests__/withLoginRequired.test.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -4,7 +4,7 @@ import withAuthenticationRequired from '../src/with-login-required';
import { render, screen, waitFor } from '@testing-library/react';
import { Auth0Client } from '@auth0/auth0-spa-js';
import Auth0Provider from '../src/auth0-provider';
import { mocked } from 'ts-jest';
import { mocked } from 'ts-jest/utils';
Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Fix warning from ts-jest


const mockClient = mocked(new Auth0Client({ client_id: '', domain: '' }));

Expand All @@ -13,7 +13,7 @@ describe('withAuthenticationRequired', () => {
const MyComponent = (): JSX.Element => <>Private</>;
const WrappedComponent = withAuthenticationRequired(MyComponent);
render(
<Auth0Provider client_id="__test_client_id__" domain="__test_domain__">
<Auth0Provider clientId="__test_client_id__" domain="__test_domain__">
<WrappedComponent />
</Auth0Provider>
);
Expand All @@ -28,7 +28,7 @@ describe('withAuthenticationRequired', () => {
const MyComponent = (): JSX.Element => <>Private</>;
const WrappedComponent = withAuthenticationRequired(MyComponent);
render(
<Auth0Provider client_id="__test_client_id__" domain="__test_domain__">
<Auth0Provider clientId="__test_client_id__" domain="__test_domain__">
<WrappedComponent />
</Auth0Provider>
);
Expand All @@ -47,7 +47,7 @@ describe('withAuthenticationRequired', () => {
OnRedirecting
);
render(
<Auth0Provider client_id="__test_client_id__" domain="__test_domain__">
<Auth0Provider clientId="__test_client_id__" domain="__test_domain__">
<WrappedComponent />
</Auth0Provider>
);
Expand Down
7 changes: 7 additions & 0 deletions jest.config.js
Original file line number Diff line number Diff line change
Expand Up @@ -9,6 +9,13 @@ module.exports = {
],
testURL: 'https://www.example.com/',
testRegex: '/__tests__/.+test.tsx?$',
globals: {
Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Coverage get's messed up when you transpile to es5 in the tests microsoft/TypeScript#13029

'ts-jest': {
tsConfig: {
target: 'es6',
},
},
},
coverageThreshold: {
global: {
branches: 100,
Expand Down
20 changes: 19 additions & 1 deletion src/auth0-context.tsx
Original file line number Diff line number Diff line change
@@ -1,15 +1,33 @@
import {
BaseLoginOptions,
GetIdTokenClaimsOptions,
GetTokenSilentlyOptions,
GetTokenWithPopupOptions,
IdToken,
LogoutOptions,
PopupLoginOptions,
RedirectLoginOptions,
} from '@auth0/auth0-spa-js';
import { createContext } from 'react';
import { AuthState, initialAuthState } from './auth-state';

export interface RedirectLoginOptions extends BaseLoginOptions {
/**
* The URL where Auth0 will redirect your browser to with
* the authentication result. It must be whitelisted in
* the "Allowed Callback URLs" field in your Auth0 Application's
* settings.
*/
redirectUri?: string;
/**
* Used to store state before doing the redirect
*/
appState?: any; // eslint-disable-line @typescript-eslint/no-explicit-any
/**
* Used to add to the URL fragment before redirecting
*/
fragment?: string;
}

export interface Auth0ContextInterface extends AuthState {
/**
* Get an access token.
Expand Down
Loading