-
Notifications
You must be signed in to change notification settings - Fork 97
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
Implement Random Selection of Register Operations When Not Using selectName #1108
base: 31-register
Are you sure you want to change the base?
Conversation
Random selection의 경우 의도한대로 등록한 순서가 아닌, 무작위 순서로 구현을 하였습니다.
우선 순위의 경우 Open Question에도 작성하긴 했지만, 우선순위를 배정한 register 연산과 배정하지 않은 register 연산이 있을 때 어떻게 처리를 해야 할 지 정해야 할 것 같습니다. 아래의 두 가지 방법이 떠오르긴 하는데, 한 번 의견 부탁드립니다!
또한 우선순위를 |
477d170
to
06111d8
Compare
넵, 말씀주신 방법이 좋아보입니다. 우선순위가 있는 register 연산이 있다면 우선순위가 있는 연산끼리 경쟁하고, 등록한 register 연산이 모두 우선순위가 없다면 모두 랜덤하게 처리하는 방향이 좋아보입니다. 이 정책은 정리해서 문서나 JavaDoc으로 남겨두면 좋을 것 같습니다.
두가지 옵션 |
@@ -204,13 +206,18 @@ private void initializeRegisteredArbitraryBuilders( | |||
private void initializeNamedArbitraryBuilderMap( |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
이러면 registerName
이 아닌 register
에는 적용이 안되지 않나요??
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
register
연산에 모두 적용해야 하는데 registeredName
을 통해 등록하는 register
연산만 적용하는 것으로 착각했네요.😅
수정하겠습니다.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
해당 commit에서 작업했습니다. b2abc78
검토 부탁드립니다 :)
넵!
|
넵, 예를 들어, reflection으로 등록하는 registerGroup에서는 등록할 때 클래스에 어노테이션으로 마킹해서 우선순위를 지정할 수 있을텐데요, 다른 입력지점에서는 어노테이션으로 제공하기가 어렵습니다. 입력 방법이 일관적이지 않더라도, 사람들이 자주 쓰는 패턴이여서 인지적 비용이 적다면 일관적이지 않아도 괜찮을 것 같긴 합니다. |
Signed-off-by: yongjunhong <[email protected]>
06111d8
to
b2abc78
Compare
우선순위 기능을 해당 커밋에서 작업했습니다. c143264 간략하게 설명하면,
이때 우선순위는 0보다 커야합니다.
낱개로 등록하는 방법입니다.
아래의 테스트 코드를 실행한 결과 예상한대로 작동함을 확인했습니다.
|
import com.navercorp.fixturemonkey.FixtureMonkey; | ||
import com.navercorp.fixturemonkey.api.matcher.MatcherOperator; | ||
|
||
public interface MatcherInfo { |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
이게 필요한 이유는 뭔가요??
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
다음과 같이 처리를 설계하였으며, 이를 위해 인터페이스를 통해 접근하도록 구성했습니다.
Line 21 in e508ba8
public interface MatcherMetadata { |
하지만 MatcherMetadata
와 다르게 이번 경우에는 내부 구현을 감출 필요가 없다는 생각이 드네요.
따라서 인터페이스를 삭제하고 구현 클래스에 바로 접근을 하도록 수정하려고 하는데 어떻게 생각하시나요?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
새로운 인터페이스를 추가하는 이유가 뭔가요?? 새로운 관심사라고 보시는건가요??
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
관심사를 프로그램의 특정 기능이나 책임을 나타내는 독립적인 부분으로 정의한다면, 우선순위가 적용된 register 연산
을 하나의 관심사로 간주할 수 있다고 생각합니다.
우선 제가 생각하는 관심사 분리의 장점은 재사용성
과 유지보수성
향상입니다.
우선, 우선순위가 적용된 register 연산은 재사용
이 가능합니다.
예를 들어, 다른 모듈에서 유사한 우선순위 기반 등록 메커니즘이 필요할 때, 이 로직을 그대로 활용할 수 있습니다.
만약 다른 case에도 사용을 하려고 한다면
Generic
를 사용해우선순위가 적용된 Object
를 받는 것이 좋을 것 같습니다.
또한, 우선순위가 적용된 register 연산에서 우선순위 입력 정책을 변경하려고 할 때 시스템의 다른 부분에 영향을 주지 않고 독립적
으로 변경할 수 있습니다.
예를 들어, 0
미만의 숫자를 입력하면 예외를 발생시킨다는 정책을 -100
으로 수정할 때, 해당 로직만 수정하면 되므로 다른 코드에 영향을 주지 않습니다.
하지만 MatcherMetadata와 다르게 이번 경우에는 내부 구현을 감출 필요가 없다는 생각이 드네요.
해당 코멘트를 작성한 이유는, MatcherMetadata
와 달리 MatcherInfo
는 단일 구현체만 존재할 가능성이 높아 보였기 때문입니다. 따라서 불필요한 추상화 계층을 제거하는 것이 더 적합하다고 판단했습니다.
@seongahjo 어떻게 생각하시나요?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
인터페이스를 분리하면 구현에 따라서 register 연산
과 우선순위를 가지는 register 연산
이 다르게 처리되는 문제가 생길수도 있을 것 같습니다. 실행흐름이 두 개로 분리되는데 오히려 유지보수가 어려워지는 게 아닐까요? 코드가 기존 흐름대로 실행되지 않고 계속 새로운 필드를 추가하고 독립적인 실행흐름을 만드는 코드는 피하시는 게 좋아보입니다. 개인적인 생각에는, 기존 흐름을 유지하면서 새로운 기능을 제공할 수 있을지 고민이 필요해보입니다.
인터페이스를 새로 추가하는 건 큰 비용이므로 정말 불가피하지 않는 생각하시지 않는 게 좋아보입니다. 인터페이스를 나누는 기준에 대해서 다시 생각해보시는 게 좋을 것 같습니다.
예를 들어, 다른 모듈에서 유사한 우선순위 기반 등록 메커니즘이 필요할 때, 이 로직을 그대로 활용할 수 있습니다.
재사용이 될지도 모르는 코드를 추상화하는 시도는 위험한 것 같은데요, 최소한 3번 이상 쓰여야 추상화를 할지 결정하는 방향이 좋은 것 같습니다. 특히, 고려해야할 요소가 많고 유즈 케이스가 명백하지 않은 경우인 것 같아서 지금 단계에서 추상화는 위험해보입니다.
https://en.wikipedia.org/wiki/Rule_of_three_(computer_programming)
저는 MatcherInfo
라는 이름에서 우선순위가 적용된 register 연산
이라는 정보를 유추하기가 어려운 것 같은데요, 이건 어떻게 알 수 있을까요??
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
실행흐름이 두 개로 분리되는데 오히려 유지보수가 어려워지는 게 아닐까요?
사용 흐름을 고려하지 않았네요,, 좋은 지적 감사합니다.
재사용이 될지도 모르는 코드를 추상화하는 시도는 위험한 것 같은데요, 최소한 3번 이상 쓰여야 추상화를 할지 결정하는 방향이 좋은 것 같습니다.
동의합니다. 재사용 여부가 불확실한 코드는 추상화하기에 위험할 것 같네요,,
건설적인 피드백 감사합니다.
- 앞으로 리팩토링 할 때 꼭 참고하겠습니다!🙂
저는 MatcherInfo라는 이름에서 우선순위가 적용된 register 연산 이라는 정보를 유추하기가 어려운 것 같은데요, 이건 어떻게 알 수 있을까요??
해당 네이밍은 성아님과 함께 대화를 하며 수정하려고 했습니다!
코드 설계 만큼 네이밍 어렵네요,,🥲
인터페이스를 삭제하고 클래스를 참조하도록 변경하겠습니다.
- Class name도 한 번 더 고민해보겠습니다..
덕분에 인사이트가 넓어진 것 같습니다. 감사합니다 :)
import com.navercorp.fixturemonkey.api.matcher.MatcherOperator; | ||
|
||
public interface MatcherInfo { | ||
Integer getPriority(); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
왜 primitive 타입이 아니라 wrapper 타입을 반환하는 걸까요??
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
처음에 null
을 고려하고 설계를 해서 wrapper
클래스로 작성을 했습니다.
하지만, null
을 허용하지 않기 때문에 primitive
타입으로 변경하겠습니다. 😅
아래의 코멘트에서 방향이 결정된 뒤 수정하겠습니다.
Signed-off-by: yongjunhong <[email protected]>
c143264
to
6e64284
Compare
private final Map<Integer, List<MatcherOperator<? extends ArbitraryBuilder<?>>>> | ||
priorityGroupedMatchers = new HashMap<>(); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
priorityGroupedMatchers
와 registeredArbitraryBuilders
중 하나만 남기는 방향이 좋을 것 같습니다.
priorityGroupedMatchers
의 이름 안에 register를 포함해야할 것 같습니다.
import com.navercorp.fixturemonkey.FixtureMonkey; | ||
import com.navercorp.fixturemonkey.api.matcher.MatcherOperator; | ||
|
||
public final class PriorityMatcherOperator { |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
MatcherOperator는 단순히 Matcher
와 Operator
를 가지고 Operator 적용 여부(match)는 외부에서 판단합니다.
적용 여부를 내부에서 판단하는 새로운 인터페이스를 설계하면 PriorityMatcherOperator
도 그 인터페이스의 구현체 중 하나로 내부 구조 노출없이 일관성있게 제공할 수 있지 않을까 싶습니다.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
궁금한 점이 있어 코멘트 남깁니다!
현재 PriorityMatcherOperator
의 경우 아래의 경우로 사용되는 것으로 알고 있습니다.
Collectors.mapping(
it -> new MatcherOperator<>(
it.getMatcherOperator(), it.getMatcherOperator().getOperator().apply(this)
),
register
연산을 입력 받을 때 파라미터로 입력 받은 Function
함수형 인터페이스를 가지고 있다가 apply
를 통해 ArbitrartyBuilder<?>
을 생성하는 것으로 알고 있습니다.
그런 뒤, registeredArbitraryBuilders
에 추가를 하고 giveMeBuilder
에서 match
를 통해 적용 여부를 판단하는 것으로 확인이 됩니다.
이렇게만 본다면 단순히 MatcherOperator
을 우선순위
와 함께 정보를 가지고 있다가 초기화를 해주는 것으로 보이는데요, 아래와 같이 PriorityMatcherOperator
가 내부에서 match
를 해야 하는 case가 있는지 궁금합니다.
적용 여부를 내부에서 판단하는 새로운 인터페이스를 설계하면 PriorityMatcherOperator 도 그 인터페이스의 구현체 중 하나로 내부 구조 노출없이 일관성있게 제공할 수 있지 않을까 싶습니다.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
질문을 잘 이해 못했습니다.
초기화는 match랑 상관이 없는 로직인데 어떤 연관성이 있다고 생각하시는 걸까요?
제가 드린 의견은 우선순위 정보를 외부에 노출할 필요가 없게 만드는 게 좋다는 의견이라고 보시면 될 것 같습니다.
우선순위 정보는 구현에 강하게 의존한 정보라고 보고 있습니다.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
아아 확인했습니다.
제가 성아님의 코멘트를 잘못 이해한 것 같네요,,😅
작업 후 리뷰 요청하겠습니다!
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
제가 드린 의견은 우선순위 정보를 외부에 노출할 필요가 없게 만드는 게 좋다는 의견이라고 보시면 될 것 같습니다.
해당 방법으로 설계를 진행하고 있는데 어려움을 겪고 있습니다.
혹시 성아님께서 생각하신 구체적인 구현 방안이 있다면 간단히 설명해주실 수 있을까요?
제가 어려움을 겪는 부분을 자세히 설명드리자면, 아래의 코드에서 getPriority
메서드를 통해 우선순위 정보를 노출하지 않고 진행하는 것에 어려움을 겪고 있습니다.
현재는
우선순위
:List<연산>
의 Key-Value 형태로 저장을 한 뒤, 각Key
에 해당하는Value
를 섞는 방법으로 진행을 하고있어, Key에 해당하는 우선순위 정보를 가져와야 하는 상황입니다.
private Map<Integer, List<MatcherOperator<? extends ArbitraryBuilder<?>>>> groupMatchersByPriority(
List<PriorityMatcherOperator> registeredArbitraryBuildersWithPriority
) {
return new HashMap<>(registeredArbitraryBuildersWithPriority
.stream()
.collect(Collectors.groupingBy(
PriorityMatcherOperator::getPriority,
Collectors.mapping(
it -> new MatcherOperator<>(
it.getMatcherOperator(), it.getMatcherOperator().getOperator().apply(this)
),
Collectors.toList()
)
))
);
}
private void groupMatchersByPriorityFromNamedMap(
Map<String, PriorityMatcherOperator> registeredPriorityMatchersByName,
Map<Integer, List<MatcherOperator<? extends ArbitraryBuilder<?>>>> registeredGroupMatchersListByPriority
) {
registeredGroupMatchersListByPriority.putAll(
registeredPriorityMatchersByName.entrySet().stream()
.collect(Collectors.groupingBy(
entry -> entry.getValue().getPriority(),
Collectors.mapping(
entry -> new MatcherOperator<>(
new NamedMatcher(entry.getValue().getMatcherOperator().getMatcher(), entry.getKey()),
entry.getValue().getMatcherOperator().getOperator().apply(this)
),
Collectors.toList()
)
))
);
}
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
성아님의 의견과 같이 정렬을 통해 해결할 수도 있으나, 아래의 경우와 같이 최우선으로 선택되는 우선순위가 동일한 경우 그룹핑을 통해 해당 우선순위를 가진 연산들만 shuffle을 해야 할 것 같아 그룹핑을 해야 한다고 표현했습니다!
FixtureMonkey sut = FixtureMonkey.builder()
.register(String.class, monkey -> monkey.giveMeBuilder("test1"), 2)
.register(String.class, monkey -> monkey.giveMeBuilder("test2"), 1)
.register(String.class, monkey -> monkey.giveMeBuilder("test3"), 1)
.register(String.class, monkey -> monkey.giveMeBuilder("test4"), 1)
.register(String.class, monkey -> monkey.giveMeBuilder("test5"), 1)
.build();
추가적으로 정렬은 어디에서 하는게 적절하다고 생각하시나요??
- 저는 아래 링크와 같이 값을 생성하는 시점에 사용자가 입력한 타입에 대해 정렬하면 좋을 것 같다고 생각합니다!
fixture-monkey/fixture-monkey/src/main/java/com/navercorp/fixturemonkey/FixtureMonkey.java
Line 152 in b28ddd7
return Stream.generate(() -> this.giveMeBuilder(type).sample());
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
아래의 경우와 같이 최우선으로 선택되는 우선순위가 동일한 경우 그룹핑을 통해 해당 우선순위를 가진 연산들만 shuffle을 해야 할 것 같아 그룹핑을 해야 한다고 표현했습니다
지금 구현을 보니 그룹핑하면 아래와 같은 두가지 문제가 있을 것 같습니다.
- 타입이 다른데 우선순위가 같은 경우 불필요한 필터링 필요
- 우선순위가 순차적이지 않을경우 순회 방식(1,2,4)
위 두 가지 경우를 어떻게 처리할 생각이신가요??
추가적으로 정렬은 어디에서 하는게 적절하다고 생각하시나요??
구현을 어떻게 하냐에 따라 다를 것 같은데요, Fixture Monkey 인스턴스가 초기화한 후에 해도 되지 않을까 싶네요.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
생각을 해보니 그룹핑을 하는 것은 불필요한 비용이 될 수 있을뿐더러 지적해주신 내용과 같이 두 가지의 경우에서 문제가 있을 것 같습니다. 지적 감사합니다 :D
2주 전에 작성해주신 코멘트와 같이 현재는 하나의 리스트로 연산을 관리하고 있습니다.
그로 인해 다음 단계로 넘어가려 하는데 혹시 생각하신 단계가 있다면 여쭤봐도 될까요?
저는 무작위로 섞는 연산이나, 초기화된 인스턴스를 정렬 하는 것을 생각 하고 있습니다.
그러나 아직 이러한 작업에 익숙치 않아 불필요한 비용이 발생할까 여쭤봅니다..!
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
저는 무작위로 섞는 연산이나, 초기화된 인스턴스를 정렬 하는 것을 생각 하고 있습니다.
넵, 저도 말씀주신 단계로 진행하는 게 좋을 것 같습니다!
위에 두 가지 경우가 각각 trade-off가 있을텐데요, 두 경우 중 어떤 경우가 기본 옵션으로 제공하는 게 좋을지 고민해보시고 진행하시면 좋을 것 같다는 생각이 듭니다.
한 가지 더 공유드리고 싶은 점은, 시드 값이 동일하면 정렬 결과가 항상 같은지 (결정적인지) 확인하는 과정도 필요할 것 같습니다.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
해당 커밋에서 작업 했습니다! d00d7a7
넵, 말씀주신 방법이 좋아보입니다. 우선순위가 있는 register 연산이 있다면 우선순위가 있는 연산끼리 경쟁하고, 등록한 register 연산이 모두 우선순위가 없다면 모두 랜덤하게 처리하는 방향이 좋아보입니다.
위 코멘트와 같이 구현을 했습니다.
저는 연산을 초기화 하는 로직을 총 2가지로 판단을 했습니다.
- 자식 노드가 있는 경우 (
List<ObjectNode> objectNodes = nodeByType.getValue();
리스트의 크기가 1 이상인 경우) - 자식 노드가 없는 경우
자식 노드가 있는 경우, MonkeyManipulatorFactory
에서 자식 노드를 초기화 할 때마다 우선순위를 통해 정렬을 수행했습니다.
자식 노드가 없는 경우, FixtureMonkey
의 giveMeBuilder
에서 정렬을 수행했습니다.
불필요한 cost를 줄이기 위해 match 연산을 통해 초기화 할 수 있는 연산에만 정렬을 수행하려 노력했습니다,,
test를 수행한 결과 모두 통과 하는 것을 확인했습니다.
한 가지 더 공유드리고 싶은 점은, 시드 값이 동일하면 정렬 결과가 항상 같은지 (결정적인지) 확인하는 과정도 필요할 것 같습니다.
정렬 관련 논의가 끝난 뒤, 문서화와 함께 테스트도 함께 작성하겠습니다 :)
동일한 정렬 로직이 두 군데에 적용됨으로 인해 중복 코드가 발생하였습니다.
이를 줄이기 위해 한 번 적용함으로 인해 자식 노드가 있는 경우와 없는 경우에도 모두 정렬을 수행할 수 있는 지점을 찾아보았으나 찾지 못 했습니다..
혹시 성아님이 의도하신 정렬과 제가 적용한 부분에 다른 점이 있다면 말씀 부탁드립니다!🙂
감사합니다.
5faa668
to
0a87f4e
Compare
Signed-off-by: yongjunhong <[email protected]>
0a87f4e
to
6fa32d7
Compare
Signed-off-by: yongjunhong <[email protected]>
@Property | ||
void registerSetFirst() { | ||
String expected = "test2"; | ||
void registerWithPriority() { | ||
FixtureMonkey sut = FixtureMonkey.builder() | ||
.register(String.class, monkey -> monkey.giveMeBuilder("test")) | ||
.register(String.class, monkey -> monkey.giveMeBuilder("test"), 1) | ||
.register(String.class, monkey -> monkey.giveMeBuilder("test2"), 2) | ||
.build(); | ||
|
||
String actual = sut.giveMeBuilder(String.class) | ||
.set("$", expected) | ||
.sample(); | ||
|
||
then(actual).isEqualTo(expected); | ||
then(actual).isEqualTo("test"); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
아래 케이스에서도 잘 동작하는지 확인하는 테스트가 있으면 좋을 것 같습니다!
FixtureMonkey sut = FixtureMonkey.builder()
.register(String.class, monkey -> monkey.giveMeBuilder("test2"), 2)
.register(String.class, monkey -> monkey.giveMeBuilder("test"), 1)
.build();
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
아직 우선순위, 무작위 적용 관련 부분이 구현되지 않아서 해당 코드를 구현한 뒤 테스트 하겠습니다 :)
Signed-off-by: yongjunhong <[email protected]>
Summary
resolved #1081
Open Question
registeredName
, if there are register operations with assigned priorities and others without, how should the operations without assigned priorities be handled?null
?How Has This Been Tested?
Is the Document updated?
Later