Skip to content

Commit

Permalink
test: header
Browse files Browse the repository at this point in the history
  • Loading branch information
jbrunton committed Aug 31, 2024
1 parent 5b69d0c commit 59d7d0a
Show file tree
Hide file tree
Showing 7 changed files with 103 additions and 11 deletions.
11 changes: 8 additions & 3 deletions client/src/features/auth/context.ts
Original file line number Diff line number Diff line change
Expand Up @@ -6,6 +6,7 @@ type LoadingState = {
signIn?: never
signOut?: never
token?: never
userName?: never
}

type SignedInState = {
Expand All @@ -14,6 +15,7 @@ type SignedInState = {
signIn?: never
signOut: () => void
token: string
userName?: string
}

type SignedOutState = {
Expand All @@ -22,6 +24,7 @@ type SignedOutState = {
signIn: () => void
signOut?: never
token?: never
userName?: never
}

export type AuthContextInterface = LoadingState | SignedInState | SignedOutState
Expand All @@ -37,29 +40,31 @@ const signedOutState = (signIn: () => void): AuthContextInterface => ({
signIn,
})

const signedInState = (token: string, signOut: () => void): AuthContextInterface => ({
const signedInState = (token: string, userName: string | undefined, signOut: () => void): AuthContextInterface => ({
isLoading: false,
isAuthenticated: true,
signOut,
token,
userName,
})

export const AuthContext = createContext<AuthContextInterface>(loadingState)

type GetAuthStateParams = {
isLoading: boolean
token?: string
userName?: string
signOut: () => void
signIn: () => void
}

export const getAuthState = ({ isLoading, token, signIn, signOut }: GetAuthStateParams) => {
export const getAuthState = ({ isLoading, token, userName, signIn, signOut }: GetAuthStateParams) => {
if (isLoading) {
return loadingState
}

if (token) {
return signedInState(token, signOut)
return signedInState(token, userName, signOut)
}

return signedOutState(signIn)
Expand Down
9 changes: 3 additions & 6 deletions client/src/features/auth/organisms/sign-in-button.tsx
Original file line number Diff line number Diff line change
@@ -1,14 +1,11 @@
import React from 'react'
import { useAuth0 } from '@auth0/auth0-react'
import { Button, Spinner } from '@chakra-ui/react'
import { AiOutlineLogout } from 'react-icons/ai'
import { DefaultUserIcon } from '../../room/atoms/default-user-icon'
import { useAuth } from '..'

export const SignInButton = () => {
const { user, isAuthenticated, isLoading, loginWithRedirect, logout } = useAuth0()

const signIn = () => loginWithRedirect()
const signOut = () => logout({ returnTo: window.location.origin })
const { isAuthenticated, isLoading, signIn, signOut, userName } = useAuth()

if (isLoading) {
return (
Expand All @@ -28,7 +25,7 @@ export const SignInButton = () => {

return (
<Button leftIcon={<AiOutlineLogout />} variant='drawer' onClick={signOut}>
Sign Out {user?.name}
Sign Out {userName}
</Button>
)
}
5 changes: 3 additions & 2 deletions client/src/features/auth/provider.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -8,7 +8,7 @@ import debug from 'debug'
const log = debug('auth')

export const AuthProvider: React.FC<{ children: React.ReactNode }> = ({ children }) => {
const { isLoading: isAuth0Loading, loginWithRedirect, logout } = useAuth0()
const { isLoading: isAuth0Loading, loginWithRedirect, logout, user } = useAuth0()
const { accessToken, isLoading: isTokenLoading } = useAccessToken()
const [isLoading, setIsLoading] = useState(true)

Expand All @@ -28,7 +28,7 @@ export const AuthProvider: React.FC<{ children: React.ReactNode }> = ({ children
log('signOut called')
setIsLoading(true)
clearBearerToken()
logout()
logout({ returnTo: window.location.origin })
}

const signIn = () => {
Expand All @@ -40,6 +40,7 @@ export const AuthProvider: React.FC<{ children: React.ReactNode }> = ({ children
const state = getAuthState({
isLoading,
token: accessToken,
userName: user?.name,
signIn,
signOut,
})
Expand Down
6 changes: 6 additions & 0 deletions client/src/mocks/handlers.ts
Original file line number Diff line number Diff line change
Expand Up @@ -14,6 +14,12 @@ export const handlers = [
const response = loadResponse(`./messages/get/${roomId}.json`)
return res(ctx.json(response))
}),

rest.get('users/:userId', (req, res, ctx) => {
const userId = req.params['userId']
const response = loadResponse(`./users/get/${userId}.json`)
return res(ctx.json(response))
}),
]

const loadResponse = (relativePath: string): JSON => {
Expand Down
17 changes: 17 additions & 0 deletions client/src/mocks/users/me.json
Original file line number Diff line number Diff line change
@@ -0,0 +1,17 @@
{
"user": {
"id": "user:google-oauth2|123",
"name": "Test User",
"picture": "https://ui-avatars.com/api/?name=Joe+Bloggs",
"email": "[email protected]"
},
"rooms": [
{
"id": "room:100",
"name": "Test Room",
"ownerId": "user:google-oauth2|123",
"contentPolicy": "private",
"joinPolicy": "invite"
}
]
}
1 change: 1 addition & 0 deletions client/src/mocks/users/system.json
Original file line number Diff line number Diff line change
@@ -0,0 +1 @@
{ "id": "system", "name": "System", "email": "[email protected]" }
65 changes: 65 additions & 0 deletions client/src/shared/navigation/organisms/header.spec.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,65 @@
import React from 'react'
import { render } from '../../../test/fixtures'
import { waitFor, screen } from '@testing-library/react'
import { AuthContext } from '../../../features/auth/context'
import { Header } from './header'
import userEvent from '@testing-library/user-event'

describe('Header', () => {
const signIn = vitest.fn()
const signOut = vitest.fn()

describe('when signed out', () => {
it('has a sign in button', async () => {
render(
<AuthContext.Provider value={{ isAuthenticated: false, isLoading: false, signIn }}>
<Header />
</AuthContext.Provider>,
{
path: '/',
initialEntry: '/',
},
)

const openMenu = screen.getByRole('button', { name: 'Open Menu' })

await waitFor(() => {
expect(openMenu).toBeVisible()
})

userEvent.click(openMenu)

await waitFor(() => {
expect(screen.getByRole('button', { name: 'Sign In' })).toBeVisible()
})
})
})

describe('when signed in', () => {
it('has a sign out button', async () => {
render(
<AuthContext.Provider
value={{ isAuthenticated: true, isLoading: false, signOut, token: 'a1b2c3', userName: 'Test User' }}
>
<Header />
</AuthContext.Provider>,
{
path: '/',
initialEntry: '/',
},
)

const openMenu = screen.getByRole('button', { name: 'Open Menu' })

await waitFor(() => {
expect(openMenu).toBeVisible()
})

userEvent.click(openMenu)

await waitFor(() => {
expect(screen.getByRole('button', { name: 'Sign Out Test User' })).toBeVisible()
})
})
})
})

0 comments on commit 59d7d0a

Please sign in to comment.