Skip to content

Commit

Permalink
prevent concurrent builds of same artifact
Browse files Browse the repository at this point in the history
Signed-off-by: Pablo Chacin <[email protected]>
  • Loading branch information
pablochacin committed Sep 19, 2024
1 parent 99c7bae commit 6e7173f
Show file tree
Hide file tree
Showing 2 changed files with 84 additions and 0 deletions.
20 changes: 20 additions & 0 deletions pkg/local/local.go
Original file line number Diff line number Diff line change
Expand Up @@ -9,6 +9,7 @@ import (
"fmt"
"os"
"sort"
"sync"

"github.com/grafana/k6build"
"github.com/grafana/k6build/pkg/cache"
Expand Down Expand Up @@ -44,6 +45,7 @@ type localBuildSrv struct {
catalog k6catalog.Catalog
builder k6foundry.Builder
cache cache.Cache
mutexes sync.Map
}

// NewBuildService creates a local build service using the given configuration
Expand Down Expand Up @@ -158,6 +160,9 @@ func (b *localBuildSrv) Build(
}
id := fmt.Sprintf("%x", sha1.Sum(hashData.Bytes())) //nolint:gosec

unlock := b.lockArtifact(id)
defer unlock()

artifactObject, err := b.cache.Get(ctx, id)
if err == nil {
return k6build.Artifact{
Expand Down Expand Up @@ -192,3 +197,18 @@ func (b *localBuildSrv) Build(
Platform: platform,
}, nil
}

// lockArtifact obtains a mutex used to prevent concurrent builds of the same artifact and
// returns a function that will unlock the mutex associated to the given id in the cache.
// The lock is also removed from the map. Subsequent calls will get another lock on the same
// id but this is safe as the object should already be in the cache and no further builds are needed.
func (b *localBuildSrv) lockArtifact(id string) func() {
value, _ := b.mutexes.LoadOrStore(id, &sync.Mutex{})
mtx, _ := value.(*sync.Mutex)
mtx.Lock()

return func() {
b.mutexes.Delete(id)
mtx.Unlock()
}
}
64 changes: 64 additions & 0 deletions pkg/local/local_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -3,6 +3,7 @@ package local
import (
"context"
"errors"
"sync"
"testing"

"github.com/grafana/k6build"
Expand Down Expand Up @@ -242,3 +243,66 @@ func TestIdempotentBuild(t *testing.T) {
}
})
}

// TestConcurrentBuilds tests that is sage to build the same artifact concurrently and that
// concurrent builds of different artifacts are not affected.
// The test uses a local test setup backed by a file cache.
// Attempting to write the same artifact twice will return an error.
func TestConcurrentBuilds(t *testing.T) {
t.Parallel()
buildsrv, err := SetupTestLocalBuildService(t)
if err != nil {
t.Fatalf("test setup %v", err)
}

builds := []struct {
k6Ver string
deps []k6build.Dependency
}{
{
k6Ver: "v0.1.0",
deps: []k6build.Dependency{
{Name: "k6/x/ext", Constraints: "v0.1.0"},
},
},
{
k6Ver: "v0.1.0",
deps: []k6build.Dependency{
{Name: "k6/x/ext", Constraints: "v0.1.0"},
},
},
{
k6Ver: "v0.2.0",
deps: []k6build.Dependency{
{Name: "k6/x/ext", Constraints: "v0.1.0"},
},
},
}

errch := make(chan error, len(builds))

wg := sync.WaitGroup{}
for _, b := range builds {
wg.Add(1)
go func() {
defer wg.Done()

if _, err := buildsrv.Build(
context.TODO(),
"linux/amd64",
b.k6Ver,
b.deps,
); err != nil {
errch <- err
}
}()
}

wg.Wait()

select {
case err := <-errch:
t.Fatalf("unexpected %v", err)
default:
}
}

0 comments on commit 6e7173f

Please sign in to comment.