2020-07-26 17:09:05 +08:00
|
|
|
package syncx
|
|
|
|
|
|
|
|
import (
|
|
|
|
"io"
|
|
|
|
"sync"
|
|
|
|
|
2022-01-04 15:51:32 +08:00
|
|
|
"github.com/zeromicro/go-zero/core/errorx"
|
2020-07-26 17:09:05 +08:00
|
|
|
)
|
|
|
|
|
2021-02-28 16:16:22 +08:00
|
|
|
// A ResourceManager is a manager that used to manage resources.
|
2020-07-26 17:09:05 +08:00
|
|
|
type ResourceManager struct {
|
2021-09-09 18:06:27 +08:00
|
|
|
resources map[string]io.Closer
|
|
|
|
singleFlight SingleFlight
|
|
|
|
lock sync.RWMutex
|
2020-07-26 17:09:05 +08:00
|
|
|
}
|
|
|
|
|
2021-02-28 16:16:22 +08:00
|
|
|
// NewResourceManager returns a ResourceManager.
|
2020-07-26 17:09:05 +08:00
|
|
|
func NewResourceManager() *ResourceManager {
|
|
|
|
return &ResourceManager{
|
2021-09-09 18:06:27 +08:00
|
|
|
resources: make(map[string]io.Closer),
|
|
|
|
singleFlight: NewSingleFlight(),
|
2020-07-26 17:09:05 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-02-28 16:16:22 +08:00
|
|
|
// Close closes the manager.
|
2021-10-01 10:03:56 +08:00
|
|
|
// Don't use the ResourceManager after Close() called.
|
2020-07-26 17:09:05 +08:00
|
|
|
func (manager *ResourceManager) Close() error {
|
|
|
|
manager.lock.Lock()
|
|
|
|
defer manager.lock.Unlock()
|
|
|
|
|
|
|
|
var be errorx.BatchError
|
|
|
|
for _, resource := range manager.resources {
|
|
|
|
if err := resource.Close(); err != nil {
|
|
|
|
be.Add(err)
|
|
|
|
}
|
|
|
|
}
|
2021-10-01 10:03:56 +08:00
|
|
|
|
|
|
|
// release resources to avoid using it later
|
|
|
|
manager.resources = nil
|
|
|
|
|
2020-07-26 17:09:05 +08:00
|
|
|
return be.Err()
|
|
|
|
}
|
|
|
|
|
2021-02-28 16:16:22 +08:00
|
|
|
// GetResource returns the resource associated with given key.
|
2020-07-26 17:09:05 +08:00
|
|
|
func (manager *ResourceManager) GetResource(key string, create func() (io.Closer, error)) (io.Closer, error) {
|
2023-01-24 16:32:02 +08:00
|
|
|
val, err := manager.singleFlight.Do(key, func() (any, error) {
|
2020-07-26 17:09:05 +08:00
|
|
|
manager.lock.RLock()
|
|
|
|
resource, ok := manager.resources[key]
|
|
|
|
manager.lock.RUnlock()
|
|
|
|
if ok {
|
|
|
|
return resource, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
resource, err := create()
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
manager.lock.Lock()
|
2022-08-26 20:07:25 +08:00
|
|
|
defer manager.lock.Unlock()
|
2020-07-26 17:09:05 +08:00
|
|
|
manager.resources[key] = resource
|
|
|
|
|
|
|
|
return resource, nil
|
|
|
|
})
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return val.(io.Closer), nil
|
|
|
|
}
|
2022-04-19 14:03:04 +08:00
|
|
|
|
|
|
|
// Inject injects the resource associated with given key.
|
|
|
|
func (manager *ResourceManager) Inject(key string, resource io.Closer) {
|
|
|
|
manager.lock.Lock()
|
|
|
|
manager.resources[key] = resource
|
|
|
|
manager.lock.Unlock()
|
|
|
|
}
|