mirror of
https://github.com/gogf/gf.git
synced 2025-04-05 03:05:05 +08:00
database/gdb: fix deadlock when orm operations perform in cache function from gcache (#3585)
This commit is contained in:
parent
2e471662f4
commit
23df83cb0b
4
.github/workflows/golangci-lint.yml
vendored
4
.github/workflows/golangci-lint.yml
vendored
@ -47,8 +47,8 @@ jobs:
|
|||||||
with:
|
with:
|
||||||
go-version: ${{ matrix.go-version }}
|
go-version: ${{ matrix.go-version }}
|
||||||
- name: golangci-lint
|
- name: golangci-lint
|
||||||
uses: golangci/golangci-lint-action@v4
|
uses: golangci/golangci-lint-action@v6
|
||||||
with:
|
with:
|
||||||
# Required: the version of golangci-lint is required and must be specified without patch version: we always use the latest patch version.
|
# Required: the version of golangci-lint is required and must be specified without patch version: we always use the latest patch version.
|
||||||
version: v1.56.2
|
version: v1.58.2
|
||||||
args: --timeout 3m0s
|
args: --timeout 3m0s
|
||||||
|
@ -38,7 +38,7 @@ linters:
|
|||||||
# Custom enable linters we want to use.
|
# Custom enable linters we want to use.
|
||||||
enable:
|
enable:
|
||||||
- errcheck # Errcheck is a program for checking for unchecked errors in go programs.
|
- errcheck # Errcheck is a program for checking for unchecked errors in go programs.
|
||||||
- errchkjson # Checks types passed to the json encoding functions. Reports unsupported types and optionally reports occasions, where the check for the returned error can be omitted.
|
- errchkjson # Checks types passed to the JSON encoding functions. Reports unsupported types and optionally reports occasions, where the check for the returned error can be omitted.
|
||||||
- funlen # Tool for detection of long functions
|
- funlen # Tool for detection of long functions
|
||||||
- goconst # Finds repeated strings that could be replaced by a constant
|
- goconst # Finds repeated strings that could be replaced by a constant
|
||||||
- gocritic # Provides diagnostics that check for bugs, performance and style issues.
|
- gocritic # Provides diagnostics that check for bugs, performance and style issues.
|
||||||
@ -172,10 +172,6 @@ linters-settings:
|
|||||||
|
|
||||||
# https://golangci-lint.run/usage/linters/#gosimple
|
# https://golangci-lint.run/usage/linters/#gosimple
|
||||||
gosimple:
|
gosimple:
|
||||||
# Select the Go version to target.
|
|
||||||
# Default: 1.13
|
|
||||||
# Deprecated: use the global `run.go` instead.
|
|
||||||
go: "1.15"
|
|
||||||
# Sxxxx checks in https://staticcheck.io/docs/configuration/options/#checks
|
# Sxxxx checks in https://staticcheck.io/docs/configuration/options/#checks
|
||||||
# Default: ["*"]
|
# Default: ["*"]
|
||||||
checks: [
|
checks: [
|
||||||
@ -184,9 +180,6 @@ linters-settings:
|
|||||||
|
|
||||||
# https://golangci-lint.run/usage/linters/#govet
|
# https://golangci-lint.run/usage/linters/#govet
|
||||||
govet:
|
govet:
|
||||||
# Report about shadowed variables.
|
|
||||||
# Default: false
|
|
||||||
check-shadowing: true
|
|
||||||
# Settings per analyzer.
|
# Settings per analyzer.
|
||||||
settings:
|
settings:
|
||||||
# Analyzer name, run `go tool vet help` to see all analyzers.
|
# Analyzer name, run `go tool vet help` to see all analyzers.
|
||||||
@ -259,10 +252,6 @@ linters-settings:
|
|||||||
|
|
||||||
# https://golangci-lint.run/usage/linters/#staticcheck
|
# https://golangci-lint.run/usage/linters/#staticcheck
|
||||||
staticcheck:
|
staticcheck:
|
||||||
# Select the Go version to target.
|
|
||||||
# Default: "1.13"
|
|
||||||
# Deprecated: use the global `run.go` instead.
|
|
||||||
go: "1.20"
|
|
||||||
# SAxxxx checks in https://staticcheck.io/docs/configuration/options/#checks
|
# SAxxxx checks in https://staticcheck.io/docs/configuration/options/#checks
|
||||||
# Default: ["*"]
|
# Default: ["*"]
|
||||||
checks: [ "all","-SA1019","-SA4015","-SA1029","-SA1016","-SA9003","-SA4006","-SA6003" ]
|
checks: [ "all","-SA1019","-SA4015","-SA1029","-SA1016","-SA9003","-SA4006","-SA6003" ]
|
||||||
|
@ -11,13 +11,14 @@ import (
|
|||||||
"context"
|
"context"
|
||||||
"fmt"
|
"fmt"
|
||||||
|
|
||||||
|
"github.com/hashicorp/consul/api"
|
||||||
|
"github.com/hashicorp/consul/api/watch"
|
||||||
|
|
||||||
"github.com/gogf/gf/v2/encoding/gjson"
|
"github.com/gogf/gf/v2/encoding/gjson"
|
||||||
"github.com/gogf/gf/v2/errors/gerror"
|
"github.com/gogf/gf/v2/errors/gerror"
|
||||||
"github.com/gogf/gf/v2/frame/g"
|
"github.com/gogf/gf/v2/frame/g"
|
||||||
"github.com/gogf/gf/v2/os/gcfg"
|
"github.com/gogf/gf/v2/os/gcfg"
|
||||||
"github.com/gogf/gf/v2/os/glog"
|
"github.com/gogf/gf/v2/os/glog"
|
||||||
"github.com/hashicorp/consul/api"
|
|
||||||
"github.com/hashicorp/consul/api/watch"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
// Config is the configuration object for consul client.
|
// Config is the configuration object for consul client.
|
||||||
@ -156,22 +157,28 @@ func (c *Client) addWatcher() (err error) {
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
if err := c.doUpdate(v.Value); err != nil {
|
if err = c.doUpdate(v.Value); err != nil {
|
||||||
c.config.Logger.Errorf(context.Background(),
|
c.config.Logger.Errorf(
|
||||||
|
context.Background(),
|
||||||
"watch config from consul path %+v update failed: %s",
|
"watch config from consul path %+v update failed: %s",
|
||||||
c.config.Path, err)
|
c.config.Path, err,
|
||||||
|
)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
plan.Datacenter = c.config.ConsulConfig.Datacenter
|
plan.Datacenter = c.config.ConsulConfig.Datacenter
|
||||||
plan.Token = c.config.ConsulConfig.Token
|
plan.Token = c.config.ConsulConfig.Token
|
||||||
|
|
||||||
go func() {
|
go c.startAsynchronousWatch(plan)
|
||||||
if err := plan.Run(c.config.ConsulConfig.Address); err != nil {
|
|
||||||
c.config.Logger.Errorf(context.Background(),
|
|
||||||
"watch config from consul path %+v plan start failed: %s",
|
|
||||||
c.config.Path, err)
|
|
||||||
}
|
|
||||||
}()
|
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (c *Client) startAsynchronousWatch(plan *watch.Plan) {
|
||||||
|
if err := plan.Run(c.config.ConsulConfig.Address); err != nil {
|
||||||
|
c.config.Logger.Errorf(
|
||||||
|
context.Background(),
|
||||||
|
"watch config from consul path %+v plan start failed: %s",
|
||||||
|
c.config.Path, err,
|
||||||
|
)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
@ -164,14 +164,16 @@ func (c *Client) doWatch(ctx context.Context, namespace string) (err error) {
|
|||||||
c.config.ConfigMap, namespace,
|
c.config.ConfigMap, namespace,
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
go func() {
|
go c.startAsynchronousWatch(ctx, namespace, watchHandler)
|
||||||
for {
|
|
||||||
event := <-watchHandler.ResultChan()
|
|
||||||
switch event.Type {
|
|
||||||
case watch.Modified:
|
|
||||||
_ = c.doUpdate(ctx, namespace)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}()
|
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (c *Client) startAsynchronousWatch(ctx context.Context, namespace string, watchHandler watch.Interface) {
|
||||||
|
for {
|
||||||
|
event := <-watchHandler.ResultChan()
|
||||||
|
switch event.Type {
|
||||||
|
case watch.Modified:
|
||||||
|
_ = c.doUpdate(ctx, namespace)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
@ -158,14 +158,18 @@ func (c *Client) doWatch(ctx context.Context) (err error) {
|
|||||||
if !c.config.Watch {
|
if !c.config.Watch {
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
var changeChan = c.client.AddChangeListenerWithChannel()
|
go c.startAsynchronousWatch(
|
||||||
go func() {
|
ctx,
|
||||||
for {
|
c.client.AddChangeListenerWithChannel(),
|
||||||
select {
|
)
|
||||||
case <-changeChan:
|
|
||||||
_ = c.doUpdate(ctx)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}()
|
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (c *Client) startAsynchronousWatch(ctx context.Context, changeChan <-chan model.ConfigFileChangeEvent) {
|
||||||
|
for {
|
||||||
|
select {
|
||||||
|
case <-changeChan:
|
||||||
|
_ = c.doUpdate(ctx)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
@ -108,11 +108,7 @@ func (s *GrpcServer) Run() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Start listening.
|
// Start listening.
|
||||||
go func() {
|
go s.doServeAsynchronously(ctx)
|
||||||
if err = s.Server.Serve(s.listener); err != nil {
|
|
||||||
s.Logger().Fatalf(ctx, `%+v`, err)
|
|
||||||
}
|
|
||||||
}()
|
|
||||||
|
|
||||||
// Service register.
|
// Service register.
|
||||||
s.doServiceRegister()
|
s.doServiceRegister()
|
||||||
@ -124,6 +120,12 @@ func (s *GrpcServer) Run() {
|
|||||||
s.doSignalListen()
|
s.doSignalListen()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (s *GrpcServer) doServeAsynchronously(ctx context.Context) {
|
||||||
|
if err := s.Server.Serve(s.listener); err != nil {
|
||||||
|
s.Logger().Fatalf(ctx, `%+v`, err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// doSignalListen does signal listening and handling for gracefully shutdown.
|
// doSignalListen does signal listening and handling for gracefully shutdown.
|
||||||
func (s *GrpcServer) doSignalListen() {
|
func (s *GrpcServer) doSignalListen() {
|
||||||
var ctx = context.Background()
|
var ctx = context.Background()
|
||||||
@ -204,10 +206,12 @@ func (s *GrpcServer) doServiceDeregister() {
|
|||||||
// Start starts the server in no-blocking way.
|
// Start starts the server in no-blocking way.
|
||||||
func (s *GrpcServer) Start() {
|
func (s *GrpcServer) Start() {
|
||||||
s.waitGroup.Add(1)
|
s.waitGroup.Add(1)
|
||||||
go func() {
|
go s.doStartAsynchronously()
|
||||||
defer s.waitGroup.Done()
|
}
|
||||||
s.Run()
|
|
||||||
}()
|
func (s *GrpcServer) doStartAsynchronously() {
|
||||||
|
defer s.waitGroup.Done()
|
||||||
|
s.Run()
|
||||||
}
|
}
|
||||||
|
|
||||||
// Wait works with Start, which blocks current goroutine until the server stops.
|
// Wait works with Start, which blocks current goroutine until the server stops.
|
||||||
|
@ -139,16 +139,15 @@ const (
|
|||||||
)
|
)
|
||||||
|
|
||||||
func wrapClientStream(s grpc.ClientStream, desc *grpc.StreamDesc) *clientStream {
|
func wrapClientStream(s grpc.ClientStream, desc *grpc.StreamDesc) *clientStream {
|
||||||
events := make(chan streamEvent)
|
var (
|
||||||
eventsDone := make(chan struct{})
|
events = make(chan streamEvent)
|
||||||
finished := make(chan error)
|
eventsDone = make(chan struct{})
|
||||||
|
finished = make(chan error)
|
||||||
|
)
|
||||||
go func() {
|
go func() {
|
||||||
defer close(eventsDone)
|
defer close(eventsDone)
|
||||||
|
|
||||||
// Both streams have to be closed
|
// Both streams have to be closed
|
||||||
state := byte(0)
|
state := byte(0)
|
||||||
|
|
||||||
for event := range events {
|
for event := range events {
|
||||||
switch event.Type {
|
switch event.Type {
|
||||||
case closeEvent:
|
case closeEvent:
|
||||||
|
@ -268,6 +268,7 @@ type Core struct {
|
|||||||
logger glog.ILogger // Logger for logging functionality.
|
logger glog.ILogger // Logger for logging functionality.
|
||||||
config *ConfigNode // Current config node.
|
config *ConfigNode // Current config node.
|
||||||
dynamicConfig dynamicConfig // Dynamic configurations, which can be changed in runtime.
|
dynamicConfig dynamicConfig // Dynamic configurations, which can be changed in runtime.
|
||||||
|
innerMemCache *gcache.Cache
|
||||||
}
|
}
|
||||||
|
|
||||||
type dynamicConfig struct {
|
type dynamicConfig struct {
|
||||||
@ -525,9 +526,6 @@ var (
|
|||||||
// allDryRun sets dry-run feature for all database connections.
|
// allDryRun sets dry-run feature for all database connections.
|
||||||
// It is commonly used for command options for convenience.
|
// It is commonly used for command options for convenience.
|
||||||
allDryRun = false
|
allDryRun = false
|
||||||
|
|
||||||
// tableFieldsMap caches the table information retrieved from database.
|
|
||||||
tableFieldsMap = gmap.NewStrAnyMap(true)
|
|
||||||
)
|
)
|
||||||
|
|
||||||
func init() {
|
func init() {
|
||||||
@ -587,12 +585,13 @@ func newDBByConfigNode(node *ConfigNode, group string) (db DB, err error) {
|
|||||||
node = parseConfigNodeLink(node)
|
node = parseConfigNodeLink(node)
|
||||||
}
|
}
|
||||||
c := &Core{
|
c := &Core{
|
||||||
group: group,
|
group: group,
|
||||||
debug: gtype.NewBool(),
|
debug: gtype.NewBool(),
|
||||||
cache: gcache.New(),
|
cache: gcache.New(),
|
||||||
links: gmap.New(true),
|
links: gmap.New(true),
|
||||||
logger: glog.New(),
|
logger: glog.New(),
|
||||||
config: node,
|
config: node,
|
||||||
|
innerMemCache: gcache.New(),
|
||||||
dynamicConfig: dynamicConfig{
|
dynamicConfig: dynamicConfig{
|
||||||
MaxIdleConnCount: node.MaxIdleConnCount,
|
MaxIdleConnCount: node.MaxIdleConnCount,
|
||||||
MaxOpenConnCount: node.MaxOpenConnCount,
|
MaxOpenConnCount: node.MaxOpenConnCount,
|
||||||
|
@ -22,6 +22,7 @@ import (
|
|||||||
"github.com/gogf/gf/v2/internal/intlog"
|
"github.com/gogf/gf/v2/internal/intlog"
|
||||||
"github.com/gogf/gf/v2/internal/reflection"
|
"github.com/gogf/gf/v2/internal/reflection"
|
||||||
"github.com/gogf/gf/v2/internal/utils"
|
"github.com/gogf/gf/v2/internal/utils"
|
||||||
|
"github.com/gogf/gf/v2/os/gcache"
|
||||||
"github.com/gogf/gf/v2/text/gregex"
|
"github.com/gogf/gf/v2/text/gregex"
|
||||||
"github.com/gogf/gf/v2/text/gstr"
|
"github.com/gogf/gf/v2/text/gstr"
|
||||||
"github.com/gogf/gf/v2/util/gconv"
|
"github.com/gogf/gf/v2/util/gconv"
|
||||||
@ -737,20 +738,27 @@ func (c *Core) HasTable(name string) (bool, error) {
|
|||||||
return false, nil
|
return false, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (c *Core) GetInnerMemCache() *gcache.Cache {
|
||||||
|
return c.innerMemCache
|
||||||
|
}
|
||||||
|
|
||||||
// GetTablesWithCache retrieves and returns the table names of current database with cache.
|
// GetTablesWithCache retrieves and returns the table names of current database with cache.
|
||||||
func (c *Core) GetTablesWithCache() ([]string, error) {
|
func (c *Core) GetTablesWithCache() ([]string, error) {
|
||||||
var (
|
var (
|
||||||
ctx = c.db.GetCtx()
|
ctx = c.db.GetCtx()
|
||||||
cacheKey = fmt.Sprintf(`Tables: %s`, c.db.GetGroup())
|
cacheKey = fmt.Sprintf(`Tables:%s`, c.db.GetGroup())
|
||||||
|
cacheDuration = gcache.DurationNoExpire
|
||||||
|
innerMemCache = c.GetInnerMemCache()
|
||||||
)
|
)
|
||||||
result, err := c.GetCache().GetOrSetFuncLock(
|
result, err := innerMemCache.GetOrSetFuncLock(
|
||||||
ctx, cacheKey, func(ctx context.Context) (interface{}, error) {
|
ctx, cacheKey,
|
||||||
|
func(ctx context.Context) (interface{}, error) {
|
||||||
tableList, err := c.db.Tables(ctx)
|
tableList, err := c.db.Tables(ctx)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return false, err
|
return nil, err
|
||||||
}
|
}
|
||||||
return tableList, nil
|
return tableList, nil
|
||||||
}, 0,
|
}, cacheDuration,
|
||||||
)
|
)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
|
@ -11,12 +11,10 @@ import (
|
|||||||
"context"
|
"context"
|
||||||
"fmt"
|
"fmt"
|
||||||
|
|
||||||
"github.com/gogf/gf/v2/crypto/gmd5"
|
|
||||||
"github.com/gogf/gf/v2/errors/gcode"
|
"github.com/gogf/gf/v2/errors/gcode"
|
||||||
"github.com/gogf/gf/v2/errors/gerror"
|
"github.com/gogf/gf/v2/errors/gerror"
|
||||||
"github.com/gogf/gf/v2/text/gregex"
|
"github.com/gogf/gf/v2/text/gregex"
|
||||||
"github.com/gogf/gf/v2/text/gstr"
|
"github.com/gogf/gf/v2/text/gstr"
|
||||||
"github.com/gogf/gf/v2/util/gconv"
|
|
||||||
"github.com/gogf/gf/v2/util/gutil"
|
"github.com/gogf/gf/v2/util/gutil"
|
||||||
)
|
)
|
||||||
|
|
||||||
@ -144,22 +142,40 @@ func (c *Core) TableFields(ctx context.Context, table string, schema ...string)
|
|||||||
|
|
||||||
// ClearTableFields removes certain cached table fields of current configuration group.
|
// ClearTableFields removes certain cached table fields of current configuration group.
|
||||||
func (c *Core) ClearTableFields(ctx context.Context, table string, schema ...string) (err error) {
|
func (c *Core) ClearTableFields(ctx context.Context, table string, schema ...string) (err error) {
|
||||||
tableFieldsMap.Remove(fmt.Sprintf(
|
tableFieldsCacheKey := genTableFieldsCacheKey(
|
||||||
`%s%s@%s#%s`,
|
|
||||||
cachePrefixTableFields,
|
|
||||||
c.db.GetGroup(),
|
c.db.GetGroup(),
|
||||||
gutil.GetOrDefaultStr(c.db.GetSchema(), schema...),
|
gutil.GetOrDefaultStr(c.db.GetSchema(), schema...),
|
||||||
table,
|
table,
|
||||||
))
|
)
|
||||||
|
_, err = c.innerMemCache.Remove(ctx, tableFieldsCacheKey)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
// ClearTableFieldsAll removes all cached table fields of current configuration group.
|
// ClearTableFieldsAll removes all cached table fields of current configuration group.
|
||||||
func (c *Core) ClearTableFieldsAll(ctx context.Context) (err error) {
|
func (c *Core) ClearTableFieldsAll(ctx context.Context) (err error) {
|
||||||
var (
|
var (
|
||||||
keys = tableFieldsMap.Keys()
|
keys, _ = c.innerMemCache.KeyStrings(ctx)
|
||||||
cachePrefix = fmt.Sprintf(`%s%s`, cachePrefixTableFields, c.db.GetGroup())
|
cachePrefix = cachePrefixTableFields
|
||||||
removedKeys = make([]string, 0)
|
removedKeys = make([]any, 0)
|
||||||
|
)
|
||||||
|
for _, key := range keys {
|
||||||
|
if gstr.HasPrefix(key, cachePrefix) {
|
||||||
|
removedKeys = append(removedKeys, key)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if len(removedKeys) > 0 {
|
||||||
|
err = c.innerMemCache.Removes(ctx, removedKeys)
|
||||||
|
}
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// ClearCache removes cached sql result of certain table.
|
||||||
|
func (c *Core) ClearCache(ctx context.Context, table string) (err error) {
|
||||||
|
var (
|
||||||
|
keys, _ = c.db.GetCache().KeyStrings(ctx)
|
||||||
|
cachePrefix = fmt.Sprintf(`%s%s@`, cachePrefixSelectCache, table)
|
||||||
|
removedKeys = make([]any, 0)
|
||||||
)
|
)
|
||||||
for _, key := range keys {
|
for _, key := range keys {
|
||||||
if gstr.HasPrefix(key, cachePrefix) {
|
if gstr.HasPrefix(key, cachePrefix) {
|
||||||
@ -167,32 +183,20 @@ func (c *Core) ClearTableFieldsAll(ctx context.Context) (err error) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
if len(removedKeys) > 0 {
|
if len(removedKeys) > 0 {
|
||||||
tableFieldsMap.Removes(removedKeys)
|
err = c.db.GetCache().Removes(ctx, removedKeys)
|
||||||
}
|
}
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
// ClearCache removes cached sql result of certain table.
|
|
||||||
func (c *Core) ClearCache(ctx context.Context, table string) (err error) {
|
|
||||||
return c.db.GetCache().Clear(ctx)
|
|
||||||
}
|
|
||||||
|
|
||||||
// ClearCacheAll removes all cached sql result from cache
|
// ClearCacheAll removes all cached sql result from cache
|
||||||
func (c *Core) ClearCacheAll(ctx context.Context) (err error) {
|
func (c *Core) ClearCacheAll(ctx context.Context) (err error) {
|
||||||
return c.db.GetCache().Clear(ctx)
|
if err = c.db.GetCache().Clear(ctx); err != nil {
|
||||||
}
|
return err
|
||||||
|
|
||||||
func (c *Core) makeSelectCacheKey(name, schema, table, sql string, args ...interface{}) string {
|
|
||||||
if name == "" {
|
|
||||||
name = fmt.Sprintf(
|
|
||||||
`%s@%s#%s:%s`,
|
|
||||||
c.db.GetGroup(),
|
|
||||||
schema,
|
|
||||||
table,
|
|
||||||
gmd5.MustEncryptString(sql+", @PARAMS:"+gconv.String(args)),
|
|
||||||
)
|
|
||||||
}
|
}
|
||||||
return fmt.Sprintf(`%s%s`, cachePrefixSelectCache, name)
|
if err = c.GetInnerMemCache().Clear(ctx); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
// HasField determine whether the field exists in the table.
|
// HasField determine whether the field exists in the table.
|
||||||
|
@ -11,10 +11,12 @@ import (
|
|||||||
"database/sql"
|
"database/sql"
|
||||||
"fmt"
|
"fmt"
|
||||||
|
|
||||||
|
"github.com/gogf/gf/v2/container/gvar"
|
||||||
"github.com/gogf/gf/v2/encoding/gjson"
|
"github.com/gogf/gf/v2/encoding/gjson"
|
||||||
"github.com/gogf/gf/v2/errors/gcode"
|
"github.com/gogf/gf/v2/errors/gcode"
|
||||||
"github.com/gogf/gf/v2/errors/gerror"
|
"github.com/gogf/gf/v2/errors/gerror"
|
||||||
"github.com/gogf/gf/v2/internal/intlog"
|
"github.com/gogf/gf/v2/internal/intlog"
|
||||||
|
"github.com/gogf/gf/v2/os/gcache"
|
||||||
"github.com/gogf/gf/v2/text/gstr"
|
"github.com/gogf/gf/v2/text/gstr"
|
||||||
"github.com/gogf/gf/v2/util/gutil"
|
"github.com/gogf/gf/v2/util/gutil"
|
||||||
)
|
)
|
||||||
@ -68,25 +70,29 @@ func (d *DriverWrapperDB) TableFields(
|
|||||||
)
|
)
|
||||||
}
|
}
|
||||||
var (
|
var (
|
||||||
|
innerMemCache = d.GetCore().GetInnerMemCache()
|
||||||
// prefix:group@schema#table
|
// prefix:group@schema#table
|
||||||
cacheKey = fmt.Sprintf(
|
cacheKey = genTableFieldsCacheKey(
|
||||||
`%s%s@%s#%s`,
|
|
||||||
cachePrefixTableFields,
|
|
||||||
d.GetGroup(),
|
d.GetGroup(),
|
||||||
gutil.GetOrDefaultStr(d.GetSchema(), schema...),
|
gutil.GetOrDefaultStr(d.GetSchema(), schema...),
|
||||||
table,
|
table,
|
||||||
)
|
)
|
||||||
value = tableFieldsMap.GetOrSetFuncLock(cacheKey, func() interface{} {
|
cacheFunc = func(ctx context.Context) (interface{}, error) {
|
||||||
ctx = context.WithValue(ctx, ctxKeyInternalProducedSQL, struct{}{})
|
return d.DB.TableFields(
|
||||||
fields, err = d.DB.TableFields(ctx, table, schema...)
|
context.WithValue(ctx, ctxKeyInternalProducedSQL, struct{}{}),
|
||||||
if err != nil {
|
table, schema...,
|
||||||
return nil
|
)
|
||||||
}
|
}
|
||||||
return fields
|
value *gvar.Var
|
||||||
})
|
|
||||||
)
|
)
|
||||||
if value != nil {
|
value, err = innerMemCache.GetOrSetFuncLock(
|
||||||
fields = value.(map[string]*TableField)
|
ctx, cacheKey, cacheFunc, gcache.DurationNoExpire,
|
||||||
|
)
|
||||||
|
if err != nil {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
if !value.IsNil() {
|
||||||
|
fields = value.Val().(map[string]*TableField)
|
||||||
}
|
}
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
@ -16,6 +16,7 @@ import (
|
|||||||
"time"
|
"time"
|
||||||
|
|
||||||
"github.com/gogf/gf/v2/container/garray"
|
"github.com/gogf/gf/v2/container/garray"
|
||||||
|
"github.com/gogf/gf/v2/encoding/ghash"
|
||||||
"github.com/gogf/gf/v2/encoding/gjson"
|
"github.com/gogf/gf/v2/encoding/gjson"
|
||||||
"github.com/gogf/gf/v2/internal/empty"
|
"github.com/gogf/gf/v2/internal/empty"
|
||||||
"github.com/gogf/gf/v2/internal/intlog"
|
"github.com/gogf/gf/v2/internal/intlog"
|
||||||
@ -944,3 +945,26 @@ func FormatMultiLineSqlToSingle(sql string) (string, error) {
|
|||||||
}
|
}
|
||||||
return sql, nil
|
return sql, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func genTableFieldsCacheKey(group, schema, table string) string {
|
||||||
|
return fmt.Sprintf(
|
||||||
|
`%s%s@%s#%s`,
|
||||||
|
cachePrefixTableFields,
|
||||||
|
group,
|
||||||
|
schema,
|
||||||
|
table,
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
func genSelectCacheKey(table, group, schema, name, sql string, args ...interface{}) string {
|
||||||
|
if name == "" {
|
||||||
|
name = fmt.Sprintf(
|
||||||
|
`%s@%s#%s:%d`,
|
||||||
|
table,
|
||||||
|
group,
|
||||||
|
schema,
|
||||||
|
ghash.BKDR64([]byte(sql+", @PARAMS:"+gconv.String(args))),
|
||||||
|
)
|
||||||
|
}
|
||||||
|
return fmt.Sprintf(`%s%s`, cachePrefixSelectCache, name)
|
||||||
|
}
|
||||||
|
@ -142,10 +142,17 @@ func (m *Model) saveSelectResultToCache(
|
|||||||
}
|
}
|
||||||
|
|
||||||
func (m *Model) makeSelectCacheKey(sql string, args ...interface{}) string {
|
func (m *Model) makeSelectCacheKey(sql string, args ...interface{}) string {
|
||||||
return m.db.GetCore().makeSelectCacheKey(
|
var (
|
||||||
m.cacheOption.Name,
|
table = m.db.GetCore().guessPrimaryTableName(m.tables)
|
||||||
m.db.GetSchema(),
|
group = m.db.GetGroup()
|
||||||
m.db.GetCore().guessPrimaryTableName(m.tables),
|
schema = m.db.GetSchema()
|
||||||
|
customName = m.cacheOption.Name
|
||||||
|
)
|
||||||
|
return genSelectCacheKey(
|
||||||
|
table,
|
||||||
|
group,
|
||||||
|
schema,
|
||||||
|
customName,
|
||||||
sql,
|
sql,
|
||||||
args...,
|
args...,
|
||||||
)
|
)
|
||||||
|
@ -190,43 +190,51 @@ func (m *softTimeMaintainer) getSoftFieldNameAndType(
|
|||||||
schema string, table string, checkFiledNames []string,
|
schema string, table string, checkFiledNames []string,
|
||||||
) (fieldName string, fieldType LocalType) {
|
) (fieldName string, fieldType LocalType) {
|
||||||
var (
|
var (
|
||||||
cacheKey = fmt.Sprintf(`getSoftFieldNameAndType:%s#%s#%s`, schema, table, strings.Join(checkFiledNames, "_"))
|
innerMemCache = m.db.GetCore().GetInnerMemCache()
|
||||||
|
cacheKey = fmt.Sprintf(
|
||||||
|
`getSoftFieldNameAndType:%s#%s#%s`,
|
||||||
|
schema, table, strings.Join(checkFiledNames, "_"),
|
||||||
|
)
|
||||||
cacheDuration = gcache.DurationNoExpire
|
cacheDuration = gcache.DurationNoExpire
|
||||||
cacheFunc = func(ctx context.Context) (value interface{}, err error) {
|
cacheFunc = func(ctx context.Context) (value interface{}, err error) {
|
||||||
// Ignore the error from TableFields.
|
// Ignore the error from TableFields.
|
||||||
fieldsMap, _ := m.TableFields(table, schema)
|
fieldsMap, err := m.TableFields(table, schema)
|
||||||
if len(fieldsMap) > 0 {
|
if err != nil {
|
||||||
for _, checkFiledName := range checkFiledNames {
|
return nil, err
|
||||||
fieldName, _ = gutil.MapPossibleItemByKey(
|
}
|
||||||
gconv.Map(fieldsMap), checkFiledName,
|
if len(fieldsMap) == 0 {
|
||||||
|
return nil, nil
|
||||||
|
}
|
||||||
|
for _, checkFiledName := range checkFiledNames {
|
||||||
|
fieldName, _ = gutil.MapPossibleItemByKey(
|
||||||
|
gconv.Map(fieldsMap), checkFiledName,
|
||||||
|
)
|
||||||
|
if fieldName != "" {
|
||||||
|
fieldType, _ = m.db.CheckLocalTypeForField(
|
||||||
|
ctx, fieldsMap[fieldName].Type, nil,
|
||||||
)
|
)
|
||||||
if fieldName != "" {
|
var cacheItem = getSoftFieldNameAndTypeCacheItem{
|
||||||
fieldType, _ = m.db.CheckLocalTypeForField(
|
FieldName: fieldName,
|
||||||
ctx, fieldsMap[fieldName].Type, nil,
|
FieldType: fieldType,
|
||||||
)
|
|
||||||
var cacheItem = getSoftFieldNameAndTypeCacheItem{
|
|
||||||
FieldName: fieldName,
|
|
||||||
FieldType: fieldType,
|
|
||||||
}
|
|
||||||
return cacheItem, nil
|
|
||||||
}
|
}
|
||||||
|
return cacheItem, nil
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
)
|
)
|
||||||
result, err := gcache.GetOrSetFunc(ctx, cacheKey, cacheFunc, cacheDuration)
|
result, err := innerMemCache.GetOrSetFunc(
|
||||||
|
ctx, cacheKey, cacheFunc, cacheDuration,
|
||||||
|
)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
intlog.Error(ctx, err)
|
return
|
||||||
}
|
}
|
||||||
if result != nil {
|
if result == nil {
|
||||||
var cacheItem getSoftFieldNameAndTypeCacheItem
|
return
|
||||||
if err = result.Scan(&cacheItem); err != nil {
|
|
||||||
return "", ""
|
|
||||||
}
|
|
||||||
fieldName = cacheItem.FieldName
|
|
||||||
fieldType = cacheItem.FieldType
|
|
||||||
}
|
}
|
||||||
|
cacheItem := result.Val().(getSoftFieldNameAndTypeCacheItem)
|
||||||
|
fieldName = cacheItem.FieldName
|
||||||
|
fieldType = cacheItem.FieldType
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -569,43 +569,45 @@ func (s *Server) startServer(fdMap listenerFdMap) {
|
|||||||
}
|
}
|
||||||
// Start listening asynchronously.
|
// Start listening asynchronously.
|
||||||
serverRunning.Add(1)
|
serverRunning.Add(1)
|
||||||
var wg = sync.WaitGroup{}
|
var wg = &sync.WaitGroup{}
|
||||||
for _, v := range s.servers {
|
for _, gs := range s.servers {
|
||||||
wg.Add(1)
|
wg.Add(1)
|
||||||
go func(server *gracefulServer) {
|
go s.startGracefulServer(ctx, wg, gs)
|
||||||
s.serverCount.Add(1)
|
|
||||||
var err error
|
|
||||||
// Create listener.
|
|
||||||
if server.isHttps {
|
|
||||||
err = server.CreateListenerTLS(
|
|
||||||
s.config.HTTPSCertPath, s.config.HTTPSKeyPath, s.config.TLSConfig,
|
|
||||||
)
|
|
||||||
} else {
|
|
||||||
err = server.CreateListener()
|
|
||||||
}
|
|
||||||
if err != nil {
|
|
||||||
s.Logger().Fatalf(ctx, `%+v`, err)
|
|
||||||
}
|
|
||||||
wg.Done()
|
|
||||||
// Start listening and serving in blocking way.
|
|
||||||
err = server.Serve(ctx)
|
|
||||||
// The process exits if the server is closed with none closing error.
|
|
||||||
if err != nil && !strings.EqualFold(http.ErrServerClosed.Error(), err.Error()) {
|
|
||||||
s.Logger().Fatalf(ctx, `%+v`, err)
|
|
||||||
}
|
|
||||||
// If all the underlying servers' shutdown, the process exits.
|
|
||||||
if s.serverCount.Add(-1) < 1 {
|
|
||||||
s.closeChan <- struct{}{}
|
|
||||||
if serverRunning.Add(-1) < 1 {
|
|
||||||
serverMapping.Remove(s.instance)
|
|
||||||
allShutdownChan <- struct{}{}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}(v)
|
|
||||||
}
|
}
|
||||||
wg.Wait()
|
wg.Wait()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (s *Server) startGracefulServer(ctx context.Context, wg *sync.WaitGroup, server *gracefulServer) {
|
||||||
|
s.serverCount.Add(1)
|
||||||
|
var err error
|
||||||
|
// Create listener.
|
||||||
|
if server.isHttps {
|
||||||
|
err = server.CreateListenerTLS(
|
||||||
|
s.config.HTTPSCertPath, s.config.HTTPSKeyPath, s.config.TLSConfig,
|
||||||
|
)
|
||||||
|
} else {
|
||||||
|
err = server.CreateListener()
|
||||||
|
}
|
||||||
|
if err != nil {
|
||||||
|
s.Logger().Fatalf(ctx, `%+v`, err)
|
||||||
|
}
|
||||||
|
wg.Done()
|
||||||
|
// Start listening and serving in blocking way.
|
||||||
|
err = server.Serve(ctx)
|
||||||
|
// The process exits if the server is closed with none closing error.
|
||||||
|
if err != nil && !strings.EqualFold(http.ErrServerClosed.Error(), err.Error()) {
|
||||||
|
s.Logger().Fatalf(ctx, `%+v`, err)
|
||||||
|
}
|
||||||
|
// If all the underlying servers' shutdown, the process exits.
|
||||||
|
if s.serverCount.Add(-1) < 1 {
|
||||||
|
s.closeChan <- struct{}{}
|
||||||
|
if serverRunning.Add(-1) < 1 {
|
||||||
|
serverMapping.Remove(s.instance)
|
||||||
|
allShutdownChan <- struct{}{}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// Status retrieves and returns the server status.
|
// Status retrieves and returns the server status.
|
||||||
func (s *Server) Status() ServerStatus {
|
func (s *Server) Status() ServerStatus {
|
||||||
if serverRunning.Val() == 0 {
|
if serverRunning.Val() == 0 {
|
||||||
|
@ -96,8 +96,8 @@ func New() (*Watcher, error) {
|
|||||||
intlog.Printf(context.TODO(), "New watcher failed: %v", err)
|
intlog.Printf(context.TODO(), "New watcher failed: %v", err)
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
w.watchLoop()
|
go w.watchLoop()
|
||||||
w.eventLoop()
|
go w.eventLoop()
|
||||||
return w, nil
|
return w, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -18,136 +18,132 @@ import (
|
|||||||
|
|
||||||
// watchLoop starts the loop for event listening from underlying inotify monitor.
|
// watchLoop starts the loop for event listening from underlying inotify monitor.
|
||||||
func (w *Watcher) watchLoop() {
|
func (w *Watcher) watchLoop() {
|
||||||
go func() {
|
for {
|
||||||
for {
|
select {
|
||||||
select {
|
// Close event.
|
||||||
// Close event.
|
case <-w.closeChan:
|
||||||
case <-w.closeChan:
|
return
|
||||||
|
|
||||||
|
// Event listening.
|
||||||
|
case ev, ok := <-w.watcher.Events:
|
||||||
|
if !ok {
|
||||||
return
|
return
|
||||||
|
}
|
||||||
// Event listening.
|
// Filter the repeated event in custom duration.
|
||||||
case ev, ok := <-w.watcher.Events:
|
_, err := w.cache.SetIfNotExist(
|
||||||
if !ok {
|
context.Background(),
|
||||||
return
|
ev.String(),
|
||||||
}
|
func(ctx context.Context) (value interface{}, err error) {
|
||||||
// Filter the repeated event in custom duration.
|
w.events.Push(&Event{
|
||||||
_, err := w.cache.SetIfNotExist(
|
event: ev,
|
||||||
context.Background(),
|
Path: ev.Name,
|
||||||
ev.String(),
|
Op: Op(ev.Op),
|
||||||
func(ctx context.Context) (value interface{}, err error) {
|
Watcher: w,
|
||||||
w.events.Push(&Event{
|
})
|
||||||
event: ev,
|
return struct{}{}, nil
|
||||||
Path: ev.Name,
|
}, repeatEventFilterDuration,
|
||||||
Op: Op(ev.Op),
|
)
|
||||||
Watcher: w,
|
if err != nil {
|
||||||
})
|
|
||||||
return struct{}{}, nil
|
|
||||||
}, repeatEventFilterDuration,
|
|
||||||
)
|
|
||||||
if err != nil {
|
|
||||||
intlog.Errorf(context.TODO(), `%+v`, err)
|
|
||||||
}
|
|
||||||
|
|
||||||
case err := <-w.watcher.Errors:
|
|
||||||
intlog.Errorf(context.TODO(), `%+v`, err)
|
intlog.Errorf(context.TODO(), `%+v`, err)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
case err := <-w.watcher.Errors:
|
||||||
|
intlog.Errorf(context.TODO(), `%+v`, err)
|
||||||
}
|
}
|
||||||
}()
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// eventLoop is the core event handler.
|
// eventLoop is the core event handler.
|
||||||
func (w *Watcher) eventLoop() {
|
func (w *Watcher) eventLoop() {
|
||||||
go func() {
|
for {
|
||||||
for {
|
if v := w.events.Pop(); v != nil {
|
||||||
if v := w.events.Pop(); v != nil {
|
event := v.(*Event)
|
||||||
event := v.(*Event)
|
// If there's no any callback of this path, it removes it from monitor.
|
||||||
// If there's no any callback of this path, it removes it from monitor.
|
callbacks := w.getCallbacks(event.Path)
|
||||||
callbacks := w.getCallbacks(event.Path)
|
if len(callbacks) == 0 {
|
||||||
if len(callbacks) == 0 {
|
_ = w.watcher.Remove(event.Path)
|
||||||
_ = w.watcher.Remove(event.Path)
|
continue
|
||||||
continue
|
|
||||||
}
|
|
||||||
switch {
|
|
||||||
case event.IsRemove():
|
|
||||||
// It should check again the existence of the path.
|
|
||||||
// It adds it back to the monitor if it still exists.
|
|
||||||
if fileExists(event.Path) {
|
|
||||||
// It adds the path back to monitor.
|
|
||||||
// We need no worry about the repeat adding.
|
|
||||||
if err := w.watcher.Add(event.Path); err != nil {
|
|
||||||
intlog.Errorf(context.TODO(), `%+v`, err)
|
|
||||||
} else {
|
|
||||||
intlog.Printf(context.TODO(), "fake remove event, watcher re-adds monitor for: %s", event.Path)
|
|
||||||
}
|
|
||||||
// Change the event to RENAME, which means it renames itself to its origin name.
|
|
||||||
event.Op = RENAME
|
|
||||||
}
|
|
||||||
|
|
||||||
case event.IsRename():
|
|
||||||
// It should check again the existence of the path.
|
|
||||||
// It adds it back to the monitor if it still exists.
|
|
||||||
// Especially Some editors might do RENAME and then CHMOD when it's editing file.
|
|
||||||
if fileExists(event.Path) {
|
|
||||||
// It might lost the monitoring for the path, so we add the path back to monitor.
|
|
||||||
// We need no worry about the repeat adding.
|
|
||||||
if err := w.watcher.Add(event.Path); err != nil {
|
|
||||||
intlog.Errorf(context.TODO(), `%+v`, err)
|
|
||||||
} else {
|
|
||||||
intlog.Printf(context.TODO(), "fake rename event, watcher re-adds monitor for: %s", event.Path)
|
|
||||||
}
|
|
||||||
// Change the event to CHMOD.
|
|
||||||
event.Op = CHMOD
|
|
||||||
}
|
|
||||||
|
|
||||||
case event.IsCreate():
|
|
||||||
// =========================================
|
|
||||||
// Note that it here just adds the path to monitor without any callback registering,
|
|
||||||
// because its parent already has the callbacks.
|
|
||||||
// =========================================
|
|
||||||
if fileIsDir(event.Path) {
|
|
||||||
// If it's a folder, it then does adding recursively to monitor.
|
|
||||||
for _, subPath := range fileAllDirs(event.Path) {
|
|
||||||
if fileIsDir(subPath) {
|
|
||||||
if err := w.watcher.Add(subPath); err != nil {
|
|
||||||
intlog.Errorf(context.TODO(), `%+v`, err)
|
|
||||||
} else {
|
|
||||||
intlog.Printf(context.TODO(), "folder creation event, watcher adds monitor for: %s", subPath)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
// If it's a file, it directly adds it to monitor.
|
|
||||||
if err := w.watcher.Add(event.Path); err != nil {
|
|
||||||
intlog.Errorf(context.TODO(), `%+v`, err)
|
|
||||||
} else {
|
|
||||||
intlog.Printf(context.TODO(), "file creation event, watcher adds monitor for: %s", event.Path)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
// Calling the callbacks in order.
|
|
||||||
for _, callback := range callbacks {
|
|
||||||
go func(callback *Callback) {
|
|
||||||
defer func() {
|
|
||||||
if err := recover(); err != nil {
|
|
||||||
switch err {
|
|
||||||
case callbackExitEventPanicStr:
|
|
||||||
w.RemoveCallback(callback.Id)
|
|
||||||
default:
|
|
||||||
if e, ok := err.(error); ok {
|
|
||||||
panic(gerror.WrapCode(gcode.CodeInternalPanic, e))
|
|
||||||
}
|
|
||||||
panic(err)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}()
|
|
||||||
callback.Func(event)
|
|
||||||
}(callback)
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
break
|
|
||||||
}
|
}
|
||||||
|
switch {
|
||||||
|
case event.IsRemove():
|
||||||
|
// It should check again the existence of the path.
|
||||||
|
// It adds it back to the monitor if it still exists.
|
||||||
|
if fileExists(event.Path) {
|
||||||
|
// It adds the path back to monitor.
|
||||||
|
// We need no worry about the repeat adding.
|
||||||
|
if err := w.watcher.Add(event.Path); err != nil {
|
||||||
|
intlog.Errorf(context.TODO(), `%+v`, err)
|
||||||
|
} else {
|
||||||
|
intlog.Printf(context.TODO(), "fake remove event, watcher re-adds monitor for: %s", event.Path)
|
||||||
|
}
|
||||||
|
// Change the event to RENAME, which means it renames itself to its origin name.
|
||||||
|
event.Op = RENAME
|
||||||
|
}
|
||||||
|
|
||||||
|
case event.IsRename():
|
||||||
|
// It should check again the existence of the path.
|
||||||
|
// It adds it back to the monitor if it still exists.
|
||||||
|
// Especially Some editors might do RENAME and then CHMOD when it's editing file.
|
||||||
|
if fileExists(event.Path) {
|
||||||
|
// It might lost the monitoring for the path, so we add the path back to monitor.
|
||||||
|
// We need no worry about the repeat adding.
|
||||||
|
if err := w.watcher.Add(event.Path); err != nil {
|
||||||
|
intlog.Errorf(context.TODO(), `%+v`, err)
|
||||||
|
} else {
|
||||||
|
intlog.Printf(context.TODO(), "fake rename event, watcher re-adds monitor for: %s", event.Path)
|
||||||
|
}
|
||||||
|
// Change the event to CHMOD.
|
||||||
|
event.Op = CHMOD
|
||||||
|
}
|
||||||
|
|
||||||
|
case event.IsCreate():
|
||||||
|
// =========================================
|
||||||
|
// Note that it here just adds the path to monitor without any callback registering,
|
||||||
|
// because its parent already has the callbacks.
|
||||||
|
// =========================================
|
||||||
|
if fileIsDir(event.Path) {
|
||||||
|
// If it's a folder, it then does adding recursively to monitor.
|
||||||
|
for _, subPath := range fileAllDirs(event.Path) {
|
||||||
|
if fileIsDir(subPath) {
|
||||||
|
if err := w.watcher.Add(subPath); err != nil {
|
||||||
|
intlog.Errorf(context.TODO(), `%+v`, err)
|
||||||
|
} else {
|
||||||
|
intlog.Printf(context.TODO(), "folder creation event, watcher adds monitor for: %s", subPath)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
// If it's a file, it directly adds it to monitor.
|
||||||
|
if err := w.watcher.Add(event.Path); err != nil {
|
||||||
|
intlog.Errorf(context.TODO(), `%+v`, err)
|
||||||
|
} else {
|
||||||
|
intlog.Printf(context.TODO(), "file creation event, watcher adds monitor for: %s", event.Path)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// Calling the callbacks in order.
|
||||||
|
for _, callback := range callbacks {
|
||||||
|
go func(callback *Callback) {
|
||||||
|
defer func() {
|
||||||
|
if err := recover(); err != nil {
|
||||||
|
switch err {
|
||||||
|
case callbackExitEventPanicStr:
|
||||||
|
w.RemoveCallback(callback.Id)
|
||||||
|
default:
|
||||||
|
if e, ok := err.(error); ok {
|
||||||
|
panic(gerror.WrapCode(gcode.CodeInternalPanic, e))
|
||||||
|
}
|
||||||
|
panic(err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}()
|
||||||
|
callback.Func(event)
|
||||||
|
}(callback)
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
break
|
||||||
}
|
}
|
||||||
}()
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// getCallbacks searches and returns all callbacks with given `path`.
|
// getCallbacks searches and returns all callbacks with given `path`.
|
||||||
|
@ -99,21 +99,23 @@ func (p *Pool) checkAndForkNewGoroutineWorker() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Create job function in goroutine.
|
// Create job function in goroutine.
|
||||||
go func() {
|
go p.asynchronousWorker()
|
||||||
defer p.count.Add(-1)
|
}
|
||||||
|
|
||||||
var (
|
func (p *Pool) asynchronousWorker() {
|
||||||
listItem interface{}
|
defer p.count.Add(-1)
|
||||||
poolItem *localPoolItem
|
|
||||||
)
|
var (
|
||||||
// Harding working, one by one, job never empty, worker never die.
|
listItem interface{}
|
||||||
for !p.closed.Val() {
|
poolItem *localPoolItem
|
||||||
listItem = p.list.PopBack()
|
)
|
||||||
if listItem == nil {
|
// Harding working, one by one, job never empty, worker never die.
|
||||||
return
|
for !p.closed.Val() {
|
||||||
}
|
listItem = p.list.PopBack()
|
||||||
poolItem = listItem.(*localPoolItem)
|
if listItem == nil {
|
||||||
poolItem.Func(poolItem.Ctx)
|
return
|
||||||
}
|
}
|
||||||
}()
|
poolItem = listItem.(*localPoolItem)
|
||||||
|
poolItem.Func(poolItem.Ctx)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
@ -8,6 +8,7 @@ package gtimer
|
|||||||
|
|
||||||
import (
|
import (
|
||||||
"context"
|
"context"
|
||||||
|
|
||||||
"github.com/gogf/gf/v2/errors/gcode"
|
"github.com/gogf/gf/v2/errors/gcode"
|
||||||
|
|
||||||
"github.com/gogf/gf/v2/container/gtype"
|
"github.com/gogf/gf/v2/container/gtype"
|
||||||
@ -45,26 +46,29 @@ func (entry *Entry) Run() {
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
go func() {
|
go entry.callJobFunc()
|
||||||
defer func() {
|
}
|
||||||
if exception := recover(); exception != nil {
|
|
||||||
if exception != panicExit {
|
// callJobFunc executes the job function in entry.
|
||||||
if v, ok := exception.(error); ok && gerror.HasStack(v) {
|
func (entry *Entry) callJobFunc() {
|
||||||
panic(v)
|
defer func() {
|
||||||
} else {
|
if exception := recover(); exception != nil {
|
||||||
panic(gerror.NewCodef(gcode.CodeInternalPanic, "exception recovered: %+v", exception))
|
if exception != panicExit {
|
||||||
}
|
if v, ok := exception.(error); ok && gerror.HasStack(v) {
|
||||||
|
panic(v)
|
||||||
} else {
|
} else {
|
||||||
entry.Close()
|
panic(gerror.NewCodef(gcode.CodeInternalPanic, "exception recovered: %+v", exception))
|
||||||
return
|
|
||||||
}
|
}
|
||||||
|
} else {
|
||||||
|
entry.Close()
|
||||||
|
return
|
||||||
}
|
}
|
||||||
if entry.Status() == StatusRunning {
|
}
|
||||||
entry.SetStatus(StatusReady)
|
if entry.Status() == StatusRunning {
|
||||||
}
|
entry.SetStatus(StatusReady)
|
||||||
}()
|
}
|
||||||
entry.job(entry.ctx)
|
|
||||||
}()
|
}()
|
||||||
|
entry.job(entry.ctx)
|
||||||
}
|
}
|
||||||
|
|
||||||
// doCheckAndRunByTicks checks the if job can run in given timer ticks,
|
// doCheckAndRunByTicks checks the if job can run in given timer ticks,
|
||||||
|
@ -10,40 +10,36 @@ import "time"
|
|||||||
|
|
||||||
// loop starts the ticker using a standalone goroutine.
|
// loop starts the ticker using a standalone goroutine.
|
||||||
func (t *Timer) loop() {
|
func (t *Timer) loop() {
|
||||||
go func() {
|
var (
|
||||||
var (
|
currentTimerTicks int64
|
||||||
currentTimerTicks int64
|
timerIntervalTicker = time.NewTicker(t.options.Interval)
|
||||||
timerIntervalTicker = time.NewTicker(t.options.Interval)
|
)
|
||||||
)
|
defer timerIntervalTicker.Stop()
|
||||||
defer timerIntervalTicker.Stop()
|
for {
|
||||||
for {
|
select {
|
||||||
select {
|
case <-timerIntervalTicker.C:
|
||||||
case <-timerIntervalTicker.C:
|
// Check the timer status.
|
||||||
// Check the timer status.
|
switch t.status.Val() {
|
||||||
switch t.status.Val() {
|
case StatusRunning:
|
||||||
case StatusRunning:
|
// Timer proceeding.
|
||||||
// Timer proceeding.
|
if currentTimerTicks = t.ticks.Add(1); currentTimerTicks >= t.queue.NextPriority() {
|
||||||
if currentTimerTicks = t.ticks.Add(1); currentTimerTicks >= t.queue.NextPriority() {
|
t.proceed(currentTimerTicks)
|
||||||
t.proceed(currentTimerTicks)
|
|
||||||
}
|
|
||||||
|
|
||||||
case StatusStopped:
|
|
||||||
// Do nothing.
|
|
||||||
|
|
||||||
case StatusClosed:
|
|
||||||
// Timer exits.
|
|
||||||
return
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
case StatusStopped:
|
||||||
|
// Do nothing.
|
||||||
|
|
||||||
|
case StatusClosed:
|
||||||
|
// Timer exits.
|
||||||
|
return
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}()
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// proceed function proceeds the timer job checking and running logic.
|
// proceed function proceeds the timer job checking and running logic.
|
||||||
func (t *Timer) proceed(currentTimerTicks int64) {
|
func (t *Timer) proceed(currentTimerTicks int64) {
|
||||||
var (
|
var value interface{}
|
||||||
value interface{}
|
|
||||||
)
|
|
||||||
for {
|
for {
|
||||||
value = t.queue.Pop()
|
value = t.queue.Pop()
|
||||||
if value == nil {
|
if value == nil {
|
||||||
|
Loading…
x
Reference in New Issue
Block a user