Skip to content
Merged
Show file tree
Hide file tree
Changes from all 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
10 changes: 9 additions & 1 deletion apps/server/src/problems/problems.module.ts
Original file line number Diff line number Diff line change
@@ -1,9 +1,17 @@
import { Module } from '@nestjs/common';
import { ProblemsController } from './problems.controller';
import { ProblemsService } from './problems.service';
import { ValidationService } from './validation/validation.service';
import { HandlerResolver } from './validation/handler-resolver';
import { UnitValidationHandler } from './validation/handlers/unit-validation.handler';

@Module({
controllers: [ProblemsController],
providers: [ProblemsService],
providers: [
ProblemsService,
ValidationService,
HandlerResolver,
UnitValidationHandler,
],
})
export class ProblemsModule {}
31 changes: 15 additions & 16 deletions apps/server/src/problems/problems.service.ts
Original file line number Diff line number Diff line change
@@ -1,27 +1,26 @@
import { Injectable } from '@nestjs/common';
import { SubmitRequestDto } from './dto/submit-request.dto';
import { SubmitResponseDto } from './dto/submit-response.dto';
import { ValidationService } from './validation/validation.service';
import { ProblemType } from './types/problem-type.enum';

@Injectable()
export class ProblemsService {
constructor(private readonly validationService: ValidationService) {}

submit(problemId: number, body: SubmitRequestDto): SubmitResponseDto {
const problem_type = body.submitConfig?.[0];
const answer = (problem_type?.configInfo?.answer ?? '') as string;
// MOCK 문제 데이터
const problemData = {
problemType: ProblemType.UNIT,
answer: '1234',
};

// test: 정답이 1234이면 PASS, 아니면 FAIL
const isCorrect = answer === '1234';
const result = this.validationService.validate(
problemData.problemType,
body.submitConfig[0],
problemData,
);

return isCorrect
? { result: 'PASS', feedback: [] }
: {
result: 'FAIL',
feedback: [
{
field: 'answer',
code: 'WRONG_ANSWER',
message: '틀렸습니다.',
},
],
};
return result;
}
}
4 changes: 4 additions & 0 deletions apps/server/src/problems/types/problem-type.enum.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,4 @@
export enum ProblemType {
UNIT = 'unit',
SCENARIO = 'scenario',
}
21 changes: 21 additions & 0 deletions apps/server/src/problems/validation/handler-resolver.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,21 @@
import { Injectable } from '@nestjs/common';
import { ValidationHandler } from './handlers/validation.handler';
import { UnitValidationHandler } from './handlers/unit-validation.handler';
import { ProblemType } from '../types/problem-type.enum';

@Injectable()
export class HandlerResolver {
private handlers: Map<ProblemType, ValidationHandler>;

constructor(private readonly unitHandler: UnitValidationHandler) {
this.handlers = new Map([[ProblemType.UNIT, this.unitHandler]]);
}

resolve(problemType: ProblemType): ValidationHandler {
const handler = this.handlers.get(problemType);
if (!handler) {
throw new Error(`${problemType} 에 대한 ValidationHandler가 없습니다.`);
}
return handler;
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,32 @@
import { ValidationHandler } from './validation.handler';
import { SubmitResponseDto } from 'src/problems/dto/submit-response.dto';
import { ConfigDto } from 'src/problems/dto/submit-request.dto';
import { ProblemType } from 'src/problems/types/problem-type.enum';

export class UnitValidationHandler implements ValidationHandler {
support(problemType: ProblemType): boolean {
return problemType === ProblemType.UNIT;
}

validate(submitConfig: ConfigDto, problemData: any): SubmitResponseDto {
// test 용 검증 로직: 제출한 풀이가 problemData.answer와 일치하는지 확인
const answer = (submitConfig.configInfo?.answer ?? '') as string;
// eslint-disable-next-line @typescript-eslint/no-unsafe-member-access
const expectedAnswer = problemData.answer as string;

if (answer === expectedAnswer) {
return { result: 'PASS', feedback: [] };
}

return {
result: 'FAIL',
feedback: [
{
field: 'answer',
code: 'WRONG_ANSWER',
message: '틀렸습니다.',
},
],
};
}
}
20 changes: 20 additions & 0 deletions apps/server/src/problems/validation/handlers/validation.handler.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,20 @@
import { ConfigDto } from 'src/problems/dto/submit-request.dto';
import { SubmitResponseDto } from 'src/problems/dto/submit-response.dto';
import { ProblemType } from 'src/problems/types/problem-type.enum';

export interface ValidationHandler {
/**
* 문제 타입(ex. unit)에 대해 핸들러가 지원되는지 여부를 반환하는 메서드
* @param problemType 문제 타입
* @returns 지원 여부 (true/false)
*/
support(problemType: ProblemType): boolean;

/**
* 제출된 풀이를 검증하고 결과를 반환하는 메서드
* @param submitConfig 제출된 풀이 객체
* @param problemData 문제의 메타데이터 (우선은 이렇게 둠)
* @returns 검증 결과 객체
*/
validate(submitConfig: ConfigDto, problemData: any): SubmitResponseDto;
}
18 changes: 18 additions & 0 deletions apps/server/src/problems/validation/validation.service.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,18 @@
import { Injectable } from '@nestjs/common';
import { ConfigDto } from '../dto/submit-request.dto';
import { HandlerResolver } from './handler-resolver';
import { ProblemType } from '../types/problem-type.enum';

@Injectable()
export class ValidationService {
constructor(private readonly handlerResolver: HandlerResolver) {}

validate(
problemType: ProblemType,
submitConfig: ConfigDto,
problemData: any,
) {
const handler = this.handlerResolver.resolve(problemType);
return handler.validate(submitConfig, problemData);
}
}