mirror of
https://github.com/opentofu/opentofu.git
synced 2025-01-18 04:32:59 -06:00
6b6b5a43c3
Because `aws_security_group_rule` resources are an abstraction on top of Security Groups, they must interact with the AWS Security Group APIs in a pattern that often results in lots of parallel requests interacting with the same security group. We've found that this pattern can trigger race conditions resulting in inconsistent behavior, including: * Rules that report as created but don't actually exist on AWS's side * Rules that show up in AWS but don't register as being created locally, resulting in follow up attempts to authorize the rule failing w/ Duplicate errors Here, we introduce a per-SG mutex that must be held by any security group before it is allowed to interact with AWS APIs. This protects the space between `DescribeSecurityGroup` and `Authorize*` / `Revoke*` calls, ensuring that no other rules interact with the SG during that span. The included test exposes the race by applying a security group with lots of rules, which based on the dependency graph can all be handled in parallel. This fails most of the time without the new locking behavior. I've omitted the mutex from `Read`, since it is only called during the Refresh walk when no changes are being made, meaning a bunch of parallel `DescribeSecurityGroup` API calls should be consistent in that case.
68 lines
1.0 KiB
Go
68 lines
1.0 KiB
Go
package mutexkv
|
|
|
|
import (
|
|
"testing"
|
|
"time"
|
|
)
|
|
|
|
func TestMutexKVLock(t *testing.T) {
|
|
mkv := NewMutexKV()
|
|
|
|
mkv.Lock("foo")
|
|
|
|
doneCh := make(chan struct{})
|
|
|
|
go func() {
|
|
mkv.Lock("foo")
|
|
close(doneCh)
|
|
}()
|
|
|
|
select {
|
|
case <-doneCh:
|
|
t.Fatal("Second lock was able to be taken. This shouldn't happen.")
|
|
case <-time.After(50 * time.Millisecond):
|
|
// pass
|
|
}
|
|
}
|
|
|
|
func TestMutexKVUnlock(t *testing.T) {
|
|
mkv := NewMutexKV()
|
|
|
|
mkv.Lock("foo")
|
|
mkv.Unlock("foo")
|
|
|
|
doneCh := make(chan struct{})
|
|
|
|
go func() {
|
|
mkv.Lock("foo")
|
|
close(doneCh)
|
|
}()
|
|
|
|
select {
|
|
case <-doneCh:
|
|
// pass
|
|
case <-time.After(50 * time.Millisecond):
|
|
t.Fatal("Second lock blocked after unlock. This shouldn't happen.")
|
|
}
|
|
}
|
|
|
|
func TestMutexKVDifferentKeys(t *testing.T) {
|
|
mkv := NewMutexKV()
|
|
|
|
mkv.Lock("foo")
|
|
|
|
doneCh := make(chan struct{})
|
|
|
|
go func() {
|
|
mkv.Lock("bar")
|
|
close(doneCh)
|
|
}()
|
|
|
|
select {
|
|
case <-doneCh:
|
|
// pass
|
|
case <-time.After(50 * time.Millisecond):
|
|
t.Fatal("Second lock on a different key blocked. This shouldn't happen.")
|
|
}
|
|
}
|