fix: do not pass around sync.Map (#15250)

it is not safe to pass around sync.Map
through pointers, as it may be concurrently
updated by different callers.

this PR simplifies by avoiding sync.Map
altogether, we do not need sync.Map
to keep object->erasureMap association.

This PR fixes a crash when concurrently
using this value when audit logs are
configured.

```
fatal error: concurrent map iteration and map write

goroutine 247651580 [running]:
runtime.throw({0x277a6c1?, 0xc002381400?})
        runtime/panic.go:992 +0x71 fp=0xc004d29b20 sp=0xc004d29af0 pc=0x438671
runtime.mapiternext(0xc0d6e87f18?)
        runtime/map.go:871 +0x4eb fp=0xc004d29b90 sp=0xc004d29b20 pc=0x41002b
```
This commit is contained in:
Harshavardhana 2022-07-07 17:04:25 -07:00 committed by GitHub
parent ae92521310
commit 32b2f6117e
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
2 changed files with 7 additions and 37 deletions

View File

@ -20,7 +20,6 @@ package cmd
import ( import (
"context" "context"
"encoding/binary" "encoding/binary"
"encoding/json"
"errors" "errors"
"fmt" "fmt"
"hash/crc32" "hash/crc32"
@ -549,29 +548,13 @@ func (s *erasureSets) cleanupStaleUploads(ctx context.Context) {
} }
} }
const objectErasureMapKey = "objectErasureMap"
type auditObjectOp struct { type auditObjectOp struct {
Name string `json:"name"`
Pool int `json:"poolId"` Pool int `json:"poolId"`
Set int `json:"setId"` Set int `json:"setId"`
Disks []string `json:"disks"` Disks []string `json:"disks"`
} }
type auditObjectErasureMap struct {
sync.Map
}
// Define how to marshal auditObjectErasureMap so it can be
// printed in the audit webhook notification request.
func (a *auditObjectErasureMap) MarshalJSON() ([]byte, error) {
mapCopy := make(map[string]auditObjectOp)
a.Range(func(k, v interface{}) bool {
mapCopy[k.(string)] = v.(auditObjectOp)
return true
})
return json.Marshal(mapCopy)
}
// Add erasure set information to the current context // Add erasure set information to the current context
func auditObjectErasureSet(ctx context.Context, object string, set *erasureObjects) { func auditObjectErasureSet(ctx context.Context, object string, set *erasureObjects) {
if len(logger.AuditTargets()) == 0 { if len(logger.AuditTargets()) == 0 {
@ -579,33 +562,20 @@ func auditObjectErasureSet(ctx context.Context, object string, set *erasureObjec
} }
object = decodeDirObject(object) object = decodeDirObject(object)
endpoints := set.getEndpoints()
var disksEndpoints []string disksEndpoints := make([]string, 0, len(endpoints))
for _, endpoint := range set.getEndpoints() { for _, endpoint := range endpoints {
disksEndpoints = append(disksEndpoints, endpoint.String()) disksEndpoints = append(disksEndpoints, endpoint.String())
} }
op := auditObjectOp{ op := auditObjectOp{
Name: object,
Pool: set.poolIndex + 1, Pool: set.poolIndex + 1,
Set: set.setIndex + 1, Set: set.setIndex + 1,
Disks: disksEndpoints, Disks: disksEndpoints,
} }
var objectErasureSetTag *auditObjectErasureMap logger.GetReqInfo(ctx).AppendTags("objectLocation", op)
reqInfo := logger.GetReqInfo(ctx)
for _, kv := range reqInfo.GetTags() {
if kv.Key == objectErasureMapKey {
objectErasureSetTag = kv.Val.(*auditObjectErasureMap)
break
}
}
if objectErasureSetTag == nil {
objectErasureSetTag = &auditObjectErasureMap{}
}
objectErasureSetTag.Store(object, op)
reqInfo.SetTags(objectErasureMapKey, objectErasureSetTag)
} }
// NewNSLock - initialize a new namespace RWLocker instance. // NewNSLock - initialize a new namespace RWLocker instance.

View File

@ -77,7 +77,7 @@ func (c *Target) Send(e interface{}, logKind string) error {
if tagString != "" { if tagString != "" {
tagString += ", " tagString += ", "
} }
tagString += fmt.Sprintf("%s=%v", key, value) tagString += fmt.Sprintf("%s=%#v", key, value)
} }
} }