From 59ddbcc97d8d9aeb278011a26765c5e9a7b0962b Mon Sep 17 00:00:00 2001 From: lizrabuya <115472349+lizrabuya@users.noreply.github.com> Date: Thu, 20 Jun 2024 11:06:27 +1000 Subject: [PATCH 1/6] create a build resolver based on user id --- internal/build/resolver/current_user.go | 32 ++++ internal/build/resolver/current_user_test.go | 145 ++++++++++++++++++ internal/build/resolver/current_userid.go | 24 +++ .../build/resolver/current_userid_test.go | 99 ++++++++++++ internal/build/resolver/user_builds.go | 82 ++++------ internal/build/resolver/user_builds_test.go | 140 ++--------------- 6 files changed, 346 insertions(+), 176 deletions(-) create mode 100644 internal/build/resolver/current_user.go create mode 100644 internal/build/resolver/current_user_test.go create mode 100644 internal/build/resolver/current_userid.go create mode 100644 internal/build/resolver/current_userid_test.go diff --git a/internal/build/resolver/current_user.go b/internal/build/resolver/current_user.go new file mode 100644 index 00000000..3ab658aa --- /dev/null +++ b/internal/build/resolver/current_user.go @@ -0,0 +1,32 @@ +package resolver + +import ( + "context" + + "github.com/buildkite/cli/v3/internal/build" + pipelineResolver "github.com/buildkite/cli/v3/internal/pipeline/resolver" + "github.com/buildkite/cli/v3/pkg/cmd/factory" + "github.com/buildkite/go-buildkite/v3/buildkite" +) + +// ResolveBuildFromCurrentBranch Finds the most recent build for the branch in the current working directory +func ResolveBuildForCurrentUser(branch string, pipelineResolver pipelineResolver.PipelineResolverFn, f *factory.Factory) BuildResolverFn { + return func(ctx context.Context) (*build.Build, error) { + var user *buildkite.User + + user, _, err := f.RestAPIClient.User.Get() + if err != nil { + return nil, err + } + + opt := &buildkite.BuildsListOptions{ + Creator: *user.Email, + Branch: []string{branch}, + ListOptions: buildkite.ListOptions{ + PerPage: 1, + }, + } + + return ResolveBuildForUser(ctx, *user.Email, opt, pipelineResolver, f) + } +} diff --git a/internal/build/resolver/current_user_test.go b/internal/build/resolver/current_user_test.go new file mode 100644 index 00000000..7dccbd45 --- /dev/null +++ b/internal/build/resolver/current_user_test.go @@ -0,0 +1,145 @@ +package resolver_test + +import ( + "bytes" + "context" + "io" + "net/http" + "os" + "strings" + "testing" + + "github.com/buildkite/cli/v3/internal/build/resolver" + "github.com/buildkite/cli/v3/internal/config" + "github.com/buildkite/cli/v3/internal/pipeline" + "github.com/buildkite/cli/v3/pkg/cmd/factory" + "github.com/buildkite/go-buildkite/v3/buildkite" + "github.com/spf13/afero" +) + +func TestResolveBuildForCurrentUser(t *testing.T) { + t.Parallel() + + pipelineResolver := func(context.Context) (*pipeline.Pipeline, error) { + return &pipeline.Pipeline{ + Name: "testing", + Org: "test org", + }, nil + } + + t.Run("Errors if user cannot be found", func(t *testing.T) { + t.Parallel() + + // mock a failed repsonse + transport := roundTripperFunc(func(r *http.Request) (*http.Response, error) { + return &http.Response{ + StatusCode: http.StatusNotFound, + }, nil + }) + client := &http.Client{Transport: transport} + f := &factory.Factory{ + RestAPIClient: buildkite.NewClient(client), + } + + r := resolver.ResolveBuildForCurrentUser("main", pipelineResolver, f) + _, err := r(context.Background()) + + if err == nil { + t.Fatal("Resolver should return error if user not found") + } + }) + + t.Run("Returns first build found", func(t *testing.T) { + t.Parallel() + + in, _ := os.ReadFile("../../../fixtures/build.json") + callIndex := 0 + responses := []http.Response{ + { + StatusCode: http.StatusOK, + Body: io.NopCloser(strings.NewReader(`{ + "id": "abc123-4567-8910-...", + "graphql_id": "VXNlci0tLWU1N2ZiYTBmLWFiMTQtNGNjMC1iYjViLTY5NTc3NGZmYmZiZQ==", + "name": "John Smith", + "email": "john.smith@example.com", + "avatar_url": "https://www.gravatar.com/avatar/abc123...", + "created_at": "2012-03-04T06:07:08.910Z" + } + `)), + }, + { + StatusCode: http.StatusOK, + Body: io.NopCloser(bytes.NewReader(in)), + }, + } + // mock a failed repsonse + transport := roundTripperFunc(func(r *http.Request) (*http.Response, error) { + resp := responses[callIndex] + callIndex++ + return &resp, nil + }) + client := &http.Client{Transport: transport} + fs := afero.NewMemMapFs() + f := &factory.Factory{ + RestAPIClient: buildkite.NewClient(client), + Config: config.New(fs, nil), + } + + r := resolver.ResolveBuildForCurrentUser("main", pipelineResolver, f) + build, err := r(context.Background()) + if err != nil { + t.Fatal(err) + } + + if build.BuildNumber != 584 { + t.Fatalf("Expected build 584, got %d", build.BuildNumber) + } + }) + + t.Run("Errors if no matching builds found", func(t *testing.T) { + t.Parallel() + + callIndex := 0 + responses := []http.Response{ + { + StatusCode: http.StatusOK, + Body: io.NopCloser(strings.NewReader(`{ + "id": "abc123-4567-8910-...", + "graphql_id": "VXNlci0tLWU1N2ZiYTBmLWFiMTQtNGNjMC1iYjViLTY5NTc3NGZmYmZiZQ==", + "name": "John Smith", + "email": "john.smith@example.com", + "avatar_url": "https://www.gravatar.com/avatar/abc123...", + "created_at": "2012-03-04T06:07:08.910Z" + } + `)), + }, + { + StatusCode: http.StatusOK, + Body: io.NopCloser(strings.NewReader("[]")), + }, + } + // mock a failed repsonse + transport := roundTripperFunc(func(r *http.Request) (*http.Response, error) { + resp := responses[callIndex] + callIndex++ + return &resp, nil + }) + client := &http.Client{Transport: transport} + fs := afero.NewMemMapFs() + f := &factory.Factory{ + RestAPIClient: buildkite.NewClient(client), + Config: config.New(fs, nil), + } + + r := resolver.ResolveBuildForCurrentUser("main", pipelineResolver, f) + build, err := r(context.Background()) + + if err == nil { + t.Fatal("Should return an error when no build is found") + } + + if build != nil { + t.Fatal("Expected no build to be found") + } + }) +} diff --git a/internal/build/resolver/current_userid.go b/internal/build/resolver/current_userid.go new file mode 100644 index 00000000..ef70f7b1 --- /dev/null +++ b/internal/build/resolver/current_userid.go @@ -0,0 +1,24 @@ +package resolver + +import ( + "context" + + "github.com/buildkite/cli/v3/internal/build" + pipelineResolver "github.com/buildkite/cli/v3/internal/pipeline/resolver" + "github.com/buildkite/cli/v3/pkg/cmd/factory" + "github.com/buildkite/go-buildkite/v3/buildkite" +) + +func ResolveBuildFromCurrentUserId(uuid string, pipelineResolver pipelineResolver.PipelineResolverFn, f *factory.Factory) BuildResolverFn { + return func(ctx context.Context) (*build.Build, error) { + + opt := &buildkite.BuildsListOptions{ + Creator: uuid, + ListOptions: buildkite.ListOptions{ + PerPage: 1, + }, + } + + return ResolveBuildForUser(ctx, uuid, opt, pipelineResolver, f) + } +} diff --git a/internal/build/resolver/current_userid_test.go b/internal/build/resolver/current_userid_test.go new file mode 100644 index 00000000..1def02db --- /dev/null +++ b/internal/build/resolver/current_userid_test.go @@ -0,0 +1,99 @@ +package resolver_test + +import ( + "bytes" + "context" + "io" + "net/http" + "os" + "strings" + "testing" + + "github.com/buildkite/cli/v3/internal/build/resolver" + "github.com/buildkite/cli/v3/internal/config" + "github.com/buildkite/cli/v3/internal/pipeline" + "github.com/buildkite/cli/v3/pkg/cmd/factory" + "github.com/buildkite/go-buildkite/v3/buildkite" + "github.com/spf13/afero" +) + +func TestResolveBuildFromUserUUID(t *testing.T) { + t.Parallel() + + pipelineResolver := func(context.Context) (*pipeline.Pipeline, error) { + return &pipeline.Pipeline{ + Name: "testing", + Org: "test org", + }, nil + } + + t.Run("Errors when user id is not a member of the organization", func(t *testing.T) { + t.Parallel() + // mock a failed repsonse + transport := roundTripperFunc(func(r *http.Request) (*http.Response, error) { + return &http.Response{ + StatusCode: http.StatusNotFound, + }, nil + }) + client := &http.Client{Transport: transport} + fs := afero.NewMemMapFs() + f := &factory.Factory{ + RestAPIClient: buildkite.NewClient(client), + Config: config.New(fs, nil), + } + + r := resolver.ResolveBuildFromCurrentUserId("1234", pipelineResolver, f) + _, err := r(context.Background()) + + if err == nil { + t.Fatal("Resolver should return error if user not found") + } + }) + + t.Run("Returns first build found", func(t *testing.T) { + t.Parallel() + + in, _ := os.ReadFile("../../../fixtures/build.json") + callIndex := 0 + responses := []http.Response{ + { + StatusCode: http.StatusOK, + Body: io.NopCloser(strings.NewReader(`[{ + "id": "abc123-4567-8910-...", + "number": 584, + "creator": { + "id": "0183c4e6-c88c-xxxx-b15e-7801077a9181", + "graphql_id": "VXNlci0tLTAxODNjNGU2LWM4OGxxxxxxxxxiMTVlLTc4MDEwNzdhOTE4MQ==" + } + }]`)), + }, + { + StatusCode: http.StatusOK, + Body: io.NopCloser(bytes.NewReader(in)), + }, + } + // mock a failed repsonse + transport := roundTripperFunc(func(r *http.Request) (*http.Response, error) { + resp := responses[callIndex] + callIndex++ + return &resp, nil + }) + + client := &http.Client{Transport: transport} + fs := afero.NewMemMapFs() + f := &factory.Factory{ + RestAPIClient: buildkite.NewClient(client), + Config: config.New(fs, nil), + } + + r := resolver.ResolveBuildFromCurrentUserId("0183c4e6-c88c-xxxx-b15e-7801077a9181", pipelineResolver, f) + build, err := r(context.Background()) + if err != nil { + t.Fatal(err) + } + + if build.BuildNumber != 584 { + t.Fatalf("Expected build 584, got %d", build.BuildNumber) + } + }) +} diff --git a/internal/build/resolver/user_builds.go b/internal/build/resolver/user_builds.go index 16924852..ec8ee70c 100644 --- a/internal/build/resolver/user_builds.go +++ b/internal/build/resolver/user_builds.go @@ -12,57 +12,43 @@ import ( "golang.org/x/sync/errgroup" ) -// ResolveBuildFromCurrentBranch Finds the most recent build for the branch in the current working directory -func ResolveBuildForCurrentUser(branch string, pipelineResolver pipelineResolver.PipelineResolverFn, f *factory.Factory) BuildResolverFn { - return func(ctx context.Context) (*build.Build, error) { - var pipeline *pipeline.Pipeline - var user *buildkite.User +// ResolveBuildForUser Finds the most recent build for the user based on the provided options +func ResolveBuildForUser(ctx context.Context, userInfo string, opt *buildkite.BuildsListOptions, pipelineResolver pipelineResolver.PipelineResolverFn, f *factory.Factory) (*build.Build, error) { - // use an errgroup so a few API calls can be done in parallel - // and then we check for any errors that occurred - g, _ := errgroup.WithContext(ctx) - g.Go(func() error { - p, e := pipelineResolver(ctx) - if p != nil { - pipeline = p - } - return e - }) - g.Go(func() error { - u, _, e := f.RestAPIClient.User.Get() - if u != nil { - user = u - } - return e - }) - err := g.Wait() - if err != nil { - return nil, err - } - if pipeline == nil { - return nil, fmt.Errorf("failed to resolve a pipeline to query builds on.") - } + var pipeline *pipeline.Pipeline - builds, _, err := f.RestAPIClient.Builds.ListByPipeline(f.Config.OrganizationSlug(), pipeline.Name, &buildkite.BuildsListOptions{ - Creator: *user.Email, - Branch: []string{branch}, - ListOptions: buildkite.ListOptions{ - PerPage: 1, - }, - }) - if err != nil { - return nil, err - } - if len(builds) == 0 { - // we error here because this resolver is explicitly used so any case where it doesn't resolve a build is a - // problem - return nil, fmt.Errorf("failed to find a build for current user (email: %s)", *user.Email) + // use an errgroup so a few API calls can be done in parallel + // and then we check for any errors that occurred + g, _ := errgroup.WithContext(ctx) + g.Go(func() error { + p, e := pipelineResolver(ctx) + if p != nil { + pipeline = p } + return e + }) + err := g.Wait() + if err != nil { + return nil, err + } + if pipeline == nil { + return nil, fmt.Errorf("failed to resolve a pipeline to query builds on") + } + + builds, _, err := f.RestAPIClient.Builds.ListByPipeline(f.Config.OrganizationSlug(), pipeline.Name, opt) - return &build.Build{ - Organization: f.Config.OrganizationSlug(), - Pipeline: pipeline.Name, - BuildNumber: *builds[0].Number, - }, nil + if err != nil { + return nil, err } + if len(builds) == 0 { + // we error here because this resolver is explicitly used so any case where it doesn't resolve a build is a + // problem + return nil, fmt.Errorf("failed to find a build for current user (%s)", userInfo) + } + + return &build.Build{ + Organization: f.Config.OrganizationSlug(), + Pipeline: pipeline.Name, + BuildNumber: *builds[0].Number, + }, nil } diff --git a/internal/build/resolver/user_builds_test.go b/internal/build/resolver/user_builds_test.go index 9d58e5f7..9a95dcf2 100644 --- a/internal/build/resolver/user_builds_test.go +++ b/internal/build/resolver/user_builds_test.go @@ -1,24 +1,20 @@ package resolver_test import ( - "bytes" "context" "errors" "io" "net/http" - "os" "strings" "testing" "github.com/buildkite/cli/v3/internal/build/resolver" - "github.com/buildkite/cli/v3/internal/config" "github.com/buildkite/cli/v3/internal/pipeline" "github.com/buildkite/cli/v3/pkg/cmd/factory" "github.com/buildkite/go-buildkite/v3/buildkite" - "github.com/spf13/afero" ) -func TestResolveBuildForCurrentUser(t *testing.T) { +func TestResolveBuildFromUserId(t *testing.T) { t.Parallel() nilPipelineResolver := func(context.Context) (*pipeline.Pipeline, error) { @@ -27,12 +23,6 @@ func TestResolveBuildForCurrentUser(t *testing.T) { errorPipelineResolver := func(context.Context) (*pipeline.Pipeline, error) { return nil, errors.New("") } - pipelineResolver := func(context.Context) (*pipeline.Pipeline, error) { - return &pipeline.Pipeline{ - Name: "testing", - Org: "test org", - }, nil - } transport := roundTripperFunc(func(r *http.Request) (*http.Response, error) { return &http.Response{ @@ -48,138 +38,32 @@ func TestResolveBuildForCurrentUser(t *testing.T) { t.Run("Errors if pipeline cannot be resolved", func(t *testing.T) { t.Parallel() - r := resolver.ResolveBuildForCurrentUser("main", nilPipelineResolver, f) - _, err := r(context.Background()) - - if err == nil { - t.Fatal("Resolver should return error if no pipeline resolved") + opt := &buildkite.BuildsListOptions{ + ListOptions: buildkite.ListOptions{ + PerPage: 1, + }, } - }) - - t.Run("Errors if pipeline resolver errors", func(t *testing.T) { - t.Parallel() - r := resolver.ResolveBuildForCurrentUser("main", errorPipelineResolver, f) - _, err := r(context.Background()) + _, err := resolver.ResolveBuildForUser(context.Background(), "", opt, nilPipelineResolver, f) if err == nil { t.Fatal("Resolver should return error if no pipeline resolved") } }) - t.Run("Errors if user cannot be found", func(t *testing.T) { - t.Parallel() - - // mock a failed repsonse - transport := roundTripperFunc(func(r *http.Request) (*http.Response, error) { - return &http.Response{ - StatusCode: http.StatusNotFound, - }, nil - }) - client := &http.Client{Transport: transport} - f := &factory.Factory{ - RestAPIClient: buildkite.NewClient(client), - } - - r := resolver.ResolveBuildForCurrentUser("main", pipelineResolver, f) - _, err := r(context.Background()) - - if err == nil { - t.Fatal("Resolver should return error if user not found") - } - }) - - t.Run("Returns first build found", func(t *testing.T) { - t.Parallel() - - in, _ := os.ReadFile("../../../fixtures/build.json") - callIndex := 0 - responses := []http.Response{ - { - StatusCode: http.StatusOK, - Body: io.NopCloser(strings.NewReader(`{ - "id": "abc123-4567-8910-...", - "graphql_id": "VXNlci0tLWU1N2ZiYTBmLWFiMTQtNGNjMC1iYjViLTY5NTc3NGZmYmZiZQ==", - "name": "John Smith", - "email": "john.smith@example.com", - "avatar_url": "https://www.gravatar.com/avatar/abc123...", - "created_at": "2012-03-04T06:07:08.910Z" - } - `)), - }, - { - StatusCode: http.StatusOK, - Body: io.NopCloser(bytes.NewReader(in)), - }, - } - // mock a failed repsonse - transport := roundTripperFunc(func(r *http.Request) (*http.Response, error) { - resp := responses[callIndex] - callIndex++ - return &resp, nil - }) - client := &http.Client{Transport: transport} - fs := afero.NewMemMapFs() - f := &factory.Factory{ - RestAPIClient: buildkite.NewClient(client), - Config: config.New(fs, nil), - } - - r := resolver.ResolveBuildForCurrentUser("main", pipelineResolver, f) - build, err := r(context.Background()) - if err != nil { - t.Fatal(err) - } - - if build.BuildNumber != 584 { - t.Fatalf("Expected build 584, got %d", build.BuildNumber) - } - }) - - t.Run("Errors if no matching builds found", func(t *testing.T) { + t.Run("Errors if pipeline resolver errors", func(t *testing.T) { t.Parallel() - callIndex := 0 - responses := []http.Response{ - { - StatusCode: http.StatusOK, - Body: io.NopCloser(strings.NewReader(`{ - "id": "abc123-4567-8910-...", - "graphql_id": "VXNlci0tLWU1N2ZiYTBmLWFiMTQtNGNjMC1iYjViLTY5NTc3NGZmYmZiZQ==", - "name": "John Smith", - "email": "john.smith@example.com", - "avatar_url": "https://www.gravatar.com/avatar/abc123...", - "created_at": "2012-03-04T06:07:08.910Z" - } - `)), - }, - { - StatusCode: http.StatusOK, - Body: io.NopCloser(strings.NewReader("[]")), + opt := &buildkite.BuildsListOptions{ + ListOptions: buildkite.ListOptions{ + PerPage: 1, }, } - // mock a failed repsonse - transport := roundTripperFunc(func(r *http.Request) (*http.Response, error) { - resp := responses[callIndex] - callIndex++ - return &resp, nil - }) - client := &http.Client{Transport: transport} - fs := afero.NewMemMapFs() - f := &factory.Factory{ - RestAPIClient: buildkite.NewClient(client), - Config: config.New(fs, nil), - } - r := resolver.ResolveBuildForCurrentUser("main", pipelineResolver, f) - build, err := r(context.Background()) + _, err := resolver.ResolveBuildForUser(context.Background(), "", opt, errorPipelineResolver, f) if err == nil { - t.Fatal("Should return an error when no build is found") - } - - if build != nil { - t.Fatal("Expected no build to be found") + t.Fatal("Resolver should return error if no pipeline resolved") } }) } From 3d4c46a2f7b36b6748968f99cf74a206ff6e8614 Mon Sep 17 00:00:00 2001 From: lizrabuya <115472349+lizrabuya@users.noreply.github.com> Date: Thu, 20 Jun 2024 13:20:07 +1000 Subject: [PATCH 2/6] various changes based on PR feedback --- internal/build/resolver/current_user.go | 1 - internal/build/resolver/current_userid.go | 12 ++----- .../build/resolver/current_userid_test.go | 4 +-- internal/build/resolver/user_builds.go | 31 +++++++++---------- 4 files changed, 18 insertions(+), 30 deletions(-) diff --git a/internal/build/resolver/current_user.go b/internal/build/resolver/current_user.go index 3ab658aa..ccd0c916 100644 --- a/internal/build/resolver/current_user.go +++ b/internal/build/resolver/current_user.go @@ -9,7 +9,6 @@ import ( "github.com/buildkite/go-buildkite/v3/buildkite" ) -// ResolveBuildFromCurrentBranch Finds the most recent build for the branch in the current working directory func ResolveBuildForCurrentUser(branch string, pipelineResolver pipelineResolver.PipelineResolverFn, f *factory.Factory) BuildResolverFn { return func(ctx context.Context) (*build.Build, error) { var user *buildkite.User diff --git a/internal/build/resolver/current_userid.go b/internal/build/resolver/current_userid.go index ef70f7b1..eab85baa 100644 --- a/internal/build/resolver/current_userid.go +++ b/internal/build/resolver/current_userid.go @@ -6,19 +6,11 @@ import ( "github.com/buildkite/cli/v3/internal/build" pipelineResolver "github.com/buildkite/cli/v3/internal/pipeline/resolver" "github.com/buildkite/cli/v3/pkg/cmd/factory" - "github.com/buildkite/go-buildkite/v3/buildkite" ) -func ResolveBuildFromCurrentUserId(uuid string, pipelineResolver pipelineResolver.PipelineResolverFn, f *factory.Factory) BuildResolverFn { +func ResolveBuildByUserId(uuid string, pipelineResolver pipelineResolver.PipelineResolverFn, f *factory.Factory) BuildResolverFn { return func(ctx context.Context) (*build.Build, error) { - opt := &buildkite.BuildsListOptions{ - Creator: uuid, - ListOptions: buildkite.ListOptions{ - PerPage: 1, - }, - } - - return ResolveBuildForUser(ctx, uuid, opt, pipelineResolver, f) + return ResolveBuildForUser(ctx, uuid, "", pipelineResolver, f) } } diff --git a/internal/build/resolver/current_userid_test.go b/internal/build/resolver/current_userid_test.go index 1def02db..81299c66 100644 --- a/internal/build/resolver/current_userid_test.go +++ b/internal/build/resolver/current_userid_test.go @@ -42,7 +42,7 @@ func TestResolveBuildFromUserUUID(t *testing.T) { Config: config.New(fs, nil), } - r := resolver.ResolveBuildFromCurrentUserId("1234", pipelineResolver, f) + r := resolver.ResolveBuildByUserId("1234", pipelineResolver, f) _, err := r(context.Background()) if err == nil { @@ -86,7 +86,7 @@ func TestResolveBuildFromUserUUID(t *testing.T) { Config: config.New(fs, nil), } - r := resolver.ResolveBuildFromCurrentUserId("0183c4e6-c88c-xxxx-b15e-7801077a9181", pipelineResolver, f) + r := resolver.ResolveBuildByUserId("0183c4e6-c88c-xxxx-b15e-7801077a9181", pipelineResolver, f) build, err := r(context.Background()) if err != nil { t.Fatal(err) diff --git a/internal/build/resolver/user_builds.go b/internal/build/resolver/user_builds.go index ec8ee70c..aa5964ae 100644 --- a/internal/build/resolver/user_builds.go +++ b/internal/build/resolver/user_builds.go @@ -5,29 +5,15 @@ import ( "fmt" "github.com/buildkite/cli/v3/internal/build" - "github.com/buildkite/cli/v3/internal/pipeline" pipelineResolver "github.com/buildkite/cli/v3/internal/pipeline/resolver" "github.com/buildkite/cli/v3/pkg/cmd/factory" "github.com/buildkite/go-buildkite/v3/buildkite" - "golang.org/x/sync/errgroup" ) // ResolveBuildForUser Finds the most recent build for the user based on the provided options -func ResolveBuildForUser(ctx context.Context, userInfo string, opt *buildkite.BuildsListOptions, pipelineResolver pipelineResolver.PipelineResolverFn, f *factory.Factory) (*build.Build, error) { - - var pipeline *pipeline.Pipeline - - // use an errgroup so a few API calls can be done in parallel - // and then we check for any errors that occurred - g, _ := errgroup.WithContext(ctx) - g.Go(func() error { - p, e := pipelineResolver(ctx) - if p != nil { - pipeline = p - } - return e - }) - err := g.Wait() +func ResolveBuildForUser(ctx context.Context, userInfo string, branch string, pipelineResolver pipelineResolver.PipelineResolverFn, f *factory.Factory) (*build.Build, error) { + + pipeline, err := pipelineResolver(ctx) if err != nil { return nil, err } @@ -35,6 +21,17 @@ func ResolveBuildForUser(ctx context.Context, userInfo string, opt *buildkite.Bu return nil, fmt.Errorf("failed to resolve a pipeline to query builds on") } + opt := &buildkite.BuildsListOptions{ + Creator: userInfo, + ListOptions: buildkite.ListOptions{ + PerPage: 1, + }, + } + + if len(branch) > 0 { + opt.Branch = []string{branch} + } + builds, _, err := f.RestAPIClient.Builds.ListByPipeline(f.Config.OrganizationSlug(), pipeline.Name, opt) if err != nil { From b2fdab4f88a6a37a03ef1341bd2ab90c0df2c8c2 Mon Sep 17 00:00:00 2001 From: lizrabuya <115472349+lizrabuya@users.noreply.github.com> Date: Thu, 20 Jun 2024 13:27:15 +1000 Subject: [PATCH 3/6] fix fn call to pass branch name instead of build options --- internal/build/resolver/current_user.go | 10 +--------- internal/build/resolver/user_builds_test.go | 16 ++-------------- 2 files changed, 3 insertions(+), 23 deletions(-) diff --git a/internal/build/resolver/current_user.go b/internal/build/resolver/current_user.go index ccd0c916..909213a1 100644 --- a/internal/build/resolver/current_user.go +++ b/internal/build/resolver/current_user.go @@ -18,14 +18,6 @@ func ResolveBuildForCurrentUser(branch string, pipelineResolver pipelineResolver return nil, err } - opt := &buildkite.BuildsListOptions{ - Creator: *user.Email, - Branch: []string{branch}, - ListOptions: buildkite.ListOptions{ - PerPage: 1, - }, - } - - return ResolveBuildForUser(ctx, *user.Email, opt, pipelineResolver, f) + return ResolveBuildForUser(ctx, *user.Email, branch, pipelineResolver, f) } } diff --git a/internal/build/resolver/user_builds_test.go b/internal/build/resolver/user_builds_test.go index 9a95dcf2..96e60cdc 100644 --- a/internal/build/resolver/user_builds_test.go +++ b/internal/build/resolver/user_builds_test.go @@ -38,13 +38,7 @@ func TestResolveBuildFromUserId(t *testing.T) { t.Run("Errors if pipeline cannot be resolved", func(t *testing.T) { t.Parallel() - opt := &buildkite.BuildsListOptions{ - ListOptions: buildkite.ListOptions{ - PerPage: 1, - }, - } - - _, err := resolver.ResolveBuildForUser(context.Background(), "", opt, nilPipelineResolver, f) + _, err := resolver.ResolveBuildForUser(context.Background(), "", "", nilPipelineResolver, f) if err == nil { t.Fatal("Resolver should return error if no pipeline resolved") @@ -54,13 +48,7 @@ func TestResolveBuildFromUserId(t *testing.T) { t.Run("Errors if pipeline resolver errors", func(t *testing.T) { t.Parallel() - opt := &buildkite.BuildsListOptions{ - ListOptions: buildkite.ListOptions{ - PerPage: 1, - }, - } - - _, err := resolver.ResolveBuildForUser(context.Background(), "", opt, errorPipelineResolver, f) + _, err := resolver.ResolveBuildForUser(context.Background(), "", "", errorPipelineResolver, f) if err == nil { t.Fatal("Resolver should return error if no pipeline resolved") From 0877ff759939a7b703c9ab504ffc338b24bed47d Mon Sep 17 00:00:00 2001 From: lizrabuya <115472349+lizrabuya@users.noreply.github.com> Date: Thu, 20 Jun 2024 13:31:46 +1000 Subject: [PATCH 4/6] rename resolver filename for userid --- internal/build/resolver/{current_userid.go => userid.go} | 0 .../build/resolver/{current_userid_test.go => userid_test.go} | 0 2 files changed, 0 insertions(+), 0 deletions(-) rename internal/build/resolver/{current_userid.go => userid.go} (100%) rename internal/build/resolver/{current_userid_test.go => userid_test.go} (100%) diff --git a/internal/build/resolver/current_userid.go b/internal/build/resolver/userid.go similarity index 100% rename from internal/build/resolver/current_userid.go rename to internal/build/resolver/userid.go diff --git a/internal/build/resolver/current_userid_test.go b/internal/build/resolver/userid_test.go similarity index 100% rename from internal/build/resolver/current_userid_test.go rename to internal/build/resolver/userid_test.go From e53b228b1871e5ebf7a8a62f0bc000b5e556768d Mon Sep 17 00:00:00 2001 From: lizrabuya <115472349+lizrabuya@users.noreply.github.com> Date: Thu, 20 Jun 2024 14:13:10 +1000 Subject: [PATCH 5/6] rename fn name for resolver for user id --- internal/build/resolver/current_user.go | 1 + internal/build/resolver/user_builds.go | 2 +- internal/build/resolver/userid.go | 3 ++- 3 files changed, 4 insertions(+), 2 deletions(-) diff --git a/internal/build/resolver/current_user.go b/internal/build/resolver/current_user.go index 909213a1..aa45a79a 100644 --- a/internal/build/resolver/current_user.go +++ b/internal/build/resolver/current_user.go @@ -9,6 +9,7 @@ import ( "github.com/buildkite/go-buildkite/v3/buildkite" ) +// ResolveBuildForCurrentUser Finds the most recent build for the current user and branch func ResolveBuildForCurrentUser(branch string, pipelineResolver pipelineResolver.PipelineResolverFn, f *factory.Factory) BuildResolverFn { return func(ctx context.Context) (*build.Build, error) { var user *buildkite.User diff --git a/internal/build/resolver/user_builds.go b/internal/build/resolver/user_builds.go index aa5964ae..7f715d27 100644 --- a/internal/build/resolver/user_builds.go +++ b/internal/build/resolver/user_builds.go @@ -10,7 +10,7 @@ import ( "github.com/buildkite/go-buildkite/v3/buildkite" ) -// ResolveBuildForUser Finds the most recent build for the user based on the provided options +// ResolveBuildForUser Finds the most recent build for the user and branch func ResolveBuildForUser(ctx context.Context, userInfo string, branch string, pipelineResolver pipelineResolver.PipelineResolverFn, f *factory.Factory) (*build.Build, error) { pipeline, err := pipelineResolver(ctx) diff --git a/internal/build/resolver/userid.go b/internal/build/resolver/userid.go index eab85baa..8d10bf5a 100644 --- a/internal/build/resolver/userid.go +++ b/internal/build/resolver/userid.go @@ -8,7 +8,8 @@ import ( "github.com/buildkite/cli/v3/pkg/cmd/factory" ) -func ResolveBuildByUserId(uuid string, pipelineResolver pipelineResolver.PipelineResolverFn, f *factory.Factory) BuildResolverFn { +// ResolveBuildForUserID Finds the most recent build for the user based on the user's UUID +func ResolveBuildForUserID(uuid string, pipelineResolver pipelineResolver.PipelineResolverFn, f *factory.Factory) BuildResolverFn { return func(ctx context.Context) (*build.Build, error) { return ResolveBuildForUser(ctx, uuid, "", pipelineResolver, f) From 67c3073e29b10486531c7a712858b3229c30f4b4 Mon Sep 17 00:00:00 2001 From: lizrabuya <115472349+lizrabuya@users.noreply.github.com> Date: Thu, 20 Jun 2024 14:24:29 +1000 Subject: [PATCH 6/6] rename fn name for resolver for user id --- internal/build/resolver/userid_test.go | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/internal/build/resolver/userid_test.go b/internal/build/resolver/userid_test.go index 81299c66..0522e067 100644 --- a/internal/build/resolver/userid_test.go +++ b/internal/build/resolver/userid_test.go @@ -42,7 +42,7 @@ func TestResolveBuildFromUserUUID(t *testing.T) { Config: config.New(fs, nil), } - r := resolver.ResolveBuildByUserId("1234", pipelineResolver, f) + r := resolver.ResolveBuildForUserID("1234", pipelineResolver, f) _, err := r(context.Background()) if err == nil { @@ -86,7 +86,7 @@ func TestResolveBuildFromUserUUID(t *testing.T) { Config: config.New(fs, nil), } - r := resolver.ResolveBuildByUserId("0183c4e6-c88c-xxxx-b15e-7801077a9181", pipelineResolver, f) + r := resolver.ResolveBuildForUserID("0183c4e6-c88c-xxxx-b15e-7801077a9181", pipelineResolver, f) build, err := r(context.Background()) if err != nil { t.Fatal(err)