-
Notifications
You must be signed in to change notification settings - Fork 880
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: remove key parameter from testvar methods #7040
Refactor: remove key parameter from testvar methods #7040
Conversation
318b87f
to
499f31c
Compare
98c8720
to
6689596
Compare
@@ -1279,7 +1271,7 @@ func makeCurrentWorkflowConditionFailedError( | |||
lastWriteVersion := common.EmptyVersion | |||
return &persistence.CurrentWorkflowConditionFailedError{ | |||
Msg: "random message", | |||
RequestID: tv.String("PrevRequestID"), | |||
RequestID: tv.RequestID(), |
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.
I think it doesn't worth to wrap every parameter into tv function.
For example this code
startRequest.StartRequest.RequestId = tv.At("PrevRequestID")
lets me know that I'm re-using prev request ID.
Intentions are more clear.
tv.RequestID()
leaves me with the question "why this will fail/not fail"?
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.
Well, I think it does. So I removed that String()
method at all. Let's create methods for every entity type.
require.False(t, existed) | ||
require.Equal(t, 1, reg.Len()) | ||
|
||
t.Run("deny new update since it is exceeding the limit", func(t *testing.T) { | ||
_, _, err = reg.FindOrCreate(context.Background(), tv.UpdateID("2")) | ||
_, _, err = reg.FindOrCreate(context.Background(), tv2.UpdateID()) |
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.
When reading this code it is very hard to instantly grasp why this should be denied.
_, _, err = reg.FindOrCreate(context.Background(), "second_update_id")
is easier to understand.
or
tv.At("second_update_id") if we want to provide TV guaranties.
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.
with the new approach, it will be more readable as:
_, _, err = reg.FindOrCreate(context.Background(), tv.WithUpdateID("second_update"))
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.
Sorry, I don't understand how new version is harder to understand than previous one.
var resExh *serviceerror.ResourceExhausted | ||
require.ErrorAs(t, err, &resExh, "creating update #2 should be denied") | ||
require.Equal(t, 1, reg.Len()) | ||
}) | ||
|
||
t.Run("increasing limit allows new updated to be created", func(t *testing.T) { | ||
_, _, err = reg.FindOrCreate(context.Background(), tv.UpdateID("2")) | ||
_, _, err = reg.FindOrCreate(context.Background(), tv2.UpdateID()) | ||
var resExh *serviceerror.ResourceExhausted | ||
require.ErrorAs(t, err, &resExh) | ||
require.Equal(t, 1, reg.Len()) | ||
|
||
limit += 1 |
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.
limit
is defined in another test, as internal variable, and used in a closure.
this was super hard to read.
@@ -458,9 +461,9 @@ func TestSendMessages(t *testing.T) { | |||
|
|||
t.Run("registry with 2 created updates has no messages to send", func(t *testing.T) { | |||
var err error | |||
upd1, _, err = reg.FindOrCreate(context.Background(), tv.UpdateID("1")) | |||
upd1, _, err = reg.FindOrCreate(context.Background(), tv.UpdateID()+"_1") |
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.
and here for some reason you are not using tv1.
even though you are replacing the same code.
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.
and same thoughts - tv.At("update_id_1") reads better.
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.
I changed all cases like this to use tv.WithUpdateIDN(1).UpdateID()
. I agree it is a better, uniform way.
@@ -589,14 +592,14 @@ func TestAbort(t *testing.T) { | |||
reg := update.NewRegistry(&mockUpdateStore{ | |||
VisitUpdatesFunc: func(visitor func(updID string, updInfo *persistencespb.UpdateInfo)) { | |||
visitor( | |||
tv.UpdateID("1"), | |||
tv.UpdateID()+"_1", |
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.
same here, why you are not using:
tv1 := tv.WithUpdateID("1")
?
tests/update_workflow_sdk_test.go
Outdated
tv := testvars.New(s.T()). | ||
WithTaskQueue(s.TaskQueue()). | ||
WithNamespaceName(namespace.Name(s.Namespace())). | ||
WithRunID("") |
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.
this basically "reset run ID".
Why?
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.
Because default behavior is to generate random one, but in these tests I need WorkflowExecution()
with empty RunID()
.
@@ -97,7 +100,10 @@ WorkflowExecutionTerminated // This can be EventID=3 if WF is terminated before | |||
func (s *UpdateWorkflowSdkSuite) TestUpdateWorkflow_TimeoutWorkflowAfterUpdateAccepted() { | |||
ctx, cancel := context.WithTimeout(context.Background(), time.Minute) | |||
defer cancel() | |||
tv := testvars.New(s.T()).WithTaskQueue(s.TaskQueue()).WithNamespaceName(namespace.Name(s.Namespace())) | |||
tv := testvars.New(s.T()). |
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.
same question
@@ -166,7 +166,7 @@ func (s *UpdateWorkflowSuite) TestUpdateWorkflow_EmptySpeculativeWorkflowTask_Ac | |||
5 WorkflowTaskScheduled // Speculative WT events are not written to the history yet. | |||
6 WorkflowTaskStarted | |||
`, task.History) | |||
return s.UpdateAcceptCompleteCommands(tv, "1"), nil | |||
return s.UpdateAcceptCompleteCommands(tv), nil |
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.
this "1" means that all messages has "1". You removed it, but I don't see any WithUpdateId above. So UpdateID() will be without "1". Is that ok? If so - what was the purpose of "1" suffix before?
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.
Well... this is a primary goal for entire PR. "1" is now encapsulated in tv
.
tests/update_workflow_test.go
Outdated
@@ -1131,7 +1131,7 @@ func (s *UpdateWorkflowSuite) TestUpdateWorkflow_ValidateWorkerMessages() { | |||
updRequest := protoutils.UnmarshalAny[*updatepb.Request](s.T(), reqMsg.GetBody()) | |||
return []*protocolpb.Message{ | |||
{ | |||
Id: tv.MessageID("update-accepted", "1"), | |||
Id: tv.MessageID() + "_1", |
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.
shouldn't it be tv.MessageID()+"_update-accepted_1"
?
func (tv *TestVars) Entity() string { | ||
return getOrCreate(tv, "entity", tv.uniqueString) | ||
} | ||
func (tv *TestVars) WithEntity(entity string) *TestVars { |
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.
usually "With" semantic means you are operating within the same object, rather then creating new object. It is kind of part of the constructor ("chaining" pattern).
Code should look like
tv:=testvars.New(testvars.WithWorkflowId(), testvars.WithWorkflowId(), ...)
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.
those are main comments, rest are oservations.
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.
Using options pattern seems like an overkill here for me. This is a simple test helper which needs to be simple to use. Micro performance and memory allocations optimization is not important here.
As for naming convention, I believe, it is exactly the opposite. With
indicates that you are getting new object "with specific property set to value". Otherwise it would be in go setter format: tv.UpdateID("new_update_id")
.
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.
I'll defer to Yuri here, the general direction LGTM.
6689596
to
4dcfe71
Compare
require.False(t, existed) | ||
require.Equal(t, 1, reg.Len()) | ||
|
||
t.Run("deny new update since it is exceeding the limit", func(t *testing.T) { | ||
_, _, err = reg.FindOrCreate(context.Background(), tv.UpdateID("2")) | ||
_, _, err = reg.FindOrCreate(context.Background(), tv2.UpdateID()) |
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.
with the new approach, it will be more readable as:
_, _, err = reg.FindOrCreate(context.Background(), tv.WithUpdateID("second_update"))
common/testing/testvars/test_vars.go
Outdated
func (tv *TestVars) BuildId(key ...string) string { | ||
//revive:disable-next-line:unchecked-type-assertion | ||
return tv.getOrCreate("build_id", key).(string) | ||
func (tv *TestVars) WithWorkflowIDN(n int) *TestVars { |
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.
What this N means at the end?
Why input parameter is int, and WorkflowID() return parameter is string? Why I can't add arbitrary string?
Why behavior of WithRunID is different from WithWorkflowIDN? (in WithRunID you call cloneSetVal, int WithWorkflowIDN you call cloneSetN)?
Why WithRunID accepting string, but WithWorkflowIDN accepting int?
This is very inconsistent and hard to understand.
I would (pretty strongly) suggest to
- remove N
- make all input parameters for WithXXXX strings instead of int if XXXX return string
- use the same approach for every WithXXX functions (either cloneSetN, or cloneSetVal)
common/testing/testvars/test_vars.go
Outdated
} | ||
|
||
func (tv *TestVars) RunID(key ...string) string { | ||
return tv.getOrCreate("run_id", key, "").(string) | ||
func (tv *TestVars) WithBuildID(buildId string) *TestVars { |
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.
tv.WithBuildID("1").BuildID() != tv.WithBuildIDN(1).BuildID()
I don't think this is predictable and intuitive, I wouldn't expect that.
What is the reason for that? Why we have 2 such function with different behaviour?
84d4e62
to
8dae771
Compare
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.
I assume you decide that those (like me) who want "WithWorkflowID(string)" will add that function themself?
a5a3022
to
302fa66
Compare
Yes, and I added doc on how to do it to |
What changed?
Remove
key
parameter fromtestvar
methods.Why?
It turn out to be confusing for people on how to use optional
key
parameter. @bergundy got a great idea to remove this parameter and use different instance oftv
when more than one entity is needed.How did you test it?
Run all tests. Refactoring affects test code only.
Potential risks
No risks.
Documentation
Yes, updated testing docs.
Is hotfix candidate?
No.