-
Notifications
You must be signed in to change notification settings - Fork 2.6k
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
fix(ovh): cache refresh and duplicates processing #4932
base: master
Are you sure you want to change the base?
Changes from all commits
95d9c3c
4fc9519
89aa8e1
f674154
4e899ca
b7c7123
ae4ebf5
4b1ee06
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -137,18 +137,32 @@ func (p *OVHProvider) Records(ctx context.Context) ([]*endpoint.Endpoint, error) | |
} | ||
|
||
// ApplyChanges applies a given set of changes in a given zone. | ||
func (p *OVHProvider) ApplyChanges(ctx context.Context, changes *plan.Changes) error { | ||
func (p *OVHProvider) ApplyChanges(ctx context.Context, changes *plan.Changes) (err error) { | ||
zones, records, err := p.zonesRecords(ctx) | ||
zonesChangeUniques := map[string]bool{} | ||
if err != nil { | ||
return err | ||
return provider.NewSoftError(err) | ||
} | ||
|
||
allChanges := make([]ovhChange, 0, countTargets(changes.Create, changes.UpdateNew, changes.UpdateOld, changes.Delete)) | ||
zonesChangeUniques := map[string]bool{} | ||
|
||
// Always refresh zones even in case of errors. | ||
defer func() { | ||
log.Debugf("OVH: %d zones will be refreshed", len(zonesChangeUniques)) | ||
|
||
eg, _ := errgroup.WithContext(ctx) | ||
for zone := range zonesChangeUniques { | ||
zone := zone | ||
eg.Go(func() error { return p.refresh(zone) }) | ||
} | ||
|
||
if e := eg.Wait(); e != nil && err == nil { // return the error only if there is no error during the changes | ||
err = provider.NewSoftError(e) | ||
} | ||
}() | ||
|
||
allChanges := make([]ovhChange, 0, countTargets(changes.Create, changes.UpdateNew, changes.UpdateOld, changes.Delete)) | ||
allChanges = append(allChanges, newOvhChange(ovhCreate, changes.Create, zones, records)...) | ||
allChanges = append(allChanges, newOvhChange(ovhCreate, changes.UpdateNew, zones, records)...) | ||
|
||
allChanges = append(allChanges, newOvhChange(ovhDelete, changes.UpdateOld, zones, records)...) | ||
allChanges = append(allChanges, newOvhChange(ovhDelete, changes.Delete, zones, records)...) | ||
|
||
|
@@ -161,27 +175,24 @@ func (p *OVHProvider) ApplyChanges(ctx context.Context, changes *plan.Changes) e | |
eg.Go(func() error { return p.change(change) }) | ||
} | ||
if err := eg.Wait(); err != nil { | ||
return err | ||
return provider.NewSoftError(err) | ||
} | ||
|
||
log.Infof("OVH: %d zones will be refreshed", len(zonesChangeUniques)) | ||
|
||
eg, _ = errgroup.WithContext(ctx) | ||
for zone := range zonesChangeUniques { | ||
zone := zone | ||
eg.Go(func() error { return p.refresh(zone) }) | ||
} | ||
if err := eg.Wait(); err != nil { | ||
return err | ||
} | ||
return nil | ||
} | ||
|
||
func (p *OVHProvider) refresh(zone string) error { | ||
log.Debugf("OVH: Refresh %s zone", zone) | ||
|
||
// Zone has been altered so we invalidate the cache | ||
// so that the next run will reload it. | ||
p.invalidateCache(zone) | ||
|
||
p.apiRateLimiter.Take() | ||
return p.client.Post(fmt.Sprintf("/domain/zone/%s/refresh", zone), nil, nil) | ||
if err := p.client.Post(fmt.Sprintf("/domain/zone/%s/refresh", zone), nil, nil); err != nil { | ||
return provider.NewSoftError(err) | ||
} | ||
return nil | ||
} | ||
|
||
func (p *OVHProvider) change(change ovhChange) error { | ||
|
@@ -201,11 +212,15 @@ func (p *OVHProvider) change(change ovhChange) error { | |
return nil | ||
} | ||
|
||
func (p *OVHProvider) invalidateCache(zone string) { | ||
p.cacheInstance.Delete(zone + "#soa") | ||
} | ||
|
||
func (p *OVHProvider) zonesRecords(ctx context.Context) ([]string, []ovhRecord, error) { | ||
var allRecords []ovhRecord | ||
zones, err := p.zones() | ||
if err != nil { | ||
return nil, nil, err | ||
return nil, nil, provider.NewSoftError(err) | ||
} | ||
|
||
chRecords := make(chan []ovhRecord, len(zones)) | ||
|
@@ -215,7 +230,7 @@ func (p *OVHProvider) zonesRecords(ctx context.Context) ([]string, []ovhRecord, | |
eg.Go(func() error { return p.records(&ctx, &zone, chRecords) }) | ||
} | ||
if err := eg.Wait(); err != nil { | ||
return nil, nil, err | ||
return nil, nil, provider.NewSoftError(err) | ||
} | ||
close(chRecords) | ||
for records := range chRecords { | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. The line for records := range chRecords {
if records == null {
continue
}
} Or it will never happen? |
||
|
@@ -270,7 +285,7 @@ func (p *OVHProvider) records(ctx *context.Context, zone *string, records chan<- | |
} | ||
} | ||
|
||
p.cacheInstance.Delete(*zone + "#soa") | ||
p.invalidateCache(*zone) | ||
} | ||
} | ||
|
||
|
@@ -359,6 +374,10 @@ func ovhGroupByNameAndType(records []ovhRecord) []*endpoint.Endpoint { | |
} | ||
|
||
func newOvhChange(action int, endpoints []*endpoint.Endpoint, zones []string, records []ovhRecord) []ovhChange { | ||
// Copy the records because we need to mutate the list. | ||
newRecords := make([]ovhRecord, len(records)) | ||
copy(newRecords, records) | ||
|
||
zoneNameIDMapper := provider.ZoneIDName{} | ||
ovhChanges := make([]ovhChange, 0, countTargets(endpoints)) | ||
for _, zone := range zones { | ||
|
@@ -390,9 +409,16 @@ func newOvhChange(action int, endpoints []*endpoint.Endpoint, zones []string, re | |
if e.RecordTTL.IsConfigured() { | ||
change.TTL = int64(e.RecordTTL) | ||
} | ||
for _, record := range records { | ||
if record.Zone == change.Zone && record.SubDomain == change.SubDomain && record.FieldType == change.FieldType && record.Target == change.Target { | ||
change.ID = record.ID | ||
|
||
// The Zone might have multiple records with the same target. In order to avoid applying the action to the | ||
// same OVH record, we remove a record from the list when a match is found. | ||
for i := 0; i < len(newRecords); i++ { | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. This is a third nested for-each loop. Matching and removing records worth to consider optimising to reduce complexity and performance. Something like that, just an idea, not a working code recordMap := make(map[string]ovhRecord, len(records))
for _, rec := range records {
key := fmt.Sprintf("%s|%s|%s|%s", rec.Zone, rec.SubDomain, rec.FieldType, rec.Target)
recordMap[key] = rec
}
for _, target := range e.Targets {
.....
// Use (datastructure) like the map to find and remove the record
key := fmt.Sprintf("%s|%s|%s|%s", change.Zone, change.SubDomain, change.FieldType, change.Target)
if rec, found := recordMap[key]; found {
change.ID = rec.ID
delete(recordMap, key)
}
} |
||
rec := newRecords[i] | ||
if rec.Zone == change.Zone && rec.SubDomain == change.SubDomain && rec.FieldType == change.FieldType && rec.Target == change.Target { | ||
change.ID = rec.ID | ||
// Deleting this record from the list to avoid retargetting it later if a change with a similar target exists. | ||
newRecords = append(newRecords[:i], newRecords[i+1:]...) | ||
break | ||
} | ||
} | ||
ovhChanges = append(ovhChanges, change) | ||
|
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.
Probably worth to add a comment why this
zone := zone
shadowing is required here. Something likeAt first look, I thought is not needed