From 0c4e2665bae89b682783074df4c9203a10874578 Mon Sep 17 00:00:00 2001 From: Dmitrii Stepanov Date: Tue, 19 Mar 2024 17:18:22 +0300 Subject: [PATCH] [#131] registry: Allow to create cycled/forward selectors Signed-off-by: Dmitrii Stepanov --- cmd/xk6-registry-exporter/root.go | 2 +- internal/registry/obj_selector.go | 9 +++++++-- internal/registry/registry.go | 10 +++++++++- scenarios/grpc.js | 2 +- scenarios/grpc_car.js | 2 +- scenarios/s3.js | 2 +- scenarios/s3_car.js | 2 +- scenarios/s3local.js | 2 +- 8 files changed, 22 insertions(+), 9 deletions(-) diff --git a/cmd/xk6-registry-exporter/root.go b/cmd/xk6-registry-exporter/root.go index e0488dc..b7d8227 100644 --- a/cmd/xk6-registry-exporter/root.go +++ b/cmd/xk6-registry-exporter/root.go @@ -78,7 +78,7 @@ func rootCmdRun(cmd *cobra.Command, args []string) error { } objRegistry := registry.NewObjRegistry(cmd.Context(), args[0]) - objSelector := registry.NewObjSelector(objRegistry, 0, ®istry.ObjFilter{ + objSelector := registry.NewObjSelector(objRegistry, 0, false, ®istry.ObjFilter{ Status: status, Age: age, }) diff --git a/internal/registry/obj_selector.go b/internal/registry/obj_selector.go index 5476a27..02624cc 100644 --- a/internal/registry/obj_selector.go +++ b/internal/registry/obj_selector.go @@ -20,6 +20,7 @@ type ObjSelector struct { boltDB *bbolt.DB filter *ObjFilter cacheSize int + looped bool } // objectSelectCache is the default maximum size of a batch to select from DB. @@ -27,7 +28,7 @@ const objectSelectCache = 1000 // NewObjSelector creates a new instance of object selector that can iterate over // objects in the specified registry. -func NewObjSelector(registry *ObjRegistry, selectionSize int, filter *ObjFilter) *ObjSelector { +func NewObjSelector(registry *ObjRegistry, selectionSize int, looped bool, filter *ObjFilter) *ObjSelector { if selectionSize <= 0 { selectionSize = objectSelectCache } @@ -40,6 +41,7 @@ func NewObjSelector(registry *ObjRegistry, selectionSize int, filter *ObjFilter) filter: filter, objChan: make(chan *ObjectInfo, selectionSize*2), cacheSize: selectionSize, + looped: looped, } go objSelector.selectLoop() @@ -160,13 +162,16 @@ func (o *ObjSelector) selectLoop() { } } - if len(cache) != o.cacheSize { + if !o.looped && len(cache) != o.cacheSize { // no more objects, wait a little; the logic could be improved. select { case <-time.After(time.Second): case <-o.ctx.Done(): return } + } + + if o.looped && len(cache) != o.cacheSize { lastID = 0 } diff --git a/internal/registry/registry.go b/internal/registry/registry.go index dcab0de..a635424 100644 --- a/internal/registry/registry.go +++ b/internal/registry/registry.go @@ -75,6 +75,14 @@ func (r *Registry) open(dbFilePath string) *ObjRegistry { } func (r *Registry) GetSelector(dbFilePath string, name string, cacheSize int, filter map[string]string) *ObjSelector { + return r.getSelectorInternal(dbFilePath, name, cacheSize, false, filter) +} + +func (r *Registry) GetLoopedSelector(dbFilePath string, name string, cacheSize int, filter map[string]string) *ObjSelector { + return r.getSelectorInternal(dbFilePath, name, cacheSize, true, filter) +} + +func (r *Registry) getSelectorInternal(dbFilePath string, name string, cacheSize int, looped bool, filter map[string]string) *ObjSelector { objFilter, err := parseFilter(filter) if err != nil { panic(err) @@ -86,7 +94,7 @@ func (r *Registry) GetSelector(dbFilePath string, name string, cacheSize int, fi selector := r.root.selectors[name] if selector == nil { registry := r.open(dbFilePath) - selector = NewObjSelector(registry, cacheSize, objFilter) + selector = NewObjSelector(registry, cacheSize, looped, objFilter) r.root.selectors[name] = selector } else if !reflect.DeepEqual(selector.filter, objFilter) { panic(fmt.Sprintf("selector %s already has been created with a different filter", name)) diff --git a/scenarios/grpc.js b/scenarios/grpc.js index 65d11d9..75d9622 100644 --- a/scenarios/grpc.js +++ b/scenarios/grpc.js @@ -57,7 +57,7 @@ if (registry_enabled && delete_age) { const read_age = __ENV.READ_AGE ? parseInt(__ENV.READ_AGE) : 10; let obj_to_read_selector = undefined; if (registry_enabled) { - obj_to_read_selector = registry.getSelector( + obj_to_read_selector = registry.getLoopedSelector( __ENV.REGISTRY_FILE, "obj_to_read", __ENV.SELECTION_SIZE ? parseInt(__ENV.SELECTION_SIZE) : 0, diff --git a/scenarios/grpc_car.js b/scenarios/grpc_car.js index c69ee7e..752e76a 100644 --- a/scenarios/grpc_car.js +++ b/scenarios/grpc_car.js @@ -57,7 +57,7 @@ if (registry_enabled && delete_age) { const read_age = __ENV.READ_AGE ? parseInt(__ENV.READ_AGE) : 10; let obj_to_read_selector = undefined; if (registry_enabled) { - obj_to_read_selector = registry.getSelector( + obj_to_read_selector = registry.getLoopedSelector( __ENV.REGISTRY_FILE, "obj_to_read", __ENV.SELECTION_SIZE ? parseInt(__ENV.SELECTION_SIZE) : 0, diff --git a/scenarios/s3.js b/scenarios/s3.js index 51eed76..7da53b9 100644 --- a/scenarios/s3.js +++ b/scenarios/s3.js @@ -56,7 +56,7 @@ if (registry_enabled && delete_age) { const read_age = __ENV.READ_AGE ? parseInt(__ENV.READ_AGE) : 10; let obj_to_read_selector = undefined; if (registry_enabled) { - obj_to_read_selector = registry.getSelector( + obj_to_read_selector = registry.getLoopedSelector( __ENV.REGISTRY_FILE, "obj_to_read", __ENV.SELECTION_SIZE ? parseInt(__ENV.SELECTION_SIZE) : 0, diff --git a/scenarios/s3_car.js b/scenarios/s3_car.js index 74eba7a..6666eee 100644 --- a/scenarios/s3_car.js +++ b/scenarios/s3_car.js @@ -56,7 +56,7 @@ if (registry_enabled && delete_age) { const read_age = __ENV.READ_AGE ? parseInt(__ENV.READ_AGE) : 10; let obj_to_read_selector = undefined; if (registry_enabled) { - obj_to_read_selector = registry.getSelector( + obj_to_read_selector = registry.getLoopedSelector( __ENV.REGISTRY_FILE, "obj_to_read", __ENV.SELECTION_SIZE ? parseInt(__ENV.SELECTION_SIZE) : 0, diff --git a/scenarios/s3local.js b/scenarios/s3local.js index 17dee46..34b7562 100644 --- a/scenarios/s3local.js +++ b/scenarios/s3local.js @@ -54,7 +54,7 @@ const obj_registry = registry_enabled ? registry.open(__ENV.REGISTRY_FILE) : und let obj_to_read_selector = undefined; if (registry_enabled) { - obj_to_read_selector = registry.getSelector( + obj_to_read_selector = registry.getLoopedSelector( __ENV.REGISTRY_FILE, "obj_to_read", __ENV.SELECTION_SIZE ? parseInt(__ENV.SELECTION_SIZE) : 0,