expo-popcore-app/hooks/use-template-like.test.ts

412 lines
12 KiB
TypeScript

import { renderHook, act } from '@testing-library/react-native'
import { useTemplateLike } from './use-template-like'
import { root } from '@repo/core'
import { TemplateSocialController } from '@repo/sdk'
import { handleError } from './use-error'
import { templateSocialStore } from '@/stores/templateSocialStore'
jest.mock('@repo/core', () => ({
root: {
get: jest.fn(),
},
}))
jest.mock('@/stores/templateSocialStore', () => ({
templateSocialStore: {
getLikeCount: jest.fn().mockReturnValue(0),
setLiked: jest.fn(),
setLikeCount: jest.fn(),
isLiked: jest.fn().mockReturnValue(false),
},
}))
jest.mock('./use-error', () => ({
handleError: jest.fn(async (cb) => {
try {
const data = await cb()
return { data, error: null }
} catch (e) {
return { data: null, error: e }
}
}),
}))
describe('useTemplateLike', () => {
beforeEach(() => {
jest.clearAllMocks()
})
afterEach(() => {
jest.restoreAllMocks()
})
describe('initial state', () => {
it('should return initial state with templateId', () => {
const { result } = renderHook(() => useTemplateLike('template-1'))
expect(result.current.loading).toBe(false)
expect(result.current.error).toBeNull()
})
it('should return initial state without templateId', () => {
const { result } = renderHook(() => useTemplateLike())
expect(result.current.loading).toBe(false)
expect(result.current.error).toBeNull()
})
})
describe('like function', () => {
it('should like template successfully', async () => {
const mockController = {
like: jest.fn().mockResolvedValue(undefined),
}
;(root.get as jest.Mock).mockReturnValue(mockController)
const { result } = renderHook(() => useTemplateLike('template-1'))
let response
await act(async () => {
response = await result.current.like()
})
expect(mockController.like).toHaveBeenCalledWith({ templateId: 'template-1' })
expect(templateSocialStore.setLiked).toHaveBeenCalledWith('template-1', true)
expect(result.current.error).toBeNull()
})
it('should handle API errors when liking', async () => {
const mockError = {
status: 500,
statusText: 'Internal Server Error',
message: 'Failed to like template',
}
const mockController = {
like: jest.fn().mockRejectedValue(mockError),
}
;(root.get as jest.Mock).mockReturnValue(mockController)
const { result } = renderHook(() => useTemplateLike('template-1'))
let response
await act(async () => {
response = await result.current.like()
})
expect(result.current.error).toEqual(mockError)
expect(response).toEqual({ error: mockError })
})
it('should do nothing if templateId is not provided', async () => {
const { result } = renderHook(() => useTemplateLike())
let response
await act(async () => {
response = await result.current.like()
})
expect(response).toEqual({ error: { message: 'TemplateId is required' } })
})
})
describe('unlike function', () => {
it('should unlike template successfully', async () => {
const mockController = {
unlike: jest.fn().mockResolvedValue(undefined),
}
;(root.get as jest.Mock).mockReturnValue(mockController)
const { result } = renderHook(() => useTemplateLike('template-1'))
let response
await act(async () => {
response = await result.current.unlike()
})
expect(mockController.unlike).toHaveBeenCalledWith({ templateId: 'template-1' })
expect(templateSocialStore.setLiked).toHaveBeenCalledWith('template-1', false)
expect(result.current.error).toBeNull()
})
it('should handle API errors when unliking', async () => {
const mockError = {
status: 500,
statusText: 'Internal Server Error',
message: 'Failed to unlike template',
}
const mockController = {
unlike: jest.fn().mockRejectedValue(mockError),
}
;(root.get as jest.Mock).mockReturnValue(mockController)
const { result } = renderHook(() => useTemplateLike('template-1'))
let response
await act(async () => {
response = await result.current.unlike()
})
expect(result.current.error).toEqual(mockError)
expect(response).toEqual({ error: mockError })
})
it('should do nothing if templateId is not provided', async () => {
const { result } = renderHook(() => useTemplateLike())
let response
await act(async () => {
response = await result.current.unlike()
})
expect(response).toEqual({ error: { message: 'TemplateId is required' } })
})
})
describe('checkLiked function', () => {
it('should check liked status successfully', async () => {
const mockData = { liked: true }
const mockController = {
checkLiked: jest.fn().mockResolvedValue(mockData),
}
;(root.get as jest.Mock).mockReturnValue(mockController)
const { result } = renderHook(() => useTemplateLike('template-1'))
let response
await act(async () => {
response = await result.current.checkLiked()
})
expect(mockController.checkLiked).toHaveBeenCalledWith({ templateId: 'template-1' })
expect(templateSocialStore.setLiked).toHaveBeenCalledWith('template-1', true)
expect(result.current.error).toBeNull()
})
it('should handle not liked status', async () => {
const mockData = { liked: false }
const mockController = {
checkLiked: jest.fn().mockResolvedValue(mockData),
}
;(root.get as jest.Mock).mockReturnValue(mockController)
const { result } = renderHook(() => useTemplateLike('template-1'))
await act(async () => {
await result.current.checkLiked()
})
expect(templateSocialStore.setLiked).toHaveBeenCalledWith('template-1', false)
})
it('should handle API errors when checking liked status', async () => {
const mockError = {
status: 500,
statusText: 'Internal Server Error',
message: 'Failed to check liked status',
}
const mockController = {
checkLiked: jest.fn().mockRejectedValue(mockError),
}
;(root.get as jest.Mock).mockReturnValue(mockController)
const { result } = renderHook(() => useTemplateLike('template-1'))
let response
await act(async () => {
response = await result.current.checkLiked()
})
expect(result.current.error).toEqual(mockError)
expect(response).toEqual({ error: mockError })
})
it('should do nothing if templateId is not provided', async () => {
const { result } = renderHook(() => useTemplateLike())
let response
await act(async () => {
response = await result.current.checkLiked()
})
expect(response).toEqual({ error: { message: 'TemplateId is required' } })
})
})
describe('loading state', () => {
it('should set loading to true during like operation', async () => {
let resolveLike: (value: any) => void
const likePromise = new Promise((resolve) => {
resolveLike = resolve
})
const mockController = {
like: jest.fn().mockReturnValue(likePromise),
}
;(root.get as jest.Mock).mockReturnValue(mockController)
const { result } = renderHook(() => useTemplateLike('template-1'))
act(() => {
result.current.like()
})
expect(result.current.loading).toBe(true)
await act(async () => {
resolveLike!(undefined)
await likePromise
})
expect(result.current.loading).toBe(false)
})
it('should set loading to true during unlike operation', async () => {
let resolveUnlike: (value: any) => void
const unlikePromise = new Promise((resolve) => {
resolveUnlike = resolve
})
const mockController = {
unlike: jest.fn().mockReturnValue(unlikePromise),
}
;(root.get as jest.Mock).mockReturnValue(mockController)
const { result } = renderHook(() => useTemplateLike('template-1'))
act(() => {
result.current.unlike()
})
expect(result.current.loading).toBe(true)
await act(async () => {
resolveUnlike!(undefined)
await unlikePromise
})
expect(result.current.loading).toBe(false)
})
it('should set loading to true during checkLiked operation', async () => {
let resolveCheck: (value: any) => void
const checkPromise = new Promise((resolve) => {
resolveCheck = resolve
})
const mockController = {
checkLiked: jest.fn().mockReturnValue(checkPromise),
}
;(root.get as jest.Mock).mockReturnValue(mockController)
const { result } = renderHook(() => useTemplateLike('template-1'))
act(() => {
result.current.checkLiked()
})
expect(result.current.loading).toBe(true)
await act(async () => {
resolveCheck!({ liked: false })
await checkPromise
})
expect(result.current.loading).toBe(false)
})
it('should set loading to false after error', async () => {
const mockError = { status: 500, message: 'Error' }
const mockController = {
like: jest.fn().mockRejectedValue(mockError),
}
;(root.get as jest.Mock).mockReturnValue(mockController)
const { result } = renderHook(() => useTemplateLike('template-1'))
await act(async () => {
await result.current.like()
})
expect(result.current.loading).toBe(false)
})
})
describe('error handling', () => {
it('should clear error on successful like', async () => {
const mockError = { status: 500, message: 'Error' }
const mockController = {
like: jest.fn()
.mockRejectedValueOnce(mockError)
.mockResolvedValueOnce(undefined),
}
;(root.get as jest.Mock).mockReturnValue(mockController)
const { result } = renderHook(() => useTemplateLike('template-1'))
await act(async () => {
await result.current.like()
})
expect(result.current.error).toEqual(mockError)
await act(async () => {
await result.current.like()
})
expect(result.current.error).toBeNull()
})
it('should clear error on successful unlike', async () => {
const mockError = { status: 500, message: 'Error' }
const mockController = {
unlike: jest.fn()
.mockRejectedValueOnce(mockError)
.mockResolvedValueOnce(undefined),
}
;(root.get as jest.Mock).mockReturnValue(mockController)
const { result } = renderHook(() => useTemplateLike('template-1'))
await act(async () => {
await result.current.unlike()
})
expect(result.current.error).toEqual(mockError)
await act(async () => {
await result.current.unlike()
})
expect(result.current.error).toBeNull()
})
it('should clear error on successful checkLiked', async () => {
const mockError = { status: 500, message: 'Error' }
const mockController = {
checkLiked: jest.fn()
.mockRejectedValueOnce(mockError)
.mockResolvedValueOnce({ liked: false }),
}
;(root.get as jest.Mock).mockReturnValue(mockController)
const { result } = renderHook(() => useTemplateLike('template-1'))
await act(async () => {
await result.current.checkLiked()
})
expect(result.current.error).toEqual(mockError)
await act(async () => {
await result.current.checkLiked()
})
expect(result.current.error).toBeNull()
})
})
})