728x90

 

지난글 https://nataekoon.tistory.com/170

 

spring boot 의 에러 페이지에 대하여

spring boot에서 에러 페에지이 대해서 알아보자 프로젝트 진행시 에러페이지가 나올 수 있는 요구 사항은 적어도 5가지 이상있을 것 같다 1. 잘못된 요청에 대한 에러페이지2. 로그인을 하지 않

nataekoon.tistory.com

 

이번에는 실습과 함께 좀 보자

 

환경은 대충 이렇다

스프링 부트 3
jdk 17
view thymeleaf

 

 

일부로 에러를 내기위한 /500 

post요청을 처리하는 /post 을 기억하자

 

 

 

 

일반적으로 에러만 안나면 index.html 를 리턴하도록 했다

 

 

 

다음은 폴더구조 이다

 

 

error 디렉터리 밑에 다음처럼 만들어놧다 

 

 

파일의 내용은 다 똑같다 

 

 

(* 참고로 trace는 숨기는게 안전하다 웹 소스경로나 클래스이름이 노출된다)

 

 

 

자 그럼 /500 을 호출해보자 에러가 발생했으므로 index.html 를 리턴하지 않았다

 

 

상황을 해석해보자

사용자가 /500 로 컨트롤러를 호출했고,

해당 컨트롤러는 무조건 예외를 터트리기 때문에 이 예외가 서블릿까지 올라간다

그다음 서블릿에서 forward로 /error 로 요청하고

지난번 포스트에서 보던데로 BasicErrorController 가 이를 처리한다

 

따라서 forward 이기떄문에

브라우저 url을 보면 최초 요청 localhost:8080/500 이 그대로있고

디버깅으로 스레드를 봐도 똑같은 번호를 가진다

 

 

1. MainController 에서 스레드

 

 

2. BasicErrorController 에서 스레드

 

 

둘다 같은걸 볼 수 있다.

 

 

728x90
728x90

spring boot에서 에러 페에지이 대해서 알아보자

 

프로젝트 진행시 에러페이지가 나올 수 있는 요구 사항은 

적어도 5가지 이상있을 것 같다

 

1. 잘못된 요청에 대한 에러페이지

2. 로그인을 하지 않는 자의 요청에 대한 에러페이지

3. 권한을 뛰어넘는 요청에 대한 에러 페이지

4. 시스템의 에러페이지

5. 허용하지 않는 http 메서드에 대한 에러 페이지 (Get,Post 등등 또는 Http Method에 존재하지않는 단어)

 

 

아마 위의 상황에 대한 http 상태 코드는 4xx ~ 5xx 사이이다.

따라서 모든 경우의 상황에 대한 에러 페이지를 만들것인지,

특정 몇개만 만들어서 사용할지 정해야한다

 

하지만 스프링 부트에서는 이 걱정을 말끔히 해결해준다

 

 

스프링 부트의 마법을 보자

 

 

 

실제 ErrorMvcAutoConfiguration 이 bean을 등록하는 코드를 보자

 

 

생성자 쪽을 보면 리졸버 까지 넘기는것을 볼 수 있다.

 

 

 

 

 

우선 컨트롤러이니까 어떤 url에 맵핑되는지 보자 

특별한값을 설정하지 않으면 /error 를 사용한다

 

 

 

다음은 BasicErrorController 이다

 

 

다음은 DefaultErrorViewResolver 이다, 에러 컨트롤러가 위임한 그 리졸버이다.

 

 

여기 코드 문맥을 보자

 

this.resolve에 현재 에러상태코드를 넘긴다 (여기선 500 이라치자)

그렇다면 error/500 이 있는지 확인하고 있다면 해당 ModelAndView 를 리턴하고 

없으면 정적리소스로서 체크하는것다

(이떄문에 타임리프쪽의 에러페이지를 먼저 찾고 없으면 정적파일에서 찾는 우선순위가 정의된다)

 

(스프링은 참 대단한것 같다)

 

 

다시 돌아와서 

그렇게 this.resolve 로 시작 한 코드는 다음과 같은 조건에 따라 결과가 두개이다

1. 준비된 http 상태 코드의 파일이 존재하면 not null

2. 존재하지않으면 null

 

존재하지않으면 노랑색 영역의 코드가 평가되는데

다음과 같은 상황에서 동작할 수 있다

 

만약 405 (Method Not Allowed) 에러 코드가 나왓는데 error/405.html 은 존재하지 않지만, error/4xx.html 이 존재한다면

4xx.html로 modelAndView 생성된다.

 

 

따라서 처음 요구사항을 다시 보면

1. 잘못된 요청에 대한 에러페이지

2. 로그인을 하지 않는 자의 요청에 대한 에러페이지

3. 권한을 뛰어넘는 요청에 대한 에러 페이지

4. 시스템의 에러페이지

5. 허용하지 않는 http 메서드에 대한 에러 페이지 (Get,Post 등등 또는 Http Method에 존재하지않는 단어)

 

4xx, 5xx 두개로 처리할 수있으니 

 

4xx.html, 404.html, 500.html 이면 크게 부족하진 않는다 (요구사항에 따라 다르겠지만)

 

 

 

 

 

728x90
728x90

spring boot 를 사용하다보면 자동으로 

org.assertj.core.api 를 추가해준다.

 

그래서 다음과 같이 static import 해야 사용하기 편리하다

 

 

 

 

메서드명과 코드를 보면 쉽게 이해할 수 있으니 캡쳐로 충분히 이해할 수 있다

 

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
782
783
784
785
786
787
788
789
790
791
792
793
794
795
796
797
798
799
800
801
 
import org.assertj.core.api.SoftAssertions;
import org.junit.jupiter.api.Test;
 
import java.io.IOException;
import java.time.LocalDate;
import java.time.LocalDateTime;
import java.time.ZoneId;
import java.time.ZonedDateTime;
import java.util.Date;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
 
import static org.assertj.core.api.Assertions.*;
 
public class AssertJ_Test {
 
    @Test
    public void 값이_같은지_검증() throws Exception {
 
        int number = 1;
        // 실제값, 기대값
        assertThat(number).isEqualTo(1);
    }
 
    @Test
    public void 값이_다른지_검증() throws Exception {
 
        int number = 1;
        // 실제값, 기대값
        assertThat(number).isNotEqualTo(10);
    }
 
    @Test
    public void null_인지_검증() throws Exception {
 
        String isNull = null;
 
        // 실제값, 기대값
        assertThat(isNull).isNull();
 
    }
    
    @Test
    public void null이_아닌지_검증() throws Exception {
 
        String isNotNull = "나는존재한다";
 
        // 실제값, 기대값
        assertThat(isNotNull).isNotNull();
 
    }
 
    @Test
    public void 값_목록에_포함되는지_검증() throws Exception {
 
        int number = 1;
        List<Integer> list = List.of(123456);
 
        // 실제값, 기대값
        assertThat(number).isIn(list);
 
    }
 
    @Test
    public void 값_목록에_포함_되지_않는지_검증() throws Exception {
 
        int number = 10;
        List<Integer> list = List.of(123456);
 
        // 실제값, 기대값
        assertThat(number).isNotIn(list);
 
    }
 
    @Test
    public void 값보다_작은지_검증() throws Exception {
 
        int number = 10;
 
        // 실제값, 기대값
        assertThat(number).isLessThan(100);
 
    }
    
    @Test
    public void 값보다_작은지_검증2() throws Exception {
 
 
        int nowYear = LocalDate.now().getYear();
        int afterYear = LocalDate.now().plusYears(5L).getYear();
 
        // 실제값, 기대값
        assertThat(nowYear).isLessThan(afterYear);
 
    }
 
    @Test
    public void 값보다_작은지_검증3() throws Exception {
 
        LocalDateTime localDateTime = LocalDateTime.now();
 
        ZonedDateTime zonedDateTime = LocalDateTime.now().plusDays(15).atZone(ZoneId.systemDefault());
 
        Date nowDate = new Date();
        Date afterDate = Date.from(zonedDateTime.toInstant());
 
 
        // 실제값, 기대값
        assertThat(nowDate.getTime()).isLessThan(afterDate.getTime());
 
    }
    
    @Test
    public void 값보다_작거나_같은지_검증() throws Exception {
 
 
        // 실제값, 기대값
        // => 10 은 20보다 작거나 같다
        assertThat(10).isLessThanOrEqualTo(20);
 
    }
 
    @Test
    public void 값보다_큰지_검증() throws Exception {
 
 
        // 실제값, 기대값
        // => 20 은 10 보다 크다
        assertThat(20).isGreaterThan(10);
 
    }
 
    @Test
    public void 값보다_크거나_같은지_검증() throws Exception {
 
 
        // 실제값, 기대값
        // => 20 은 20 보다 크거나 같다
        assertThat(20).isGreaterThanOrEqualTo(20);
 
    }
 
    @Test
    public void 값_A와_값_B_사이에_포함된다() throws Exception {
 
 
        // 실제값, 기대값
        // => 값 50은 1과 100사이에 포함된다
        assertThat(50).isBetween(1100);
 
    }
    
    @Test
    public void True_인지_검증() throws Exception {
 
        LocalDateTime localDateTime = LocalDateTime.now();
 
        ZonedDateTime zonedDateTime = LocalDateTime.now().plusDays(15).atZone(ZoneId.systemDefault());
 
        Date nowDate = new Date();
        Date afterDate = Date.from(zonedDateTime.toInstant());
 
 
        // 실제값, 기대값
        // nowDate 가 afterDate 보다 과거이냐?
        assertThat(nowDate.before(afterDate)).isTrue();
 
    }
 
    @Test
    public void False_인지_검증() throws Exception {
 
        LocalDateTime localDateTime = LocalDateTime.now();
 
        ZonedDateTime zonedDateTime = LocalDateTime.now().plusDays(15).atZone(ZoneId.systemDefault());
 
        Date nowDate = new Date();
        Date afterDate = Date.from(zonedDateTime.toInstant());
 
 
        // 실제값, 기대값
        // afterDate 가 nowDate 보다 과거이냐?
        assertThat(afterDate.before(nowDate)).isFalse();
 
    }
 
    /**
     * String
     */
    @Test
    public void 문자열을_다_포함_하는지_검증() throws Exception {
 
 
        // 실제값, 기대값
        // "피자나라치킨공주" 글자에는 "피자"와 "치킨"이라는 글자가 포함된다
        assertThat("피자나라치킨공주").contains("피자""치킨");
 
    }
 
    @Test
    public void 해당_문자열이_딱_한번만_존재_하는지_검증() throws Exception {
 
 
        // 실제값, 기대값
        // "피자나라치킨공주" 글자에는 "피자" 라는 글자가 딱 한번만 포함된다
        assertThat("피자나라치킨공주").containsOnlyOnce("피자");
 
    }
 
 
    @Test
    public void 문자열에_오직_숫자만_있는지_검증() throws Exception {
 
 
        String phone = "01012345678";
 
        // 실제값, 기대값
        // phone 이라는 문자열 변수에 오직 숫자로 이루어져 있다
        assertThat(phone).containsOnlyDigits();
 
    }
    
    @Test
    public void 문자열에_공백이_있는지_검증() throws Exception {
 
 
        String text = "white space";
 
        // 실제값, 기대값
        // text 이라는 문자열 변수에 공백 문자가 포함된다
        assertThat(text).containsWhitespaces();
 
    }
    
    @Test
    public void 공백_으로만_이루어진_문자열_인지_검증() throws Exception {
 
 
        String whiteSpaceText = "             ";
 
        // 실제값, 기대값
        // whiteSpaceText 이라는 문자열 변수는 공백문자로만 이루어져 있다
        assertThat(whiteSpaceText).containsOnlyWhitespaces();
 
    }
 
    @Test
    public void 정규_표현식을_만족하는지_검증() throws Exception {
 
 
        String phone = "01012345678";
 
        // 실제값, 기대값
        // phone 이라는 문자열 변수는 숫자로만 이루어져 있다
        assertThat(phone).containsPattern("^[0-9]+$");
 
    }
 
    @Test
    public void 문자열이_글자를_포함하지_않는지_검증() throws Exception {
 
 
        String foodNames = "치킨과피자";
 
        // 실제값, 기대값
        // foodNames 이라는 문자열 변수는 "햄버거", "스파게티" 라는 글자가 없다
        assertThat(foodNames).doesNotContain("햄버거""스파게티");
 
    }
 
    @Test
    public void 공백_문자_를_포함하지_않은지_검증() throws Exception {
 
 
        String foodNames = "치킨과피자";
 
        // 실제값, 기대값
        // foodNames 이라는 문자열 변수는 공백문자가 없다
        assertThat(foodNames).doesNotContainAnyWhitespaces();
 
    }
 
    @Test
    public void 공백_문자_로만_이루어진_문자가_아닌지_검증() throws Exception {
 
 
        String message = "Hello            ";
 
        // 실제값, 기대값
        // message 이라는 문자열 변수는 공백문자로만 이루어져 있지 않다 (즉, 글자가 있다)
        assertThat(message).doesNotContainOnlyWhitespaces();
 
    }
    
    @Test
    public void 정규_표현식을_만족하지_않는지_검증() throws Exception {
 
 
        String phone = "홍길동01012345678";
 
        // 실제값, 기대값
        // phone 이라는 문자열 변수는 숫자로만 이루어져 있지 않다
        assertThat(phone).doesNotContainPattern("^[0-9]+$");
 
    }
    
    @Test
    public void 특정_문자로_시작_하는지_검증() throws Exception {
 
 
        String frameWork = "spring data jpa";
 
        // 실제값, 기대값
        // frameWork 이라는 문자열 변수는 "spring" 이라는 글자로 시작한다
        assertThat(frameWork).startsWith("spring");
 
    }
 
    @Test
    public void 특정_문자로_시작_하지_않는지_검증() throws Exception {
 
 
        String application = "egov spring application";
 
        // 실제값, 기대값
        // application 이라는 문자열 변수는 "spring" 이라는 글자로 시작한다
        assertThat(application).doesNotStartWith("spring");
 
    }
    
    @Test
    public void 특정_문자로_끝나는지_하는지_검증() throws Exception {
 
 
        String frameWork = "spring data jpa";
 
        // 실제값, 기대값
        // frameWork 이라는 문자열 변수는 "jpa" 이라는 글자로 끝난다
        assertThat(frameWork).endsWith("jpa");
 
    }
    
    @Test
    public void 특정_문자로_끝나지_않는지_검증() throws Exception {
 
 
        String application = "egov spring application";
 
        // 실제값, 기대값
        // application 이라는 문자열 변수는 "egov" 이라는 글자로 끝나지 않는다
        assertThat(application).doesNotEndWith("egov");
 
    }
 
 
    /**
     * 숫자에 대한 추가 검증
     */
 
    @Test
    public void 값이_0_인지_검증() throws Exception {
 
        int zeroCount = 0;
 
        // 실제값, 기대값
        // zeroCount 은 0 이다
        assertThat(zeroCount).isZero();
 
    }
    
    @Test
    public void 값이_0_이_아닌지_검증() throws Exception {
 
        int count = 10;
 
        // 실제값, 기대값
        // count 은 0 이 아니다
        assertThat(count).isNotZero();
 
    }
 
    @Test
    public void 값이_0_이_양수_인지_검증() throws Exception {
 
        int age = 132;
 
        // 실제값, 기대값
        // age 은 양수이다
        assertThat(age).isPositive();
        assertThat(age).isNotNegative();
 
    }
 
    @Test
    public void 값이_0_이_음수_인지_검증() throws Exception {
 
        int myCash = -1;
 
        // 실제값, 기대값
        // myCash 은 음수이다
        assertThat(myCash).isNotPositive();
        assertThat(myCash).isNegative();
 
    }
 
 
    /**
     * 날짜에 대한 검증
     */
    @Test
    public void 비교할_값_보다_이전_날짜_인지_검증() throws Exception {
 
        LocalDateTime nowDateTime = LocalDateTime.now();
        LocalDateTime afterDateTime = LocalDateTime.of(20301231235959);
 
        // 실제값, 기대값
        // nowDateTime 은 afterDateTime 보다 이전이다
        assertThat(nowDateTime).isBefore(afterDateTime);
 
 
    }
 
    @Test
    public void 비교할_값_보다_이전_이거나_같은_날짜_인지_검증() throws Exception {
 
        LocalDateTime nowDateTime = LocalDateTime.now();
        LocalDateTime afterDateTime = LocalDateTime.of(
                nowDateTime.getYear(),
                nowDateTime.getMonthValue(),
                nowDateTime.getDayOfMonth(),
                nowDateTime.getHour(),
                nowDateTime.getMinute(),
                nowDateTime.getSecond(),
                nowDateTime.getNano()
        );
        LocalDateTime beforeDateTime = LocalDateTime.now().minusDays(1);
 
        // 실제값, 기대값
        // nowDateTime 은 afterDateTime 보다 이전 이거나 같다
        assertThat(nowDateTime).isBeforeOrEqualTo(afterDateTime);
 
        // beforeDateTime 은 nowDateTime 보다 이전 이거나 같다
        assertThat(beforeDateTime).isBeforeOrEqualTo(nowDateTime);
 
 
    }
 
    @Test
    public void 비교할_값_보다_이후_날짜_인지_검증() throws Exception {
 
        LocalDateTime nowDateTime = LocalDateTime.now();
        LocalDateTime afterDateTime = LocalDateTime.now().plusDays(10);
 
        // 실제값, 기대값
        // afterDateTime 은 nowDateTime 보다 이후 이다(미래)
        assertThat(afterDateTime).isAfter(nowDateTime);
 
 
    }
 
    @Test
    public void 비교할_값_보다_이후_이거나_같은_날짜_인지_검증() throws Exception {
 
        LocalDateTime nowDateTime = LocalDateTime.now();
 
        LocalDateTime afterDateTime = LocalDateTime.now().plusYears(1);
 
        // 실제값, 기대값
 
        // afterDateTime 은 nowDateTime 보다 이후 이거나 같다
        assertThat(afterDateTime).isAfterOrEqualTo(nowDateTime);
 
 
    }
 
 
    @Test
    public void 날짜_의_특정_칼럼_을_제외하고_나머지_값이_같은지_검증() throws Exception {
 
        LocalDateTime nowDateTime = LocalDateTime.now();
 
        LocalDateTime beforeDateTime1 = LocalDateTime.now().minusNanos(1);
        LocalDateTime beforeDateTime2 = LocalDateTime.now().minusSeconds(1);
        LocalDateTime beforeDateTime3 = LocalDateTime.now().minusMinutes(1);
        LocalDateTime beforeDateTime4 = LocalDateTime.now().minusHours(1);
 
        // 실제값, 기대값
 
        // nowDateTime 은 beforeDateTime1 의 나노초를 제외한 시간대는 같다
        assertThat(nowDateTime).isEqualToIgnoringNanos(beforeDateTime1);
 
        // nowDateTime 은 beforeDateTime2 의 초 이하를 제외한 시간대는 같다
        assertThat(nowDateTime).isEqualToIgnoringSeconds(beforeDateTime2);
 
        // nowDateTime 은 beforeDateTime3 의 분 이하를 제외한 시간대는 같다
        assertThat(nowDateTime).isEqualToIgnoringMinutes(beforeDateTime3);
 
        // nowDateTime 은 beforeDateTime3 의 시간 이하를 제외한 시간대는 같다
        assertThat(nowDateTime).isEqualToIgnoringHours(beforeDateTime4);
 
 
 
    }
 
    /**
     * 컬렉션에 대한 검증
     */
 
    // List Or Set
    @Test
    public void 컬렉션의_크기가_기대한_값과_같은지_검증() throws Exception {
 
        List<Integer> list = List.of(12345);
 
        // 실제값, 기대값
        // list 은 크기가 5 이다
        assertThat(list).hasSize(5);
 
 
    }
 
    @Test
    public void 컬렉션이_해당_값을_포함_하는지_검증() throws Exception {
 
        List<Integer> list = List.of(12345);
 
        // 실제값, 기대값
        // list 은 1,2,3 을 포함한다
        assertThat(list).contains(123);
 
 
    }
 
    @Test
    public void 컬렉션이_지정한_값만_포함_하는지_검증() throws Exception {
 
        List<Integer> list = List.of(1234);
 
        // 실제값, 기대값
        // list 은 정확히 1,2,3,4 만 가지고 있다
        assertThat(list).containsOnly(1234);
 
 
    }
 
 
    @Test
    public void 컬렉션이_지정한_값의_일부를_포함_하는지_검증() throws Exception {
 
        List<Integer> list = List.of(1234);
 
        // 실제값, 기대값
        // list 은 1, 3, 5, 7, 9 중에 일부를 포함한다 (1, 3)
        assertThat(list).containsAnyOf(13579);
 
    }
 
    @Test
    public void 컬렉션이_지정한값을_딱_한번만_포함_하는지_검증() throws Exception {
 
        List<Integer> list = List.of(1234);
 
        // 실제값, 기대값
        // list 은 1, 3 을 딱 한번만 포함한다
        assertThat(list).containsOnlyOnce(1,3);
 
    }
 
 
    // Map
    @Test
    public void Map이_지정한_키를_포함_하는지_검증() throws Exception {
 
        Map<String, Object> map = new HashMap<>();
        map.put("age"132);
        map.put("name""ksj");
 
 
        // 실제값, 기대값
        // map 은 age 라는 키를 가지고 있다
        assertThat(map).containsKey("age");
 
    }
 
    @Test
    public void Map이_지정한_키들을_포함_하는지_검증() throws Exception {
 
        Map<String, Object> map = new HashMap<>();
        map.put("age"132);
        map.put("name""ksj");
 
 
        // 실제값, 기대값
        // map 은 age, name 라는 키들을 가지고 있다
        assertThat(map).containsKeys("age""name");
 
    }
 
 
 
    @Test
    public void Map이_지정한_키만_포함_하는지_검증() throws Exception {
 
        Map<String, Object> map = new HashMap<>();
        map.put("age"132);
        map.put("name""ksj");
//        map.put("name2", "ksj");
 
 
        // 실제값, 기대값
        // map 은 age, name 라는 키들만 가지고 있다
        assertThat(map).containsOnlyKeys("age""name");
 
    }
 
 
    @Test
    public void Map이_지정한_키를_포함_하지_않는지_검증() throws Exception {
 
        Map<String, Object> map = new HashMap<>();
        map.put("age"132);
        map.put("name""ksj");
 
 
        // 실제값, 기대값
        // map 은 title 라는 키를 갖지 않는다
        assertThat(map).doesNotContainKey("title");
 
    }
 
    @Test
    public void Map이_지정한_키들을_포함_하지_않는지_검증() throws Exception {
 
        Map<String, Object> map = new HashMap<>();
        map.put("age"132);
        map.put("name""ksj");
 
 
        // 실제값, 기대값
        // map 은 title 라는 키를 갖지 않는다
        assertThat(map).doesNotContainKeys("title""content");
 
    }
 
    @Test
    public void Map이_지정한_엔트리_포함_하는지_검증() throws Exception {
 
        Map<String, Object> map = new HashMap<>();
        map.put("age"132);
        map.put("name""ksj");
 
        Map.Entry<String, Integer> entry = Map.entry("age"132);
 
        // 실제값, 기대값
        // map 은 Map.entry("age", 132) 라는 엔트리를 갖는지 검증
        assertThat(map).contains(entry);
 
    }
 
    @Test
    public void Map이_지정한_엔트리_포함_하는지_검증2() throws Exception {
 
        Map<String, Object> map = new HashMap<>();
        map.put("age"132);
        map.put("name""ksj");
 
        // 실제값, 기대값
        // map 은 키는 "age" 이고 값은 132 인 엔트리를 갖는지 검증
        assertThat(map).containsEntry("age"132);
 
    }
 
    /**
     * 익셉션 관련 검증
     */
    @Test
    public void 익셉션이_발생_하는지_검증() throws Exception {
 
        assertThatThrownBy(() -> {
            throw new IllegalArgumentException();
        });
 
    }
 
    @Test
    public void 해당_타입의_익셉션이_발생_하는지_검증() throws Exception {
 
        assertThatThrownBy(() -> {
            throw new IllegalArgumentException();
        }).isInstanceOf(RuntimeException.class);
 
    }
 
    @Test
    public void 해당_타입의_익셉션이_발생_하는지_검증_두번째_방법() throws Exception {
 
 
        assertThatExceptionOfType(RuntimeException.class)
                .isThrownBy(() -> {
                    throw new IllegalArgumentException();
                });
    }
 
    @Test
    public void IOException_익셉션이_발생_하는지_검증() throws Exception {
 
        assertThatIOException().isThrownBy(() -> {
            throw new IOException();
        });
 
    }
 
    @Test
    public void 익셉션이_발생_하지_않는지_검증() throws Exception {
 
        assertThatCode(() -> {
            // code...
            System.out.println("예외가 발생하지 않습니다");
        }).doesNotThrowAnyException();
 
    }
 
    @Test
    public void 여러_검증을_모아서_검증하기() throws Exception {
 
        // 테스트가 실패해야 정상, 실패했을때 나오는 콘솔을 확인
 
        SoftAssertions soft = new SoftAssertions();
        soft.assertThat(1).isEqualTo(1);
        soft.assertThat(10).isGreaterThan(20);  // <- 거짓
        soft.assertThat(10).isGreaterThan(30);  // <- 거짓
        soft.assertThat("1234567890").containsOnlyDigits();
        soft.assertAll();
 
 
    }
 
    @Test
    public void 여러_검증을_모아서_검증하기_두번째_방법() throws Exception {
 
        // 테스트가 실패해야 정상, 실패했을때 나오는 콘솔을 확인
 
        SoftAssertions.assertSoftly((soft) -> {
            soft.assertThat(1).isEqualTo(1);
            soft.assertThat(10).isGreaterThan(20);  // <- 거짓
            soft.assertThat(10).isGreaterThan(30);  // <- 거짓
            soft.assertThat("1234567890").containsOnlyDigits();
        });
 
    }
 
    /**
     * 테스트 설명달기
     */
    @Test
    public void 테스트에_설명_달기_1() throws Exception {
 
        // 실패해야 정상
        assertThat(1)
                .as("1은 양의 정수이다")
                .isNegative();
    }
 
    @Test
    public void 테스트에_설명_달기_2() throws Exception {
 
        String message = "1은 양의 정수이다";
 
        // 실패해야 정상
        assertThat(1)
                .as("[정수 검사] %s", message)
                .isNegative();
    }
    
    @Test
    public void 테스트에_설명_달기_3() throws Exception {
 
 
        // 실패해야 정상
        assertThat(1)
                .describedAs("1은 양의 정수이다")
                .isNegative();
    }
 
    @Test
    public void 테스트에_설명_달기_4() throws Exception {
 
 
        // 실패해야 정상
        String message = "1은 양의 정수이다";
 
        // 실패해야 정상
        assertThat(1)
                .describedAs("[정수 검사] %s", message)
                .isNegative();
    }
 
}
 
cs

 

 

 

728x90
728x90

트랜잭션의 ACID 라는걸 알아보자

 

✨ 원자성 Atomicity

트랜잭션에서 여러 데이터 베이스에 변경 사항을 주는데

마치 하나의 작업처럼 다루어 져야 한다.

모든 작업이 성공하든지, 모두 실패하든지

특정 어느 하나만 성공하거나 실패해선 안된다

 

일관성 Consistency

데이터 베이스에서 정한 규칙을 지켜야한다 무결성 제약조건이라 든지

 

 

격리성 Isolation

동시에 실행되는 트랜잭션들이 서로에게 영향을 미치지 않아야 한다

한마디로, 동시에 같은 데이터를 수정하면 안된다

 

 

지속성 Durability

트랜잭션이 성공적으로 마쳤으면,

어딘가에 남겨야한다

예를 들면 로그가 되겠다

그래야 에러가 났을때 복구 시킬 수 있다

728x90
728x90

spring boot 3 버전에서 

 

일반적인 jedis client 를 추가하고

 

JedisPool 을 Bean 으로 등록하려하면

 

 

 

MXBean already registered with name org.apache.commons.pool2:type=GenericObjectPool,name=pool

 

에러가 발생한다

 

이미 등록된게 있다는건데

이를 해결하기위해서

 

다음과 같이 설정하거나

 

그래도 안된다면, 

 

구성편집 > 실행 옵션 

 

다음을 추가한다

 

 

 

728x90

'개발 > spring' 카테고리의 다른 글

AssertJ 기본 문법 정리  (0) 2024.07.02
트랜잭션 ACID  (0) 2024.06.29
JUnit5 배워보자  (0) 2024.05.05
Junit 5 설정하기  (0) 2024.04.28
yaml(yml)에 있는 값을 실시간으로 가져오고 싶다(프로퍼티)  (0) 2024.04.02
728x90

 

지난 포스팅에서 JUnit5 설정 방법을 알아봤다 (gradle)

설정을 아직 안했다면 이걸 보자

 

https://nataekoon.tistory.com/119

 

Junit 5 설정하기

builder.gradle 파일에 다음과 같이 설정해주면 된다

nataekoon.tistory.com

 

 

어노테이션에 대한 설명은 하단에서 한다, 충분히 유추 가능하니 

우선 사용법 부터 보자 

 

 

1. True 을 단정하기

말 그대로 참을 단정한다

 

 

 

여기서 assertTrue 는 static import를 한것이니 패키지 명에 주의하자

junit의 꺼다 

 

 

 

 

 

2. 거짓을 단정하기

(1) 의 을  거짓을 부정해서 참으로 단정할 수 있겠지만

따로 존재한다

 

 

 

3. Null 단정하기

4. Not Null 단정하기

 

 

 

 

5. 테스트가 실패했음을 단정하기

 

fail() 이라는 메서드로 실패를 단정한다

예제에서는 랜덤 값이 10일 때로 조건을 걸었다

 

예외 메세지 콘솔은 다음과 같다 AssertionFailedError 를 확인했다

 

 

 

6. 같음을 단정하기

7. 다름을 단정하기

주의 할건 

assertEquals / assertNotEquals 메서드의 파라미터 순서인데

첫번째 파라미터 - 기대값

두번째 파라미터 - 검증할 값 (실제값)

즉, B가 A이길 바라는거다

 

 

 

8. 객체가 같음을 단정

 

 

 

 

9. 객체가 다름을 단정

 

 

 

10. 특정 예외가 발생함을 단정

assertThrows 메서드를 사용하는데 파라미터 순서에 주의 하자

 

첫번쨰 파라미터: 발생할 예외 클래스 명시

두번째 파라미터: 실행할 로직  (편하게 람다식으로 작성한다)

 

 

 

11. 예외가 발생하지 않음을 단정

assertDoesNotThrow 메서드로 파라미터는 하나만 받는다 (편하게 람다식으로 작성한다)

 

 

 

10번과 11번의 람다식은 Executable 함수형 인터페이스로

execute() 라는 메서드를 잘보면 예외를 던지게 되어있다


어노테이션에 대해서 알아보자

 

 

첫번쨰 - @Test 어노테이션

 

테스트 메서드에 반드시 적용해야 하는 어노테이션이고,

메서드는 private 이면 안된다.

 

 

 

 

 

두번쨰 -  @DisplayName("")  어노테이션

JUnit은 메서드를 한글로 짓기엔 거시기한게 있어서 설명을 달 수 있는 어노테이션을 제공한다

 

@DisplayName("") : 테스트에 설명을 추가한다, 클래스나 메서드에 사용할 수 있다

 

 

 

 

 

 

이렇게 추가하고 테스트를 실행하면 다음과 같이 표시된다 (사용하지않으면 메서드 이름으로 대체된다)

 

 

 

 

 

세번쨰 -  @BeforeXXX , @AfterXXX 선후 관계 어노테이션

예를 들어 테스트 마다 시점에 대하여 처리할 로직이 있다면 

유용하게 쓰일 수 있다 

 

1. @BeforeAll  모든 테스트 시작 전에 호출 한다 

2. @BeforeEach  각 테스트 시작 전에 호출 한다 

3. @AfterEach  각 테스트 종료 후에 호출 한다 

4. @AfterAll  모든 테스트 종료 후에 호출 한다 

 

 

 

그렇다면 테스트 코드를 실행해보고 출력을 봐보자

 

 

 

 

너무 많으니까 하나만 실행해서 봐보자

콘솔을 보면 바로 이해할 수 있을 것이다

 

 

 

 

 

네번쨰 -  @Disabled 어노테이션

어떤 이유로 해당 테스트를 실행하고 싶지 않을때 사용 할 수 있다

 

 

 

 

전체 테스트를 실행해보면 무시되는 것을 볼 수 있다

 

 

 

우아 힘들다... 이만 마쳐보겠다

728x90
728x90

 

builder.gradle 파일에 다음과 같이 설정해주면 된다

 

 

728x90
728x90

 

이 친구와 친해지고싶다

 

 

우선 다음과 같이 yml 파일에 server의 port를 적어놧다 치자

 

 

property나 yml(yaml) 두 방식 다

@Value("${}") 이 문법을 써서 프로퍼티 값을 가져오는건 잘안다

 

 

 

 

근데 이게 많아질 수록 클래스에는 필드가 많아지는 샘이다(인스턴스 멤버)

 

물론 스프링은 싱글톤 이긴 하지만

 

여러 서비스에서 서버의 포트를 사용한다면 그것도 중복이긴 할 것 같다(우기기)

 

그래서 원하는것은 yml 파일에서  실시간으로 "server.port" 와 같이

문자열을 파라미터로 넘겨서 리턴 받고싶었다.

 

또한

 

@PropertySource 라는 어노테이션에 application.properties 나 특정 프로퍼티를 지정하고

클래스의 멤버에 매핑하는방법도 있는데 (가끔 써봣다)

 

스프링 공식 문서에서는 yml(yaml)에서는 사용할 수 없다 나왓다  

 

 

그리고 YAML 파일을 직접 로드하려면 

 

 

이런 내용이 있다

 

여기서 눈여겨 봐야할건 YamlPropertiesFactoryBean 이다

내용은 YamlPropertiesFactoryBean은 yaml 파일을properties 파일로 로드한다는 것 같다

 

내용은 얼마안되니 보자

 

 

 

일단 생성자로 생성할 수 있고, 기본 전략은 싱글톤이다.

YamlProcessor을 확장했기에 YamlProcessor에서 열려있는것은 사용할 수 있다

 

그래서 다음과 같은 코드를 작성했다

 

우선 결론적으로 사용할 내가만든 클래스이다, 키를 주면 꺼내는 메서드가 있다

 

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
import org.springframework.beans.factory.config.YamlPropertiesFactoryBean;
 
import java.util.*;
 
public class MyYmlProperty {
 
 
    private final YamlPropertiesFactoryBean yamlBean;
 
    public MyYmlProperty(YamlPropertiesFactoryBean yamlBean) {
        this.yamlBean = yamlBean;
    }
 
 
    public String getKey(String keyName) {
        Properties properties = yamlBean.getObject();
        return properties.getProperty(keyName);
    }
 
 
}
cs

 

 

 

여기가 실제로 Bean을 등록하는 부분이다

setter로 Resource 위치를 알려주면 되는데 application.yml 파일명을 지정했다 (클래스패스)

(참고로 ClassPathResource는 Resource의 구현체이다)

 

그리고 위에 있는 MyYmlProperty에 생성자 주입하여 Bean을 등록했다

 

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
import com.myjwt.ksj.common.MyYmlProperty;
import org.springframework.beans.factory.config.YamlPropertiesFactoryBean;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.core.io.ClassPathResource;
 
@Configuration
public class TestConfig {
 
    @Bean
    public MyYmlProperty myYmlProperty() {
 
        YamlPropertiesFactoryBean propertiesFactoryBean = new YamlPropertiesFactoryBean();
//        Resource classPathResource = new ClassPathResource("application.yml");
        ClassPathResource classPathResource = new ClassPathResource("application.yml");
        propertiesFactoryBean.setResources(classPathResource);
        MyYmlProperty myYmlProperty = new MyYmlProperty(propertiesFactoryBean);
 
        return myYmlProperty;
 
    }
 
}
cs

 

 

 

그렇다면 다음과 같이 사용할수있다

 

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
@Controller
@RequiredArgsConstructor
public class MainController {
 
    // 첫번째 방법
    @Value("${server.port}")
    private Integer port;
 
    // 두번째 방법
    private final MyYmlProperty myYmlProperty;
 
    @GetMapping("/")
    public String main() {
        // 첫번째 방법의 필드에서 가져오기
        System.out.println("port = " + port);
 
        // 두번째 방법의 메서드로 가져오기
        String serverPort = myYmlProperty.getKey("server.port");
        System.out.println("serverPort = " + serverPort);
 
        return "main";
    }
}
cs

 

 

 

 

잘 출력 되는것을 볼 수 있다

 

새벽에 갑자기 즉흥적으로 알아봣는데 해소되었다..

 

간다...

 

728x90

'개발 > spring' 카테고리의 다른 글

JUnit5 배워보자  (0) 2024.05.05
Junit 5 설정하기  (0) 2024.04.28
Test 코드에서 lombok 사용하기  (0) 2024.02.13
[트랜잭션] 트랜잭션을 위한 Exception 종류 알아보기  (0) 2022.12.28
message source사용하기  (0) 2021.03.05

+ Recent posts