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

refactor(be): admin problem and submission exception handling #2130

Closed
Show file tree
Hide file tree
Changes from 3 commits
Commits
Show all changes
85 commits
Select commit Hold shift + click to select a range
ce3eb88
refactor(be): improve admin problem module error handling
mnseok Sep 30, 2024
1b503bd
refactor(be): improve admin submission module error handling
mnseok Sep 30, 2024
94a24d0
refactor(be): improve admin problem module error handling in excluded…
mnseok Sep 30, 2024
aa72481
fix(fe): prevent horizontal scrollbar from appearing (#2097)
eunnbi Sep 19, 2024
b578c0e
fix(be): add image url for production env (#2101)
Jaehyeon1020 Sep 20, 2024
7c89f7c
refactor(fe): admin problem create page refactoring (#2100)
eunnbi Sep 20, 2024
c196b16
fix(be): add condition for compile-error when run test (#2102)
Jaehyeon1020 Sep 20, 2024
921e9cf
fix(fe): block testcase and limit edit when submission exist (#2084)
Kohminchae Sep 22, 2024
c8a5805
feat(be): implement search in contest overall apis (#2088)
Jaehyeon1020 Sep 22, 2024
9ec5b43
fix(infra): remove aws elastic ip for nat gateway (#2113)
bocklag Sep 23, 2024
8b297ad
revert: revert typescript-eslint monorepo from v8 to v7 (#2114)
jimin9038 Sep 23, 2024
fcb2502
refactor(fe): add hide spin button util style (#2109)
B0XERCAT Sep 24, 2024
dbe9918
fix(be): set testcases on cache before publiush message (#2120)
jspark2000 Sep 24, 2024
7512f7e
fix(fe): delete scrollbar (#2116)
jihorobert Sep 25, 2024
7fdaeb6
feat(be): handle prisma exception in global exception filter (#2103)
Jaehyeon1020 Oct 1, 2024
df607ad
refactor(be): improve client submission module error handling logic (…
jimin9038 Oct 1, 2024
bb2be54
refactor(be): improve client problem error handling logic (#2117)
jimin9038 Oct 1, 2024
9365fe9
feat(infra): add lifecycle policy to ecr repositories (#2132)
bocklag Oct 1, 2024
6c13f7c
chore(deps): update codemirror (#2105)
renovate[bot] Oct 1, 2024
53e9edd
fix(deps): update tiptap monorepo to ^2.8.0 (#2106)
renovate[bot] Oct 1, 2024
b30867e
chore(fe): change blind text color from red to neutral (#2131)
B0XERCAT Oct 1, 2024
996819a
refactor(fe): refactor settings page (#2128)
jihorobert Oct 2, 2024
6597996
fix(deps): update all non-major dependencies (#2058)
renovate[bot] Oct 2, 2024
9d25076
chore(deps): lock file maintenance (#2091)
renovate[bot] Oct 2, 2024
52d29c4
fix(deps): update dependency date-fns to v4 (#2110)
renovate[bot] Oct 2, 2024
d44dee9
chore(deps): update dependency @types/express to v5 (#2126)
renovate[bot] Oct 2, 2024
2ff513a
chore(deps): lock file maintenance (#2134)
renovate[bot] Oct 2, 2024
513f962
feat(infra): created gateway vpc endpoint for communication between E…
junhyunlee123 Oct 2, 2024
3751886
fix(fe): allocate more memory to node in devcontainer for smooth fron…
jimin9038 Oct 2, 2024
96004cd
refactor(be): apply global exception filter to admin contest and admi…
Jaehyeon1020 Oct 3, 2024
1f3b16b
feat(iris): pass run output when test submission (#2137)
jspark2000 Oct 4, 2024
2f0762d
fix(fe): fix updateinfo modal (#2129)
juhyeong0505 Oct 4, 2024
ac79773
feat(fe): create submission detail modal (#2104)
jwoojin9 Oct 4, 2024
060b857
refactor(be): refactor client group and admin notice module (#2133)
cho-to Oct 4, 2024
04ae9ad
feat(fe): add code manually save button (#2095)
Kohminchae Oct 5, 2024
05c11e5
feat(fe): implement test in code editor (#2093)
B0XERCAT Oct 6, 2024
60bbeee
fix(fe): show warning when editing contest if submission exists (#2139)
B0XERCAT Oct 6, 2024
d2ef053
chore(deps): lock file maintenance (#2142)
renovate[bot] Oct 7, 2024
33ddbf9
fix(fe): fix scrollbar (#2145)
jihorobert Oct 8, 2024
a4a4fcb
chore(fe): wrap the line of the hint automatically (#2147)
eunnbi Oct 8, 2024
aee2a83
chore(fe): fix admin contest time format (#2146)
jwoojin9 Oct 8, 2024
d3f1492
fix(be): use is-optional (#2149)
Jaehyeon1020 Oct 9, 2024
6b43c88
fix(fe): resolve code disappearing problem after test/submit (#2144)
Kohminchae Oct 9, 2024
ebcebd4
feat(fe): poc - new data table for admin (#2136)
eunnbi Oct 10, 2024
2e4dd44
fix(fe): change language's storekey to include problemId and contestI…
Kohminchae Oct 10, 2024
3fe2567
fix(fe): restore the score of the problems when the new problems are …
eunnbi Oct 10, 2024
4b39136
fix(fe): not show add/delete buttons when testcase cannot be edited (…
eunnbi Oct 11, 2024
18c3e48
fix(fe): fix test bugs (#2148)
B0XERCAT Oct 12, 2024
f157b92
fix(fe): block editing problem when contest has submission (#2158)
B0XERCAT Oct 12, 2024
5c85e18
fix(fe): prevent submission detail modal overflow (#2157)
jwoojin9 Oct 12, 2024
be9eaf5
fix(fe): fix CreateProblem modal (#2156)
Kimhyojung0810 Oct 12, 2024
9ded02d
chore(fe): create signup hover (#2143)
juhyeong0505 Oct 13, 2024
eb959a5
chore(fe): fix updatenow text (#2153)
juhyeong0505 Oct 13, 2024
2f6781e
fix(fe): fix contestdenied message (#2155)
Kimhyojung0810 Oct 13, 2024
39e3bfb
fix(fe): hotfix build error (#2160)
eunnbi Oct 13, 2024
44deed7
fix(fe): restore the imported problem order (#2159)
eunnbi Oct 13, 2024
90b6baa
fix(fe): differentiate toast messages for save and submit actions (#2…
B0XERCAT Oct 13, 2024
6fec43c
chore(fe): edit update information dialog content (#2162)
B0XERCAT Oct 13, 2024
694ff46
fix(fe): fix save button bug (#2163)
jihorobert Oct 13, 2024
d35afe5
feat(infra): created gateway vpc endpoint for communication between E…
junhyunlee123 Oct 14, 2024
c1c8e84
fix(be): no not-found exception in get-score-summaries api (#2167)
Jaehyeon1020 Oct 14, 2024
dc1c378
feat(infra): created gateway vpc endpoint for communication between E…
junhyunlee123 Oct 14, 2024
5ab0220
fix(infra): set portgres version to 14 to avoid downgrade errors and …
bocklag Oct 14, 2024
37d104a
fix(fe): render katex in editor input, output description (#2170)
jimin9038 Oct 16, 2024
7522b80
chore(deps): update all non-major dependencies (#2140)
renovate[bot] Oct 19, 2024
9cd71f8
chore(deps): lock file maintenance (#2166)
renovate[bot] Oct 19, 2024
cf568a6
fix(fe): fix submission take number (#2173)
youznn Oct 23, 2024
d7d2f01
chore(deps): update all non-major dependencies (#2171)
renovate[bot] Oct 26, 2024
e3db67a
chore(deps): update all non-major dependencies (#2174)
renovate[bot] Oct 29, 2024
32d1ec0
fix(deps): update tiptap monorepo to ^2.9.1 (#2175)
renovate[bot] Oct 29, 2024
5610818
refactor(fe): create getResultColor and replace related codes (#2178)
B0XERCAT Nov 1, 2024
866172d
refactor(fe): replace data table admin component to new components (1…
eunnbi Nov 3, 2024
a696c05
refactor(fe): replace data table admin component to new components (2…
eunnbi Nov 3, 2024
6f6b922
fix(deps): update all non-major dependencies (#2180)
renovate[bot] Nov 5, 2024
a731801
chore(deps): lock file maintenance (#2185)
renovate[bot] Nov 6, 2024
6c3ff41
chore(fe): use coolify as stage frontend (#2184)
jimin9038 Nov 6, 2024
9c603d4
refactor(fe): app folder structure refactoring (#2187)
eunnbi Nov 7, 2024
d3cbd37
refactor(be): adm user cli auth exception refator (#2115)
hwantae Nov 8, 2024
5f074ff
fix(be): update redis test key for race condition (#2190)
donghun1214 Nov 8, 2024
5e46e22
refactor(fe): refactor public directory (#2188)
youznn Nov 9, 2024
7b8c762
fix(fe): hotfix params prop type (#2195)
eunnbi Nov 11, 2024
4c13f69
feat(fe): improve editor problems dropdown list ui (#2196)
jwoojin9 Nov 11, 2024
fc5e4eb
refactor(be): remove useless internalserver exception
mnseok Nov 12, 2024
a89e559
refactor(be): remove useless internalserverexceptions
mnseok Nov 12, 2024
e7cdd8d
refactor(be): remove useless internalserverexceptions in admin submis…
mnseok Nov 12, 2024
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
230 changes: 24 additions & 206 deletions apps/backend/apps/admin/src/problem/problem.resolver.ts
Original file line number Diff line number Diff line change
@@ -1,12 +1,4 @@
import {
InternalServerErrorException,
Logger,
NotFoundException,
ParseArrayPipe,
UnprocessableEntityException,
UsePipes,
ValidationPipe
} from '@nestjs/common'
import { ParseArrayPipe, UsePipes, ValidationPipe } from '@nestjs/common'
import {
Args,
Context,
Expand All @@ -24,15 +16,8 @@ import {
ProblemTestcase,
WorkbookProblem
} from '@generated'
import { Prisma } from '@prisma/client'
import { PrismaClientKnownRequestError } from '@prisma/client/runtime/library'
import { AuthenticatedRequest } from '@libs/auth'
import { OPEN_SPACE_ID } from '@libs/constants'
import {
ConflictFoundException,
ForbiddenAccessException,
UnprocessableDataException
} from '@libs/exception'
import { CursorValidationPipe, GroupIDPipe, RequiredIntPipe } from '@libs/pipe'
import { ImageSource } from './model/image.output'
import {
Expand All @@ -46,8 +31,6 @@ import { ProblemService } from './problem.service'

@Resolver(() => ProblemWithIsVisible)
export class ProblemResolver {
private readonly logger = new Logger(ProblemResolver.name)

constructor(private readonly problemService: ProblemService) {}

@Mutation(() => ProblemWithIsVisible)
Expand All @@ -61,24 +44,7 @@ export class ProblemResolver {
groupId: number,
@Args('input') input: CreateProblemInput
) {
try {
return await this.problemService.createProblem(
input,
req.user.id,
groupId
)
} catch (error) {
if (error instanceof UnprocessableDataException) {
throw error.convert2HTTPException()
} else if (
error instanceof Prisma.PrismaClientKnownRequestError &&
error.code === 'P2003'
) {
throw new UnprocessableEntityException(error.message)
}
this.logger.error(error)
throw new InternalServerErrorException()
}
return await this.problemService.createProblem(input, req.user.id, groupId)
}

@Mutation(() => [ProblemWithIsVisible])
Expand All @@ -93,56 +59,23 @@ export class ProblemResolver {
groupId: number,
@Args('input') input: UploadFileInput
) {
try {
return await this.problemService.uploadProblems(
input,
req.user.id,
groupId
)
} catch (error) {
if (error instanceof UnprocessableDataException) {
throw error.convert2HTTPException()
}
this.logger.error(error)
throw new InternalServerErrorException()
}
return await this.problemService.uploadProblems(input, req.user.id, groupId)
}

@Mutation(() => ImageSource)
async uploadImage(
@Args('input') input: UploadFileInput,
@Context('req') req: AuthenticatedRequest
) {
try {
return await this.problemService.uploadImage(input, req.user.id)
} catch (error) {
if (error instanceof UnprocessableDataException) {
throw error.convert2HTTPException()
}
this.logger.error(error)
throw new InternalServerErrorException()
}
return await this.problemService.uploadImage(input, req.user.id)
}

@Mutation(() => Image)
async deleteImage(
@Args('filename') filename: string,
@Context('req') req: AuthenticatedRequest
) {
try {
return await this.problemService.deleteImage(filename, req.user.id)
} catch (error) {
if (error instanceof UnprocessableDataException) {
throw error.convert2HTTPException()
} else if (
error instanceof PrismaClientKnownRequestError &&
error.code == 'P2025'
) {
throw new NotFoundException(error.message)
}
this.logger.error(error)
throw new InternalServerErrorException()
}
return await this.problemService.deleteImage(filename, req.user.id)
}

@Query(() => [ProblemWithIsVisible])
Expand Down Expand Up @@ -171,38 +104,17 @@ export class ProblemResolver {
groupId: number,
@Args('id', { type: () => Int }, new RequiredIntPipe('id')) id: number
) {
try {
return await this.problemService.getProblem(id, groupId)
} catch (error) {
if (
error instanceof Prisma.PrismaClientKnownRequestError &&
error.name == 'NotFoundError'
) {
throw new NotFoundException(error.message)
}
this.logger.error(error)
throw new InternalServerErrorException()
}
return await this.problemService.getProblem(id, groupId)
}

@ResolveField('tag', () => [ProblemTag])
async getProblemTags(@Parent() problem: ProblemWithIsVisible) {
try {
return await this.problemService.getProblemTags(problem.id)
} catch (error) {
this.logger.error(error)
throw new InternalServerErrorException()
}
return await this.problemService.getProblemTags(problem.id)
}

@ResolveField('testcase', () => [ProblemTestcase])
async getProblemTestCases(@Parent() problem: ProblemWithIsVisible) {
try {
return await this.problemService.getProblemTestcases(problem.id)
} catch (error) {
this.logger.error(error)
throw new InternalServerErrorException()
}
return await this.problemService.getProblemTestcases(problem.id)
}

@Mutation(() => ProblemWithIsVisible)
Expand All @@ -215,24 +127,7 @@ export class ProblemResolver {
groupId: number,
@Args('input') input: UpdateProblemInput
) {
try {
return await this.problemService.updateProblem(input, groupId)
} catch (error) {
if (
error instanceof UnprocessableDataException ||
error instanceof ConflictFoundException
) {
throw error.convert2HTTPException()
} else if (error instanceof Prisma.PrismaClientKnownRequestError) {
if (error.name == 'NotFoundError') {
throw new NotFoundException(error.message)
} else if (error.code === 'P2003') {
throw new UnprocessableEntityException(error.message)
}
}
this.logger.error(error)
throw new InternalServerErrorException()
}
return await this.problemService.updateProblem(input, groupId)
}

@Mutation(() => ProblemWithIsVisible)
Expand All @@ -245,25 +140,12 @@ export class ProblemResolver {
groupId: number,
@Args('id', { type: () => Int }, new RequiredIntPipe('id')) id: number
) {
try {
return await this.problemService.deleteProblem(id, groupId)
} catch (error) {
if (
error instanceof Prisma.PrismaClientKnownRequestError &&
error.name == 'NotFoundError'
) {
throw new NotFoundException(error.message)
}
this.logger.error(error)
throw new InternalServerErrorException()
}
return await this.problemService.deleteProblem(id, groupId)
}
}

@Resolver(() => ContestProblem)
export class ContestProblemResolver {
private readonly logger = new Logger(ProblemResolver.name)

constructor(private readonly problemService: ProblemService) {}

@Query(() => [ContestProblem], { name: 'getContestProblems' })
Expand All @@ -277,20 +159,7 @@ export class ContestProblemResolver {
@Args('contestId', { type: () => Int }, new RequiredIntPipe('contestId'))
contestId: number
) {
try {
return await this.problemService.getContestProblems(groupId, contestId)
} catch (error) {
if (
error instanceof UnprocessableDataException ||
error instanceof ForbiddenAccessException
) {
throw error.convert2HTTPException()
} else if (error.code == 'P2025') {
throw new NotFoundException(error.message)
}
this.logger.error(error)
throw new InternalServerErrorException(error.message)
}
return await this.problemService.getContestProblems(groupId, contestId)
}

@Mutation(() => [ContestProblem])
Expand All @@ -305,41 +174,21 @@ export class ContestProblemResolver {
contestId: number,
@Args('orders', { type: () => [Int] }, ParseArrayPipe) orders: number[]
) {
try {
return await this.problemService.updateContestProblemsOrder(
groupId,
contestId,
orders
)
} catch (error) {
if (
error instanceof UnprocessableDataException ||
error instanceof ForbiddenAccessException
) {
throw error.convert2HTTPException()
} else if (error.code == 'P2025') {
throw new NotFoundException(error.message)
}
this.logger.error(error)
throw new InternalServerErrorException(error.message)
}
return await this.problemService.updateContestProblemsOrder(
groupId,
contestId,
orders
)
}

@ResolveField('problem', () => ProblemWithIsVisible)
async getProblem(@Parent() contestProblem: ContestProblem) {
try {
return await this.problemService.getProblemById(contestProblem.problemId)
} catch (error) {
this.logger.error(error)
throw new InternalServerErrorException()
}
return await this.problemService.getProblemById(contestProblem.problemId)
}
}

@Resolver(() => WorkbookProblem)
export class WorkbookProblemResolver {
private readonly logger = new Logger(ProblemResolver.name)

constructor(private readonly problemService: ProblemService) {}

@Query(() => [WorkbookProblem], { name: 'getWorkbookProblems' })
Expand All @@ -352,20 +201,7 @@ export class WorkbookProblemResolver {
groupId: number,
@Args('workbookId', { type: () => Int }) workbookId: number
) {
try {
return await this.problemService.getWorkbookProblems(groupId, workbookId)
} catch (error) {
if (
error instanceof UnprocessableDataException ||
error instanceof ForbiddenAccessException
) {
throw error.convert2HTTPException()
} else if (error.code == 'P2025') {
throw new NotFoundException(error.message)
}
this.logger.error(error)
throw new InternalServerErrorException(error.message)
}
return await this.problemService.getWorkbookProblems(groupId, workbookId)
}

@Mutation(() => [WorkbookProblem])
Expand All @@ -380,33 +216,15 @@ export class WorkbookProblemResolver {
// orders는 항상 workbookId에 해당하는 workbookProblems들이 모두 딸려 온다.
@Args('orders', { type: () => [Int] }, ParseArrayPipe) orders: number[]
) {
try {
return await this.problemService.updateWorkbookProblemsOrder(
groupId,
workbookId,
orders
)
} catch (error) {
if (
error instanceof UnprocessableDataException ||
error instanceof ForbiddenAccessException
) {
throw error.convert2HTTPException()
} else if (error.code == 'P2025') {
throw new NotFoundException(error.message)
}
this.logger.error(error)
throw new InternalServerErrorException(error.message)
}
return await this.problemService.updateWorkbookProblemsOrder(
groupId,
workbookId,
orders
)
}

@ResolveField('problem', () => ProblemWithIsVisible)
async getProblem(@Parent() workbookProblem: WorkbookProblem) {
try {
return await this.problemService.getProblemById(workbookProblem.problemId)
} catch (error) {
console.log(error)
throw new InternalServerErrorException()
}
return await this.problemService.getProblemById(workbookProblem.problemId)
}
}
Loading