@@ -5,13 +5,15 @@ package iampolicy
5
5
6
6
import (
7
7
"context"
8
+ "log"
8
9
"sync"
9
10
"time"
10
11
11
12
iamModels "github.com/hashicorp/hcp-sdk-go/clients/cloud-iam/stable/2019-12-10/models"
12
13
"github.com/hashicorp/hcp-sdk-go/clients/cloud-resource-manager/stable/2019-12-10/models"
13
14
"github.com/hashicorp/terraform-plugin-framework/diag"
14
15
"github.com/hashicorp/terraform-provider-hcp/internal/clients"
16
+ "github.com/hashicorp/terraform-provider-hcp/internal/customdiags"
15
17
"golang.org/x/exp/maps"
16
18
)
17
19
@@ -184,45 +186,72 @@ func (f *policyFuture) executeModifers(ctx context.Context, u ResourceIamUpdater
184
186
return
185
187
}
186
188
187
- // Get the existing policy
188
- ep , diags := u .GetResourceIamPolicy (ctx )
189
- if diags .HasError () {
190
- f .set (nil , diags )
191
- return
192
- }
189
+ backoff := time .Second
190
+ maxBackoff := 30 * time .Second
193
191
194
- // Determine the principal's we need to lookup their type.
195
- principalSet , diags := f .getPrincipals (ctx , client )
196
- if diags .HasError () {
197
- f .set (nil , diags )
198
- return
199
- }
192
+ for {
193
+ // Get the existing policy
194
+ ep , diags := u .GetResourceIamPolicy (ctx )
195
+ if diags .HasError () {
196
+ f .set (nil , diags )
197
+ return
198
+ }
200
199
201
- // Remove any bindings needed
202
- bindings := ToMap (ep )
203
- for _ , rm := range f .removers {
204
- if members , ok := bindings [rm .RoleID ]; ok {
205
- delete (members , rm .Members [0 ].MemberID )
206
- if len (members ) == 0 {
207
- delete (bindings , rm .RoleID )
208
- }
200
+ // Determine the principal's we need to lookup their type.
201
+ principalSet , diags := f .getPrincipals (ctx , client )
202
+ if diags .HasError () {
203
+ f .set (nil , diags )
204
+ return
209
205
}
210
- }
211
206
212
- // Go through the setters and apply them
213
- for _ , s := range f .setters {
214
- members , ok := bindings [s .RoleID ]
215
- if ! ok {
216
- members = make (map [string ]* models.HashicorpCloudResourcemanagerPolicyBindingMemberType , 4 )
217
- bindings [s .RoleID ] = members
207
+ // Remove any bindings needed
208
+ bindings := ToMap (ep )
209
+ for _ , rm := range f .removers {
210
+ if members , ok := bindings [rm .RoleID ]; ok {
211
+ delete (members , rm .Members [0 ].MemberID )
212
+ if len (members ) == 0 {
213
+ delete (bindings , rm .RoleID )
214
+ }
215
+ }
218
216
}
219
217
220
- members [s .Members [0 ].MemberID ] = principalSet [s .Members [0 ].MemberID ]
221
- }
218
+ // Go through the setters and apply them
219
+ for _ , s := range f .setters {
220
+ members , ok := bindings [s .RoleID ]
221
+ if ! ok {
222
+ members = make (map [string ]* models.HashicorpCloudResourcemanagerPolicyBindingMemberType , 4 )
223
+ bindings [s .RoleID ] = members
224
+ }
222
225
223
- // Apply the policy
224
- f . set ( u . SetResourceIamPolicy ( ctx , FromMap ( ep . Etag , bindings )))
226
+ members [ s . Members [ 0 ]. MemberID ] = principalSet [ s . Members [ 0 ]. MemberID ]
227
+ }
225
228
229
+ // Apply the policy
230
+ ep , diags = u .SetResourceIamPolicy (ctx , FromMap (ep .Etag , bindings ))
231
+ if diags .HasError () {
232
+ if customdiags .HasConflictError (diags ) {
233
+ // Policy object has changed since it was last gotten and the etag is now different.
234
+ // Continuously retry getting and setting the policy with an increasing backoff period until the maximum backoff period is reached.
235
+ if backoff > maxBackoff {
236
+ log .Printf ("[DEBUG]: Maximum backoff time reached. Aborting operation." )
237
+ f .set (nil , diags )
238
+ return
239
+ }
240
+ log .Printf ("[DEBUG]: Operation failed due to conflicts. Operation will be restarted after %s" , backoff )
241
+ // Pause the execution for the duration specified by the current backoff time.
242
+ time .Sleep (backoff )
243
+ // Double the backoff time to increase the delay for the next retry.
244
+ backoff *= 2
245
+ continue
246
+ }
247
+ f .set (nil , diags )
248
+ return
249
+ }
250
+
251
+ // Successfully applied policy
252
+ f .set (ep , nil )
253
+ return
254
+ }
226
255
}
227
256
228
257
// getPrincipals returns a map of principal_id to binding type. The binding type
0 commit comments