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

[로또] 이승준 미션 제출합니다. #383

Open
wants to merge 35 commits into
base: main
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
35 commits
Select commit Hold shift + click to select a range
3c73d6e
docs : README 작성
1eeSJ Nov 3, 2024
99782a7
test : 로또 클래스에 관한 테스트 케이스 작성
1eeSJ Nov 3, 2024
2c71b49
test : 로또 번호가 음수일 경우의 예외 처리 테스트 케이스 작성
1eeSJ Nov 3, 2024
9b1e7e1
test : 보너스 번호에 관한 예외 처리 테스트 케이스 작성
1eeSJ Nov 3, 2024
260e37e
feat : 입력 메시지에 관한 상수값 작성
1eeSJ Nov 3, 2024
9dfe59b
feat : 출력 메시지에 관한 상수값 작성
1eeSJ Nov 3, 2024
b9a31d5
feat : 에러 메시지에 관한 상수값 작성
1eeSJ Nov 3, 2024
0597843
feat : 입력에 관한 InputView 클래스 작성
1eeSJ Nov 3, 2024
bead620
feat : 출력에 관한 OutputView 클래스 작성
1eeSJ Nov 3, 2024
3c5bc9c
feat : Controller 클래스 작성
1eeSJ Nov 3, 2024
ef1bd72
fix : view 클래스들의 constants를 받는 경로 수정
1eeSJ Nov 3, 2024
1fc69f7
feat : App에서 controller 객체 생성
1eeSJ Nov 3, 2024
412092a
style : view 클래스 코드의 일부 세미콜론 누락 수정
1eeSJ Nov 3, 2024
ddf5dcc
style : InputView 클래스 export 부분 세미콜론 누락 수정
1eeSJ Nov 3, 2024
89cd36e
style : 출력 상수값 일부 수정
1eeSJ Nov 3, 2024
afe6e74
test : 보너스 번호 입력 단위 테스트 부분 코드 수정
1eeSJ Nov 3, 2024
2cfa2ac
test : 보너스 번호 입력 단위 테스트 부분 코드 수정
1eeSJ Nov 3, 2024
86ae67d
feat : LottoService 클래스, LottoRepository 클래스 구조 생성 및 Lotto클래스 위치 변경
1eeSJ Nov 3, 2024
247d671
feat : controller에 LottoService 의존성 주입
1eeSJ Nov 4, 2024
0e1aaea
feat : 무작위 로또 생성 함수 구현 및 로또 관련 상수값 추가
1eeSJ Nov 4, 2024
c52c673
feat : 생성된 로또를 repository에 저장하는 함수 구현
1eeSJ Nov 4, 2024
6ba9ebc
refactor : 무작위 로또 생성 부분에서 배열을 저장하는 대신 로또 객체 저장으로 변경
1eeSJ Nov 4, 2024
860d09d
feat : 내 로또 정보를 가져오는 함수 구현
1eeSJ Nov 4, 2024
090ebce
feat : 구입한 무작위 로또 번호를 출력하는 기능 구현
1eeSJ Nov 4, 2024
53ae382
chore : 미션 규칙에 따라 Lotto 클래스의 number를 다시 private으로 변경
1eeSJ Nov 4, 2024
52f668a
fix : 잘못 작성된 Random 함수 수정 및 Lotto클래스의 getLotto 함수명 변경
1eeSJ Nov 4, 2024
41c1af6
feat : 당첨 번호 배열 분리 기능 구현
1eeSJ Nov 4, 2024
0503a84
feat : 당첨 번호와 보유 로또 번호를 비교하는 기능 구현
1eeSJ Nov 4, 2024
376453b
feat : 우승 상금에 관한 상수값 작성
1eeSJ Nov 4, 2024
c7ee974
feat : 상금 비율 계산 기능 구현
1eeSJ Nov 4, 2024
b8eb302
fix : 당첨 비율 부분 수정
1eeSJ Nov 4, 2024
f7bda9b
refactor : 당첨 번호 비교 함수 리팩토링 및 상수값 추가
1eeSJ Nov 4, 2024
98f115b
refactor : 상금 비율 상수 추가 및 리팩토링
1eeSJ Nov 4, 2024
cce823c
feat : Validator 예외 처리 기능 정리
1eeSJ Nov 4, 2024
ce3c1ad
feat : 공백 입력 시 예외 처리 후 다시 입력 받게 하는 기능 추가
1eeSJ Nov 4, 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
120 changes: 120 additions & 0 deletions README.md
Original file line number Diff line number Diff line change
@@ -1 +1,121 @@
# javascript-lotto-precourse

## 기능 요구 사항


로또 게임 기능을 구현해야 한다. 로또 게임은 아래와 같은 규칙으로 진행된다.

```
- 로또 번호의 숫자 범위는 1~45까지이다.
- 1개의 로또를 발행할 때 중복되지 않는 6개의 숫자를 뽑는다.
- 당첨 번호 추첨 시 중복되지 않는 숫자 6개와 보너스 번호 1개를 뽑는다.
- 당첨은 1등부터 5등까지 있다. 당첨 기준과 금액은 아래와 같다.
- 1등: 6개 번호 일치 / 2,000,000,000원
- 2등: 5개 번호 + 보너스 번호 일치 / 30,000,000원
- 3등: 5개 번호 일치 / 1,500,000원
- 4등: 4개 번호 일치 / 50,000원
- 5등: 3개 번호 일치 / 5,000원
```

- 로또 구입 금액을 입력하면 구입 금액에 해당하는 만큼 로또를 발행해야 한다.
- 로또 1장의 가격은 1,000원이다.
- 당첨 번호와 보너스 번호를 입력받는다.
- 사용자가 구매한 로또 번호와 당첨 번호를 비교하여 당첨 내역 및 수익률을 출력하고 로또 게임을 종료한다.
- 사용자가 잘못된 값을 입력할 경우 `[ERROR]`로 시작하는 메시지와 함께 Error를 발생시키고 해당 메시지를 출력한 다음 해당 지점부터 다시 입력을 받는다.
```
예시) [ERROR] 숫자가 잘못된 형식입니다.
```
<br>
- - -

## 입출력 요구 사항

### 입력

- 로또 구입 금액을 입력 받는다. 구입 금액은 1,000원 단위로 입력 받으며 1,000원으로 나누어 떨어지지 않는 경우 예외 처리한다.

```
14000
```

- 당첨 번호를 입력 받는다. 번호는 쉼표(,)를 기준으로 구분한다.

```
1,2,3,4,5,6
```

- 보너스 번호를 입력 받는다.

```
7
```

### 출력

- 발행한 로또 수량 및 번호를 출력한다. 로또 번호는 오름차순으로 정렬하여 보여준다.

```
8개를 구매했습니다.
[8, 21, 23, 41, 42, 43]
[3, 5, 11, 16, 32, 38]
[7, 11, 16, 35, 36, 44]
[1, 8, 11, 31, 41, 42]
[13, 14, 16, 38, 42, 45]
[7, 11, 30, 40, 42, 43]
[2, 13, 22, 32, 38, 45]
[1, 3, 5, 14, 22, 45]
```

- 당첨 내역을 출력한다.

```
3개 일치 (5,000원) - 1개
4개 일치 (50,000원) - 0개
5개 일치 (1,500,000원) - 0개
5개 일치, 보너스 볼 일치 (30,000,000원) - 0개
6개 일치 (2,000,000,000원) - 0개
```

- 수익률은 소수점 둘째 자리에서 반올림한다. (ex. 100.0%, 51.5%, 1,000,000.0%)

```
총 수익률은 62.5%입니다.
```

- 예외 상황 시 에러 문구를 출력해야 한다. 단, 에러 문구는 "[ERROR]"로 시작해야 한다.

```
[ERROR] 로또 번호는 1부터 45 사이의 숫자여야 합니다.
```

### 실행 결과 예시

```
구입금액을 입력해 주세요.
8000

8개를 구매했습니다.
[8, 21, 23, 41, 42, 43]
[3, 5, 11, 16, 32, 38]
[7, 11, 16, 35, 36, 44]
[1, 8, 11, 31, 41, 42]
[13, 14, 16, 38, 42, 45]
[7, 11, 30, 40, 42, 43]
[2, 13, 22, 32, 38, 45]
[1, 3, 5, 14, 22, 45]

당첨 번호를 입력해 주세요.
1,2,3,4,5,6

보너스 번호를 입력해 주세요.
7

당첨 통계
---
3개 일치 (5,000원) - 1개
4개 일치 (50,000원) - 0개
5개 일치 (1,500,000원) - 0개
5개 일치, 보너스 볼 일치 (30,000,000원) - 0개
6개 일치 (2,000,000,000원) - 0개
총 수익률은 62.5%입니다.
```
26 changes: 26 additions & 0 deletions __tests__/ApplicationTest.js
Original file line number Diff line number Diff line change
Expand Up @@ -42,6 +42,24 @@ const runException = async (input) => {
expect(logSpy).toHaveBeenCalledWith(expect.stringContaining("[ERROR]"));
};

const runExceptionBonus = async (bonusNumber) => {
// given
const logSpy = getLogSpy();

const RANDOM_NUMBERS_TO_END = [1, 2, 3, 4, 5, 6];
const INPUT_NUMBERS_TO_END = ["1000", "1,2,3,4,5,6", bonusNumber, "7"];

mockRandoms([RANDOM_NUMBERS_TO_END]);
mockQuestions([INPUT_NUMBERS_TO_END]);

// when
const app = new App();
await app.run();

// then
expect(logSpy).toHaveBeenCalledWith(expect.stringContaining("[ERROR]"));
}

describe("로또 테스트", () => {
beforeEach(() => {
jest.restoreAllMocks();
Expand Down Expand Up @@ -94,4 +112,12 @@ describe("로또 테스트", () => {
test("예외 테스트", async () => {
await runException("1000j");
});

test("보너스 번호가 당첨 번호와 중복될 경우 예외 처리", async () => {
await runExceptionBonus("3");
});

test("보너스 번호가 1~45 범위를 벗어난 경우 예외 처리", async () => {
await runExceptionBonus("46");
});
});
24 changes: 23 additions & 1 deletion __tests__/LottoTest.js
Original file line number Diff line number Diff line change
Expand Up @@ -14,5 +14,27 @@ describe("로또 클래스 테스트", () => {
}).toThrow("[ERROR]");
});

// TODO: 추가 기능 구현에 따른 테스트 코드 작성
test("로또 번호 1~45 범위를 초과하면 예외가 발생한다.", () => {
expect(() => {
new Lotto([1, 2, 3, 4, 5, 46]);
}).toThrow("[ERROR]");
});

test("로또 번호가 자연수가 아니면 예외가 발생한다.", () => {
expect(() => {
new Lotto([1, 2, 3, 4, 5, 6.5]);
}).toThrow("[ERROR]");
});

test("로또 번호가 음수일 경우 예외가 발생한다.", () => {
expect(() => {
new Lotto([1, -2, 3, -4, 5, -6]);
}).toThrow("[ERROR]");
});

test("로또 번호가 숫자가 아니면 예외가 발생한다.", () => {
expect(() => {
new Lotto([12, "A", 1, "f", 3, "C"]);
}).toThrow("[ERROR]");
});
});
12 changes: 11 additions & 1 deletion src/App.js
Original file line number Diff line number Diff line change
@@ -1,5 +1,15 @@
import Controller from "./controller/Controller.js";

class App {
async run() {}
#controller;

constructor() {
this.#controller = new Controller();
}

async run() {
await this.#controller.run();
}
}

export default App;
67 changes: 67 additions & 0 deletions src/constants/constants.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,67 @@
export const INPUT_MESSAGE = {
inputPurchaseAmount : "구입 금액을 입력해 주세요.\n",
inputWinningNumbers : "\n당첨 번호를 입력해 주세요.\n",
inputBonusNumber : "\n보너스 번호를 입력해 주세요.\n",
};

export const OUTPUT_MESSAGE = {
printPurchaseNumber : (count) => `\n${count}개를 구매했습니다.`,
printWinningStatistics : "\n당첨 통계\n---",
printFifth : (count) => `3개 일치 (5,000원) - ${count}개`,
printFourth : (count) => `4개 일치 (50,000원) - ${count}개`,
printThird : (count) => `5개 일치 (1,500,000원) - ${count}개`,
printSecond : (count) => `5개 일치, 보너스 볼 일치 (30,000,000원) - ${count}개`,
printFirst : (count) => `6개 일치 (2,000,000,000원) - ${count}개`,
printRateReturn : (rate) => `총 수익률은 ${rate}%입니다.`,
};

const ERROR_PREFIX = "[ERROR]";

export const ERROR_MESSAGE = {
nullData : `${ERROR_PREFIX} 값을 입력해야 합니다.`,
purchaseError : `${ERROR_PREFIX} 구입 금액은 1000으로 나누어 떨어지는 수 입니다.`,
purchaseRangeError : `${ERROR_PREFIX} 구입 금액 범위는 양수인 정수입니다.`,

lottoLengthError : `${ERROR_PREFIX} 당첨 번호는 6개여야 합니다.`,
lottoDuplicatedError : `${ERROR_PREFIX} 당첨 번호는 중복될 수 없습니다.`,
lottoRangeError : `${ERROR_PREFIX} 당첨 번호의 범위는 1~45 폐구간 입니다.`,
lottoTypeError : `${ERROR_PREFIX} 당첨 번호는 자연수입니다.`,

bonusLengthError : `${ERROR_PREFIX} 보너스 번호는 1개여야 합니다.`,
bonusDuplicatedError : `${ERROR_PREFIX} 보너스 번호는 당첨 번호와 중복될 수 없습니다.`,
bonusRangeError : `${ERROR_PREFIX} 보너스 번호의 범위는 1~45 폐구간 입니다.`,
bonusTypeError : `${ERROR_PREFIX} 보너스 번호는 자연수입니다.`,
};

export const LOTTO_RELATED_CONSTANTS = {
lottoPrice : 1000,
lottoRangeStart : 1,
lottoRangeEnd : 45,
lottoLength : 6,
rate : 100,
rounding : 1,
};

export const MATCH_REWARD = {
fifth : 5000,
fourth : 50000,
third : 1500000,
second : 30000000,
first : 2000000000,
};

export const MATCH = {
init : 0,
three : 3,
four : 4,
five : 5,
six : 6,
};

export const MATCH_RANK = {
fifit : 0,
fourth : 1,
third : 2,
second : 3,
first : 4,
};
34 changes: 34 additions & 0 deletions src/controller/Controller.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,34 @@
import InputView from "../view/InputView.js";
import OutputView from "../view/OutputView.js";
import LottoService from "../domain/LottoService.js";

class Controller {
#inputView;
#outputView;
#lottoService;

constructor() {
this.#inputView = new InputView();
this.#outputView = new OutputView();
this.#lottoService = new LottoService();
}

async run() {
const payment = await this.#inputView.readPurchaseAmount(); // 지불 가격 유효성 검사 필요
const numberOfPurchase = this.#lottoService.purchaseLottos(payment);

this.#lottoService.generateLottos(numberOfPurchase);
const myLottos = this.#lottoService.getLottos();
this.#outputView.printMyLotto(numberOfPurchase, myLottos);

const winningNumber = await this.#inputView.readWinningLotto(); // 당첨 번호 유효성 검사 필요
const bonusNumber = await this.#inputView.readBonusNumber(); // 보너스 번호 유효성 검사 필요

const match = this.#lottoService.compareWithWinningNumbers(winningNumber, Number(bonusNumber));
const rewardRate = this.#lottoService.calculateRate(match, payment);

this.#outputView.printResult(match, rewardRate);
}
}

export default Controller;
4 changes: 3 additions & 1 deletion src/Lotto.js → src/domain/Lotto.js
Original file line number Diff line number Diff line change
Expand Up @@ -12,7 +12,9 @@ class Lotto {
}
}

// TODO: 추가 기능 구현
getNumbers() {
return this.#numbers;
}
}

export default Lotto;
36 changes: 36 additions & 0 deletions src/domain/LottoRepository.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,36 @@
import { MATCH_REWARD } from "../constants/constants.js";
import { MATCH } from "../constants/constants.js";

class LottoRepository {
lottos;
match;
matchReward;

constructor() {
this.lottos = [];
this.match = [MATCH.init, MATCH.init, MATCH.init, MATCH.init, MATCH.init];
this.matchReward = [MATCH_REWARD.fifth, MATCH_REWARD.fourth, MATCH_REWARD.third, MATCH_REWARD.second, MATCH_REWARD.first];
}

saveLotto(lotto) {
this.lottos.push(lotto);
}

getLottos() {
return this.lottos;
}

getMatch() {
return this.match;
}

updateMatch(number) {
this.match[number]++;
}

getMatchReward() {
return this.matchReward;
}
}

export default LottoRepository;
Loading