-
Notifications
You must be signed in to change notification settings - Fork 89
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
[cmds][scd/store] Add 'db-evictor' command enabling deletion of expir…
…ed op intents and subscriptions
- Loading branch information
Showing
6 changed files
with
244 additions
and
2 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,71 @@ | ||
# db-evictor | ||
|
||
CLI tool that lists and deletes expired entities in the DSS store. | ||
At the time of writing this README, the entities supported by this tool are: | ||
- SCD operational intents; | ||
- SCD subscriptions. | ||
|
||
The usage of this tool is potentially dangerous: inputting wrong parameters may result in loss of data. | ||
As such it is strongly recommended to always review and validate the list of entities identified as expired, and to | ||
ensure that a backup of the data is available before deleting anything using the `-delete` flag | ||
|
||
## Usage | ||
Extract from running `db-evictor --help`: | ||
``` | ||
Usage of db-evictor: | ||
-cockroach_application_name string | ||
application name for tagging the connection to cockroach (default "dss") | ||
-cockroach_db_name string | ||
application name for tagging the connection to cockroach (default "dss") | ||
-cockroach_host string | ||
cockroach host to connect to | ||
-cockroach_max_retries int | ||
maximum number of attempts to retry a query in case of contention, default is 100 (default 100) | ||
-cockroach_port int | ||
cockroach port to connect to (default 26257) | ||
-cockroach_ssl_dir string | ||
directory to ssl certificates. Must contain files: ca.crt, client.<user>.crt, client.<user>.key | ||
-cockroach_ssl_mode string | ||
cockroach sslmode (default "disable") | ||
-cockroach_user string | ||
cockroach user to authenticate as (default "root") | ||
-delete | ||
set this flag to true to delete the expired entities | ||
-max_conn_idle_secs int | ||
maximum amount of time in seconds a connection may be idle, default is 30 seconds (default 30) | ||
-max_open_conns int | ||
maximum number of open connections to the database, default is 4 (default 4) | ||
-op_intents | ||
set this flag to true to list expired operational intents (default true) | ||
-scd_subs | ||
set this flag to true to list expired SCD subscriptions (default true) | ||
-ttl duration | ||
time-to-live duration used for determining expiration (default 2688h0m0s) | ||
``` | ||
|
||
Do note: | ||
- by default expired entities are only listed, not deleted, the flag `-delete` is required for deleting entities; | ||
- expiration of entities is preferably determined through their end times, however when they do not have end times, the last update times are used; | ||
- the flag `-ttl` accepts durations formatted as [Go `time.Duration` strings](https://pkg.go.dev/time#ParseDuration), e.g. `24h`; | ||
- the CockroachDB cluster connection flags are the same than [the `core-service` command](../core-service/README.md). | ||
|
||
## Examples | ||
The following examples assume a running DSS deployed locally through [the `run_locally.sh` script](../../build/dev/standalone_instance.md). | ||
|
||
### List all entities older than 1 week | ||
```shell | ||
docker compose -f docker-compose_dss.yaml -p dss_sandbox exec local-dss-core-service db-evictor \ | ||
-cockroach_host=local-dss-crdb -ttl=168h | ||
``` | ||
|
||
### List operational intents older than 1 week | ||
```shell | ||
docker compose -f docker-compose_dss.yaml -p dss_sandbox exec local-dss-core-service db-evictor \ | ||
-cockroach_host=local-dss-crdb -ttl=168h -op_intents=true -scd_subs=false | ||
``` | ||
|
||
### Delete all entities older than 30 days | ||
```shell | ||
docker compose -f docker-compose_dss.yaml -p dss_sandbox exec local-dss-core-service db-evictor \ | ||
-cockroach_host=local-dss-crdb -ttl=720h -delete | ||
``` |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,107 @@ | ||
package main | ||
|
||
import ( | ||
"context" | ||
"flag" | ||
"fmt" | ||
"log" | ||
"time" | ||
|
||
"github.com/interuss/dss/pkg/cockroach" | ||
"github.com/interuss/dss/pkg/cockroach/flags" | ||
dssmodels "github.com/interuss/dss/pkg/models" | ||
scdmodels "github.com/interuss/dss/pkg/scd/models" | ||
"github.com/interuss/dss/pkg/scd/repos" | ||
scdc "github.com/interuss/dss/pkg/scd/store/cockroach" | ||
) | ||
|
||
var ( | ||
listOpIntents = flag.Bool("op_intents", true, "set this flag to true to list expired operational intents") | ||
listScdSubs = flag.Bool("scd_subs", true, "set this flag to true to list expired SCD subscriptions") | ||
ttl = flag.Duration("ttl", time.Hour*24*112, "time-to-live duration used for determining expiration") | ||
deleteExpired = flag.Bool("delete", false, "set this flag to true to delete the expired entities") | ||
) | ||
|
||
func main() { | ||
flag.Parse() | ||
|
||
var ( | ||
ctx = context.Background() | ||
threshold = time.Now().Add(-*ttl) | ||
) | ||
|
||
connectParameters := flags.ConnectParameters() | ||
connectParameters.ApplicationName = "db-evictor" | ||
connectParameters.DBName = scdc.DatabaseName | ||
scdCrdb, err := cockroach.Dial(ctx, connectParameters) | ||
if err != nil { | ||
log.Panicf("Failed to connect to database with %+v: %v", connectParameters, err) | ||
} | ||
|
||
scdStore, err := scdc.NewStore(ctx, scdCrdb) | ||
if err != nil { | ||
log.Panicf("Failed to create strategic conflict detection store with %+v: %v", connectParameters, err) | ||
} | ||
|
||
var ( | ||
expiredOpIntents []*scdmodels.OperationalIntent | ||
expiredSubs []*scdmodels.Subscription | ||
) | ||
action := func(ctx context.Context, r repos.Repository) (err error) { | ||
if *listOpIntents { | ||
expiredOpIntents, err = r.ListExpiredOperationalIntents(ctx, threshold) | ||
if err != nil { | ||
return fmt.Errorf("listing expired operational intents: %w", err) | ||
} | ||
if *deleteExpired { | ||
for _, opIntent := range expiredOpIntents { | ||
if err = r.DeleteOperationalIntent(ctx, opIntent.ID); err != nil { | ||
return fmt.Errorf("deleting expired operational intents: %w", err) | ||
} | ||
} | ||
} | ||
} | ||
|
||
if *listScdSubs { | ||
expiredSubs, err = r.ListExpiredSubscriptions(ctx, threshold) | ||
if err != nil { | ||
return fmt.Errorf("listing expired subscriptions: %w", err) | ||
} | ||
if *deleteExpired { | ||
for _, sub := range expiredSubs { | ||
if err = r.DeleteSubscription(ctx, sub.ID); err != nil { | ||
return fmt.Errorf("deleting expired subscriptions: %w", err) | ||
} | ||
} | ||
} | ||
} | ||
|
||
return nil | ||
} | ||
if err = scdStore.Transact(ctx, action); err != nil { | ||
log.Panicf("Failed to execute CRDB transaction: %v", err) | ||
} | ||
|
||
for _, opIntent := range expiredOpIntents { | ||
logExpiredEntity("operational intent", opIntent.ID, threshold, *deleteExpired, opIntent.EndTime != nil) | ||
} | ||
for _, sub := range expiredSubs { | ||
logExpiredEntity("subscription", sub.ID, threshold, *deleteExpired, sub.EndTime != nil) | ||
} | ||
if len(expiredOpIntents) == 0 && len(expiredSubs) == 0 { | ||
log.Printf("no entity older than %s found", threshold.String()) | ||
} | ||
} | ||
|
||
func logExpiredEntity(entity string, entityID dssmodels.ID, threshold time.Time, deleted, hasEndTime bool) { | ||
logMsg := "found" | ||
if deleted { | ||
logMsg = "deleted" | ||
} | ||
|
||
expMsg := "last update before %s (missing end time)" | ||
if hasEndTime { | ||
expMsg = "end time before %s" | ||
} | ||
log.Printf("%s %s %s; expired due to %s", logMsg, entity, entityID.String(), fmt.Sprintf(expMsg, threshold.String())) | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters