feat(state): introduce SQLite based state store
Allow modules to keep a state of their latest execution and skp if not necessary
This commit is contained in:
parent
7c60d0f951
commit
fee941a0e4
|
@ -19,7 +19,7 @@ build "go_build" "linux" {
|
|||
|
||||
ldflags = [
|
||||
"-w -s",
|
||||
"-X 'code.icb4dc0.de/buildr/buildr/cmd.CurrentVersion=v0.1.0'"
|
||||
"-X 'code.icb4dc0.de/buildr/buildr/cmd.CurrentVersion=${buildr.repo.git.tag == "" ? buildr.repo.git.branch : buildr.repo.git.tag}'"
|
||||
]
|
||||
|
||||
environment = {
|
||||
|
|
1
.gitignore
vendored
1
.gitignore
vendored
|
@ -68,6 +68,7 @@ fabric.properties
|
|||
.buildr/out
|
||||
.buildr/logs
|
||||
.buildr/.vaultpw
|
||||
.buildr/state.sqlite
|
||||
|
||||
# protobuf
|
||||
|
||||
|
|
|
@ -18,8 +18,9 @@ message Buildr {
|
|||
|
||||
message ModuleReference {
|
||||
string task_id = 1;
|
||||
string module_type = 2;
|
||||
string module_name = 3;
|
||||
string module_category = 2;
|
||||
string module_type = 3;
|
||||
string module_name = 4;
|
||||
}
|
||||
|
||||
message StartTaskRequest {
|
||||
|
@ -56,21 +57,52 @@ message TaskLog {
|
|||
repeated LogAttribute attributes = 4;
|
||||
}
|
||||
|
||||
message ExecuteTaskRequest {
|
||||
message SetState {
|
||||
bytes key = 1;
|
||||
bytes data = 2;
|
||||
}
|
||||
|
||||
message GetStateRequest {
|
||||
bytes key = 1;
|
||||
}
|
||||
|
||||
message GetStateResponse {
|
||||
bytes key = 1;
|
||||
bytes data = 2;
|
||||
}
|
||||
|
||||
message Result {
|
||||
string error = 1;
|
||||
}
|
||||
|
||||
message ExecutionClientMessage {
|
||||
oneof meta {
|
||||
bytes message_id = 1;
|
||||
bytes replies_to = 2;
|
||||
}
|
||||
|
||||
oneof envelope {
|
||||
StartTaskRequest start_task = 11;
|
||||
GetStateResponse get_state = 12;
|
||||
Result error = 13;
|
||||
}
|
||||
}
|
||||
|
||||
message ExecuteTaskResponse {
|
||||
bytes replies_to = 1;
|
||||
message ExecutionServerMessage {
|
||||
oneof meta {
|
||||
bytes message_id = 1;
|
||||
bytes replies_to = 2;
|
||||
}
|
||||
|
||||
oneof envelope {
|
||||
TaskResult task_result = 11;
|
||||
TaskLog task_log = 12;
|
||||
TaskOutput task_output = 13;
|
||||
GetStateRequest get_state = 14;
|
||||
SetState set_state = 15;
|
||||
}
|
||||
}
|
||||
|
||||
service ExecutorService {
|
||||
rpc ExecuteTask(stream ExecuteTaskRequest) returns (stream ExecuteTaskResponse);
|
||||
rpc ExecuteStream(stream ExecutionClientMessage) returns (stream ExecutionServerMessage);
|
||||
}
|
35
go.mod
35
go.mod
|
@ -4,6 +4,7 @@ go 1.20
|
|||
|
||||
require (
|
||||
code.icb4dc0.de/prskr/go-pwgen v0.0.0-20230427131724-8ef26fd9749e
|
||||
entgo.io/ent v0.12.3
|
||||
github.com/docker/docker v23.0.5+incompatible
|
||||
github.com/docker/go-connections v0.4.0
|
||||
github.com/fsnotify/fsnotify v1.6.0
|
||||
|
@ -20,20 +21,20 @@ require (
|
|||
github.com/klauspost/pgzip v1.2.5
|
||||
github.com/opencontainers/image-spec v1.1.0-rc3
|
||||
github.com/spf13/cobra v1.7.0
|
||||
github.com/spf13/pflag v1.0.5
|
||||
github.com/spf13/viper v1.15.0
|
||||
github.com/stretchr/testify v1.8.1
|
||||
github.com/stretchr/testify v1.8.2
|
||||
github.com/zclconf/go-cty v1.13.1
|
||||
golang.org/x/crypto v0.8.0
|
||||
golang.org/x/exp v0.0.0-20230425010034-47ecfdc1ba53
|
||||
golang.org/x/sync v0.1.0
|
||||
google.golang.org/grpc v1.54.0
|
||||
google.golang.org/protobuf v1.30.0
|
||||
modernc.org/sqlite v1.22.1
|
||||
)
|
||||
|
||||
replace golang.org/x/net => golang.org/x/net v0.8.0
|
||||
|
||||
require (
|
||||
ariga.io/atlas v0.10.2-0.20230427182402-87a07dfb83bf // indirect
|
||||
github.com/Microsoft/go-winio v0.6.1 // indirect
|
||||
github.com/ProtonMail/go-crypto v0.0.0-20230426101702-58e86b294756 // indirect
|
||||
github.com/acomagu/bufpipe v1.0.4 // indirect
|
||||
|
@ -48,25 +49,25 @@ require (
|
|||
github.com/docker/distribution v2.8.1+incompatible // indirect
|
||||
github.com/docker/docker-credential-helpers v0.7.0 // indirect
|
||||
github.com/docker/go-units v0.5.0 // indirect
|
||||
github.com/dustin/go-humanize v1.0.1 // indirect
|
||||
github.com/emirpasic/gods v1.18.1 // indirect
|
||||
github.com/go-git/gcfg v1.5.0 // indirect
|
||||
github.com/go-git/go-billy/v5 v5.4.1 // indirect
|
||||
github.com/go-openapi/inflect v0.19.0 // indirect
|
||||
github.com/gogo/protobuf v1.3.2 // indirect
|
||||
github.com/golang/protobuf v1.5.3 // indirect
|
||||
github.com/google/go-cmp v0.5.9 // indirect
|
||||
github.com/google/go-querystring v1.1.0 // indirect
|
||||
github.com/hashicorp/hcl v1.0.0 // indirect
|
||||
github.com/imdario/mergo v0.3.15 // indirect
|
||||
github.com/inconshreveable/mousetrap v1.1.0 // indirect
|
||||
github.com/jbenet/go-context v0.0.0-20150711004518-d14ea06fba99 // indirect
|
||||
github.com/kballard/go-shellquote v0.0.0-20180428030007-95032a82bc51 // indirect
|
||||
github.com/kevinburke/ssh_config v1.2.0 // indirect
|
||||
github.com/magiconair/properties v1.8.7 // indirect
|
||||
github.com/mattn/go-isatty v0.0.18 // indirect
|
||||
github.com/matttproud/golang_protobuf_extensions v1.0.4 // indirect
|
||||
github.com/mitchellh/go-homedir v1.1.0 // indirect
|
||||
github.com/mitchellh/go-wordwrap v1.0.1 // indirect
|
||||
github.com/mitchellh/mapstructure v1.5.0 // indirect
|
||||
github.com/opencontainers/go-digest v1.0.0 // indirect
|
||||
github.com/pelletier/go-toml/v2 v2.0.7 // indirect
|
||||
github.com/pjbgf/sha1cd v0.3.0 // indirect
|
||||
github.com/pkg/errors v0.9.1 // indirect
|
||||
github.com/pmezard/go-difflib v1.0.0 // indirect
|
||||
|
@ -74,13 +75,12 @@ require (
|
|||
github.com/prometheus/client_model v0.3.0 // indirect
|
||||
github.com/prometheus/common v0.42.0 // indirect
|
||||
github.com/prometheus/procfs v0.9.0 // indirect
|
||||
github.com/remyoudompheng/bigfft v0.0.0-20230129092748-24d4a6f8daec // indirect
|
||||
github.com/rogpeppe/go-internal v1.6.1 // indirect
|
||||
github.com/sergi/go-diff v1.3.1 // indirect
|
||||
github.com/sirupsen/logrus v1.9.0 // indirect
|
||||
github.com/skeema/knownhosts v1.1.0 // indirect
|
||||
github.com/spf13/afero v1.9.5 // indirect
|
||||
github.com/spf13/cast v1.5.0 // indirect
|
||||
github.com/spf13/jwalterweatherman v1.1.0 // indirect
|
||||
github.com/subosito/gotenv v1.4.2 // indirect
|
||||
github.com/spf13/pflag v1.0.5 // indirect
|
||||
github.com/vbatts/tar-split v0.11.3 // indirect
|
||||
github.com/xanzy/ssh-agent v0.3.3 // indirect
|
||||
golang.org/x/mod v0.10.0 // indirect
|
||||
|
@ -88,10 +88,19 @@ require (
|
|||
golang.org/x/oauth2 v0.7.0 // indirect
|
||||
golang.org/x/sys v0.7.0 // indirect
|
||||
golang.org/x/text v0.9.0 // indirect
|
||||
golang.org/x/tools v0.8.0 // indirect
|
||||
golang.org/x/time v0.1.0 // indirect
|
||||
golang.org/x/tools v0.8.1-0.20230428195545-5283a0178901 // indirect
|
||||
google.golang.org/appengine v1.6.7 // indirect
|
||||
google.golang.org/genproto v0.0.0-20230410155749-daa745c078e1 // indirect
|
||||
gopkg.in/ini.v1 v1.67.0 // indirect
|
||||
gopkg.in/warnings.v0 v0.1.2 // indirect
|
||||
gopkg.in/yaml.v3 v3.0.1 // indirect
|
||||
lukechampine.com/uint128 v1.3.0 // indirect
|
||||
modernc.org/cc/v3 v3.40.0 // indirect
|
||||
modernc.org/ccgo/v3 v3.16.13 // indirect
|
||||
modernc.org/libc v1.22.5 // indirect
|
||||
modernc.org/mathutil v1.5.0 // indirect
|
||||
modernc.org/memory v1.5.0 // indirect
|
||||
modernc.org/opt v0.1.3 // indirect
|
||||
modernc.org/strutil v1.1.3 // indirect
|
||||
modernc.org/token v1.1.0 // indirect
|
||||
)
|
||||
|
|
385
go.sum
385
go.sum
|
@ -1,47 +1,14 @@
|
|||
ariga.io/atlas v0.10.2-0.20230427182402-87a07dfb83bf h1:Tq2DRB39ZHScIwWACjPKLv5oEErv7zv6PBb5RTz5CKA=
|
||||
ariga.io/atlas v0.10.2-0.20230427182402-87a07dfb83bf/go.mod h1:+TR129FJZ5Lvzms6dvCeGWh1yR6hMvmXBhug4hrNIGk=
|
||||
cloud.google.com/go v0.26.0/go.mod h1:aQUYkXzVsufM+DwF1aE+0xfcU+56JwCaLick0ClmMTw=
|
||||
cloud.google.com/go v0.34.0/go.mod h1:aQUYkXzVsufM+DwF1aE+0xfcU+56JwCaLick0ClmMTw=
|
||||
cloud.google.com/go v0.38.0/go.mod h1:990N+gfupTy94rShfmMCWGDn0LpTmnzTp2qbd1dvSRU=
|
||||
cloud.google.com/go v0.44.1/go.mod h1:iSa0KzasP4Uvy3f1mN/7PiObzGgflwredwwASm/v6AU=
|
||||
cloud.google.com/go v0.44.2/go.mod h1:60680Gw3Yr4ikxnPRS/oxxkBccT6SA1yMk63TGekxKY=
|
||||
cloud.google.com/go v0.44.3/go.mod h1:60680Gw3Yr4ikxnPRS/oxxkBccT6SA1yMk63TGekxKY=
|
||||
cloud.google.com/go v0.45.1/go.mod h1:RpBamKRgapWJb87xiFSdk4g1CME7QZg3uwTez+TSTjc=
|
||||
cloud.google.com/go v0.46.3/go.mod h1:a6bKKbmY7er1mI7TEI4lsAkts/mkhTSZK8w33B4RAg0=
|
||||
cloud.google.com/go v0.50.0/go.mod h1:r9sluTvynVuxRIOHXQEHMFffphuXHOMZMycpNR5e6To=
|
||||
cloud.google.com/go v0.52.0/go.mod h1:pXajvRH/6o3+F9jDHZWQ5PbGhn+o8w9qiu/CffaVdO4=
|
||||
cloud.google.com/go v0.53.0/go.mod h1:fp/UouUEsRkN6ryDKNW/Upv/JBKnv6WDthjR6+vze6M=
|
||||
cloud.google.com/go v0.54.0/go.mod h1:1rq2OEkV3YMf6n/9ZvGWI3GWw0VoqH/1x2nd8Is/bPc=
|
||||
cloud.google.com/go v0.56.0/go.mod h1:jr7tqZxxKOVYizybht9+26Z/gUq7tiRzu+ACVAMbKVk=
|
||||
cloud.google.com/go v0.57.0/go.mod h1:oXiQ6Rzq3RAkkY7N6t3TcE6jE+CIBBbA36lwQ1JyzZs=
|
||||
cloud.google.com/go v0.62.0/go.mod h1:jmCYTdRCQuc1PHIIJ/maLInMho30T/Y0M4hTdTShOYc=
|
||||
cloud.google.com/go v0.65.0/go.mod h1:O5N8zS7uWy9vkA9vayVHs65eM1ubvY4h553ofrNHObY=
|
||||
cloud.google.com/go v0.72.0/go.mod h1:M+5Vjvlc2wnp6tjzE102Dw08nGShTscUx2nZMufOKPI=
|
||||
cloud.google.com/go v0.74.0/go.mod h1:VV1xSbzvo+9QJOxLDaJfTjx5e+MePCpCWwvftOeQmWk=
|
||||
cloud.google.com/go v0.75.0/go.mod h1:VGuuCn7PG0dwsd5XPVm2Mm3wlh3EL55/79EKB6hlPTY=
|
||||
cloud.google.com/go/bigquery v1.0.1/go.mod h1:i/xbL2UlR5RvWAURpBYZTtm/cXjCha9lbfbpx4poX+o=
|
||||
cloud.google.com/go/bigquery v1.3.0/go.mod h1:PjpwJnslEMmckchkHFfq+HTD2DmtT67aNFKH1/VBDHE=
|
||||
cloud.google.com/go/bigquery v1.4.0/go.mod h1:S8dzgnTigyfTmLBfrtrhyYhwRxG72rYxvftPBK2Dvzc=
|
||||
cloud.google.com/go/bigquery v1.5.0/go.mod h1:snEHRnqQbz117VIFhE8bmtwIDY80NLUZUMb4Nv6dBIg=
|
||||
cloud.google.com/go/bigquery v1.7.0/go.mod h1://okPTzCYNXSlb24MZs83e2Do+h+VXtc4gLoIoXIAPc=
|
||||
cloud.google.com/go/bigquery v1.8.0/go.mod h1:J5hqkt3O0uAFnINi6JXValWIb1v0goeZM77hZzJN/fQ=
|
||||
cloud.google.com/go/datastore v1.0.0/go.mod h1:LXYbyblFSglQ5pkeyhO+Qmw7ukd3C+pD7TKLgZqpHYE=
|
||||
cloud.google.com/go/datastore v1.1.0/go.mod h1:umbIZjpQpHh4hmRpGhH4tLFup+FVzqBi1b3c64qFpCk=
|
||||
cloud.google.com/go/pubsub v1.0.1/go.mod h1:R0Gpsv3s54REJCy4fxDixWD93lHJMoZTyQ2kNxGRt3I=
|
||||
cloud.google.com/go/pubsub v1.1.0/go.mod h1:EwwdRX2sKPjnvnqCa270oGRyludottCI76h+R3AArQw=
|
||||
cloud.google.com/go/pubsub v1.2.0/go.mod h1:jhfEVHT8odbXTkndysNHCcx0awwzvfOlguIAii9o8iA=
|
||||
cloud.google.com/go/pubsub v1.3.1/go.mod h1:i+ucay31+CNRpDW4Lu78I4xXG+O1r/MAHgjpRVR+TSU=
|
||||
cloud.google.com/go/storage v1.0.0/go.mod h1:IhtSnM/ZTZV8YYJWCY8RULGVqBDmpoyjwiyrjsg+URw=
|
||||
cloud.google.com/go/storage v1.5.0/go.mod h1:tpKbwo567HUNpVclU5sGELwQWBDZ8gh0ZeosJ0Rtdos=
|
||||
cloud.google.com/go/storage v1.6.0/go.mod h1:N7U0C8pVQ/+NIKOBQyamJIeKQKkZ+mxpohlUTyfDhBk=
|
||||
cloud.google.com/go/storage v1.8.0/go.mod h1:Wv1Oy7z6Yz3DshWRJFhqM/UCfaWIRTdp0RXyy7KQOVs=
|
||||
cloud.google.com/go/storage v1.10.0/go.mod h1:FLPqc6j+Ki4BU591ie1oL6qBQGu2Bl/tZ9ullr3+Kg0=
|
||||
cloud.google.com/go/storage v1.14.0/go.mod h1:GrKmX003DSIwi9o29oFT7YDnHYwZoctc3fOKtUw0Xmo=
|
||||
code.icb4dc0.de/prskr/go-pwgen v0.0.0-20230427131724-8ef26fd9749e h1:N+3hdfeHRf/ndLjiZoet6h+9eZ2Zr5O/Ia5G7b3oQXU=
|
||||
code.icb4dc0.de/prskr/go-pwgen v0.0.0-20230427131724-8ef26fd9749e/go.mod h1:1MCCxqZsOgfQzV8AR2ZAVRI36MEGF+teFium6Pf2HHU=
|
||||
dmitri.shuralyov.com/gpu/mtl v0.0.0-20190408044501-666a987793e9/go.mod h1:H6x//7gZCb22OMCxBHrMx7a5I7Hp++hsVxbQ4BYO7hU=
|
||||
entgo.io/ent v0.12.3 h1:N5lO2EOrHpCH5HYfiMOCHYbo+oh5M8GjT0/cx5x6xkk=
|
||||
entgo.io/ent v0.12.3/go.mod h1:AigGGx+tbrBBYHAzGOg8ND661E5cxx1Uiu5o/otJ6Yg=
|
||||
github.com/Azure/go-ansiterm v0.0.0-20210617225240-d185dfc1b5a1 h1:UQHMgLO+TxOElx5B5HZ4hJQsoJ/PvUvKRhJHDQXO8P8=
|
||||
github.com/BurntSushi/toml v0.3.1/go.mod h1:xHWCNGjB5oqiDr8zfno3MHue2Ht5sIBksp03qcyfWMU=
|
||||
github.com/BurntSushi/toml v1.2.1/go.mod h1:CxXYINrC8qIiEnFrOxCa7Jy5BFHlXnUU2pbicEuybxQ=
|
||||
github.com/BurntSushi/xgb v0.0.0-20160522181843-27f122750802/go.mod h1:IVnqGOEym/WlBOVXweHU+Q+/VP0lqqI8lqeDx9IjBqo=
|
||||
github.com/DATA-DOG/go-sqlmock v1.5.0 h1:Shsta01QNfFxHCfpW6YH2STWB0MudeXXEWMr20OEh60=
|
||||
github.com/Microsoft/go-winio v0.5.2/go.mod h1:WpS1mjBmmwHBEWmogvA2mj8546UReBk4v8QkMxJ6pZY=
|
||||
github.com/Microsoft/go-winio v0.6.1 h1:9/kr64B9VUZrLm5YYwbGtUJnMgqWVOdUAXu6Migciow=
|
||||
github.com/Microsoft/go-winio v0.6.1/go.mod h1:LRdKpFKfdobln8UmuiYcKPot9D2v6svN5+sAH+4kjUM=
|
||||
|
@ -65,16 +32,11 @@ github.com/bwesterb/go-ristretto v1.2.0/go.mod h1:fUIoIZaG73pV5biE2Blr2xEzDoMj7N
|
|||
github.com/census-instrumentation/opencensus-proto v0.2.1/go.mod h1:f6KPmirojxKA12rnyqOA5BBL4O983OfeGPqjHWSTneU=
|
||||
github.com/cespare/xxhash/v2 v2.2.0 h1:DC2CZ1Ep5Y4k3ZQ899DldepgrayRUGE6BBZ/cd9Cj44=
|
||||
github.com/cespare/xxhash/v2 v2.2.0/go.mod h1:VGX0DQ3Q6kWi7AoAeZDth3/j3BFtOZR5XLFGgcrjCOs=
|
||||
github.com/chzyer/logex v1.1.10/go.mod h1:+Ywpsq7O8HXn0nuIou7OrIPyXbp3wmkHB+jjWRnGsAI=
|
||||
github.com/chzyer/readline v0.0.0-20180603132655-2972be24d48e/go.mod h1:nSuG5e5PlCu98SY8svDHJxuZscDgtXS6KTTbou5AhLI=
|
||||
github.com/chzyer/test v0.0.0-20180213035817-a1ea475d72b1/go.mod h1:Q3SI9o4m/ZMnBNeIyt5eFwwo7qiLfzFZmjNmxjkiQlU=
|
||||
github.com/client9/misspell v0.3.4/go.mod h1:qj6jICC3Q7zFZvVWo7KLAzC3yx5G7kyvSDkc90ppPyw=
|
||||
github.com/cloudflare/circl v1.1.0/go.mod h1:prBCrKB9DV4poKZY1l9zBXg2QJY7mvgRvtMxxK7fi4I=
|
||||
github.com/cloudflare/circl v1.3.2 h1:VWp8dY3yH69fdM7lM6A1+NhhVoDu9vqK0jOgmkQHFWk=
|
||||
github.com/cloudflare/circl v1.3.2/go.mod h1:+CauBF6R70Jqcyl8N2hC8pAXYbWkGIezuSbuGLtRhnw=
|
||||
github.com/cncf/udpa/go v0.0.0-20191209042840-269d4d468f6f/go.mod h1:M8M6+tZqaGXZJjfX53e64911xZQV5JYwmTeXPW+k8Sc=
|
||||
github.com/cncf/udpa/go v0.0.0-20200629203442-efcf912fb354/go.mod h1:WmhPx2Nbnhtbo57+VJT5O0JRkEi1Wbu0z5j0R8u5Hbk=
|
||||
github.com/cncf/udpa/go v0.0.0-20201120205902-5459f2c99403/go.mod h1:WmhPx2Nbnhtbo57+VJT5O0JRkEi1Wbu0z5j0R8u5Hbk=
|
||||
github.com/containerd/stargz-snapshotter/estargz v0.14.3 h1:OqlDCK3ZVUO6C3B/5FSkDwbkEETK84kQgEeFwDC+62k=
|
||||
github.com/containerd/stargz-snapshotter/estargz v0.14.3/go.mod h1:KY//uOCIkSuNAHhJogcZtrNHdKrA99/FCCRjE3HD36o=
|
||||
github.com/cpuguy83/go-md2man/v2 v2.0.2/go.mod h1:tgQtvFlXSQOSOSIRvRPT7W67SCa46tRHOmNcaadrF8o=
|
||||
|
@ -94,15 +56,14 @@ github.com/docker/go-connections v0.4.0 h1:El9xVISelRB7BuFusrZozjnkIM5YnzCViNKoh
|
|||
github.com/docker/go-connections v0.4.0/go.mod h1:Gbd7IOopHjR8Iph03tsViu4nIes5XhDvyHbTtUxmeec=
|
||||
github.com/docker/go-units v0.5.0 h1:69rxXcBk27SvSaaxTtLh/8llcHD8vYHT7WSdRZ/jvr4=
|
||||
github.com/docker/go-units v0.5.0/go.mod h1:fgPhTUdO+D/Jk86RDLlptpiXQzgHJF7gydDDbaIK4Dk=
|
||||
github.com/dustin/go-humanize v1.0.1 h1:GzkhY7T5VNhEkwH0PVJgjz+fX1rhBrR7pRT3mDkpeCY=
|
||||
github.com/dustin/go-humanize v1.0.1/go.mod h1:Mu1zIs6XwVuF/gI1OepvI0qD18qycQx+mFykh5fBlto=
|
||||
github.com/emirpasic/gods v1.18.1 h1:FXtiHYKDGKCW2KzwZKx0iC0PQmdlorYgdFG9jPXJ1Bc=
|
||||
github.com/emirpasic/gods v1.18.1/go.mod h1:8tpGGwCnJ5H4r6BWwaV6OrWmMoPhUl5jm/FMNAnJvWQ=
|
||||
github.com/envoyproxy/go-control-plane v0.9.0/go.mod h1:YTl/9mNaCwkRvm6d1a2C3ymFceY/DCBVvsKhRF0iEA4=
|
||||
github.com/envoyproxy/go-control-plane v0.9.1-0.20191026205805-5f8ba28d4473/go.mod h1:YTl/9mNaCwkRvm6d1a2C3ymFceY/DCBVvsKhRF0iEA4=
|
||||
github.com/envoyproxy/go-control-plane v0.9.4/go.mod h1:6rpuAdCZL397s3pYoYcLgu1mIlRU8Am5FuJP05cCM98=
|
||||
github.com/envoyproxy/go-control-plane v0.9.7/go.mod h1:cwu0lG7PUMfa9snN8LXBig5ynNVH9qI8YYLbd1fK2po=
|
||||
github.com/envoyproxy/go-control-plane v0.9.9-0.20201210154907-fd9021fe5dad/go.mod h1:cXg6YxExXjJnVBQHBLXeUAgxn2UodCpnH306RInaBQk=
|
||||
github.com/envoyproxy/protoc-gen-validate v0.1.0/go.mod h1:iSmxcyjqTsJpI2R4NaDN7+kN2VEUnK/pcBlmesArF7c=
|
||||
github.com/frankban/quicktest v1.14.3 h1:FJKSZTDHjyhriyC81FLQ0LY93eSai0ZyR/ZIkd3ZUKE=
|
||||
github.com/fsnotify/fsnotify v1.6.0 h1:n+5WquG0fcWoWp6xPWfHdbskMCQaFnG6PfBrh1Ky4HY=
|
||||
github.com/fsnotify/fsnotify v1.6.0/go.mod h1:sl3t1tCWJFWoRz9R8WJCbQihKKwmorjAbSClcnxKAGw=
|
||||
github.com/gliderlabs/ssh v0.3.5 h1:OcaySEmAQJgyYcArR+gGGTHCyE7nvhEMTlYY+Dp8CpY=
|
||||
|
@ -116,54 +77,26 @@ github.com/go-git/go-git-fixtures/v4 v4.3.1 h1:y5z6dd3qi8Hl+stezc8p3JxDkoTRqMAlK
|
|||
github.com/go-git/go-git-fixtures/v4 v4.3.1/go.mod h1:8LHG1a3SRW71ettAD/jW13h8c6AqjVSeL11RAdgaqpo=
|
||||
github.com/go-git/go-git/v5 v5.6.1 h1:q4ZRqQl4pR/ZJHc1L5CFjGA1a10u76aV1iC+nh+bHsk=
|
||||
github.com/go-git/go-git/v5 v5.6.1/go.mod h1:mvyoL6Unz0PiTQrGQfSfiLFhBH1c1e84ylC2MDs4ee8=
|
||||
github.com/go-gl/glfw v0.0.0-20190409004039-e6da0acd62b1/go.mod h1:vR7hzQXu2zJy9AVAgeJqvqgH9Q5CA+iKCZ2gyEVpxRU=
|
||||
github.com/go-gl/glfw/v3.3/glfw v0.0.0-20191125211704-12ad95a8df72/go.mod h1:tQ2UAYgL5IevRw8kRxooKSPJfGvJ9fJQFa0TUsXzTg8=
|
||||
github.com/go-gl/glfw/v3.3/glfw v0.0.0-20200222043503-6f7a984d4dc4/go.mod h1:tQ2UAYgL5IevRw8kRxooKSPJfGvJ9fJQFa0TUsXzTg8=
|
||||
github.com/go-kit/log v0.1.0/go.mod h1:zbhenjAZHb184qTLMA9ZjW7ThYL0H2mk7Q6pNt4vbaY=
|
||||
github.com/go-logfmt/logfmt v0.5.0/go.mod h1:wCYkCAKZfumFQihp8CzCvQ3paCTfi41vtzG1KdI/P7A=
|
||||
github.com/go-openapi/inflect v0.19.0 h1:9jCH9scKIbHeV9m12SmPilScz6krDxKRasNNSNPXu/4=
|
||||
github.com/go-openapi/inflect v0.19.0/go.mod h1:lHpZVlpIQqLyKwJ4N+YSc9hchQy/i12fJykb83CRBH4=
|
||||
github.com/go-stack/stack v1.8.0/go.mod h1:v0f6uXyyMGvRgIKkXu+yp6POWl0qKG85gN/melR3HDY=
|
||||
github.com/go-test/deep v1.0.3 h1:ZrJSEWsXzPOxaZnFteGEfooLba+ju3FYIbOrS+rQd68=
|
||||
github.com/gogo/protobuf v1.3.2 h1:Ov1cvc58UF3b5XjBnZv7+opcTcQFZebYjWzi34vdm4Q=
|
||||
github.com/gogo/protobuf v1.3.2/go.mod h1:P1XiOD3dCwIKUDQYPy72D8LYyHL2YPYrpS2s69NZV8Q=
|
||||
github.com/golang/glog v0.0.0-20160126235308-23def4e6c14b/go.mod h1:SBH7ygxi8pfUlaOkMMuAQtPIUF8ecWP5IEl/CR7VP2Q=
|
||||
github.com/golang/groupcache v0.0.0-20190702054246-869f871628b6/go.mod h1:cIg4eruTrX1D+g88fzRXU5OdNfaM+9IcxsU14FzY7Hc=
|
||||
github.com/golang/groupcache v0.0.0-20191227052852-215e87163ea7/go.mod h1:cIg4eruTrX1D+g88fzRXU5OdNfaM+9IcxsU14FzY7Hc=
|
||||
github.com/golang/groupcache v0.0.0-20200121045136-8c9f03a8e57e/go.mod h1:cIg4eruTrX1D+g88fzRXU5OdNfaM+9IcxsU14FzY7Hc=
|
||||
github.com/golang/mock v1.1.1/go.mod h1:oTYuIxOrZwtPieC+H1uAHpcLFnEyAGVDL/k47Jfbm0A=
|
||||
github.com/golang/mock v1.2.0/go.mod h1:oTYuIxOrZwtPieC+H1uAHpcLFnEyAGVDL/k47Jfbm0A=
|
||||
github.com/golang/mock v1.3.1/go.mod h1:sBzyDLLjw3U8JLTeZvSv8jJB+tU5PVekmnlKIyFUx0Y=
|
||||
github.com/golang/mock v1.4.0/go.mod h1:UOMv5ysSaYNkG+OFQykRIcU/QvvxJf3p21QfJ2Bt3cw=
|
||||
github.com/golang/mock v1.4.1/go.mod h1:UOMv5ysSaYNkG+OFQykRIcU/QvvxJf3p21QfJ2Bt3cw=
|
||||
github.com/golang/mock v1.4.3/go.mod h1:UOMv5ysSaYNkG+OFQykRIcU/QvvxJf3p21QfJ2Bt3cw=
|
||||
github.com/golang/mock v1.4.4/go.mod h1:l3mdAwkq5BuhzHwde/uurv3sEJeZMXNpwsxVWU71h+4=
|
||||
github.com/golang/protobuf v1.2.0/go.mod h1:6lQm79b+lXiMfvg/cZm0SGofjICqVBUtrP5yJMmIC1U=
|
||||
github.com/golang/protobuf v1.3.1/go.mod h1:6lQm79b+lXiMfvg/cZm0SGofjICqVBUtrP5yJMmIC1U=
|
||||
github.com/golang/protobuf v1.3.2/go.mod h1:6lQm79b+lXiMfvg/cZm0SGofjICqVBUtrP5yJMmIC1U=
|
||||
github.com/golang/protobuf v1.3.3/go.mod h1:vzj43D7+SQXF/4pzW/hwtAqwc6iTitCiVSaWz5lYuqw=
|
||||
github.com/golang/protobuf v1.3.4/go.mod h1:vzj43D7+SQXF/4pzW/hwtAqwc6iTitCiVSaWz5lYuqw=
|
||||
github.com/golang/protobuf v1.3.5/go.mod h1:6O5/vntMXwX2lRkT1hjjk0nAC1IDOTvTlVgjlRvqsdk=
|
||||
github.com/golang/protobuf v1.4.0-rc.1/go.mod h1:ceaxUfeHdC40wWswd/P6IGgMaK3YpKi5j83Wpe3EHw8=
|
||||
github.com/golang/protobuf v1.4.0-rc.1.0.20200221234624-67d41d38c208/go.mod h1:xKAWHe0F5eneWXFV3EuXVDTCmh+JuBKY0li0aMyXATA=
|
||||
github.com/golang/protobuf v1.4.0-rc.2/go.mod h1:LlEzMj4AhA7rCAGe4KMBDvJI+AwstrUpVNzEA03Pprs=
|
||||
github.com/golang/protobuf v1.4.0-rc.4.0.20200313231945-b860323f09d0/go.mod h1:WU3c8KckQ9AFe+yFwt9sWVRKCVIyN9cPHBJSNnbL67w=
|
||||
github.com/golang/protobuf v1.4.0/go.mod h1:jodUvKwWbYaEsadDk5Fwe5c77LiNKVO9IDvqG2KuDX0=
|
||||
github.com/golang/protobuf v1.4.1/go.mod h1:U8fpvMrcmy5pZrNK1lt4xCsGvpyWQ/VVv6QDs8UjoX8=
|
||||
github.com/golang/protobuf v1.4.2/go.mod h1:oDoupMAO8OvCJWAcko0GGGIgR6R6ocIYbsSw735rRwI=
|
||||
github.com/golang/protobuf v1.4.3/go.mod h1:oDoupMAO8OvCJWAcko0GGGIgR6R6ocIYbsSw735rRwI=
|
||||
github.com/golang/protobuf v1.5.0/go.mod h1:FsONVRAS9T7sI+LIUmWTfcYkHO4aIWwzhcaSAoJOfIk=
|
||||
github.com/golang/protobuf v1.5.3 h1:KhyjKVUg7Usr/dYsdSqoFveMYd5ko72D+zANwlG1mmg=
|
||||
github.com/golang/protobuf v1.5.3/go.mod h1:XVQd3VNwM+JqD3oG2Ue2ip4fOMUkwXdXDdiuN0vRsmY=
|
||||
github.com/google/btree v0.0.0-20180813153112-4030bb1f1f0c/go.mod h1:lNA+9X1NB3Zf8V7Ke586lFgjr2dZNuvo3lPJSGZ5JPQ=
|
||||
github.com/google/btree v1.0.0/go.mod h1:lNA+9X1NB3Zf8V7Ke586lFgjr2dZNuvo3lPJSGZ5JPQ=
|
||||
github.com/google/go-cmp v0.2.0/go.mod h1:oXzfMopK8JAjlY9xF4vHSVASa0yLyX7SntLO5aqRK0M=
|
||||
github.com/google/go-cmp v0.3.0/go.mod h1:8QqcDgzrUqlUb/G2PQTWiueGozuR1884gddMywk6iLU=
|
||||
github.com/google/go-cmp v0.3.1/go.mod h1:8QqcDgzrUqlUb/G2PQTWiueGozuR1884gddMywk6iLU=
|
||||
github.com/google/go-cmp v0.4.0/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE=
|
||||
github.com/google/go-cmp v0.4.1/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE=
|
||||
github.com/google/go-cmp v0.5.0/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE=
|
||||
github.com/google/go-cmp v0.5.1/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE=
|
||||
github.com/google/go-cmp v0.5.2/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE=
|
||||
github.com/google/go-cmp v0.5.4/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE=
|
||||
github.com/google/go-cmp v0.5.5/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE=
|
||||
github.com/google/go-cmp v0.5.9 h1:O2Tfq5qg4qc4AmwVlvv0oLiVAGB7enBSJ2x2DqQFi38=
|
||||
github.com/google/go-cmp v0.5.9/go.mod h1:17dUlkBOakJ0+DkrSSNjCkIjxS6bF9zb3elmeNGIjoY=
|
||||
|
@ -173,40 +106,17 @@ github.com/google/go-github/v50 v50.2.0 h1:j2FyongEHlO9nxXLc+LP3wuBSVU9mVxfpdYUe
|
|||
github.com/google/go-github/v50 v50.2.0/go.mod h1:VBY8FB6yPIjrtKhozXv4FQupxKLS6H4m6xFZlT43q8Q=
|
||||
github.com/google/go-querystring v1.1.0 h1:AnCroh3fv4ZBgVIf1Iwtovgjaw/GiKJo8M8yD/fhyJ8=
|
||||
github.com/google/go-querystring v1.1.0/go.mod h1:Kcdr2DB4koayq7X8pmAG4sNG59So17icRSOU623lUBU=
|
||||
github.com/google/martian v2.1.0+incompatible/go.mod h1:9I4somxYTbIHy5NJKHRl3wXiIaQGbYVAs8BPL6v8lEs=
|
||||
github.com/google/martian/v3 v3.0.0/go.mod h1:y5Zk1BBys9G+gd6Jrk0W3cC1+ELVxBWuIGO+w/tUAp0=
|
||||
github.com/google/martian/v3 v3.1.0/go.mod h1:y5Zk1BBys9G+gd6Jrk0W3cC1+ELVxBWuIGO+w/tUAp0=
|
||||
github.com/google/pprof v0.0.0-20181206194817-3ea8567a2e57/go.mod h1:zfwlbNMJ+OItoe0UupaVj+oy1omPYYDuagoSzA8v9mc=
|
||||
github.com/google/pprof v0.0.0-20190515194954-54271f7e092f/go.mod h1:zfwlbNMJ+OItoe0UupaVj+oy1omPYYDuagoSzA8v9mc=
|
||||
github.com/google/pprof v0.0.0-20191218002539-d4f498aebedc/go.mod h1:ZgVRPoUq/hfqzAqh7sHMqb3I9Rq5C59dIz2SbBwJ4eM=
|
||||
github.com/google/pprof v0.0.0-20200212024743-f11f1df84d12/go.mod h1:ZgVRPoUq/hfqzAqh7sHMqb3I9Rq5C59dIz2SbBwJ4eM=
|
||||
github.com/google/pprof v0.0.0-20200229191704-1ebb73c60ed3/go.mod h1:ZgVRPoUq/hfqzAqh7sHMqb3I9Rq5C59dIz2SbBwJ4eM=
|
||||
github.com/google/pprof v0.0.0-20200430221834-fc25d7d30c6d/go.mod h1:ZgVRPoUq/hfqzAqh7sHMqb3I9Rq5C59dIz2SbBwJ4eM=
|
||||
github.com/google/pprof v0.0.0-20200708004538-1a94d8640e99/go.mod h1:ZgVRPoUq/hfqzAqh7sHMqb3I9Rq5C59dIz2SbBwJ4eM=
|
||||
github.com/google/pprof v0.0.0-20201023163331-3e6fc7fc9c4c/go.mod h1:kpwsk12EmLew5upagYY7GY0pfYCcupk39gWOCRROcvE=
|
||||
github.com/google/pprof v0.0.0-20201203190320-1bf35d6f28c2/go.mod h1:kpwsk12EmLew5upagYY7GY0pfYCcupk39gWOCRROcvE=
|
||||
github.com/google/pprof v0.0.0-20201218002935-b9804c9f04c2/go.mod h1:kpwsk12EmLew5upagYY7GY0pfYCcupk39gWOCRROcvE=
|
||||
github.com/google/renameio v0.1.0/go.mod h1:KWCgfxg9yswjAJkECMjeO8J8rahYeXnNhOm40UhjYkI=
|
||||
github.com/google/uuid v1.1.2/go.mod h1:TIyPZe4MgqvfeYDBFedMoGGpEw/LqOeaOT+nhxU+yHo=
|
||||
github.com/google/pprof v0.0.0-20221118152302-e6195bd50e26 h1:Xim43kblpZXfIBQsbuBVKCudVG457BR2GZFIz3uw3hQ=
|
||||
github.com/google/uuid v1.3.0 h1:t6JiXgmwXMjEs8VusXIJk2BXHsn+wx8BZdTaoZ5fu7I=
|
||||
github.com/google/uuid v1.3.0/go.mod h1:TIyPZe4MgqvfeYDBFedMoGGpEw/LqOeaOT+nhxU+yHo=
|
||||
github.com/googleapis/gax-go/v2 v2.0.4/go.mod h1:0Wqv26UfaUD9n4G6kQubkQ+KchISgw+vpHVxEJEs9eg=
|
||||
github.com/googleapis/gax-go/v2 v2.0.5/go.mod h1:DWXyrwAJ9X0FpwwEdw+IPEYBICEFu5mhpdKc/us6bOk=
|
||||
github.com/googleapis/google-cloud-go-testing v0.0.0-20200911160855-bcd43fbb19e8/go.mod h1:dvDLG8qkwmyD9a/MJJN3XJcT3xFxOKAvTZGvuZmac9g=
|
||||
github.com/grpc-ecosystem/go-grpc-middleware v1.4.0 h1:UH//fgunKIs4JdUbpDl1VZCDaL56wXCB/5+wF6uHfaI=
|
||||
github.com/grpc-ecosystem/go-grpc-middleware v1.4.0/go.mod h1:g5qyo/la0ALbONm6Vbp88Yd8NsDy6rZz+RcrMPxvld8=
|
||||
github.com/grpc-ecosystem/go-grpc-prometheus v1.2.0 h1:Ovs26xHkKqVztRpIrF/92BcuyuQ/YW4NSIpoGtfXNho=
|
||||
github.com/grpc-ecosystem/go-grpc-prometheus v1.2.0/go.mod h1:8NvIoxWQoOIhqOTXgfV/d3M/q6VIi02HzZEHgUlZvzk=
|
||||
github.com/hanwen/go-fuse/v2 v2.3.0 h1:t5ivNIH2PK+zw4OBul/iJjsoG9K6kXo4nMDoBpciC8A=
|
||||
github.com/hanwen/go-fuse/v2 v2.3.0/go.mod h1:xKwi1cF7nXAOBCXujD5ie0ZKsxc8GGSA1rlMJc+8IJs=
|
||||
github.com/hashicorp/golang-lru v0.5.0/go.mod h1:/m3WP610KZHVQ1SGc6re/UDhFvYD7pJ4Ao+sR/qLZy8=
|
||||
github.com/hashicorp/golang-lru v0.5.1/go.mod h1:/m3WP610KZHVQ1SGc6re/UDhFvYD7pJ4Ao+sR/qLZy8=
|
||||
github.com/hashicorp/hcl v1.0.0 h1:0Anlzjpi4vEasTeNFn2mLJgTSwt0+6sfsiTG8qcWGx4=
|
||||
github.com/hashicorp/hcl v1.0.0/go.mod h1:E5yfLk+7swimpb2L/Alb/PJmXilQ/rhwaUYs4T20WEQ=
|
||||
github.com/hashicorp/hcl/v2 v2.16.2 h1:mpkHZh/Tv+xet3sy3F9Ld4FyI2tUpWe9x3XtPx9f1a0=
|
||||
github.com/hashicorp/hcl/v2 v2.16.2/go.mod h1:JRmR89jycNkrrqnMmvPDMd56n1rQJ2Q6KocSLCMCXng=
|
||||
github.com/ianlancetaylor/demangle v0.0.0-20181102032728-5e5cf60278f6/go.mod h1:aSSvb/t6k1mPoxDqO4vJh6VOCGPwU4O0C2/Eqndh1Sc=
|
||||
github.com/ianlancetaylor/demangle v0.0.0-20200824232613-28f6c0f3b639/go.mod h1:aSSvb/t6k1mPoxDqO4vJh6VOCGPwU4O0C2/Eqndh1Sc=
|
||||
github.com/imdario/mergo v0.3.13/go.mod h1:4lJ1jqUDcsbIECGy0RUJAXNIhg+6ocWgb1ALK2O4oXg=
|
||||
github.com/imdario/mergo v0.3.15 h1:M8XP7IuFNsqUx6VPK2P9OSmsYsI/YFaGil0uD21V3dM=
|
||||
github.com/imdario/mergo v0.3.15/go.mod h1:WBLT9ZmE3lPoWsEzCh9LPo3TiwVN+ZKEjmz+hD27ysY=
|
||||
|
@ -217,8 +127,8 @@ github.com/jbenet/go-context v0.0.0-20150711004518-d14ea06fba99/go.mod h1:1lJo3i
|
|||
github.com/jessevdk/go-flags v1.5.0/go.mod h1:Fw0T6WPc1dYxT4mKEZRfG5kJhaTDP9pj1c2EWnYs/m4=
|
||||
github.com/jinzhu/copier v0.3.5 h1:GlvfUwHk62RokgqVNvYsku0TATCF7bAHVwEXoBh3iJg=
|
||||
github.com/jinzhu/copier v0.3.5/go.mod h1:DfbEm0FYsaqBcKcFuvmOZb218JkPGtvSHsKg8S8hyyg=
|
||||
github.com/jstemmer/go-junit-report v0.0.0-20190106144839-af01ea7f8024/go.mod h1:6v2b51hI/fHJwM22ozAgKL4VKDeJcHhJFhtBdhmNjmU=
|
||||
github.com/jstemmer/go-junit-report v0.9.1/go.mod h1:Brl9GWCQeLvo8nXZwPNNblvFj/XSXhF0NWZEnDohbsk=
|
||||
github.com/kballard/go-shellquote v0.0.0-20180428030007-95032a82bc51 h1:Z9n2FFNUXsshfwJMBgNA0RU6/i7WVaAegv3PtuIHPMs=
|
||||
github.com/kballard/go-shellquote v0.0.0-20180428030007-95032a82bc51/go.mod h1:CzGEWj7cYgsdH8dAjBGEr58BoE7ScuLd+fwFZ44+/x8=
|
||||
github.com/kevinburke/ssh_config v1.2.0 h1:x584FjTGwHzMwvHx18PXxbBVzfnxogHaAReU4gf13a4=
|
||||
github.com/kevinburke/ssh_config v1.2.0/go.mod h1:CT57kijsi8u/K/BOFA39wgDQJ9CxiF4nAY/ojJ6r6mM=
|
||||
github.com/kisielk/errcheck v1.5.0/go.mod h1:pFxgyoBC7bSaBwPgfKdkLd5X25qrDl4LWUI2bnpBCr8=
|
||||
|
@ -228,7 +138,6 @@ github.com/klauspost/compress v1.16.5/go.mod h1:ntbaceVETuRiXiv4DpjP66DpAtAGkEQs
|
|||
github.com/klauspost/pgzip v1.2.5 h1:qnWYvvKqedOF2ulHpMG72XQol4ILEJ8k2wwRl/Km8oE=
|
||||
github.com/klauspost/pgzip v1.2.5/go.mod h1:Ch1tH69qFZu15pkjo5kYi6mth2Zzwzt50oCQKQE9RUs=
|
||||
github.com/konsorten/go-windows-terminal-sequences v1.0.1/go.mod h1:T0+1ngSBFLxvqU3pZ+m/2kptfBszLMUkC4ZK/EgS/cQ=
|
||||
github.com/kr/fs v0.1.0/go.mod h1:FFnZGqtBN9Gxj7eW1uZ42v5BccTP0vu6NEaFoC2HwRg=
|
||||
github.com/kr/pretty v0.1.0/go.mod h1:dAy3ld7l9f0ibDNOQOHHMYYIIbhfbHSm3C4ZsoJORNo=
|
||||
github.com/kr/pretty v0.2.1/go.mod h1:ipq/a2n7PKx3OHsz4KJII5eveXtPO4qwEXGdVfWzfnI=
|
||||
github.com/kr/pretty v0.3.1 h1:flRD4NNwYAUpkphVc1HcthR4KEIFJ65n8Mw5qdRn3LE=
|
||||
|
@ -238,37 +147,35 @@ github.com/kr/text v0.2.0 h1:5Nx0Ya0ZqY2ygV366QzturHI13Jq95ApcVaJBhpS+AY=
|
|||
github.com/kr/text v0.2.0/go.mod h1:eLer722TekiGuMkidMxC/pM04lWEeraHUUmBw8l2grE=
|
||||
github.com/kylelemons/godebug v0.0.0-20170820004349-d65d576e9348 h1:MtvEpTB6LX3vkb4ax0b5D2DHbNAUsen0Gx5wZoq3lV4=
|
||||
github.com/kylelemons/godebug v0.0.0-20170820004349-d65d576e9348/go.mod h1:B69LEHPfb2qLo0BaaOLcbitczOKLWTsrBG9LczfCD4k=
|
||||
github.com/magiconair/properties v1.8.7 h1:IeQXZAiQcpL9mgcAe1Nu6cX9LLw6ExEHKjN0VQdvPDY=
|
||||
github.com/magiconair/properties v1.8.7/go.mod h1:Dhd985XPs7jluiymwWYZ0G4Z61jb3vdS329zhj2hYo0=
|
||||
github.com/matryer/is v1.2.0 h1:92UTHpy8CDwaJ08GqLDzhhuixiBUUD1p3AU6PHddz4A=
|
||||
github.com/matryer/is v1.2.0/go.mod h1:2fLPjFQM9rhQ15aVEtbuwhJinnOqrmgXPNdZsdwlWXA=
|
||||
github.com/mattn/go-isatty v0.0.18 h1:DOKFKCQ7FNG2L1rbrmstDN4QVRdS89Nkh85u68Uwp98=
|
||||
github.com/mattn/go-isatty v0.0.18/go.mod h1:W+V8PltTTMOvKvAeJH7IuucS94S2C6jfK/D7dTCTo3Y=
|
||||
github.com/mattn/go-runewidth v0.0.9 h1:Lm995f3rfxdpd6TSmuVCHVb/QhupuXlYr8sCI/QdE+0=
|
||||
github.com/mattn/go-sqlite3 v1.14.16 h1:yOQRA0RpS5PFz/oikGwBEqvAWhWg5ufRz4ETLjwpU1Y=
|
||||
github.com/matttproud/golang_protobuf_extensions v1.0.4 h1:mmDVorXM7PCGKw94cs5zkfA9PSy5pEvNWRP0ET0TIVo=
|
||||
github.com/matttproud/golang_protobuf_extensions v1.0.4/go.mod h1:BSXmuO+STAnVfrANrmjBb36TMTDstsz7MSK+HVaYKv4=
|
||||
github.com/mitchellh/go-homedir v1.1.0 h1:lukF9ziXFxDFPkA1vsr5zpc1XuPDn/wFntq5mG+4E0Y=
|
||||
github.com/mitchellh/go-homedir v1.1.0/go.mod h1:SfyaCUpYCn1Vlf4IUYiD9fPX4A5wJrkLzIz1N1q0pr0=
|
||||
github.com/mitchellh/go-wordwrap v1.0.1 h1:TLuKupo69TCn6TQSyGxwI1EblZZEsQ0vMlAFQflz0v0=
|
||||
github.com/mitchellh/go-wordwrap v1.0.1/go.mod h1:R62XHJLzvMFRBbcrT7m7WgmE1eOyTSsCt+hzestvNj0=
|
||||
github.com/mitchellh/mapstructure v1.5.0 h1:jeMsZIYE/09sWLaz43PL7Gy6RuMjD2eJVyuac5Z2hdY=
|
||||
github.com/mitchellh/mapstructure v1.5.0/go.mod h1:bFUtVrKA4DC2yAKiSyO/QUcy7e+RRV2QTWOzhPopBRo=
|
||||
github.com/mmcloughlin/avo v0.5.0/go.mod h1:ChHFdoV7ql95Wi7vuq2YT1bwCJqiWdZrQ1im3VujLYM=
|
||||
github.com/moby/sys/mountinfo v0.6.2 h1:BzJjoreD5BMFNmD9Rus6gdd1pLuecOFPt8wC+Vygl78=
|
||||
github.com/moby/sys/mountinfo v0.6.2/go.mod h1:IJb6JQeOklcdMU9F5xQ8ZALD+CUr5VlGpwtX+VE0rpI=
|
||||
github.com/moby/term v0.0.0-20210610120745-9d4ed1856297 h1:yH0SvLzcbZxcJXho2yh7CqdENGMQe73Cw3woZBpPli0=
|
||||
github.com/morikuni/aec v1.0.0 h1:nP9CBfwrvYnBRgY6qfDQkygYDmYwOilePFkwzv4dU8A=
|
||||
github.com/niemeyer/pretty v0.0.0-20200227124842-a10e7caefd8e/go.mod h1:zD1mROLANZcx1PVRCS0qkT7pwLkGfwJo4zjcN/Tysno=
|
||||
github.com/olekukonko/tablewriter v0.0.5 h1:P2Ga83D34wi1o9J6Wh1mRuqd4mF/x/lgBS7N7AbDhec=
|
||||
github.com/opencontainers/go-digest v1.0.0 h1:apOUWs51W5PlhuyGyz9FCeeBIOUDA/6nW8Oi/yOhh5U=
|
||||
github.com/opencontainers/go-digest v1.0.0/go.mod h1:0JzlMkj0TRzQZfJkVvzbP0HBR3IKzErnv2BNG4W4MAM=
|
||||
github.com/opencontainers/image-spec v1.1.0-rc3 h1:fzg1mXZFj8YdPeNkRXMg+zb88BFV0Ys52cJydRwBkb8=
|
||||
github.com/opencontainers/image-spec v1.1.0-rc3/go.mod h1:X4pATf0uXsnn3g5aiGIsVnJBR4mxhKzfwmvK/B2NTm8=
|
||||
github.com/opentracing/opentracing-go v1.1.0/go.mod h1:UkNAQd3GIcIGf0SeVgPpRdFStlNbqXla1AfSYxPUl2o=
|
||||
github.com/pelletier/go-toml/v2 v2.0.7 h1:muncTPStnKRos5dpVKULv2FVd4bMOhNePj9CjgDb8Us=
|
||||
github.com/pelletier/go-toml/v2 v2.0.7/go.mod h1:eumQOmlWiOPt5WriQQqoM5y18pDHwha2N+QD+EUNTek=
|
||||
github.com/pjbgf/sha1cd v0.3.0 h1:4D5XXmUUBUl/xQ6IjCkEAbqXskkq/4O7LmGn0AqMDs4=
|
||||
github.com/pjbgf/sha1cd v0.3.0/go.mod h1:nZ1rrWOcGJ5uZgEEVL1VUM9iRQiZvWdbZjkKyFzPPsI=
|
||||
github.com/pkg/errors v0.8.1/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0=
|
||||
github.com/pkg/errors v0.9.1 h1:FEBLx1zS214owpjy7qsBeixbURkuhQAwrK5UwLGTwt4=
|
||||
github.com/pkg/errors v0.9.1/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0=
|
||||
github.com/pkg/sftp v1.13.1/go.mod h1:3HaPG6Dq1ILlpPZRO0HVMrsydcdLt6HRDccSgb87qRg=
|
||||
github.com/pmezard/go-difflib v1.0.0 h1:4DBwDE0NGyQoBHbLQYPwSUPoCMWR5BEzIk/f1lZbAQM=
|
||||
github.com/pmezard/go-difflib v1.0.0/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4=
|
||||
github.com/prometheus/client_golang v1.15.0 h1:5fCgGYogn0hFdhyhLbw7hEsWxufKtY9klyvdNfFlFhM=
|
||||
|
@ -280,8 +187,11 @@ github.com/prometheus/common v0.42.0 h1:EKsfXEYo4JpWMHH5cg+KOUWeuJSov1Id8zGR8eeI
|
|||
github.com/prometheus/common v0.42.0/go.mod h1:xBwqVerjNdUDjgODMpudtOMwlOwf2SaTr1yjz4b7Zbc=
|
||||
github.com/prometheus/procfs v0.9.0 h1:wzCHvIvM5SxWqYvwgVL7yJY8Lz3PKn49KQtpgMYJfhI=
|
||||
github.com/prometheus/procfs v0.9.0/go.mod h1:+pB4zwohETzFnmlpe6yd2lSc+0/46IYZRB/chUwxUZY=
|
||||
github.com/rogpeppe/go-internal v1.3.0/go.mod h1:M8bDsm7K2OlrFYOpmOWEs/qY81heoFRclV5y23lUDJ4=
|
||||
github.com/remyoudompheng/bigfft v0.0.0-20200410134404-eec4a21b6bb0/go.mod h1:qqbHyh8v60DhA7CoWK5oRCqLrMHRGoxYCSS9EjAz6Eo=
|
||||
github.com/remyoudompheng/bigfft v0.0.0-20230129092748-24d4a6f8daec h1:W09IVJc94icq4NjY3clb7Lk8O1qJ8BdBEF8z0ibU0rE=
|
||||
github.com/remyoudompheng/bigfft v0.0.0-20230129092748-24d4a6f8daec/go.mod h1:qqbHyh8v60DhA7CoWK5oRCqLrMHRGoxYCSS9EjAz6Eo=
|
||||
github.com/rogpeppe/go-internal v1.6.1 h1:/FiVV8dS/e+YqF2JvO3yXRFbBLTIuSDkuC7aBOAvL+k=
|
||||
github.com/rogpeppe/go-internal v1.6.1/go.mod h1:xXDCJY+GAPziupqXw64V24skbSoqbTEfhy4qGm1nDQc=
|
||||
github.com/russross/blackfriday/v2 v2.1.0/go.mod h1:+Rmxgy9KzJVeS9/2gXHxylqXiyQDYRxCVz55jmeOWTM=
|
||||
github.com/sergi/go-diff v1.1.0/go.mod h1:STckp+ISIX8hZLjrqAeVduY0gWCT9IjLuqbuNXdaHfM=
|
||||
github.com/sergi/go-diff v1.3.1 h1:xkr+Oxo4BOQKmkn/B9eMK0g5Kg/983T9DqqPHwYqD+8=
|
||||
|
@ -292,18 +202,10 @@ github.com/sirupsen/logrus v1.9.0 h1:trlNQbNUG3OdDrDil03MCb1H2o9nJ1x4/5LYw7byDE0
|
|||
github.com/sirupsen/logrus v1.9.0/go.mod h1:naHLuLoDiP4jHNo9R0sCBMtWGeIprob74mVsIT4qYEQ=
|
||||
github.com/skeema/knownhosts v1.1.0 h1:Wvr9V0MxhjRbl3f9nMnKnFfiWTJmtECJ9Njkea3ysW0=
|
||||
github.com/skeema/knownhosts v1.1.0/go.mod h1:sKFq3RD6/TKZkSWn8boUbDC7Qkgcv+8XXijpFO6roag=
|
||||
github.com/spf13/afero v1.9.5 h1:stMpOSZFs//0Lv29HduCmli3GUfpFoF3Y1Q/aXj/wVM=
|
||||
github.com/spf13/afero v1.9.5/go.mod h1:UBogFpq8E9Hx+xc5CNTTEpTnuHVmXDwZcZcE1eb/UhQ=
|
||||
github.com/spf13/cast v1.5.0 h1:rj3WzYc11XZaIZMPKmwP96zkFEnnAmV8s6XbB2aY32w=
|
||||
github.com/spf13/cast v1.5.0/go.mod h1:SpXXQ5YoyJw6s3/6cMTQuxvgRl3PCJiyaX9p6b155UU=
|
||||
github.com/spf13/cobra v1.7.0 h1:hyqWnYt1ZQShIddO5kBpj3vu05/++x6tJ6dg8EC572I=
|
||||
github.com/spf13/cobra v1.7.0/go.mod h1:uLxZILRyS/50WlhOIKD7W6V5bgeIt+4sICxh6uRMrb0=
|
||||
github.com/spf13/jwalterweatherman v1.1.0 h1:ue6voC5bR5F8YxI5S67j9i582FU4Qvo2bmqnqMYADFk=
|
||||
github.com/spf13/jwalterweatherman v1.1.0/go.mod h1:aNWZUN0dPAAO/Ljvb5BEdw96iTZ0EXowPYD95IqWIGo=
|
||||
github.com/spf13/pflag v1.0.5 h1:iy+VFUOCP1a+8yFto/drg2CJ5u0yRoB7fZw3DKv/JXA=
|
||||
github.com/spf13/pflag v1.0.5/go.mod h1:McXfInJRrz4CZXVZOBLb0bTZqETkiAhM9Iw0y3An2Bg=
|
||||
github.com/spf13/viper v1.15.0 h1:js3yy885G8xwJa6iOISGFwd+qlUo5AvyXb7CiihdtiU=
|
||||
github.com/spf13/viper v1.15.0/go.mod h1:fFcTBJxvhhzSJiZy8n+PeW6t8l+KeT/uTARa0jHOQLA=
|
||||
github.com/stretchr/objx v0.1.0/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME=
|
||||
github.com/stretchr/objx v0.1.1/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME=
|
||||
github.com/stretchr/objx v0.4.0/go.mod h1:YvHI0jy2hoMjB+UWwv71VJQ9isScKT/TqJzVSSt89Yw=
|
||||
|
@ -311,45 +213,31 @@ github.com/stretchr/objx v0.5.0/go.mod h1:Yh+to48EsGEfYuaHDzXPcE3xhTkx73EhmCGUpE
|
|||
github.com/stretchr/testify v1.2.2/go.mod h1:a8OnRcib4nhh0OaRAV+Yts87kKdq0PP7pXfy6kDkUVs=
|
||||
github.com/stretchr/testify v1.3.0/go.mod h1:M5WIy9Dh21IEIfnGCwXGc5bZfKNJtfHm1UVUgZn+9EI=
|
||||
github.com/stretchr/testify v1.4.0/go.mod h1:j7eGeouHqKxXV5pUuKE4zz7dFj8WfuZ+81PSLYec5m4=
|
||||
github.com/stretchr/testify v1.5.1/go.mod h1:5W2xD1RspED5o8YsWQXVCued0rvSQ+mT+I5cxcmMvtA=
|
||||
github.com/stretchr/testify v1.7.0/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg=
|
||||
github.com/stretchr/testify v1.7.1/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg=
|
||||
github.com/stretchr/testify v1.8.0/go.mod h1:yNjHg4UonilssWZ8iaSj1OCr/vHnekPRkoO+kdMU+MU=
|
||||
github.com/stretchr/testify v1.8.1 h1:w7B6lhMri9wdJUVmEZPGGhZzrYTPvgJArz7wNPgYKsk=
|
||||
github.com/stretchr/testify v1.8.1/go.mod h1:w2LPCIKwWwSfY2zedu0+kehJoqGctiVI29o6fzry7u4=
|
||||
github.com/subosito/gotenv v1.4.2 h1:X1TuBLAMDFbaTAChgCBLu3DU3UPyELpnF2jjJ2cz/S8=
|
||||
github.com/subosito/gotenv v1.4.2/go.mod h1:ayKnFf/c6rvx/2iiLrJUk1e6plDbT3edrFNGqEflhK0=
|
||||
github.com/stretchr/testify v1.8.2 h1:+h33VjcLVPDHtOdpUCuF+7gSuG3yGIftsP1YvFihtJ8=
|
||||
github.com/stretchr/testify v1.8.2/go.mod h1:w2LPCIKwWwSfY2zedu0+kehJoqGctiVI29o6fzry7u4=
|
||||
github.com/urfave/cli v1.22.12/go.mod h1:sSBEIC79qR6OvcmsD4U3KABeOTxDqQtdDnaFuUN30b8=
|
||||
github.com/vbatts/tar-split v0.11.3 h1:hLFqsOLQ1SsppQNTMpkpPXClLDfC2A3Zgy9OUU+RVck=
|
||||
github.com/vbatts/tar-split v0.11.3/go.mod h1:9QlHN18E+fEH7RdG+QAJJcuya3rqT7eXSTY7wGrAokY=
|
||||
github.com/xanzy/ssh-agent v0.3.3 h1:+/15pJfg/RsTxqYcX6fHqOXZwwMP+2VyYWJeWM2qQFM=
|
||||
github.com/xanzy/ssh-agent v0.3.3/go.mod h1:6dzNDKs0J9rVPHPhaGCukekBHKqfl+L3KghI1Bc68Uw=
|
||||
github.com/yuin/goldmark v1.1.25/go.mod h1:3hX8gzYuyVAZsxl0MRgGTJEmQBFcNTphYh9decYSb74=
|
||||
github.com/yuin/goldmark v1.1.27/go.mod h1:3hX8gzYuyVAZsxl0MRgGTJEmQBFcNTphYh9decYSb74=
|
||||
github.com/yuin/goldmark v1.1.32/go.mod h1:3hX8gzYuyVAZsxl0MRgGTJEmQBFcNTphYh9decYSb74=
|
||||
github.com/yuin/goldmark v1.2.1/go.mod h1:3hX8gzYuyVAZsxl0MRgGTJEmQBFcNTphYh9decYSb74=
|
||||
github.com/yuin/goldmark v1.4.13/go.mod h1:6yULJ656Px+3vBD8DxQVa3kxgyrAnzto9xy5taEt/CY=
|
||||
github.com/zclconf/go-cty v1.13.1 h1:0a6bRwuiSHtAmqCqNOE+c2oHgepv0ctoxU4FUe43kwc=
|
||||
github.com/zclconf/go-cty v1.13.1/go.mod h1:YKQzy/7pZ7iq2jNFzy5go57xdxdWoLLpaEp4u238AE0=
|
||||
go.opencensus.io v0.21.0/go.mod h1:mSImk1erAIZhrmZN+AvHh14ztQfjbGwt4TtuofqLduU=
|
||||
go.opencensus.io v0.22.0/go.mod h1:+kGneAE2xo2IficOXnaByMWTGM9T73dGwxeWcUqIpI8=
|
||||
go.opencensus.io v0.22.2/go.mod h1:yxeiOL68Rb0Xd1ddK5vPZ/oVn4vY4Ynel7k9FzqtOIw=
|
||||
go.opencensus.io v0.22.3/go.mod h1:yxeiOL68Rb0Xd1ddK5vPZ/oVn4vY4Ynel7k9FzqtOIw=
|
||||
go.opencensus.io v0.22.4/go.mod h1:yxeiOL68Rb0Xd1ddK5vPZ/oVn4vY4Ynel7k9FzqtOIw=
|
||||
go.opencensus.io v0.22.5/go.mod h1:5pWMHQbX5EPX2/62yrJeAkowc+lfs/XD7Uxpq3pI6kk=
|
||||
go.uber.org/atomic v1.7.0/go.mod h1:fEN4uk6kAWBTFdckzkM89CLk9XfWZrxpCo0nPH17wJc=
|
||||
go.uber.org/goleak v1.1.10/go.mod h1:8a7PlsEVH3e/a/GLqe5IIrQx6GzcnRmZEufDUTk4A7A=
|
||||
go.uber.org/multierr v1.6.0/go.mod h1:cdWPpRnG4AhwMwsgIHip0KRBQjJy5kYEpYjJxpXp9iU=
|
||||
go.uber.org/zap v1.18.1/go.mod h1:xg/QME4nWcxGxrpdeYfq7UvYrLh66cuVKdrbD1XF/NI=
|
||||
golang.org/x/arch v0.1.0/go.mod h1:5om86z9Hs0C8fWVUuoMHwpExlXzs5Tkyp9hOrfG7pp8=
|
||||
golang.org/x/crypto v0.0.0-20190510104115-cbcb75029529/go.mod h1:yigFU9vqHzYiE8UmvKecakEJjdnWj3jj499lnFckfCI=
|
||||
golang.org/x/crypto v0.0.0-20190605123033-f99c8df09eb5/go.mod h1:yigFU9vqHzYiE8UmvKecakEJjdnWj3jj499lnFckfCI=
|
||||
golang.org/x/crypto v0.0.0-20191011191535-87dc89f01550/go.mod h1:yigFU9vqHzYiE8UmvKecakEJjdnWj3jj499lnFckfCI=
|
||||
golang.org/x/crypto v0.0.0-20210421170649-83a5a9bb288b/go.mod h1:T9bdIzuCu7OtxOm1hfPfRQxPLYneinmdGuTeoZ9dtd4=
|
||||
golang.org/x/crypto v0.0.0-20210921155107-089bfa567519/go.mod h1:GvvjBRRGRdwPK5ydBHafDWAxML/pGHZbMvKqRZ5+Abc=
|
||||
golang.org/x/crypto v0.0.0-20220525230936-793ad666bf5e/go.mod h1:IxCIyHEi3zRg3s0A5j5BB6A9Jmi73HwBIUl50j+osU4=
|
||||
golang.org/x/crypto v0.0.0-20220622213112-05595931fe9d/go.mod h1:IxCIyHEi3zRg3s0A5j5BB6A9Jmi73HwBIUl50j+osU4=
|
||||
golang.org/x/crypto v0.0.0-20220722155217-630584e8d5aa/go.mod h1:IxCIyHEi3zRg3s0A5j5BB6A9Jmi73HwBIUl50j+osU4=
|
||||
golang.org/x/crypto v0.0.0-20220826181053-bd7e27e6170d/go.mod h1:IxCIyHEi3zRg3s0A5j5BB6A9Jmi73HwBIUl50j+osU4=
|
||||
golang.org/x/crypto v0.1.0/go.mod h1:RecgLatLF4+eUMCP1PoPZQb+cVrJcOPbHkTkbkB9sbw=
|
||||
golang.org/x/crypto v0.6.0/go.mod h1:OFC/31mSvZgRz0V1QTNCzfAI1aIRzbiufJtkMIlEp58=
|
||||
|
@ -357,40 +245,14 @@ golang.org/x/crypto v0.7.0/go.mod h1:pYwdfH91IfpZVANVyUOhSIPZaFoJGxTFbZhFTx+dXZU
|
|||
golang.org/x/crypto v0.8.0 h1:pd9TJtTueMTVQXzk8E2XESSMQDj/U7OUu0PqJqPXQjQ=
|
||||
golang.org/x/crypto v0.8.0/go.mod h1:mRqEX+O9/h5TFCrQhkgjo2yKi0yYA+9ecGkdQoHrywE=
|
||||
golang.org/x/exp v0.0.0-20190121172915-509febef88a4/go.mod h1:CJ0aWSM057203Lf6IL+f9T1iT9GByDxfZKAQTCR3kQA=
|
||||
golang.org/x/exp v0.0.0-20190306152737-a1d7652674e8/go.mod h1:CJ0aWSM057203Lf6IL+f9T1iT9GByDxfZKAQTCR3kQA=
|
||||
golang.org/x/exp v0.0.0-20190510132918-efd6b22b2522/go.mod h1:ZjyILWgesfNpC6sMxTJOJm9Kp84zZh5NQWvqDGG3Qr8=
|
||||
golang.org/x/exp v0.0.0-20190829153037-c13cbed26979/go.mod h1:86+5VVa7VpoJ4kLfm080zCjGlMRFzhUhsZKEZO7MGek=
|
||||
golang.org/x/exp v0.0.0-20191030013958-a1ab85dbe136/go.mod h1:JXzH8nQsPlswgeRAPE3MuO9GYsAcnJvJ4vnMwN/5qkY=
|
||||
golang.org/x/exp v0.0.0-20191129062945-2f5052295587/go.mod h1:2RIsYlXP63K8oxa1u096TMicItID8zy7Y6sNkU49FU4=
|
||||
golang.org/x/exp v0.0.0-20191227195350-da58074b4299/go.mod h1:2RIsYlXP63K8oxa1u096TMicItID8zy7Y6sNkU49FU4=
|
||||
golang.org/x/exp v0.0.0-20200119233911-0405dc783f0a/go.mod h1:2RIsYlXP63K8oxa1u096TMicItID8zy7Y6sNkU49FU4=
|
||||
golang.org/x/exp v0.0.0-20200207192155-f17229e696bd/go.mod h1:J/WKrq2StrnmMY6+EHIKF9dgMWnmCNThgcyBT1FY9mM=
|
||||
golang.org/x/exp v0.0.0-20200224162631-6cc2880d07d6/go.mod h1:3jZMyOhIsHpP37uCMkUooju7aAi5cS1Q23tOzKc+0MU=
|
||||
golang.org/x/exp v0.0.0-20230425010034-47ecfdc1ba53 h1:5llv2sWeaMSnA3w2kS57ouQQ4pudlXrR0dCgw51QK9o=
|
||||
golang.org/x/exp v0.0.0-20230425010034-47ecfdc1ba53/go.mod h1:V1LtkGg67GoY2N1AnLN78QLrzxkLyJw7RJb1gzOOz9w=
|
||||
golang.org/x/image v0.0.0-20190227222117-0694c2d4d067/go.mod h1:kZ7UVZpmo3dzQBMxlp+ypCbDeSB+sBbTgSJuh5dn5js=
|
||||
golang.org/x/image v0.0.0-20190802002840-cff245a6509b/go.mod h1:FeLwcggjj3mMvU+oOTbSwawSJRM1uh48EjtB4UJZlP0=
|
||||
golang.org/x/lint v0.0.0-20181026193005-c67002cb31c3/go.mod h1:UVdnD1Gm6xHRNCYTkRU2/jEulfH38KcIWyp/GAMgvoE=
|
||||
golang.org/x/lint v0.0.0-20190227174305-5b3e6a55c961/go.mod h1:wehouNa3lNwaWXcvxsM5YxQ5yQlVC4a0KAMCusXpPoU=
|
||||
golang.org/x/lint v0.0.0-20190301231843-5614ed5bae6f/go.mod h1:UVdnD1Gm6xHRNCYTkRU2/jEulfH38KcIWyp/GAMgvoE=
|
||||
golang.org/x/lint v0.0.0-20190313153728-d0100b6bd8b3/go.mod h1:6SW0HCj/g11FgYtHlgUYUwCkIfeOF89ocIRzGO/8vkc=
|
||||
golang.org/x/lint v0.0.0-20190409202823-959b441ac422/go.mod h1:6SW0HCj/g11FgYtHlgUYUwCkIfeOF89ocIRzGO/8vkc=
|
||||
golang.org/x/lint v0.0.0-20190909230951-414d861bb4ac/go.mod h1:6SW0HCj/g11FgYtHlgUYUwCkIfeOF89ocIRzGO/8vkc=
|
||||
golang.org/x/lint v0.0.0-20190930215403-16217165b5de/go.mod h1:6SW0HCj/g11FgYtHlgUYUwCkIfeOF89ocIRzGO/8vkc=
|
||||
golang.org/x/lint v0.0.0-20191125180803-fdd1cda4f05f/go.mod h1:5qLYkcX4OjUUV8bRuDixDT3tpyyb+LUpUlRWLxfhWrs=
|
||||
golang.org/x/lint v0.0.0-20200130185559-910be7a94367/go.mod h1:3xt1FjdF8hUf6vQPIChWIBhFzV8gjjsPE/fR3IyQdNY=
|
||||
golang.org/x/lint v0.0.0-20200302205851-738671d3881b/go.mod h1:3xt1FjdF8hUf6vQPIChWIBhFzV8gjjsPE/fR3IyQdNY=
|
||||
golang.org/x/lint v0.0.0-20201208152925-83fdc39ff7b5/go.mod h1:3xt1FjdF8hUf6vQPIChWIBhFzV8gjjsPE/fR3IyQdNY=
|
||||
golang.org/x/mobile v0.0.0-20190312151609-d3739f865fa6/go.mod h1:z+o9i4GpDbdi3rU15maQ/Ox0txvL9dWGYEHz965HBQE=
|
||||
golang.org/x/mobile v0.0.0-20190719004257-d2bd2a29d028/go.mod h1:E/iHnbuqvinMTCcRqshq8CkpyQDoeVncDDYHnLhea+o=
|
||||
golang.org/x/mod v0.0.0-20190513183733-4bf6d317e70e/go.mod h1:mXi4GBBbnImb6dmsKGUJ2LatrhH/nqhxcFungHvyanc=
|
||||
golang.org/x/mod v0.1.0/go.mod h1:0QHyrYULN0/3qlju5TqG8bIK38QM8yzMo5ekMj3DlcY=
|
||||
golang.org/x/mod v0.1.1-0.20191105210325-c90efee705ee/go.mod h1:QqPTAvyqsEbceGzBzNggFXnrqF1CaUcvgkdR5Ot7KZg=
|
||||
golang.org/x/mod v0.1.1-0.20191107180719-034126e5016b/go.mod h1:QqPTAvyqsEbceGzBzNggFXnrqF1CaUcvgkdR5Ot7KZg=
|
||||
golang.org/x/mod v0.2.0/go.mod h1:s0Qsj1ACt9ePp/hMypM3fl4fZqREWJwdYDEqhRiZZUA=
|
||||
golang.org/x/mod v0.3.0/go.mod h1:s0Qsj1ACt9ePp/hMypM3fl4fZqREWJwdYDEqhRiZZUA=
|
||||
golang.org/x/mod v0.4.0/go.mod h1:s0Qsj1ACt9ePp/hMypM3fl4fZqREWJwdYDEqhRiZZUA=
|
||||
golang.org/x/mod v0.4.1/go.mod h1:s0Qsj1ACt9ePp/hMypM3fl4fZqREWJwdYDEqhRiZZUA=
|
||||
golang.org/x/mod v0.6.0-dev.0.20220419223038-86c51ed26bb4/go.mod h1:jJ57K6gSWd91VN4djpZkiMVwK6gcyfeH4XE8wZrZaV4=
|
||||
golang.org/x/mod v0.6.0/go.mod h1:4mET923SAdbXp2ki8ey+zGs1SLqsuM2Y0uvdZR/fUNI=
|
||||
golang.org/x/mod v0.8.0/go.mod h1:iBbtSCu2XBx23ZKBPSOrRkjjQPZFPuis4dIYUhu/chs=
|
||||
|
@ -399,24 +261,13 @@ golang.org/x/mod v0.10.0/go.mod h1:iBbtSCu2XBx23ZKBPSOrRkjjQPZFPuis4dIYUhu/chs=
|
|||
golang.org/x/net v0.8.0 h1:Zrh2ngAOFYneWTAIAPethzeaQLuHwhuBkuV6ZiRnUaQ=
|
||||
golang.org/x/net v0.8.0/go.mod h1:QVkue5JL9kW//ek3r6jTKnTFis1tRmNAW2P1shuFdJc=
|
||||
golang.org/x/oauth2 v0.0.0-20180821212333-d2e6202438be/go.mod h1:N/0e6XlmueqKjAGxoOufVs8QHGRruUQn6yWY3a++T0U=
|
||||
golang.org/x/oauth2 v0.0.0-20190226205417-e64efc72b421/go.mod h1:gOpvHmFTYa4IltrdGE7lF6nIHvwfUNPOp7c8zoXwtLw=
|
||||
golang.org/x/oauth2 v0.0.0-20190604053449-0f29369cfe45/go.mod h1:gOpvHmFTYa4IltrdGE7lF6nIHvwfUNPOp7c8zoXwtLw=
|
||||
golang.org/x/oauth2 v0.0.0-20191202225959-858c2ad4c8b6/go.mod h1:gOpvHmFTYa4IltrdGE7lF6nIHvwfUNPOp7c8zoXwtLw=
|
||||
golang.org/x/oauth2 v0.0.0-20200107190931-bf48bf16ab8d/go.mod h1:gOpvHmFTYa4IltrdGE7lF6nIHvwfUNPOp7c8zoXwtLw=
|
||||
golang.org/x/oauth2 v0.0.0-20200902213428-5d25da1a8d43/go.mod h1:KelEdhl1UZF7XfJ4dDtk6s++YSgaE7mD/BuKKDLBl4A=
|
||||
golang.org/x/oauth2 v0.0.0-20201109201403-9fd604954f58/go.mod h1:KelEdhl1UZF7XfJ4dDtk6s++YSgaE7mD/BuKKDLBl4A=
|
||||
golang.org/x/oauth2 v0.0.0-20201208152858-08078c50e5b5/go.mod h1:KelEdhl1UZF7XfJ4dDtk6s++YSgaE7mD/BuKKDLBl4A=
|
||||
golang.org/x/oauth2 v0.0.0-20210218202405-ba52d332ba99/go.mod h1:KelEdhl1UZF7XfJ4dDtk6s++YSgaE7mD/BuKKDLBl4A=
|
||||
golang.org/x/oauth2 v0.7.0 h1:qe6s0zUXlPX80/dITx3440hWZ7GwMwgDDyrSGTPJG/g=
|
||||
golang.org/x/oauth2 v0.7.0/go.mod h1:hPLQkd9LyjfXTiRohC/41GhcFqxisoUQ99sCUOHO9x4=
|
||||
golang.org/x/sync v0.0.0-20180314180146-1d60e4601c6f/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
|
||||
golang.org/x/sync v0.0.0-20181108010431-42b317875d0f/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
|
||||
golang.org/x/sync v0.0.0-20181221193216-37e7f081c4d4/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
|
||||
golang.org/x/sync v0.0.0-20190227155943-e225da77a7e6/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
|
||||
golang.org/x/sync v0.0.0-20190423024810-112230192c58/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
|
||||
golang.org/x/sync v0.0.0-20190911185100-cd5d95a43a6e/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
|
||||
golang.org/x/sync v0.0.0-20200317015054-43a5402ce75a/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
|
||||
golang.org/x/sync v0.0.0-20200625203802-6e8e738ad208/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
|
||||
golang.org/x/sync v0.0.0-20201020160332-67f06af15bc9/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
|
||||
golang.org/x/sync v0.0.0-20201207232520-09787c993a3a/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
|
||||
golang.org/x/sync v0.0.0-20220722155255-886fb9371eb4/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
|
||||
|
@ -424,39 +275,13 @@ golang.org/x/sync v0.1.0 h1:wsuoTGHzEhffawBOhz5CYhcrV4IdKZbEyZjBMuTp12o=
|
|||
golang.org/x/sync v0.1.0/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
|
||||
golang.org/x/sys v0.0.0-20180830151530-49385e6e1522/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY=
|
||||
golang.org/x/sys v0.0.0-20190215142949-d0b11bdaac8a/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY=
|
||||
golang.org/x/sys v0.0.0-20190312061237-fead79001313/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
|
||||
golang.org/x/sys v0.0.0-20190412213103-97732733099d/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
|
||||
golang.org/x/sys v0.0.0-20190422165155-953cdadca894/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
|
||||
golang.org/x/sys v0.0.0-20190502145724-3ef323f4f1fd/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
|
||||
golang.org/x/sys v0.0.0-20190507160741-ecd444e8653b/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
|
||||
golang.org/x/sys v0.0.0-20190606165138-5da285871e9c/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
|
||||
golang.org/x/sys v0.0.0-20190624142023-c5567b49c5d0/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
|
||||
golang.org/x/sys v0.0.0-20190726091711-fc99dfbffb4e/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
|
||||
golang.org/x/sys v0.0.0-20191001151750-bb3f8db39f24/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
|
||||
golang.org/x/sys v0.0.0-20191026070338-33540a1f6037/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
|
||||
golang.org/x/sys v0.0.0-20191204072324-ce4227a45e2e/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
|
||||
golang.org/x/sys v0.0.0-20191228213918-04cbcbbfeed8/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
|
||||
golang.org/x/sys v0.0.0-20200113162924-86b910548bc1/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
|
||||
golang.org/x/sys v0.0.0-20200122134326-e047566fdf82/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
|
||||
golang.org/x/sys v0.0.0-20200202164722-d101bd2416d5/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
|
||||
golang.org/x/sys v0.0.0-20200212091648-12a6c2dcc1e4/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
|
||||
golang.org/x/sys v0.0.0-20200223170610-d5e6a3e2c0ae/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
|
||||
golang.org/x/sys v0.0.0-20200302150141-5c8b2ff67527/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
|
||||
golang.org/x/sys v0.0.0-20200331124033-c3d80250170d/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
|
||||
golang.org/x/sys v0.0.0-20200501052902-10377860bb8e/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
|
||||
golang.org/x/sys v0.0.0-20200511232937-7e40ca221e25/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
|
||||
golang.org/x/sys v0.0.0-20200515095857-1151b9dac4a9/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
|
||||
golang.org/x/sys v0.0.0-20200523222454-059865788121/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
|
||||
golang.org/x/sys v0.0.0-20200803210538-64077c9b5642/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
|
||||
golang.org/x/sys v0.0.0-20200905004654-be1d3432aa8f/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
|
||||
golang.org/x/sys v0.0.0-20201119102817-f84b799fce68/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
|
||||
golang.org/x/sys v0.0.0-20201201145000-ef89a241ccb3/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
|
||||
golang.org/x/sys v0.0.0-20210104204734-6f8348627aad/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
|
||||
golang.org/x/sys v0.0.0-20210119212857-b64e53b001e4/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
|
||||
golang.org/x/sys v0.0.0-20210124154548-22da62e12c0c/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
|
||||
golang.org/x/sys v0.0.0-20210225134936-a50acf3fe073/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
|
||||
golang.org/x/sys v0.0.0-20210320140829-1e4c9ba3b0c4/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
|
||||
golang.org/x/sys v0.0.0-20210423185535-09eb48e85fd7/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
|
||||
golang.org/x/sys v0.0.0-20210615035016-665e8c7367d1/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
|
||||
golang.org/x/sys v0.0.0-20211007075335-d3039528d8ac/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
|
||||
golang.org/x/sys v0.0.0-20211025201205-69cdffdb9359/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
|
||||
|
@ -478,12 +303,9 @@ golang.org/x/term v0.1.0/go.mod h1:jbD1KX2456YbFQfuXm/mYQcufACuNUgVhRMnK/tPxf8=
|
|||
golang.org/x/term v0.5.0/go.mod h1:jMB1sMXY+tzblOD4FWmEbocvup2/aLOaQEp7JmGp78k=
|
||||
golang.org/x/term v0.6.0/go.mod h1:m6U89DPEgQRMq3DNkDClhWw02AUbt2daBVO4cn4Hv9U=
|
||||
golang.org/x/term v0.7.0 h1:BEvjmm5fURWqcfbSKTdpkDXYBrUS1c0m8agp14W48vQ=
|
||||
golang.org/x/text v0.0.0-20170915032832-14c0d48ead0c/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ=
|
||||
golang.org/x/text v0.3.0/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ=
|
||||
golang.org/x/text v0.3.1-0.20180807135948-17ff2d5776d2/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ=
|
||||
golang.org/x/text v0.3.2/go.mod h1:bEr9sfX3Q8Zfm5fL9x+3itogRgK3+ptLWKqgva+5dAk=
|
||||
golang.org/x/text v0.3.3/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ=
|
||||
golang.org/x/text v0.3.4/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ=
|
||||
golang.org/x/text v0.3.6/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ=
|
||||
golang.org/x/text v0.3.7/go.mod h1:u+2+/6zg+i71rQMx5EYifcz6MCKuco9NR6JIITiCfzQ=
|
||||
golang.org/x/text v0.4.0/go.mod h1:mrYo+phRRbMaCq/xk9113O4dZlRixOauAjOtrjsXDZ8=
|
||||
|
@ -491,163 +313,42 @@ golang.org/x/text v0.7.0/go.mod h1:mrYo+phRRbMaCq/xk9113O4dZlRixOauAjOtrjsXDZ8=
|
|||
golang.org/x/text v0.8.0/go.mod h1:e1OnstbJyHTd6l/uOt8jFFHp6TRDWZR/bV3emEE/zU8=
|
||||
golang.org/x/text v0.9.0 h1:2sjJmO8cDvYveuX97RDLsxlyUxLl+GHoLxBiRdHllBE=
|
||||
golang.org/x/text v0.9.0/go.mod h1:e1OnstbJyHTd6l/uOt8jFFHp6TRDWZR/bV3emEE/zU8=
|
||||
golang.org/x/time v0.0.0-20181108054448-85acf8d2951c/go.mod h1:tRJNPiyCQ0inRvYxbN9jk5I+vvW/OXSQhTDSoE431IQ=
|
||||
golang.org/x/time v0.0.0-20190308202827-9d24e82272b4/go.mod h1:tRJNPiyCQ0inRvYxbN9jk5I+vvW/OXSQhTDSoE431IQ=
|
||||
golang.org/x/time v0.0.0-20191024005414-555d28b269f0/go.mod h1:tRJNPiyCQ0inRvYxbN9jk5I+vvW/OXSQhTDSoE431IQ=
|
||||
golang.org/x/time v0.1.0 h1:xYY+Bajn2a7VBmTM5GikTmnK8ZuX8YgnQCqZpbBNtmA=
|
||||
golang.org/x/time v0.1.0/go.mod h1:tRJNPiyCQ0inRvYxbN9jk5I+vvW/OXSQhTDSoE431IQ=
|
||||
golang.org/x/tools v0.0.0-20180917221912-90fa682c2a6e/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ=
|
||||
golang.org/x/tools v0.0.0-20190114222345-bf090417da8b/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ=
|
||||
golang.org/x/tools v0.0.0-20190226205152-f727befe758c/go.mod h1:9Yl7xja0Znq3iFh3HoIrodX9oNMXvdceNzlUR8zjMvY=
|
||||
golang.org/x/tools v0.0.0-20190311212946-11955173bddd/go.mod h1:LCzVGOaR6xXOjkQ3onu1FJEFr0SW1gC7cKk1uF8kGRs=
|
||||
golang.org/x/tools v0.0.0-20190312151545-0bb0c0a6e846/go.mod h1:LCzVGOaR6xXOjkQ3onu1FJEFr0SW1gC7cKk1uF8kGRs=
|
||||
golang.org/x/tools v0.0.0-20190312170243-e65039ee4138/go.mod h1:LCzVGOaR6xXOjkQ3onu1FJEFr0SW1gC7cKk1uF8kGRs=
|
||||
golang.org/x/tools v0.0.0-20190425150028-36563e24a262/go.mod h1:RgjU9mgBXZiqYHBnxXauZ1Gv1EHHAz9KjViQ78xBX0Q=
|
||||
golang.org/x/tools v0.0.0-20190506145303-2d16b83fe98c/go.mod h1:RgjU9mgBXZiqYHBnxXauZ1Gv1EHHAz9KjViQ78xBX0Q=
|
||||
golang.org/x/tools v0.0.0-20190524140312-2c0ae7006135/go.mod h1:RgjU9mgBXZiqYHBnxXauZ1Gv1EHHAz9KjViQ78xBX0Q=
|
||||
golang.org/x/tools v0.0.0-20190606124116-d0a3d012864b/go.mod h1:/rFqwRUd4F7ZHNgwSSTFct+R/Kf4OFW1sUzUTQQTgfc=
|
||||
golang.org/x/tools v0.0.0-20190621195816-6e04913cbbac/go.mod h1:/rFqwRUd4F7ZHNgwSSTFct+R/Kf4OFW1sUzUTQQTgfc=
|
||||
golang.org/x/tools v0.0.0-20190628153133-6cdbf07be9d0/go.mod h1:/rFqwRUd4F7ZHNgwSSTFct+R/Kf4OFW1sUzUTQQTgfc=
|
||||
golang.org/x/tools v0.0.0-20190816200558-6889da9d5479/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo=
|
||||
golang.org/x/tools v0.0.0-20190911174233-4f2ddba30aff/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo=
|
||||
golang.org/x/tools v0.0.0-20191012152004-8de300cfc20a/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo=
|
||||
golang.org/x/tools v0.0.0-20191108193012-7d206e10da11/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo=
|
||||
golang.org/x/tools v0.0.0-20191113191852-77e3bb0ad9e7/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo=
|
||||
golang.org/x/tools v0.0.0-20191115202509-3a792d9c32b2/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo=
|
||||
golang.org/x/tools v0.0.0-20191119224855-298f0cb1881e/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo=
|
||||
golang.org/x/tools v0.0.0-20191125144606-a911d9008d1f/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo=
|
||||
golang.org/x/tools v0.0.0-20191130070609-6e064ea0cf2d/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo=
|
||||
golang.org/x/tools v0.0.0-20191216173652-a0e659d51361/go.mod h1:TB2adYChydJhpapKDTa4BR/hXlZSLoq2Wpct/0txZ28=
|
||||
golang.org/x/tools v0.0.0-20191227053925-7b8e75db28f4/go.mod h1:TB2adYChydJhpapKDTa4BR/hXlZSLoq2Wpct/0txZ28=
|
||||
golang.org/x/tools v0.0.0-20200117161641-43d50277825c/go.mod h1:TB2adYChydJhpapKDTa4BR/hXlZSLoq2Wpct/0txZ28=
|
||||
golang.org/x/tools v0.0.0-20200122220014-bf1340f18c4a/go.mod h1:TB2adYChydJhpapKDTa4BR/hXlZSLoq2Wpct/0txZ28=
|
||||
golang.org/x/tools v0.0.0-20200130002326-2f3ba24bd6e7/go.mod h1:TB2adYChydJhpapKDTa4BR/hXlZSLoq2Wpct/0txZ28=
|
||||
golang.org/x/tools v0.0.0-20200204074204-1cc6d1ef6c74/go.mod h1:TB2adYChydJhpapKDTa4BR/hXlZSLoq2Wpct/0txZ28=
|
||||
golang.org/x/tools v0.0.0-20200207183749-b753a1ba74fa/go.mod h1:TB2adYChydJhpapKDTa4BR/hXlZSLoq2Wpct/0txZ28=
|
||||
golang.org/x/tools v0.0.0-20200212150539-ea181f53ac56/go.mod h1:TB2adYChydJhpapKDTa4BR/hXlZSLoq2Wpct/0txZ28=
|
||||
golang.org/x/tools v0.0.0-20200224181240-023911ca70b2/go.mod h1:TB2adYChydJhpapKDTa4BR/hXlZSLoq2Wpct/0txZ28=
|
||||
golang.org/x/tools v0.0.0-20200227222343-706bc42d1f0d/go.mod h1:TB2adYChydJhpapKDTa4BR/hXlZSLoq2Wpct/0txZ28=
|
||||
golang.org/x/tools v0.0.0-20200304193943-95d2e580d8eb/go.mod h1:o4KQGtdN14AW+yjsvvwRTJJuXz8XRtIHtEnmAXLyFUw=
|
||||
golang.org/x/tools v0.0.0-20200312045724-11d5b4c81c7d/go.mod h1:o4KQGtdN14AW+yjsvvwRTJJuXz8XRtIHtEnmAXLyFUw=
|
||||
golang.org/x/tools v0.0.0-20200331025713-a30bf2db82d4/go.mod h1:Sl4aGygMT6LrqrWclx+PTx3U+LnKx/seiNR+3G19Ar8=
|
||||
golang.org/x/tools v0.0.0-20200501065659-ab2804fb9c9d/go.mod h1:EkVYQZoAsY45+roYkvgYkIh4xh/qjgUK9TdY2XT94GE=
|
||||
golang.org/x/tools v0.0.0-20200512131952-2bc93b1c0c88/go.mod h1:EkVYQZoAsY45+roYkvgYkIh4xh/qjgUK9TdY2XT94GE=
|
||||
golang.org/x/tools v0.0.0-20200515010526-7d3b6ebf133d/go.mod h1:EkVYQZoAsY45+roYkvgYkIh4xh/qjgUK9TdY2XT94GE=
|
||||
golang.org/x/tools v0.0.0-20200618134242-20370b0cb4b2/go.mod h1:EkVYQZoAsY45+roYkvgYkIh4xh/qjgUK9TdY2XT94GE=
|
||||
golang.org/x/tools v0.0.0-20200619180055-7c47624df98f/go.mod h1:EkVYQZoAsY45+roYkvgYkIh4xh/qjgUK9TdY2XT94GE=
|
||||
golang.org/x/tools v0.0.0-20200729194436-6467de6f59a7/go.mod h1:njjCfa9FT2d7l9Bc6FUM5FLjQPp3cFF28FI3qnDFljA=
|
||||
golang.org/x/tools v0.0.0-20200804011535-6c149bb5ef0d/go.mod h1:njjCfa9FT2d7l9Bc6FUM5FLjQPp3cFF28FI3qnDFljA=
|
||||
golang.org/x/tools v0.0.0-20200825202427-b303f430e36d/go.mod h1:njjCfa9FT2d7l9Bc6FUM5FLjQPp3cFF28FI3qnDFljA=
|
||||
golang.org/x/tools v0.0.0-20200904185747-39188db58858/go.mod h1:Cj7w3i3Rnn0Xh82ur9kSqwfTHTeVxaDqrfMjpcNT6bE=
|
||||
golang.org/x/tools v0.0.0-20201110124207-079ba7bd75cd/go.mod h1:emZCQorbCU4vsT4fOWvOPXz4eW1wZW4PmDk9uLelYpA=
|
||||
golang.org/x/tools v0.0.0-20201201161351-ac6f37ff4c2a/go.mod h1:emZCQorbCU4vsT4fOWvOPXz4eW1wZW4PmDk9uLelYpA=
|
||||
golang.org/x/tools v0.0.0-20201208233053-a543418bbed2/go.mod h1:emZCQorbCU4vsT4fOWvOPXz4eW1wZW4PmDk9uLelYpA=
|
||||
golang.org/x/tools v0.0.0-20210105154028-b0ab187a4818/go.mod h1:emZCQorbCU4vsT4fOWvOPXz4eW1wZW4PmDk9uLelYpA=
|
||||
golang.org/x/tools v0.0.0-20210106214847-113979e3529a/go.mod h1:emZCQorbCU4vsT4fOWvOPXz4eW1wZW4PmDk9uLelYpA=
|
||||
golang.org/x/tools v0.0.0-20210108195828-e2f9c7f1fc8e/go.mod h1:emZCQorbCU4vsT4fOWvOPXz4eW1wZW4PmDk9uLelYpA=
|
||||
golang.org/x/tools v0.1.0/go.mod h1:xkSsbof2nBLbhDlRMhhhyNLN/zl3eTqcnHD5viDpcZ0=
|
||||
golang.org/x/tools v0.1.12/go.mod h1:hNGJHUnrk76NpqgfD5Aqm5Crs+Hm0VOH/i9J2+nxYbc=
|
||||
golang.org/x/tools v0.2.0/go.mod h1:y4OqIKeOV/fWJetJ8bXPU1sEVniLMIyDAZWeHdV+NTA=
|
||||
golang.org/x/tools v0.6.0/go.mod h1:Xwgl3UAJ/d3gWutnCtw505GrjyAbvKui8lOU390QaIU=
|
||||
golang.org/x/tools v0.8.0 h1:vSDcovVPld282ceKgDimkRSC8kpaH1dgyc9UMzlt84Y=
|
||||
golang.org/x/tools v0.8.0/go.mod h1:JxBZ99ISMI5ViVkT1tr6tdNmXeTrcpVSD3vZ1RsRdN4=
|
||||
golang.org/x/tools v0.8.1-0.20230428195545-5283a0178901 h1:0wxTF6pSjIIhNt7mo9GvjDfzyCOiWhmICgtO/Ah948s=
|
||||
golang.org/x/tools v0.8.1-0.20230428195545-5283a0178901/go.mod h1:JxBZ99ISMI5ViVkT1tr6tdNmXeTrcpVSD3vZ1RsRdN4=
|
||||
golang.org/x/xerrors v0.0.0-20190717185122-a985d3407aa7/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0=
|
||||
golang.org/x/xerrors v0.0.0-20191011141410-1b5146add898/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0=
|
||||
golang.org/x/xerrors v0.0.0-20191204190536-9bdfabe68543/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0=
|
||||
golang.org/x/xerrors v0.0.0-20200804184101-5ec99f83aff1/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0=
|
||||
google.golang.org/api v0.4.0/go.mod h1:8k5glujaEP+g9n7WNsDg8QP6cUVNI86fCNMcbazEtwE=
|
||||
google.golang.org/api v0.7.0/go.mod h1:WtwebWUNSVBH/HAw79HIFXZNqEvBhG+Ra+ax0hx3E3M=
|
||||
google.golang.org/api v0.8.0/go.mod h1:o4eAsZoiT+ibD93RtjEohWalFOjRDx6CVaqeizhEnKg=
|
||||
google.golang.org/api v0.9.0/go.mod h1:o4eAsZoiT+ibD93RtjEohWalFOjRDx6CVaqeizhEnKg=
|
||||
google.golang.org/api v0.13.0/go.mod h1:iLdEw5Ide6rF15KTC1Kkl0iskquN2gFfn9o9XIsbkAI=
|
||||
google.golang.org/api v0.14.0/go.mod h1:iLdEw5Ide6rF15KTC1Kkl0iskquN2gFfn9o9XIsbkAI=
|
||||
google.golang.org/api v0.15.0/go.mod h1:iLdEw5Ide6rF15KTC1Kkl0iskquN2gFfn9o9XIsbkAI=
|
||||
google.golang.org/api v0.17.0/go.mod h1:BwFmGc8tA3vsd7r/7kR8DY7iEEGSU04BFxCo5jP/sfE=
|
||||
google.golang.org/api v0.18.0/go.mod h1:BwFmGc8tA3vsd7r/7kR8DY7iEEGSU04BFxCo5jP/sfE=
|
||||
google.golang.org/api v0.19.0/go.mod h1:BwFmGc8tA3vsd7r/7kR8DY7iEEGSU04BFxCo5jP/sfE=
|
||||
google.golang.org/api v0.20.0/go.mod h1:BwFmGc8tA3vsd7r/7kR8DY7iEEGSU04BFxCo5jP/sfE=
|
||||
google.golang.org/api v0.22.0/go.mod h1:BwFmGc8tA3vsd7r/7kR8DY7iEEGSU04BFxCo5jP/sfE=
|
||||
google.golang.org/api v0.24.0/go.mod h1:lIXQywCXRcnZPGlsd8NbLnOjtAoL6em04bJ9+z0MncE=
|
||||
google.golang.org/api v0.28.0/go.mod h1:lIXQywCXRcnZPGlsd8NbLnOjtAoL6em04bJ9+z0MncE=
|
||||
google.golang.org/api v0.29.0/go.mod h1:Lcubydp8VUV7KeIHD9z2Bys/sm/vGKnG1UHuDBSrHWM=
|
||||
google.golang.org/api v0.30.0/go.mod h1:QGmEvQ87FHZNiUVJkT14jQNYJ4ZJjdRF23ZXz5138Fc=
|
||||
google.golang.org/api v0.35.0/go.mod h1:/XrVsuzM0rZmrsbjJutiuftIzeuTQcEeaYcSk/mQ1dg=
|
||||
google.golang.org/api v0.36.0/go.mod h1:+z5ficQTmoYpPn8LCUNVpK5I7hwkpjbcgqA7I34qYtE=
|
||||
google.golang.org/api v0.40.0/go.mod h1:fYKFpnQN0DsDSKRVRcQSDQNtqWPfM9i+zNPxepjRCQ8=
|
||||
google.golang.org/appengine v1.1.0/go.mod h1:EbEs0AVv82hx2wNQdGPgUI5lhzA/G0D9YwlJXL52JkM=
|
||||
google.golang.org/appengine v1.4.0/go.mod h1:xpcJRLb0r/rnEns0DIKYYv+WjYCduHsrkT7/EB5XEv4=
|
||||
google.golang.org/appengine v1.5.0/go.mod h1:xpcJRLb0r/rnEns0DIKYYv+WjYCduHsrkT7/EB5XEv4=
|
||||
google.golang.org/appengine v1.6.1/go.mod h1:i06prIuMbXzDqacNJfV5OdTW448YApPu5ww/cMBSeb0=
|
||||
google.golang.org/appengine v1.6.5/go.mod h1:8WjMMxjGQR8xUklV/ARdw2HLXBOI7O7uCIDZVag1xfc=
|
||||
google.golang.org/appengine v1.6.6/go.mod h1:8WjMMxjGQR8xUklV/ARdw2HLXBOI7O7uCIDZVag1xfc=
|
||||
google.golang.org/appengine v1.6.7 h1:FZR1q0exgwxzPzp/aF+VccGrSfxfPpkBqjIIEq3ru6c=
|
||||
google.golang.org/appengine v1.6.7/go.mod h1:8WjMMxjGQR8xUklV/ARdw2HLXBOI7O7uCIDZVag1xfc=
|
||||
google.golang.org/genproto v0.0.0-20180817151627-c66870c02cf8/go.mod h1:JiN7NxoALGmiZfu7CAH4rXhgtRTLTxftemlI0sWmxmc=
|
||||
google.golang.org/genproto v0.0.0-20190307195333-5fe7a883aa19/go.mod h1:VzzqZJRnGkLBvHegQrXjBqPurQTc5/KpmUdxsrq26oE=
|
||||
google.golang.org/genproto v0.0.0-20190418145605-e7d98fc518a7/go.mod h1:VzzqZJRnGkLBvHegQrXjBqPurQTc5/KpmUdxsrq26oE=
|
||||
google.golang.org/genproto v0.0.0-20190425155659-357c62f0e4bb/go.mod h1:VzzqZJRnGkLBvHegQrXjBqPurQTc5/KpmUdxsrq26oE=
|
||||
google.golang.org/genproto v0.0.0-20190502173448-54afdca5d873/go.mod h1:VzzqZJRnGkLBvHegQrXjBqPurQTc5/KpmUdxsrq26oE=
|
||||
google.golang.org/genproto v0.0.0-20190801165951-fa694d86fc64/go.mod h1:DMBHOl98Agz4BDEuKkezgsaosCRResVns1a3J2ZsMNc=
|
||||
google.golang.org/genproto v0.0.0-20190819201941-24fa4b261c55/go.mod h1:DMBHOl98Agz4BDEuKkezgsaosCRResVns1a3J2ZsMNc=
|
||||
google.golang.org/genproto v0.0.0-20190911173649-1774047e7e51/go.mod h1:IbNlFCBrqXvoKpeg0TB2l7cyZUmoaFKYIwrEpbDKLA8=
|
||||
google.golang.org/genproto v0.0.0-20191108220845-16a3f7862a1a/go.mod h1:n3cpQtvxv34hfy77yVDNjmbRyujviMdxYliBSkLhpCc=
|
||||
google.golang.org/genproto v0.0.0-20191115194625-c23dd37a84c9/go.mod h1:n3cpQtvxv34hfy77yVDNjmbRyujviMdxYliBSkLhpCc=
|
||||
google.golang.org/genproto v0.0.0-20191216164720-4f79533eabd1/go.mod h1:n3cpQtvxv34hfy77yVDNjmbRyujviMdxYliBSkLhpCc=
|
||||
google.golang.org/genproto v0.0.0-20191230161307-f3c370f40bfb/go.mod h1:n3cpQtvxv34hfy77yVDNjmbRyujviMdxYliBSkLhpCc=
|
||||
google.golang.org/genproto v0.0.0-20200115191322-ca5a22157cba/go.mod h1:n3cpQtvxv34hfy77yVDNjmbRyujviMdxYliBSkLhpCc=
|
||||
google.golang.org/genproto v0.0.0-20200122232147-0452cf42e150/go.mod h1:n3cpQtvxv34hfy77yVDNjmbRyujviMdxYliBSkLhpCc=
|
||||
google.golang.org/genproto v0.0.0-20200204135345-fa8e72b47b90/go.mod h1:GmwEX6Z4W5gMy59cAlVYjN9JhxgbQH6Gn+gFDQe2lzA=
|
||||
google.golang.org/genproto v0.0.0-20200212174721-66ed5ce911ce/go.mod h1:55QSHmfGQM9UVYDPBsyGGes0y52j32PQ3BqQfXhyH3c=
|
||||
google.golang.org/genproto v0.0.0-20200224152610-e50cd9704f63/go.mod h1:55QSHmfGQM9UVYDPBsyGGes0y52j32PQ3BqQfXhyH3c=
|
||||
google.golang.org/genproto v0.0.0-20200228133532-8c2c7df3a383/go.mod h1:55QSHmfGQM9UVYDPBsyGGes0y52j32PQ3BqQfXhyH3c=
|
||||
google.golang.org/genproto v0.0.0-20200305110556-506484158171/go.mod h1:55QSHmfGQM9UVYDPBsyGGes0y52j32PQ3BqQfXhyH3c=
|
||||
google.golang.org/genproto v0.0.0-20200312145019-da6875a35672/go.mod h1:55QSHmfGQM9UVYDPBsyGGes0y52j32PQ3BqQfXhyH3c=
|
||||
google.golang.org/genproto v0.0.0-20200331122359-1ee6d9798940/go.mod h1:55QSHmfGQM9UVYDPBsyGGes0y52j32PQ3BqQfXhyH3c=
|
||||
google.golang.org/genproto v0.0.0-20200423170343-7949de9c1215/go.mod h1:55QSHmfGQM9UVYDPBsyGGes0y52j32PQ3BqQfXhyH3c=
|
||||
google.golang.org/genproto v0.0.0-20200430143042-b979b6f78d84/go.mod h1:55QSHmfGQM9UVYDPBsyGGes0y52j32PQ3BqQfXhyH3c=
|
||||
google.golang.org/genproto v0.0.0-20200511104702-f5ebc3bea380/go.mod h1:55QSHmfGQM9UVYDPBsyGGes0y52j32PQ3BqQfXhyH3c=
|
||||
google.golang.org/genproto v0.0.0-20200515170657-fc4c6c6a6587/go.mod h1:YsZOwe1myG/8QRHRsmBRE1LrgQY60beZKjly0O1fX9U=
|
||||
google.golang.org/genproto v0.0.0-20200526211855-cb27e3aa2013/go.mod h1:NbSheEEYHJ7i3ixzK3sjbqSGDJWnxyFXZblF3eUsNvo=
|
||||
google.golang.org/genproto v0.0.0-20200618031413-b414f8b61790/go.mod h1:jDfRM7FcilCzHH/e9qn6dsT145K34l5v+OpcnNgKAAA=
|
||||
google.golang.org/genproto v0.0.0-20200729003335-053ba62fc06f/go.mod h1:FWY/as6DDZQgahTzZj3fqbO1CbirC29ZNUFHwi0/+no=
|
||||
google.golang.org/genproto v0.0.0-20200804131852-c06518451d9c/go.mod h1:FWY/as6DDZQgahTzZj3fqbO1CbirC29ZNUFHwi0/+no=
|
||||
google.golang.org/genproto v0.0.0-20200825200019-8632dd797987/go.mod h1:FWY/as6DDZQgahTzZj3fqbO1CbirC29ZNUFHwi0/+no=
|
||||
google.golang.org/genproto v0.0.0-20200904004341-0bd0a958aa1d/go.mod h1:FWY/as6DDZQgahTzZj3fqbO1CbirC29ZNUFHwi0/+no=
|
||||
google.golang.org/genproto v0.0.0-20201109203340-2640f1f9cdfb/go.mod h1:FWY/as6DDZQgahTzZj3fqbO1CbirC29ZNUFHwi0/+no=
|
||||
google.golang.org/genproto v0.0.0-20201201144952-b05cb90ed32e/go.mod h1:FWY/as6DDZQgahTzZj3fqbO1CbirC29ZNUFHwi0/+no=
|
||||
google.golang.org/genproto v0.0.0-20201210142538-e3217bee35cc/go.mod h1:FWY/as6DDZQgahTzZj3fqbO1CbirC29ZNUFHwi0/+no=
|
||||
google.golang.org/genproto v0.0.0-20201214200347-8c77b98c765d/go.mod h1:FWY/as6DDZQgahTzZj3fqbO1CbirC29ZNUFHwi0/+no=
|
||||
google.golang.org/genproto v0.0.0-20210108203827-ffc7fda8c3d7/go.mod h1:FWY/as6DDZQgahTzZj3fqbO1CbirC29ZNUFHwi0/+no=
|
||||
google.golang.org/genproto v0.0.0-20210226172003-ab064af71705/go.mod h1:FWY/as6DDZQgahTzZj3fqbO1CbirC29ZNUFHwi0/+no=
|
||||
google.golang.org/genproto v0.0.0-20230410155749-daa745c078e1 h1:KpwkzHKEF7B9Zxg18WzOa7djJ+Ha5DzthMyZYQfEn2A=
|
||||
google.golang.org/genproto v0.0.0-20230410155749-daa745c078e1/go.mod h1:nKE/iIaLqn2bQwXBg8f1g2Ylh6r5MN5CmZvuzZCgsCU=
|
||||
google.golang.org/grpc v1.19.0/go.mod h1:mqu4LbDTu4XGKhr4mRzUsmM4RtVoemTSY81AxZiDr8c=
|
||||
google.golang.org/grpc v1.20.1/go.mod h1:10oTOabMzJvdu6/UiuZezV6QK5dSlG84ov/aaiqXj38=
|
||||
google.golang.org/grpc v1.21.1/go.mod h1:oYelfM1adQP15Ek0mdvEgi9Df8B9CZIaU1084ijfRaM=
|
||||
google.golang.org/grpc v1.23.0/go.mod h1:Y5yQAOtifL1yxbo5wqy6BxZv8vAUGQwXBOALyacEbxg=
|
||||
google.golang.org/grpc v1.25.1/go.mod h1:c3i+UQWmh7LiEpx4sFZnkU36qjEYZ0imhYfXVyQciAY=
|
||||
google.golang.org/grpc v1.26.0/go.mod h1:qbnxyOmOxrQa7FizSgH+ReBfzJrCY1pSN7KXBS8abTk=
|
||||
google.golang.org/grpc v1.27.0/go.mod h1:qbnxyOmOxrQa7FizSgH+ReBfzJrCY1pSN7KXBS8abTk=
|
||||
google.golang.org/grpc v1.27.1/go.mod h1:qbnxyOmOxrQa7FizSgH+ReBfzJrCY1pSN7KXBS8abTk=
|
||||
google.golang.org/grpc v1.28.0/go.mod h1:rpkK4SK4GF4Ach/+MFLZUBavHOvF2JJB5uozKKal+60=
|
||||
google.golang.org/grpc v1.29.1/go.mod h1:itym6AZVZYACWQqET3MqgPpjcuV5QH3BxFS3IjizoKk=
|
||||
google.golang.org/grpc v1.30.0/go.mod h1:N36X2cJ7JwdamYAgDz+s+rVMFjt3numwzf/HckM8pak=
|
||||
google.golang.org/grpc v1.31.0/go.mod h1:N36X2cJ7JwdamYAgDz+s+rVMFjt3numwzf/HckM8pak=
|
||||
google.golang.org/grpc v1.31.1/go.mod h1:N36X2cJ7JwdamYAgDz+s+rVMFjt3numwzf/HckM8pak=
|
||||
google.golang.org/grpc v1.33.2/go.mod h1:JMHMWHQWaTccqQQlmk3MJZS+GWXOdAesneDmEnv2fbc=
|
||||
google.golang.org/grpc v1.34.0/go.mod h1:WotjhfgOW/POjDeRt8vscBtXq+2VjORFy659qA51WJ8=
|
||||
google.golang.org/grpc v1.35.0/go.mod h1:qjiiYl8FncCW8feJPdyg3v6XW24KsRHe+dy9BAGRRjU=
|
||||
google.golang.org/grpc v1.54.0 h1:EhTqbhiYeixwWQtAEZAxmV9MGqcjEU2mFx52xCzNyag=
|
||||
google.golang.org/grpc v1.54.0/go.mod h1:PUSEXI6iWghWaB6lXM4knEgpJNu2qUcKfDtNci3EC2g=
|
||||
google.golang.org/protobuf v0.0.0-20200109180630-ec00e32a8dfd/go.mod h1:DFci5gLYBciE7Vtevhsrf46CRTquxDuWsQurQQe4oz8=
|
||||
google.golang.org/protobuf v0.0.0-20200221191635-4d8936d0db64/go.mod h1:kwYJMbMJ01Woi6D6+Kah6886xMZcty6N08ah7+eCXa0=
|
||||
google.golang.org/protobuf v0.0.0-20200228230310-ab0ca4ff8a60/go.mod h1:cfTl7dwQJ+fmap5saPgwCLgHXTUD7jkjRqWcaiX5VyM=
|
||||
google.golang.org/protobuf v1.20.1-0.20200309200217-e05f789c0967/go.mod h1:A+miEFZTKqfCUM6K7xSMQL9OKL/b6hQv+e19PK+JZNE=
|
||||
google.golang.org/protobuf v1.21.0/go.mod h1:47Nbq4nVaFHyn7ilMalzfO3qCViNmqZ2kzikPIcrTAo=
|
||||
google.golang.org/protobuf v1.22.0/go.mod h1:EGpADcykh3NcUnDUJcl1+ZksZNG86OlYog2l/sGQquU=
|
||||
google.golang.org/protobuf v1.23.0/go.mod h1:EGpADcykh3NcUnDUJcl1+ZksZNG86OlYog2l/sGQquU=
|
||||
google.golang.org/protobuf v1.23.1-0.20200526195155-81db48ad09cc/go.mod h1:EGpADcykh3NcUnDUJcl1+ZksZNG86OlYog2l/sGQquU=
|
||||
google.golang.org/protobuf v1.24.0/go.mod h1:r/3tXBNzIEhYS9I1OUVjXDlt8tc493IdKGjtUeSXeh4=
|
||||
google.golang.org/protobuf v1.25.0/go.mod h1:9JNX74DMeImyA3h4bdi1ymwjUzf21/xIlbajtzgsN7c=
|
||||
google.golang.org/protobuf v1.26.0-rc.1/go.mod h1:jlhhOSvTdKEhbULTjvd4ARK9grFBp09yW+WbY/TyQbw=
|
||||
google.golang.org/protobuf v1.26.0/go.mod h1:9q0QmTI4eRPtz6boOQmLYwt+qCgq0jsYwAQnmE0givc=
|
||||
google.golang.org/protobuf v1.30.0 h1:kPPoIgf3TsEvrm0PFe15JQ+570QVxYzEvvHqChK+cng=
|
||||
|
@ -659,8 +360,6 @@ gopkg.in/check.v1 v1.0.0-20200227125254-8fa46927fb4f/go.mod h1:Co6ibVJAznAaIkqp8
|
|||
gopkg.in/check.v1 v1.0.0-20201130134442-10cb98267c6c h1:Hei/4ADfdWqJk1ZMxUNpqntNwaWcugrBjAiHlqqRiVk=
|
||||
gopkg.in/check.v1 v1.0.0-20201130134442-10cb98267c6c/go.mod h1:JHkPIbrfpd72SG/EVd6muEfDQjcINNoR0C8j2r3qZ4Q=
|
||||
gopkg.in/errgo.v2 v2.1.0/go.mod h1:hNsd1EY+bozCKY1Ytp96fpM3vjJbqLJn88ws8XvfDNI=
|
||||
gopkg.in/ini.v1 v1.67.0 h1:Dgnx+6+nfE+IfzjUEISNeydPJh9AXNNsWbGP9KzCsOA=
|
||||
gopkg.in/ini.v1 v1.67.0/go.mod h1:pNLf8WUiyNEtQjuu5G5vTm06TEv9tsIgeAvK8hOrP4k=
|
||||
gopkg.in/warnings.v0 v0.1.2 h1:wFXVbFY8DY5/xOe1ECiWdKCzZlxgshcYVNkBHstARME=
|
||||
gopkg.in/warnings.v0 v0.1.2/go.mod h1:jksf8JmL6Qr/oQM2OXTHunEvvTAsrWBLb6OOjuVWRNI=
|
||||
gopkg.in/yaml.v2 v2.2.2/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI=
|
||||
|
@ -674,13 +373,29 @@ gopkg.in/yaml.v3 v3.0.1 h1:fxVm/GzAzEWqLHuvctI91KS9hhNmmWOoWu0XTYJS7CA=
|
|||
gopkg.in/yaml.v3 v3.0.1/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM=
|
||||
gotest.tools/v3 v3.0.3 h1:4AuOwCGf4lLR9u3YOe2awrHygurzhO/HeQ6laiA6Sx0=
|
||||
honnef.co/go/tools v0.0.0-20190102054323-c2f93a96b099/go.mod h1:rf3lG4BRIbNafJWhAfAdb/ePZxsR/4RtNHQocxwk9r4=
|
||||
honnef.co/go/tools v0.0.0-20190106161140-3f1c8253044a/go.mod h1:rf3lG4BRIbNafJWhAfAdb/ePZxsR/4RtNHQocxwk9r4=
|
||||
honnef.co/go/tools v0.0.0-20190418001031-e561f6794a2a/go.mod h1:rf3lG4BRIbNafJWhAfAdb/ePZxsR/4RtNHQocxwk9r4=
|
||||
honnef.co/go/tools v0.0.0-20190523083050-ea95bdfd59fc/go.mod h1:rf3lG4BRIbNafJWhAfAdb/ePZxsR/4RtNHQocxwk9r4=
|
||||
honnef.co/go/tools v0.0.1-2019.2.3/go.mod h1:a3bituU0lyd329TUQxRnasdCoJDkEUEAqEt0JzvZhAg=
|
||||
honnef.co/go/tools v0.0.1-2020.1.3/go.mod h1:X/FiERA/W4tHapMX5mGpAtMSVEeEUOyHaw9vFzvIQ3k=
|
||||
honnef.co/go/tools v0.0.1-2020.1.4/go.mod h1:X/FiERA/W4tHapMX5mGpAtMSVEeEUOyHaw9vFzvIQ3k=
|
||||
rsc.io/binaryregexp v0.2.0/go.mod h1:qTv7/COck+e2FymRvadv62gMdZztPaShugOCi3I+8D8=
|
||||
lukechampine.com/uint128 v1.3.0 h1:cDdUVfRwDUDovz610ABgFD17nXD4/uDgVHl2sC3+sbo=
|
||||
lukechampine.com/uint128 v1.3.0/go.mod h1:c4eWIwlEGaxC/+H1VguhU4PHXNWDCDMUlWdIWl2j1gk=
|
||||
modernc.org/cc/v3 v3.40.0 h1:P3g79IUS/93SYhtoeaHW+kRCIrYaxJ27MFPv+7kaTOw=
|
||||
modernc.org/cc/v3 v3.40.0/go.mod h1:/bTg4dnWkSXowUO6ssQKnOV0yMVxDYNIsIrzqTFDGH0=
|
||||
modernc.org/ccgo/v3 v3.16.13 h1:Mkgdzl46i5F/CNR/Kj80Ri59hC8TKAhZrYSaqvkwzUw=
|
||||
modernc.org/ccgo/v3 v3.16.13/go.mod h1:2Quk+5YgpImhPjv2Qsob1DnZ/4som1lJTodubIcoUkY=
|
||||
modernc.org/ccorpus v1.11.6 h1:J16RXiiqiCgua6+ZvQot4yUuUy8zxgqbqEEUuGPlISk=
|
||||
modernc.org/httpfs v1.0.6 h1:AAgIpFZRXuYnkjftxTAZwMIiwEqAfk8aVB2/oA6nAeM=
|
||||
modernc.org/libc v1.22.5 h1:91BNch/e5B0uPbJFgqbxXuOnxBQjlS//icfQEGmvyjE=
|
||||
modernc.org/libc v1.22.5/go.mod h1:jj+Z7dTNX8fBScMVNRAYZ/jF91K8fdT2hYMThc3YjBY=
|
||||
modernc.org/mathutil v1.5.0 h1:rV0Ko/6SfM+8G+yKiyI830l3Wuz1zRutdslNoQ0kfiQ=
|
||||
modernc.org/mathutil v1.5.0/go.mod h1:mZW8CKdRPY1v87qxC/wUdX5O1qDzXMP5TH3wjfpga6E=
|
||||
modernc.org/memory v1.5.0 h1:N+/8c5rE6EqugZwHii4IFsaJ7MUhoWX07J5tC/iI5Ds=
|
||||
modernc.org/memory v1.5.0/go.mod h1:PkUhL0Mugw21sHPeskwZW4D6VscE/GQJOnIpCnW6pSU=
|
||||
modernc.org/opt v0.1.3 h1:3XOZf2yznlhC+ibLltsDGzABUGVx8J6pnFMS3E4dcq4=
|
||||
modernc.org/opt v0.1.3/go.mod h1:WdSiB5evDcignE70guQKxYUl14mgWtbClRi5wmkkTX0=
|
||||
modernc.org/sqlite v1.22.1 h1:P2+Dhp5FR1RlVRkQ3dDfCiv3Ok8XPxqpe70IjYVA9oE=
|
||||
modernc.org/sqlite v1.22.1/go.mod h1:OrDj17Mggn6MhE+iPbBNf7RGKODDE9NFT0f3EwDzJqk=
|
||||
modernc.org/strutil v1.1.3 h1:fNMm+oJklMGYfU9Ylcywl0CO5O6nTfaowNsh2wpPjzY=
|
||||
modernc.org/strutil v1.1.3/go.mod h1:MEHNA7PdEnEwLvspRMtWTNnp2nnyvMfkimT1NKNAGbw=
|
||||
modernc.org/tcl v1.15.2 h1:C4ybAYCGJw968e+Me18oW55kD/FexcHbqH2xak1ROSY=
|
||||
modernc.org/token v1.1.0 h1:Xl7Ap9dKaEs5kLoOQeQmPWevfnk/DM5qcLcYlA8ys6Y=
|
||||
modernc.org/token v1.1.0/go.mod h1:UGzOrNV1mAFSEB63lOFHIpNRUVMvYTc6yu1SMY/XTDM=
|
||||
modernc.org/z v1.7.3 h1:zDJf6iHjrnB+WRD88stbXokugjyc0/pB91ri1gO6LZY=
|
||||
rsc.io/pdf v0.1.1/go.mod h1:n8OzWcQ6Sp37PL01nO98y4iUCRdTGarVfzxY20ICaU4=
|
||||
rsc.io/quote/v3 v3.1.0/go.mod h1:yEA65RcK8LyAZtP9Kv3t0HmxON59tX3rD+tICJqUlj0=
|
||||
rsc.io/sampler v1.3.0/go.mod h1:T1hPZKmBbMNahiBKFy5HrXp6adAjACjK9JXDnKaTXpA=
|
||||
|
|
|
@ -1,16 +1,22 @@
|
|||
package cmd
|
||||
|
||||
import (
|
||||
"bytes"
|
||||
"context"
|
||||
"crypto/sha256"
|
||||
"encoding/json"
|
||||
"errors"
|
||||
"fmt"
|
||||
"os"
|
||||
"path/filepath"
|
||||
"runtime"
|
||||
"runtime/pprof"
|
||||
"strings"
|
||||
|
||||
"code.icb4dc0.de/buildr/buildr/internal/profiling"
|
||||
"code.icb4dc0.de/buildr/buildr/modules/state"
|
||||
|
||||
"github.com/go-git/go-git/v5/plumbing/hash"
|
||||
|
||||
"code.icb4dc0.de/buildr/buildr/internal/containers"
|
||||
"code.icb4dc0.de/buildr/buildr/internal/errs"
|
||||
"code.icb4dc0.de/buildr/buildr/internal/execution"
|
||||
|
@ -95,8 +101,8 @@ func NewApp() *App {
|
|||
app.ArgProviderFor(modules.CategoryPackage),
|
||||
WithShort("Create a package by its name"),
|
||||
),
|
||||
VaultCommand(NewVaultApp(app, app.Collection, app)),
|
||||
ServerCommand(NewServerApp(app.Collection)),
|
||||
VaultCommand(NewVaultApp(app, app, app)),
|
||||
ServerCommand(NewServerApp(app, app)),
|
||||
EnvCommand(NewEnvApp(app, app)),
|
||||
VersionCommand(),
|
||||
)
|
||||
|
@ -117,7 +123,7 @@ type App struct {
|
|||
loggingCfg logging.Config
|
||||
appCfg AppConfig
|
||||
rootCmd *cobra.Command
|
||||
profile *pprof.Profile
|
||||
recorder *profiling.Recorder
|
||||
initializers map[InitLevel]AppInitializer
|
||||
buildrInstance *buildr.Buildr
|
||||
repo *modules.Repository
|
||||
|
@ -155,8 +161,8 @@ func (a *App) SetInitializer(lvl InitLevel, init AppInitializer) {
|
|||
func (a *App) Init(ctx context.Context) (err error) {
|
||||
slog.SetDefault(a.loggingCfg.Logger())
|
||||
|
||||
if profilingCfg := a.appCfg.Profiling; profilingCfg.IsConfigured() {
|
||||
a.profile = pprof.Lookup(profilingCfg.ProfileName)
|
||||
if a.recorder, err = a.appCfg.Profiling.Setup(); err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
registry := modules.NewRegistry(
|
||||
|
@ -189,7 +195,7 @@ func (a *App) Init(ctx context.Context) (err error) {
|
|||
}
|
||||
|
||||
func (a *App) Shutdown() error {
|
||||
return errors.Join(a.persistVaultState(), a.writeProfile(), a.Collection.Close())
|
||||
return errors.Join(a.persistVaultState(), a.recorder.Close(), a.Collection.Close())
|
||||
}
|
||||
|
||||
func (a *App) AppConfig() AppConfig {
|
||||
|
@ -201,17 +207,25 @@ func (a *App) BuildR() buildr.Buildr {
|
|||
}
|
||||
|
||||
func (a *App) initBasic(ctx context.Context) (err error) {
|
||||
var v *vault.Vault
|
||||
var (
|
||||
v *vault.Vault
|
||||
s state.Store
|
||||
)
|
||||
|
||||
if v, err = a.appCfg.InitVault(); err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
return a.Collection.With(services.WithVault(v))
|
||||
if s, err = a.appCfg.InitState(ctx); err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
return a.Collection.With(services.WithVault(v), services.WithStateStore(s))
|
||||
}
|
||||
|
||||
func (a *App) RunModule(ctx context.Context, cat modules.Category, name string) error {
|
||||
if err := a.InitAt(InitLevelParseConfig); err != nil {
|
||||
return err
|
||||
}
|
||||
orchestrator, err := containers.NewOrchestrator(ctx, a.DockerClient(), a.Ignorer())
|
||||
if err != nil {
|
||||
|
@ -219,8 +233,8 @@ func (a *App) RunModule(ctx context.Context, cat modules.Category, name string)
|
|||
}
|
||||
|
||||
factory := execution.NewTaskFactory(
|
||||
execution.WithProvider(local.Provider()),
|
||||
execution.WithProvider(container.Provider(orchestrator, a.repo)),
|
||||
execution.WithProvider(local.Provider(a.StateStore())),
|
||||
execution.WithProvider(container.Provider(orchestrator, a.repo, a.StateStore())),
|
||||
)
|
||||
|
||||
plan, err := execution.NewPlanFor(cat, name, a.repo, factory)
|
||||
|
@ -313,6 +327,10 @@ func (a *App) initBuildRConfig(ctx context.Context) (err error) {
|
|||
return err
|
||||
}
|
||||
|
||||
if err = a.Collection.With(services.WithGitHubTokenClient(ctx, a.buildrInstance.GitHub.APIToken)); err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
if err = copyCurrentBinaryToBinariesDir(a.buildrInstance.Config.BinariesDirectory); err != nil {
|
||||
return err
|
||||
}
|
||||
|
@ -338,24 +356,6 @@ func (a *App) initParseConfigs(ctx context.Context) (err error) {
|
|||
return nil
|
||||
}
|
||||
|
||||
func (a *App) writeProfile() (err error) {
|
||||
profilingCfg := a.appCfg.Profiling
|
||||
if !profilingCfg.IsConfigured() || a.profile == nil {
|
||||
return nil
|
||||
}
|
||||
|
||||
f, err := os.Create(profilingCfg.OutFile)
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
defer func() {
|
||||
err = errors.Join(err, f.Close())
|
||||
}()
|
||||
|
||||
return a.profile.WriteTo(f, 0)
|
||||
}
|
||||
|
||||
func (a *App) persistVaultState() error {
|
||||
if a.Vault() == nil {
|
||||
return nil
|
||||
|
@ -376,12 +376,6 @@ func (a *App) persistVaultState() error {
|
|||
}
|
||||
|
||||
func copyCurrentBinaryToBinariesDir(binariesDir string) (err error) {
|
||||
expectedBuildrBinPath := filepath.Join(binariesDir, fmt.Sprintf("buildr_%s_%s", runtime.GOOS, runtime.GOARCH))
|
||||
|
||||
if _, err := os.Stat(expectedBuildrBinPath); err == nil {
|
||||
return nil
|
||||
}
|
||||
|
||||
currentBinaryPath, err := os.Executable()
|
||||
if err != nil {
|
||||
return err
|
||||
|
@ -396,6 +390,7 @@ func copyCurrentBinaryToBinariesDir(binariesDir string) (err error) {
|
|||
err = errors.Join(err, currentBinary.Close())
|
||||
}()
|
||||
|
||||
expectedBuildrBinPath := filepath.Join(binariesDir, fmt.Sprintf("buildr_%s_%s", runtime.GOOS, runtime.GOARCH))
|
||||
outFile, err := os.OpenFile(expectedBuildrBinPath, os.O_RDWR|os.O_CREATE|os.O_TRUNC, 0o755)
|
||||
if err != nil {
|
||||
return err
|
||||
|
@ -405,7 +400,41 @@ func copyCurrentBinaryToBinariesDir(binariesDir string) (err error) {
|
|||
err = errors.Join(err, outFile.Close())
|
||||
}()
|
||||
|
||||
_, err = ioutils.CopyWithPooledBuffer(outFile, currentBinary)
|
||||
var shouldCopy bool
|
||||
|
||||
h := sha256.New()
|
||||
expectedHash, err := fileHash(h, currentBinaryPath)
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
currentBinaryHash, err := fileHash(h, expectedBuildrBinPath)
|
||||
if err != nil {
|
||||
if os.IsNotExist(err) {
|
||||
shouldCopy = true
|
||||
} else {
|
||||
return err
|
||||
}
|
||||
}
|
||||
|
||||
if shouldCopy || !bytes.Equal(expectedHash, currentBinaryHash) {
|
||||
_, err = ioutils.CopyWithPooledBuffer(outFile, currentBinary)
|
||||
}
|
||||
|
||||
return err
|
||||
}
|
||||
|
||||
func fileHash(h hash.Hash, path string) (hash []byte, err error) {
|
||||
f, err := os.Open(path)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
|
||||
defer f.Close()
|
||||
|
||||
_, _ = ioutils.CopyWithPooledBuffer(h, f)
|
||||
|
||||
hash = h.Sum(nil)
|
||||
h.Reset()
|
||||
return hash, nil
|
||||
}
|
||||
|
|
|
@ -1,12 +0,0 @@
|
|||
package cmd
|
||||
|
||||
import (
|
||||
"runtime/pprof"
|
||||
|
||||
"code.icb4dc0.de/buildr/buildr/internal/profiling"
|
||||
)
|
||||
|
||||
type PprofApp struct {
|
||||
profilingCfg *profiling.Config
|
||||
profile *pprof.Profile
|
||||
}
|
|
@ -11,17 +11,23 @@ import (
|
|||
|
||||
var _ ServerCommander = (*ServerApp)(nil)
|
||||
|
||||
func NewServerApp(accessor services.TypeRegistryAccessor) *ServerApp {
|
||||
func NewServerApp(initializer LevelInitializer, accessor services.TypeRegistryAccessor) *ServerApp {
|
||||
return &ServerApp{
|
||||
accessor: accessor,
|
||||
initializer: initializer,
|
||||
accessor: accessor,
|
||||
}
|
||||
}
|
||||
|
||||
type ServerApp struct {
|
||||
accessor services.TypeRegistryAccessor
|
||||
initializer LevelInitializer
|
||||
accessor services.TypeRegistryAccessor
|
||||
}
|
||||
|
||||
func (s *ServerApp) ServeAPI(ctx context.Context, cfg *rpc.GrpcConfig) error {
|
||||
if err := s.initializer.InitAt(InitLevelBasic); err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
logger := slog.Default()
|
||||
|
||||
logger.Info("Starting gRPC server", slog.Group("grpc", slog.String("addr", cfg.Host.Address)))
|
||||
|
|
|
@ -1,12 +1,16 @@
|
|||
package cmd
|
||||
|
||||
import (
|
||||
"context"
|
||||
"errors"
|
||||
"flag"
|
||||
"io/fs"
|
||||
"os"
|
||||
"path/filepath"
|
||||
"strconv"
|
||||
"time"
|
||||
|
||||
"code.icb4dc0.de/buildr/buildr/modules/state"
|
||||
|
||||
"code.icb4dc0.de/buildr/buildr/internal/config"
|
||||
"code.icb4dc0.de/buildr/buildr/internal/ignore"
|
||||
|
@ -22,7 +26,13 @@ type AppConfig struct {
|
|||
VCSType vcs.Type
|
||||
BuildRDirectory string
|
||||
RepoRoot string
|
||||
Vault struct {
|
||||
Cache struct {
|
||||
TTL time.Duration
|
||||
}
|
||||
State struct {
|
||||
FilePath string
|
||||
}
|
||||
Vault struct {
|
||||
FilePath string
|
||||
Passphrase string
|
||||
PassphraseFile string
|
||||
|
@ -55,9 +65,21 @@ func (c *AppConfig) InitPaths(from string) (err error) {
|
|||
c.BuildRDirectory = filepath.Join(c.RepoRoot, c.BuildRDirectory)
|
||||
}
|
||||
|
||||
if stateFilePath := c.State.FilePath; stateFilePath != "" && !filepath.IsAbs(stateFilePath) {
|
||||
c.State.FilePath = filepath.Join(c.RepoRoot, stateFilePath)
|
||||
}
|
||||
|
||||
return nil
|
||||
}
|
||||
|
||||
func (c *AppConfig) InitState(ctx context.Context) (state.Store, error) {
|
||||
if stateFilePath := c.State.FilePath; stateFilePath != "" {
|
||||
return state.NewEntStore(ctx, c.State.FilePath)
|
||||
}
|
||||
|
||||
return nil, nil
|
||||
}
|
||||
|
||||
func (c *AppConfig) InitVault() (*vault.Vault, error) {
|
||||
var vaultOpts []vault.InitOption
|
||||
|
||||
|
@ -124,6 +146,20 @@ func (c *AppConfig) Flags() *flag.FlagSet {
|
|||
"If output of commands should be piped to stdout",
|
||||
)
|
||||
|
||||
flags.StringVar(
|
||||
&c.State.FilePath,
|
||||
"state.file-path",
|
||||
config.StringEnvOr("BUILDR_STATE_FILE_PATH", filepath.Join(".buildr", "state.sqlite")),
|
||||
"Relative file path to state file",
|
||||
)
|
||||
|
||||
flags.DurationVar(
|
||||
&c.Cache.TTL,
|
||||
"cache.ttl",
|
||||
config.EnvOr("BUILDR_CACHE_TTL", time.ParseDuration, 15*time.Minute),
|
||||
"TTL for cache entries",
|
||||
)
|
||||
|
||||
flags.Var(
|
||||
&c.VCSType,
|
||||
"vcs.type",
|
||||
|
|
|
@ -74,6 +74,8 @@ func (o *Orchestrator) BuildRContainer(ctx context.Context, spec BuildRContainer
|
|||
return nil, nil, err
|
||||
}
|
||||
|
||||
containerRepoRoot := path.Join("/", "work", spec.ID)
|
||||
|
||||
conSpec := ContainerSpec{
|
||||
Image: spec.Image,
|
||||
User: spec.User,
|
||||
|
@ -82,7 +84,8 @@ func (o *Orchestrator) BuildRContainer(ctx context.Context, spec BuildRContainer
|
|||
ExposedPorts: []string{"3000/tcp"},
|
||||
Env: map[string]string{
|
||||
"BUILDR_GRPC_SERVE_ADDRESS": "0.0.0.0:3000",
|
||||
"BUILDR_REPO_ROOT": spec.RepoRoot,
|
||||
"BUILDR_REPO_ROOT": containerRepoRoot,
|
||||
"BUILDR_STATE_FILE_PATH": "",
|
||||
},
|
||||
Entrypoint: []string{
|
||||
fmt.Sprintf("/opt/buildr/bin/buildr_%s_%s", o.dockerHostOSType, o.dockerHostArch),
|
||||
|
@ -111,7 +114,7 @@ func (o *Orchestrator) BuildRContainer(ctx context.Context, spec BuildRContainer
|
|||
spec.Content[path.Join(spec.BinariesDir, extraBin)] = path.Join("/", "opt", "buildr", "bin", extraBin)
|
||||
}
|
||||
|
||||
if err := o.copyFilesToContainer(containerSetupCtx, spec.RepoRoot, con, spec.Content, path.Join("/", "work", spec.ID)); err != nil {
|
||||
if err := o.copyFilesToContainer(containerSetupCtx, spec.RepoRoot, con, spec.Content, containerRepoRoot); err != nil {
|
||||
return nil, nil, err
|
||||
}
|
||||
|
||||
|
|
7
internal/errs/logging.go
Normal file
7
internal/errs/logging.go
Normal file
|
@ -0,0 +1,7 @@
|
|||
package errs
|
||||
|
||||
import "golang.org/x/exp/slog"
|
||||
|
||||
func Attr(err error) slog.Attr {
|
||||
return slog.String("err", err.Error())
|
||||
}
|
|
@ -3,6 +3,8 @@ package container
|
|||
import (
|
||||
"fmt"
|
||||
|
||||
"code.icb4dc0.de/buildr/buildr/modules/state"
|
||||
|
||||
"code.icb4dc0.de/buildr/buildr/internal/containers"
|
||||
"code.icb4dc0.de/buildr/buildr/internal/execution"
|
||||
"code.icb4dc0.de/buildr/buildr/modules"
|
||||
|
@ -10,16 +12,18 @@ import (
|
|||
|
||||
var _ execution.TaskProvider = (*taskProvider)(nil)
|
||||
|
||||
func Provider(orchestrator *containers.Orchestrator, repo *modules.Repository) execution.TaskProvider {
|
||||
func Provider(orchestrator *containers.Orchestrator, repo *modules.Repository, stateStore state.Store) execution.TaskProvider {
|
||||
return taskProvider{
|
||||
orchestrator: orchestrator,
|
||||
repo: repo,
|
||||
stateStore: stateStore,
|
||||
}
|
||||
}
|
||||
|
||||
type taskProvider struct {
|
||||
orchestrator *containers.Orchestrator
|
||||
repo *modules.Repository
|
||||
stateStore state.Store
|
||||
}
|
||||
|
||||
func (taskProvider) CanProvide(m modules.ModuleWithMeta) bool {
|
||||
|
@ -34,6 +38,7 @@ func (p taskProvider) Create(m modules.ModuleWithMeta) (execution.Task, error) {
|
|||
moduleWithMeta: m,
|
||||
orchestrator: p.orchestrator,
|
||||
repo: p.repo,
|
||||
stateStore: p.stateStore,
|
||||
}, nil
|
||||
}
|
||||
}
|
||||
|
|
|
@ -3,6 +3,7 @@ package container
|
|||
import (
|
||||
"archive/tar"
|
||||
"context"
|
||||
"encoding/hex"
|
||||
"encoding/json"
|
||||
"errors"
|
||||
"fmt"
|
||||
|
@ -11,6 +12,9 @@ import (
|
|||
"path/filepath"
|
||||
"sync"
|
||||
|
||||
"code.icb4dc0.de/buildr/buildr/internal/errs"
|
||||
"code.icb4dc0.de/buildr/buildr/modules/state"
|
||||
|
||||
"github.com/klauspost/compress/s2"
|
||||
"golang.org/x/exp/slog"
|
||||
|
||||
|
@ -28,6 +32,7 @@ type containerTask struct {
|
|||
once sync.Once
|
||||
orchestrator *containers.Orchestrator
|
||||
repo *modules.Repository
|
||||
stateStore state.Store
|
||||
moduleWithMeta modules.ModuleWithMeta
|
||||
}
|
||||
|
||||
|
@ -74,6 +79,15 @@ func (c *containerTask) doExecute(ctx context.Context, b buildr.Buildr) (err err
|
|||
}
|
||||
}
|
||||
|
||||
outputSink, err := logging.NewTaskOutputSink(b.Config.Logging, modules.LogFileNameFormatter(c.moduleWithMeta))
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
defer func() {
|
||||
err = errors.Join(err, outputSink.Close())
|
||||
}()
|
||||
|
||||
containerSpec := c.moduleWithMeta.ContainerSpec()
|
||||
spec := containers.BuildRContainerSpec{
|
||||
ID: c.moduleWithMeta.ID(),
|
||||
|
@ -88,15 +102,6 @@ func (c *containerTask) doExecute(ctx context.Context, b buildr.Buildr) (err err
|
|||
Mounts: containerSpec.Mounts(),
|
||||
}
|
||||
|
||||
outputSink, err := logging.NewTaskOutputSink(b.Config.Logging, modules.LogFileNameFormatter(c.moduleWithMeta))
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
defer func() {
|
||||
err = errors.Join(err, outputSink.Close())
|
||||
}()
|
||||
|
||||
logger.Debug("Preparing container")
|
||||
con, grpcConn, err := c.orchestrator.BuildRContainer(ctx, spec)
|
||||
if err != nil {
|
||||
|
@ -116,7 +121,7 @@ func (c *containerTask) doExecute(ctx context.Context, b buildr.Buildr) (err err
|
|||
executorClient := rpcv1.NewExecutorServiceClient(grpcConn)
|
||||
|
||||
logger.Debug("Start remote task execution")
|
||||
streamClient, err := executorClient.ExecuteTask(ctx)
|
||||
streamClient, err := executorClient.ExecuteStream(ctx)
|
||||
if err != nil {
|
||||
return fmt.Errorf("failed to start remote task execution: %w", err)
|
||||
}
|
||||
|
@ -130,8 +135,8 @@ func (c *containerTask) doExecute(ctx context.Context, b buildr.Buildr) (err err
|
|||
return fmt.Errorf("failed to marshal task spec as JSON: %w", err)
|
||||
}
|
||||
|
||||
startTaskReq := &rpcv1.ExecuteTaskRequest{
|
||||
Envelope: &rpcv1.ExecuteTaskRequest_StartTask{
|
||||
startTaskReq := &rpcv1.ExecutionClientMessage{
|
||||
Envelope: &rpcv1.ExecutionClientMessage_StartTask{
|
||||
StartTask: &rpcv1.StartTaskRequest{
|
||||
Buildr: &rpcv1.Buildr{
|
||||
Repo: &rpcv1.Buildr_Repo{
|
||||
|
@ -139,16 +144,16 @@ func (c *containerTask) doExecute(ctx context.Context, b buildr.Buildr) (err err
|
|||
},
|
||||
},
|
||||
Reference: &rpcv1.ModuleReference{
|
||||
TaskId: c.moduleWithMeta.ID(),
|
||||
ModuleType: c.moduleWithMeta.Category().String(),
|
||||
ModuleName: c.moduleWithMeta.Type(),
|
||||
TaskId: c.moduleWithMeta.ID(),
|
||||
ModuleCategory: c.moduleWithMeta.Category().String(),
|
||||
ModuleType: c.moduleWithMeta.Type(),
|
||||
ModuleName: c.moduleWithMeta.Name(),
|
||||
},
|
||||
RawTask: rawModule,
|
||||
},
|
||||
},
|
||||
}
|
||||
err = streamClient.Send(startTaskReq)
|
||||
|
||||
if err != nil {
|
||||
return fmt.Errorf("failed to send start task request: %w", err)
|
||||
}
|
||||
|
@ -161,14 +166,14 @@ func (c *containerTask) doExecute(ctx context.Context, b buildr.Buildr) (err err
|
|||
}
|
||||
|
||||
switch msg := ev.GetEnvelope().(type) {
|
||||
case *rpcv1.ExecuteTaskResponse_TaskLog:
|
||||
case *rpcv1.ExecutionServerMessage_TaskLog:
|
||||
c.handleTaskLog(ctx, msg.TaskLog, logger)
|
||||
case *rpcv1.ExecuteTaskResponse_TaskOutput:
|
||||
case *rpcv1.ExecutionServerMessage_TaskOutput:
|
||||
logger.Debug("Handle task output")
|
||||
if err := c.handleTaskOutput(outputSink, msg.TaskOutput); err != nil {
|
||||
logger.Error("Failed to process task output", slog.String("err", err.Error()))
|
||||
}
|
||||
case *rpcv1.ExecuteTaskResponse_TaskResult:
|
||||
case *rpcv1.ExecutionServerMessage_TaskResult:
|
||||
logger.Debug("Received task result")
|
||||
if errMsg := msg.TaskResult.Error; errMsg != "" {
|
||||
return fmt.Errorf("failed to execute task: %s", msg.TaskResult.Error)
|
||||
|
@ -181,6 +186,62 @@ func (c *containerTask) doExecute(ctx context.Context, b buildr.Buildr) (err err
|
|||
}
|
||||
|
||||
return nil
|
||||
case *rpcv1.ExecutionServerMessage_SetState:
|
||||
msgIdMeta, ok := ev.Meta.(*rpcv1.ExecutionServerMessage_MessageId)
|
||||
if !ok {
|
||||
continue
|
||||
}
|
||||
result := &rpcv1.ExecutionClientMessage_Error{
|
||||
Error: new(rpcv1.Result),
|
||||
}
|
||||
respMsg := &rpcv1.ExecutionClientMessage{
|
||||
Meta: &rpcv1.ExecutionClientMessage_RepliesTo{
|
||||
RepliesTo: msgIdMeta.MessageId,
|
||||
},
|
||||
Envelope: result,
|
||||
}
|
||||
|
||||
k := state.PlainKey(msg.SetState.Key)
|
||||
if err := c.stateStore.Set(ctx, k, msg.SetState.Data); err != nil {
|
||||
result.Error.Error = err.Error()
|
||||
logger.Error("Failed to set state", slog.String("err", err.Error()), slog.String("key", hex.EncodeToString(k.Bytes())))
|
||||
}
|
||||
|
||||
if err = streamClient.Send(respMsg); err != nil {
|
||||
logger.Error("Failed to send set status response", errs.Attr(err))
|
||||
}
|
||||
case *rpcv1.ExecutionServerMessage_GetState:
|
||||
k := state.PlainKey(msg.GetState.Key)
|
||||
msgIdMeta, ok := ev.Meta.(*rpcv1.ExecutionServerMessage_MessageId)
|
||||
if !ok {
|
||||
continue
|
||||
}
|
||||
|
||||
respMsg := &rpcv1.ExecutionClientMessage{
|
||||
Meta: &rpcv1.ExecutionClientMessage_RepliesTo{
|
||||
RepliesTo: msgIdMeta.MessageId,
|
||||
},
|
||||
}
|
||||
|
||||
data, _, err := c.stateStore.Get(ctx, k)
|
||||
if err == nil {
|
||||
respMsg.Envelope = &rpcv1.ExecutionClientMessage_GetState{
|
||||
GetState: &rpcv1.GetStateResponse{
|
||||
Key: msg.GetState.Key,
|
||||
Data: data,
|
||||
},
|
||||
}
|
||||
} else {
|
||||
respMsg.Envelope = &rpcv1.ExecutionClientMessage_Error{
|
||||
Error: &rpcv1.Result{
|
||||
Error: err.Error(),
|
||||
},
|
||||
}
|
||||
}
|
||||
|
||||
if err = streamClient.Send(respMsg); err != nil {
|
||||
logger.Error("Failed to send get status response", errs.Attr(err))
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
|
|
@ -5,6 +5,8 @@ import (
|
|||
"io"
|
||||
"path/filepath"
|
||||
|
||||
"code.icb4dc0.de/buildr/buildr/modules/state"
|
||||
|
||||
"code.icb4dc0.de/buildr/buildr/internal/logging"
|
||||
"code.icb4dc0.de/buildr/buildr/modules"
|
||||
"code.icb4dc0.de/buildr/buildr/modules/buildr"
|
||||
|
@ -32,12 +34,14 @@ func WithLoggerFactory(factory func() *slog.Logger) defaultExecutionContextOptio
|
|||
|
||||
func NewDefaultExecutionContext(
|
||||
ctx context.Context,
|
||||
store state.Store,
|
||||
mod modules.ModuleWithMeta,
|
||||
buildr buildr.Buildr,
|
||||
opts ...DefaultExecutionContextOption,
|
||||
) (DefaultExecutionContext, error) {
|
||||
execCtx := DefaultExecutionContext{
|
||||
Context: ctx,
|
||||
store: store,
|
||||
mod: mod,
|
||||
buildr: buildr,
|
||||
}
|
||||
|
@ -61,10 +65,21 @@ type DefaultExecutionContext struct {
|
|||
outputSink logging.TaskOutputSink
|
||||
mod modules.ModuleWithMeta
|
||||
buildr buildr.Buildr
|
||||
store state.Store
|
||||
|
||||
loggerFactory func() *slog.Logger
|
||||
}
|
||||
|
||||
func (d DefaultExecutionContext) GetState(ctx context.Context, key string) ([]byte, state.Metadata, error) {
|
||||
k := state.KeyOfStrings(d.mod.Category(), d.mod.Name(), key)
|
||||
return d.store.Get(ctx, k)
|
||||
}
|
||||
|
||||
func (d DefaultExecutionContext) SetState(ctx context.Context, key string, value []byte) error {
|
||||
k := state.KeyOfStrings(d.mod.Category(), d.mod.Name(), key)
|
||||
return d.store.Set(ctx, k, value)
|
||||
}
|
||||
|
||||
func (d DefaultExecutionContext) WorkingDir() string {
|
||||
return d.buildr.Repo.Root
|
||||
}
|
||||
|
|
|
@ -3,28 +3,35 @@ package local
|
|||
import (
|
||||
"fmt"
|
||||
|
||||
"code.icb4dc0.de/buildr/buildr/modules/state"
|
||||
|
||||
"code.icb4dc0.de/buildr/buildr/internal/execution"
|
||||
"code.icb4dc0.de/buildr/buildr/modules"
|
||||
)
|
||||
|
||||
var _ execution.TaskProvider = (*taskProvider)(nil)
|
||||
|
||||
func Provider() execution.TaskProvider {
|
||||
return taskProvider{}
|
||||
func Provider(stateStore state.Store) execution.TaskProvider {
|
||||
return taskProvider{
|
||||
stateStore: stateStore,
|
||||
}
|
||||
}
|
||||
|
||||
type taskProvider struct{}
|
||||
type taskProvider struct {
|
||||
stateStore state.Store
|
||||
}
|
||||
|
||||
func (taskProvider) CanProvide(m modules.ModuleWithMeta) bool {
|
||||
return m.ContainerSpec() == nil
|
||||
}
|
||||
|
||||
func (taskProvider) Create(m modules.ModuleWithMeta) (execution.Task, error) {
|
||||
func (p taskProvider) Create(m modules.ModuleWithMeta) (execution.Task, error) {
|
||||
if m.ContainerSpec() != nil {
|
||||
return nil, fmt.Errorf("module %v has a container spec - should be handled differently", m.Name())
|
||||
} else {
|
||||
return &localTask{
|
||||
module: m,
|
||||
module: m,
|
||||
stateStore: p.stateStore,
|
||||
}, nil
|
||||
}
|
||||
}
|
||||
|
|
|
@ -8,6 +8,8 @@ import (
|
|||
"path/filepath"
|
||||
"sync"
|
||||
|
||||
"code.icb4dc0.de/buildr/buildr/modules/state"
|
||||
|
||||
"code.icb4dc0.de/buildr/buildr/internal/execution"
|
||||
"code.icb4dc0.de/buildr/buildr/internal/storage"
|
||||
"code.icb4dc0.de/buildr/buildr/modules"
|
||||
|
@ -21,8 +23,9 @@ var _ execution.Task = (*localTask)(nil)
|
|||
|
||||
type localTask struct {
|
||||
execution.TaskDependencies
|
||||
once sync.Once
|
||||
module modules.ModuleWithMeta
|
||||
once sync.Once
|
||||
module modules.ModuleWithMeta
|
||||
stateStore state.Store
|
||||
}
|
||||
|
||||
func (t *localTask) Execute(ctx context.Context, b buildr.Buildr) (err error) {
|
||||
|
@ -137,6 +140,7 @@ func (t *localTask) executeIsolated(ctx context.Context, b buildr.Buildr) error
|
|||
func (t *localTask) executionContextFor(ctx context.Context, b buildr.Buildr, m modules.ModuleWithMeta) (execution.DefaultExecutionContext, error) {
|
||||
return execution.NewDefaultExecutionContext(
|
||||
ctx,
|
||||
t.stateStore,
|
||||
m,
|
||||
b,
|
||||
execution.WithLoggerFactory(func() *slog.Logger {
|
||||
|
|
|
@ -1,18 +1,46 @@
|
|||
package profiling
|
||||
|
||||
import (
|
||||
"errors"
|
||||
"flag"
|
||||
"fmt"
|
||||
"os"
|
||||
"runtime/pprof"
|
||||
"strconv"
|
||||
"strings"
|
||||
|
||||
"code.icb4dc0.de/buildr/buildr/internal/config"
|
||||
)
|
||||
|
||||
type Config struct {
|
||||
OutFile string `mapstructure:"pprof-out-file"`
|
||||
ProfileName string `mapstructure:"pprof-profile-name"`
|
||||
OutFile string
|
||||
CPUProfile bool
|
||||
ProfileName string
|
||||
}
|
||||
|
||||
func (c *Config) IsConfigured() bool {
|
||||
return c.ProfileName != "" && c.OutFile != ""
|
||||
return (c.ProfileName != "" || c.CPUProfile) && c.OutFile != ""
|
||||
}
|
||||
|
||||
func (c *Config) Setup() (*Recorder, error) {
|
||||
if !c.IsConfigured() {
|
||||
return nil, nil
|
||||
}
|
||||
|
||||
f, err := os.Create(c.OutFile)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
|
||||
if c.CPUProfile {
|
||||
if err := pprof.StartCPUProfile(f); err != nil {
|
||||
return nil, errors.Join(err, f.Close())
|
||||
}
|
||||
|
||||
return &Recorder{outFile: f, cpuProfile: true}, nil
|
||||
}
|
||||
|
||||
return &Recorder{outFile: f, profile: pprof.Lookup(c.ProfileName)}, nil
|
||||
}
|
||||
|
||||
func (c *Config) AddFlags(fs *flag.FlagSet) {
|
||||
|
@ -23,10 +51,26 @@ func (c *Config) AddFlags(fs *flag.FlagSet) {
|
|||
"Output file for PPROF profiling data.",
|
||||
)
|
||||
|
||||
fs.BoolVar(
|
||||
&c.CPUProfile,
|
||||
"pprof.cpu-profile",
|
||||
config.EnvOr("BUILDR_PPROF_CPU_PROFILE", strconv.ParseBool, false),
|
||||
"Enable CPU profiling. If enabled, any profile name will be ignored and a CPU profile will be generated at the specified path.",
|
||||
)
|
||||
|
||||
fs.StringVar(
|
||||
&c.ProfileName,
|
||||
"pprof.profile-name",
|
||||
config.StringEnvOr("BUILDR_PPROF_PROFILE_NAME", ""),
|
||||
"Name of the PPROF profiling data.",
|
||||
fmt.Sprintf("Name of the PPROF profile to use [%s]", strings.Join(profileNames(), ", ")),
|
||||
)
|
||||
}
|
||||
|
||||
func profileNames() []string {
|
||||
names := make([]string, 0)
|
||||
for _, profile := range pprof.Profiles() {
|
||||
names = append(names, profile.Name())
|
||||
}
|
||||
|
||||
return names
|
||||
}
|
||||
|
|
26
internal/profiling/recorder.go
Normal file
26
internal/profiling/recorder.go
Normal file
|
@ -0,0 +1,26 @@
|
|||
package profiling
|
||||
|
||||
import (
|
||||
"errors"
|
||||
"os"
|
||||
"runtime/pprof"
|
||||
)
|
||||
|
||||
type Recorder struct {
|
||||
cpuProfile bool
|
||||
profile *pprof.Profile
|
||||
outFile *os.File
|
||||
}
|
||||
|
||||
func (r *Recorder) Close() error {
|
||||
if r == nil {
|
||||
return nil
|
||||
}
|
||||
|
||||
if r.cpuProfile {
|
||||
pprof.StopCPUProfile()
|
||||
return r.outFile.Close()
|
||||
}
|
||||
|
||||
return errors.Join(r.profile.WriteTo(r.outFile, 0), r.outFile.Close())
|
||||
}
|
|
@ -7,3 +7,7 @@ import (
|
|||
type StreamSender[T proto.Message] interface {
|
||||
Send(msg T) error
|
||||
}
|
||||
|
||||
type StreamReceiver[T proto.Message] interface {
|
||||
Recv() (msg T, err error)
|
||||
}
|
||||
|
|
|
@ -3,6 +3,9 @@ package v1
|
|||
import (
|
||||
"context"
|
||||
"io"
|
||||
"os"
|
||||
|
||||
"code.icb4dc0.de/buildr/buildr/modules/state"
|
||||
|
||||
"golang.org/x/exp/slog"
|
||||
|
||||
|
@ -16,31 +19,42 @@ func NewContainerExecutionContext(
|
|||
ctx context.Context,
|
||||
logger *slog.Logger,
|
||||
workingDir string,
|
||||
sender StreamSender[*rpcv1.ExecuteTaskResponse],
|
||||
sender StreamSender[*rpcv1.ExecutionServerMessage],
|
||||
remoteState *RemoteStateClient,
|
||||
) *ContainerExecutionContext {
|
||||
return &ContainerExecutionContext{
|
||||
Context: ctx,
|
||||
logger: logger,
|
||||
workingDir: workingDir,
|
||||
stdOutWriter: newTaskOutputWriter(
|
||||
Context: ctx,
|
||||
remoteState: remoteState,
|
||||
logger: logger,
|
||||
workingDir: workingDir,
|
||||
stdOutWriter: io.MultiWriter(newTaskOutputWriter(
|
||||
rpcv1.TaskOutputSource_TASK_OUTPUT_SOURCE_STDOUT,
|
||||
sender,
|
||||
),
|
||||
stdErrWriter: newTaskOutputWriter(
|
||||
), os.Stdout),
|
||||
stdErrWriter: io.MultiWriter(newTaskOutputWriter(
|
||||
rpcv1.TaskOutputSource_TASK_OUTPUT_SOURCE_STDERR,
|
||||
sender,
|
||||
),
|
||||
), os.Stderr),
|
||||
}
|
||||
}
|
||||
|
||||
type ContainerExecutionContext struct {
|
||||
context.Context
|
||||
logger *slog.Logger
|
||||
remoteState *RemoteStateClient
|
||||
workingDir string
|
||||
stdOutWriter io.Writer
|
||||
stdErrWriter io.Writer
|
||||
}
|
||||
|
||||
func (c ContainerExecutionContext) GetState(ctx context.Context, key string) ([]byte, state.Metadata, error) {
|
||||
return c.remoteState.GetState(ctx, key)
|
||||
}
|
||||
|
||||
func (c ContainerExecutionContext) SetState(ctx context.Context, key string, value []byte) error {
|
||||
return c.remoteState.SetState(ctx, key, value)
|
||||
}
|
||||
|
||||
func (c ContainerExecutionContext) WorkingDir() string {
|
||||
return c.workingDir
|
||||
}
|
||||
|
|
104
internal/rpc/v1/executor.go
Normal file
104
internal/rpc/v1/executor.go
Normal file
|
@ -0,0 +1,104 @@
|
|||
package v1
|
||||
|
||||
import (
|
||||
"context"
|
||||
"errors"
|
||||
"os"
|
||||
"time"
|
||||
|
||||
"code.icb4dc0.de/buildr/buildr/internal/archive"
|
||||
rpcv1 "code.icb4dc0.de/buildr/buildr/internal/generated/rpc/v1"
|
||||
"code.icb4dc0.de/buildr/buildr/modules"
|
||||
|
||||
"golang.org/x/exp/slog"
|
||||
"google.golang.org/grpc/codes"
|
||||
"google.golang.org/grpc/status"
|
||||
)
|
||||
|
||||
type RemoteTaskExecutor struct {
|
||||
logger *slog.Logger
|
||||
registry *modules.TypeRegistry
|
||||
requestClient *RequestResponseClient
|
||||
sender StreamSender[*rpcv1.ExecutionServerMessage]
|
||||
}
|
||||
|
||||
func (e *RemoteTaskExecutor) Execute(ctx context.Context, startTask *rpcv1.StartTaskRequest) (resp chan *rpcv1.TaskResult, errs chan error) {
|
||||
resp = make(chan *rpcv1.TaskResult)
|
||||
errs = make(chan error)
|
||||
|
||||
ctx, cancel := context.WithCancelCause(ctx)
|
||||
|
||||
go func() {
|
||||
defer cancel(errors.New("stream closed"))
|
||||
|
||||
watcher, err := newFSWatcher(e.logger, startTask.Buildr.Repo.Root)
|
||||
if err != nil {
|
||||
e.logger.Error("Failed to start FS watcher", slog.String("err", err.Error()))
|
||||
errs <- status.Error(codes.Internal, err.Error())
|
||||
return
|
||||
}
|
||||
|
||||
mod, err := e.registry.CreateFromJSON(
|
||||
modules.Category(startTask.GetReference().GetModuleCategory()),
|
||||
startTask.GetReference().GetModuleType(),
|
||||
startTask.GetRawTask(),
|
||||
)
|
||||
|
||||
if err != nil {
|
||||
e.logger.Error("Failed to unmarshal module from JSON", slog.String("err", err.Error()))
|
||||
errs <- status.Error(codes.NotFound, err.Error())
|
||||
return
|
||||
} else {
|
||||
e.logger.Info("Executing module")
|
||||
go watcher.Watch(ctx)
|
||||
|
||||
remoteState := newRemoteStateClient(e.logger, mod.Category(), startTask.Reference.GetModuleName(), e.requestClient)
|
||||
|
||||
executionContext := NewContainerExecutionContext(ctx, e.logger, startTask.Buildr.Repo.Root, e.sender, remoteState)
|
||||
err = mod.Execute(executionContext)
|
||||
result := new(rpcv1.TaskResult)
|
||||
|
||||
if err != nil {
|
||||
result.Error = err.Error()
|
||||
}
|
||||
|
||||
e.logger.Debug("Waiting for file change events to propagate before canceling watcher")
|
||||
time.Sleep(250 * time.Millisecond)
|
||||
|
||||
cancel(ErrExecutionCompleted)
|
||||
|
||||
if err := e.addModifiedFilesToResult(result, watcher); err != nil {
|
||||
errs <- err
|
||||
return
|
||||
}
|
||||
|
||||
resp <- result
|
||||
}
|
||||
}()
|
||||
|
||||
return resp, errs
|
||||
}
|
||||
|
||||
func (e *RemoteTaskExecutor) addModifiedFilesToResult(result *rpcv1.TaskResult, watcher *fsWatcher) error {
|
||||
tmpFile, err := os.CreateTemp(os.TempDir(), "buildr-modified-files-*.tar.s2")
|
||||
if err != nil {
|
||||
e.logger.Error("Failed to create temporary file", slog.String("err", err.Error()))
|
||||
return status.Error(codes.Internal, err.Error())
|
||||
}
|
||||
|
||||
if err := watcher.WriteTo(tmpFile); err != nil {
|
||||
if errors.Is(err, archive.ErrEmptyArchive) {
|
||||
return nil
|
||||
}
|
||||
e.logger.Error("Failed to compress modified files", slog.String("err", err.Error()))
|
||||
return status.Error(codes.Internal, err.Error())
|
||||
}
|
||||
|
||||
if err := tmpFile.Close(); err != nil {
|
||||
e.logger.Error("Failed to close temporary file", slog.String("err", err.Error()))
|
||||
return status.Error(codes.Internal, err.Error())
|
||||
}
|
||||
|
||||
result.ModifiedFilesArchivePath = tmpFile.Name()
|
||||
return nil
|
||||
}
|
|
@ -5,16 +5,14 @@ import (
|
|||
"errors"
|
||||
"fmt"
|
||||
"io"
|
||||
"os"
|
||||
"time"
|
||||
|
||||
"code.icb4dc0.de/buildr/buildr/internal/errs"
|
||||
rpcv1 "code.icb4dc0.de/buildr/buildr/internal/generated/rpc/v1"
|
||||
"code.icb4dc0.de/buildr/buildr/modules"
|
||||
|
||||
"golang.org/x/exp/slog"
|
||||
"google.golang.org/grpc/codes"
|
||||
"google.golang.org/grpc/status"
|
||||
|
||||
"code.icb4dc0.de/buildr/buildr/internal/archive"
|
||||
rpcv1 "code.icb4dc0.de/buildr/buildr/internal/generated/rpc/v1"
|
||||
"code.icb4dc0.de/buildr/buildr/modules"
|
||||
)
|
||||
|
||||
var (
|
||||
|
@ -32,7 +30,7 @@ type ExecutorServiceServer struct {
|
|||
registry *modules.TypeRegistry
|
||||
}
|
||||
|
||||
func (e *ExecutorServiceServer) ExecuteTask(server rpcv1.ExecutorService_ExecuteTaskServer) (err error) {
|
||||
func (e *ExecutorServiceServer) ExecuteStream(server rpcv1.ExecutorService_ExecuteStreamServer) (err error) {
|
||||
ctx, cancel := context.WithCancelCause(server.Context())
|
||||
defer cancel(errors.New("stream closed"))
|
||||
|
||||
|
@ -43,83 +41,47 @@ func (e *ExecutorServiceServer) ExecuteTask(server rpcv1.ExecutorService_Execute
|
|||
}()
|
||||
|
||||
logger := slog.New(NewGrpcExecutorHandler(server))
|
||||
requestClient := NewRequestResponseClient(logger, server)
|
||||
|
||||
for ctx.Err() == nil {
|
||||
request, err := server.Recv()
|
||||
if err != nil {
|
||||
if errors.Is(err, io.EOF) {
|
||||
return nil
|
||||
} else {
|
||||
return err
|
||||
}
|
||||
}
|
||||
|
||||
switch unwrapped := request.GetEnvelope().(type) {
|
||||
case *rpcv1.ExecuteTaskRequest_StartTask:
|
||||
t := unwrapped.StartTask
|
||||
|
||||
watcher, err := newFSWatcher(logger, t.Buildr.Repo.Root)
|
||||
if err != nil {
|
||||
logger.Error("Failed to start FS watcher", slog.String("err", err.Error()))
|
||||
return status.Error(codes.Internal, err.Error())
|
||||
}
|
||||
|
||||
mod, err := e.registry.CreateFromJSON(modules.Category(t.GetReference().GetModuleType()), t.GetReference().GetModuleName(), t.GetRawTask())
|
||||
if err != nil {
|
||||
logger.Error("Failed to unmarshal module from JSON", slog.String("err", err.Error()))
|
||||
return status.Error(codes.NotFound, err.Error())
|
||||
} else {
|
||||
logger.Info("Executing module")
|
||||
go watcher.Watch(ctx)
|
||||
|
||||
executionContext := NewContainerExecutionContext(ctx, logger, t.Buildr.Repo.Root, server)
|
||||
err = mod.Execute(executionContext)
|
||||
result := new(rpcv1.TaskResult)
|
||||
|
||||
if err != nil {
|
||||
result.Error = err.Error()
|
||||
}
|
||||
|
||||
logger.Debug("Waiting for file change events to propagate before canceling watcher")
|
||||
time.Sleep(250 * time.Millisecond)
|
||||
cancel(ErrExecutionCompleted)
|
||||
|
||||
if err := addModifiedFilesToResult(result, logger, watcher); err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
_ = server.Send(&rpcv1.ExecuteTaskResponse{
|
||||
Envelope: &rpcv1.ExecuteTaskResponse_TaskResult{
|
||||
TaskResult: result,
|
||||
},
|
||||
})
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
return nil
|
||||
}
|
||||
|
||||
func addModifiedFilesToResult(result *rpcv1.TaskResult, logger *slog.Logger, watcher *fsWatcher) error {
|
||||
tmpFile, err := os.CreateTemp(os.TempDir(), "buildr-modified-files-*.tar.s2")
|
||||
request, err := server.Recv()
|
||||
if err != nil {
|
||||
logger.Error("Failed to create temporary file", slog.String("err", err.Error()))
|
||||
return status.Error(codes.Internal, err.Error())
|
||||
}
|
||||
|
||||
if err := watcher.WriteTo(tmpFile); err != nil {
|
||||
if errors.Is(err, archive.ErrEmptyArchive) {
|
||||
if errors.Is(err, io.EOF) {
|
||||
return nil
|
||||
} else {
|
||||
return err
|
||||
}
|
||||
logger.Error("Failed to compress modified files", slog.String("err", err.Error()))
|
||||
return status.Error(codes.Internal, err.Error())
|
||||
}
|
||||
|
||||
if err := tmpFile.Close(); err != nil {
|
||||
logger.Error("Failed to close temporary file", slog.String("err", err.Error()))
|
||||
return status.Error(codes.Internal, err.Error())
|
||||
unwrapped, ok := request.GetEnvelope().(*rpcv1.ExecutionClientMessage_StartTask)
|
||||
if !ok {
|
||||
return status.Error(codes.InvalidArgument, "expected StartTask as first message in stream")
|
||||
}
|
||||
|
||||
executor := RemoteTaskExecutor{
|
||||
logger: logger,
|
||||
registry: e.registry,
|
||||
requestClient: requestClient,
|
||||
sender: server,
|
||||
}
|
||||
|
||||
clientErrs := requestClient.DispatchMessages(server)
|
||||
resp, execErrs := executor.Execute(ctx, unwrapped.StartTask)
|
||||
|
||||
select {
|
||||
case <-ctx.Done():
|
||||
return status.Error(codes.Canceled, ctx.Err().Error())
|
||||
case e := <-clientErrs:
|
||||
logger.Error("Error occurred while handling client - server communicatoin", errs.Attr(e))
|
||||
case e := <-execErrs:
|
||||
logger.Error("Error occurred while executing module", errs.Attr(e))
|
||||
return status.Error(codes.Internal, e.Error())
|
||||
case r := <-resp:
|
||||
_ = server.Send(&rpcv1.ExecutionServerMessage{
|
||||
Envelope: &rpcv1.ExecutionServerMessage_TaskResult{
|
||||
TaskResult: r,
|
||||
},
|
||||
})
|
||||
}
|
||||
|
||||
result.ModifiedFilesArchivePath = tmpFile.Name()
|
||||
return nil
|
||||
}
|
||||
|
|
|
@ -11,7 +11,7 @@ import (
|
|||
|
||||
var _ slog.Handler = (*GrpcExecutorHandler)(nil)
|
||||
|
||||
func NewGrpcExecutorHandler(sender StreamSender[*rpcv1.ExecuteTaskResponse]) *GrpcExecutorHandler {
|
||||
func NewGrpcExecutorHandler(sender StreamSender[*rpcv1.ExecutionServerMessage]) *GrpcExecutorHandler {
|
||||
return &GrpcExecutorHandler{
|
||||
Level: slog.LevelDebug,
|
||||
sender: sender,
|
||||
|
@ -19,7 +19,7 @@ func NewGrpcExecutorHandler(sender StreamSender[*rpcv1.ExecuteTaskResponse]) *Gr
|
|||
}
|
||||
|
||||
type GrpcExecutorHandler struct {
|
||||
sender StreamSender[*rpcv1.ExecuteTaskResponse]
|
||||
sender StreamSender[*rpcv1.ExecutionServerMessage]
|
||||
Level slog.Level
|
||||
group string
|
||||
attributes []slog.Attr
|
||||
|
@ -46,8 +46,8 @@ func (g GrpcExecutorHandler) Handle(_ context.Context, record slog.Record) error
|
|||
return true
|
||||
})
|
||||
|
||||
resp := rpcv1.ExecuteTaskResponse{
|
||||
Envelope: &rpcv1.ExecuteTaskResponse_TaskLog{
|
||||
resp := rpcv1.ExecutionServerMessage{
|
||||
Envelope: &rpcv1.ExecutionServerMessage_TaskLog{
|
||||
TaskLog: &taskLog,
|
||||
},
|
||||
}
|
||||
|
|
133
internal/rpc/v1/request_response_client.go
Normal file
133
internal/rpc/v1/request_response_client.go
Normal file
|
@ -0,0 +1,133 @@
|
|||
package v1
|
||||
|
||||
import (
|
||||
"context"
|
||||
"errors"
|
||||
"fmt"
|
||||
"io"
|
||||
"sync"
|
||||
|
||||
rpcv1 "code.icb4dc0.de/buildr/buildr/internal/generated/rpc/v1"
|
||||
|
||||
"github.com/google/uuid"
|
||||
"golang.org/x/exp/slog"
|
||||
)
|
||||
|
||||
var (
|
||||
ErrNoReplyMsgId = errors.New("no repliesTo set - cannot handle message")
|
||||
ErrNoMatchingRequest = errors.New("no matching request for given message id")
|
||||
)
|
||||
|
||||
func NewRequestResponseClient(logger *slog.Logger, sender StreamSender[*rpcv1.ExecutionServerMessage]) *RequestResponseClient {
|
||||
return &RequestResponseClient{
|
||||
logger: logger,
|
||||
sender: sender,
|
||||
requests: make(map[uuid.UUID]chan *rpcv1.ExecutionClientMessage),
|
||||
}
|
||||
}
|
||||
|
||||
type RequestResponseClient struct {
|
||||
lock sync.Mutex
|
||||
logger *slog.Logger
|
||||
sender StreamSender[*rpcv1.ExecutionServerMessage]
|
||||
requests map[uuid.UUID]chan *rpcv1.ExecutionClientMessage
|
||||
}
|
||||
|
||||
func (rrc *RequestResponseClient) DispatchMessages(receiver StreamReceiver[*rpcv1.ExecutionClientMessage]) (errs chan error) {
|
||||
errs = make(chan error)
|
||||
|
||||
go func() {
|
||||
for {
|
||||
msg, err := receiver.Recv()
|
||||
if err != nil {
|
||||
if errors.Is(err, io.EOF) {
|
||||
rrc.logger.Info("Closing request response client")
|
||||
return
|
||||
}
|
||||
errs <- err
|
||||
return
|
||||
}
|
||||
|
||||
rrc.logger.Debug("Dispatching message")
|
||||
repliesTo, ok := msg.Meta.(*rpcv1.ExecutionClientMessage_RepliesTo)
|
||||
if !ok {
|
||||
errs <- ErrNoReplyMsgId
|
||||
continue
|
||||
}
|
||||
|
||||
msgId, err := uuid.FromBytes(repliesTo.RepliesTo)
|
||||
if err != nil {
|
||||
errs <- err
|
||||
continue
|
||||
}
|
||||
|
||||
rrc.logger.Debug("Checking for pending request", slog.String("msgId", msgId.String()))
|
||||
rrc.lock.Lock()
|
||||
waitingChan, ok := rrc.requests[msgId]
|
||||
if !ok {
|
||||
rrc.lock.Unlock()
|
||||
errs <- fmt.Errorf("%w: %s", ErrNoMatchingRequest, msgId)
|
||||
continue
|
||||
}
|
||||
|
||||
delete(rrc.requests, msgId)
|
||||
rrc.lock.Unlock()
|
||||
|
||||
rrc.logger.Debug("Replying to pending request", slog.String("msgId", msgId.String()))
|
||||
waitingChan <- msg
|
||||
}
|
||||
}()
|
||||
|
||||
return errs
|
||||
}
|
||||
|
||||
func (rrc *RequestResponseClient) Send(ctx context.Context, req *rpcv1.ExecutionServerMessage) (resp *rpcv1.ExecutionClientMessage, err error) {
|
||||
var (
|
||||
msgId = uuid.New()
|
||||
respChan = make(chan *rpcv1.ExecutionClientMessage)
|
||||
)
|
||||
|
||||
req.Meta = &rpcv1.ExecutionServerMessage_MessageId{
|
||||
MessageId: uuidBytes(msgId),
|
||||
}
|
||||
rrc.lock.Lock()
|
||||
rrc.requests[msgId] = respChan
|
||||
rrc.lock.Unlock()
|
||||
|
||||
defer rrc.cleanRequest(msgId)
|
||||
|
||||
if err := rrc.sender.Send(req); err != nil {
|
||||
return nil, err
|
||||
}
|
||||
|
||||
select {
|
||||
case <-ctx.Done():
|
||||
return nil, ctx.Err()
|
||||
case resp := <-respChan:
|
||||
switch unwrap := resp.Envelope.(type) {
|
||||
case *rpcv1.ExecutionClientMessage_Error:
|
||||
if errMsg := unwrap.Error.Error; errMsg != "" {
|
||||
return nil, errors.New(unwrap.Error.Error)
|
||||
}
|
||||
return nil, nil
|
||||
default:
|
||||
return resp, nil
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
func (rrc *RequestResponseClient) cleanRequest(id uuid.UUID) {
|
||||
rrc.lock.Lock()
|
||||
defer rrc.lock.Unlock()
|
||||
|
||||
c, ok := rrc.requests[id]
|
||||
if ok {
|
||||
close(c)
|
||||
delete(rrc.requests, id)
|
||||
}
|
||||
}
|
||||
|
||||
func uuidBytes(uuid uuid.UUID) []byte {
|
||||
b, _ := uuid.MarshalBinary()
|
||||
return b
|
||||
}
|
73
internal/rpc/v1/state.go
Normal file
73
internal/rpc/v1/state.go
Normal file
|
@ -0,0 +1,73 @@
|
|||
package v1
|
||||
|
||||
import (
|
||||
"context"
|
||||
"errors"
|
||||
|
||||
"code.icb4dc0.de/buildr/buildr/internal/errs"
|
||||
rpcv1 "code.icb4dc0.de/buildr/buildr/internal/generated/rpc/v1"
|
||||
"code.icb4dc0.de/buildr/buildr/modules"
|
||||
"code.icb4dc0.de/buildr/buildr/modules/state"
|
||||
|
||||
"golang.org/x/exp/slog"
|
||||
)
|
||||
|
||||
func newRemoteStateClient(
|
||||
logger *slog.Logger,
|
||||
cat modules.Category,
|
||||
name string,
|
||||
requestClient *RequestResponseClient,
|
||||
) *RemoteStateClient {
|
||||
return &RemoteStateClient{
|
||||
logger: logger,
|
||||
cat: cat,
|
||||
name: name,
|
||||
requestClient: requestClient,
|
||||
}
|
||||
}
|
||||
|
||||
type RemoteStateClient struct {
|
||||
logger *slog.Logger
|
||||
cat modules.Category
|
||||
name string
|
||||
requestClient *RequestResponseClient
|
||||
}
|
||||
|
||||
func (c *RemoteStateClient) GetState(ctx context.Context, key string) ([]byte, state.Metadata, error) {
|
||||
c.logger.Debug("Getting state", slog.String("category", c.cat.String()), slog.String("name", c.name), slog.String("key", key))
|
||||
resp, err := c.requestClient.Send(ctx, &rpcv1.ExecutionServerMessage{
|
||||
Envelope: &rpcv1.ExecutionServerMessage_GetState{
|
||||
GetState: &rpcv1.GetStateRequest{
|
||||
Key: state.KeyOfStrings(c.cat, c.name, key).Bytes(),
|
||||
},
|
||||
},
|
||||
})
|
||||
if err != nil {
|
||||
c.logger.Warn("Failed to get state", slog.String("key", key), errs.Attr(err))
|
||||
return nil, state.Metadata{}, err
|
||||
}
|
||||
|
||||
statusResp, ok := resp.Envelope.(*rpcv1.ExecutionClientMessage_GetState)
|
||||
if !ok {
|
||||
return nil, state.Metadata{}, errors.New("unexpected response type")
|
||||
}
|
||||
|
||||
return statusResp.GetState.Data, state.Metadata{}, nil
|
||||
}
|
||||
|
||||
func (c *RemoteStateClient) SetState(ctx context.Context, key string, value []byte) error {
|
||||
c.logger.Debug("Setting state", slog.String("category", c.cat.String()), slog.String("name", c.name), slog.String("key", key))
|
||||
_, err := c.requestClient.Send(ctx, &rpcv1.ExecutionServerMessage{
|
||||
Envelope: &rpcv1.ExecutionServerMessage_SetState{
|
||||
SetState: &rpcv1.SetState{
|
||||
Key: state.KeyOfStrings(c.cat, c.name, key).Bytes(),
|
||||
Data: value,
|
||||
},
|
||||
},
|
||||
})
|
||||
if err != nil {
|
||||
c.logger.Warn("Failed to set state", slog.String("key", key), errs.Attr(err))
|
||||
}
|
||||
|
||||
return err
|
||||
}
|
|
@ -8,7 +8,7 @@ import (
|
|||
|
||||
var _ io.Writer = (*taskOutputWriter)(nil)
|
||||
|
||||
func newTaskOutputWriter(source rpcv1.TaskOutputSource, sender StreamSender[*rpcv1.ExecuteTaskResponse]) taskOutputWriter {
|
||||
func newTaskOutputWriter(source rpcv1.TaskOutputSource, sender StreamSender[*rpcv1.ExecutionServerMessage]) taskOutputWriter {
|
||||
return taskOutputWriter{
|
||||
source: source,
|
||||
sender: sender,
|
||||
|
@ -17,12 +17,12 @@ func newTaskOutputWriter(source rpcv1.TaskOutputSource, sender StreamSender[*rpc
|
|||
|
||||
type taskOutputWriter struct {
|
||||
source rpcv1.TaskOutputSource
|
||||
sender StreamSender[*rpcv1.ExecuteTaskResponse]
|
||||
sender StreamSender[*rpcv1.ExecutionServerMessage]
|
||||
}
|
||||
|
||||
func (t taskOutputWriter) Write(p []byte) (n int, err error) {
|
||||
err = t.sender.Send(&rpcv1.ExecuteTaskResponse{
|
||||
Envelope: &rpcv1.ExecuteTaskResponse_TaskOutput{
|
||||
err = t.sender.Send(&rpcv1.ExecutionServerMessage{
|
||||
Envelope: &rpcv1.ExecutionServerMessage_TaskOutput{
|
||||
TaskOutput: &rpcv1.TaskOutput{
|
||||
Source: t.source,
|
||||
Payload: p,
|
||||
|
|
|
@ -4,6 +4,8 @@ import (
|
|||
"context"
|
||||
"net/http"
|
||||
|
||||
"code.icb4dc0.de/buildr/buildr/modules/state"
|
||||
|
||||
"github.com/docker/docker/client"
|
||||
gh "github.com/google/go-github/v50/github"
|
||||
|
||||
|
@ -43,6 +45,13 @@ func WithIgnorer(ignorer *ignore.Ignorer) CollectionOption {
|
|||
})
|
||||
}
|
||||
|
||||
func WithStateStore(store state.Store) CollectionOption {
|
||||
return collectionOptionFunc(func(svc *Collection) error {
|
||||
svc.stateStore = store
|
||||
return nil
|
||||
})
|
||||
}
|
||||
|
||||
func WithDockerClientFromEnv(ctx context.Context) CollectionOption {
|
||||
return collectionOptionFunc(func(svc *Collection) error {
|
||||
cli, err := client.NewClientWithOpts(
|
||||
|
@ -90,6 +99,7 @@ type Collection struct {
|
|||
vault *vault.Vault
|
||||
dockerClient *client.Client
|
||||
ignorer *ignore.Ignorer
|
||||
stateStore state.Store
|
||||
}
|
||||
|
||||
func (c *Collection) With(opts ...CollectionOption) error {
|
||||
|
@ -102,6 +112,10 @@ func (c *Collection) With(opts ...CollectionOption) error {
|
|||
return nil
|
||||
}
|
||||
|
||||
func (c *Collection) StateStore() state.Store {
|
||||
return c.stateStore
|
||||
}
|
||||
|
||||
func (c *Collection) GitHubClient() *gh.Client {
|
||||
return c.ghClient
|
||||
}
|
||||
|
|
|
@ -4,6 +4,8 @@ import (
|
|||
"context"
|
||||
"io"
|
||||
|
||||
"code.icb4dc0.de/buildr/buildr/modules/state"
|
||||
|
||||
"github.com/hashicorp/hcl/v2"
|
||||
"golang.org/x/exp/slog"
|
||||
)
|
||||
|
@ -16,6 +18,13 @@ type ExecutionContext interface {
|
|||
StdOut() io.Writer
|
||||
StdErr() io.Writer
|
||||
Logger() *slog.Logger
|
||||
GetState(ctx context.Context, key string) ([]byte, state.Metadata, error)
|
||||
SetState(ctx context.Context, key string, value []byte) error
|
||||
}
|
||||
|
||||
type StateEncoder[T any] interface {
|
||||
Get(ctx context.Context, key string) (val T, ok bool, meta state.Metadata, err error)
|
||||
Set(ctx context.Context, key string, val T) error
|
||||
}
|
||||
|
||||
type ModuleWithMeta interface {
|
||||
|
|
48
modules/json_state_encoder.go
Normal file
48
modules/json_state_encoder.go
Normal file
|
@ -0,0 +1,48 @@
|
|||
package modules
|
||||
|
||||
import (
|
||||
"context"
|
||||
"encoding/json"
|
||||
|
||||
"code.icb4dc0.de/buildr/buildr/modules/state"
|
||||
)
|
||||
|
||||
var _ StateEncoder[struct{}] = (*JSONStateEncoder[struct{}])(nil)
|
||||
|
||||
func NewJSONStateEncoder[T any](ctx ExecutionContext) *JSONStateEncoder[T] {
|
||||
return &JSONStateEncoder[T]{
|
||||
Context: ctx,
|
||||
}
|
||||
}
|
||||
|
||||
type JSONStateEncoder[T any] struct {
|
||||
Context ExecutionContext
|
||||
}
|
||||
|
||||
func (j JSONStateEncoder[T]) Get(ctx context.Context, key string) (val T, ok bool, meta state.Metadata, err error) {
|
||||
var data []byte
|
||||
|
||||
data, meta, err = j.Context.GetState(ctx, key)
|
||||
if err != nil {
|
||||
return val, false, state.Metadata{}, err
|
||||
}
|
||||
|
||||
if len(data) == 0 {
|
||||
return val, false, meta, nil
|
||||
}
|
||||
|
||||
if err = json.Unmarshal(data, &val); err != nil {
|
||||
return val, false, state.Metadata{}, err
|
||||
}
|
||||
|
||||
return val, true, meta, nil
|
||||
}
|
||||
|
||||
func (j JSONStateEncoder[T]) Set(ctx context.Context, key string, val T) error {
|
||||
data, err := json.Marshal(val)
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
return j.Context.SetState(ctx, key, data)
|
||||
}
|
1
modules/state/.gitignore
vendored
Normal file
1
modules/state/.gitignore
vendored
Normal file
|
@ -0,0 +1 @@
|
|||
ent/
|
34
modules/state/api.go
Normal file
34
modules/state/api.go
Normal file
|
@ -0,0 +1,34 @@
|
|||
package state
|
||||
|
||||
import (
|
||||
"context"
|
||||
"time"
|
||||
|
||||
"code.icb4dc0.de/buildr/buildr/modules/state/ent"
|
||||
)
|
||||
|
||||
type Metadata struct {
|
||||
ModifiedAt time.Time
|
||||
TTL *time.Time
|
||||
}
|
||||
|
||||
type Key interface {
|
||||
Bytes() []byte
|
||||
}
|
||||
|
||||
type EntryOption interface {
|
||||
applyToEntry(e *ent.KVEntryCreate)
|
||||
}
|
||||
|
||||
type Store interface {
|
||||
StoreWriter
|
||||
StoreReader
|
||||
}
|
||||
|
||||
type StoreWriter interface {
|
||||
Set(ctx context.Context, key Key, state []byte, opts ...EntryOption) error
|
||||
}
|
||||
|
||||
type StoreReader interface {
|
||||
Get(ctx context.Context, key Key) (state []byte, meta Metadata, err error)
|
||||
}
|
3
modules/state/generate.go
Normal file
3
modules/state/generate.go
Normal file
|
@ -0,0 +1,3 @@
|
|||
package state
|
||||
|
||||
//go:generate go run -mod=mod entgo.io/ent/cmd/ent generate --feature sql/upsert --target ./ent ./schema
|
12
modules/state/init.go
Normal file
12
modules/state/init.go
Normal file
|
@ -0,0 +1,12 @@
|
|||
package state
|
||||
|
||||
import (
|
||||
"database/sql"
|
||||
|
||||
"entgo.io/ent/dialect"
|
||||
"modernc.org/sqlite"
|
||||
)
|
||||
|
||||
func init() {
|
||||
sql.Register(dialect.SQLite, new(sqlite.Driver))
|
||||
}
|
49
modules/state/key.go
Normal file
49
modules/state/key.go
Normal file
|
@ -0,0 +1,49 @@
|
|||
package state
|
||||
|
||||
import (
|
||||
"crypto/md5"
|
||||
"fmt"
|
||||
)
|
||||
|
||||
func KeyOfStrings(parts ...any) StringsKey {
|
||||
nonEmpty := make([]string, 0, len(parts))
|
||||
for i := range parts {
|
||||
if parts[i] == nil {
|
||||
continue
|
||||
}
|
||||
|
||||
switch s := parts[i].(type) {
|
||||
case string:
|
||||
if s != "" {
|
||||
nonEmpty = append(nonEmpty, s)
|
||||
}
|
||||
case fmt.Stringer:
|
||||
if out := s.String(); out != "" {
|
||||
nonEmpty = append(nonEmpty, out)
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
return nonEmpty
|
||||
}
|
||||
|
||||
type StringsKey []string
|
||||
|
||||
func (k StringsKey) Bytes() []byte {
|
||||
if len(k) == 0 {
|
||||
return nil
|
||||
}
|
||||
|
||||
h := md5.New()
|
||||
for i := range k {
|
||||
_, _ = h.Write([]byte(k[i]))
|
||||
}
|
||||
|
||||
return h.Sum(nil)
|
||||
}
|
||||
|
||||
type PlainKey []byte
|
||||
|
||||
func (k PlainKey) Bytes() []byte {
|
||||
return k
|
||||
}
|
36
modules/state/key_test.go
Normal file
36
modules/state/key_test.go
Normal file
|
@ -0,0 +1,36 @@
|
|||
package state_test
|
||||
|
||||
import (
|
||||
"errors"
|
||||
"testing"
|
||||
|
||||
"code.icb4dc0.de/buildr/buildr/modules/state"
|
||||
)
|
||||
|
||||
func TestStringsKey_Hash(t *testing.T) {
|
||||
tests := []struct {
|
||||
name string
|
||||
k []any
|
||||
want func(k []byte) error
|
||||
}{
|
||||
{
|
||||
name: "Empty key",
|
||||
k: nil,
|
||||
want: func(k []byte) error {
|
||||
if len(k) != 0 {
|
||||
return errors.New("expected empty key")
|
||||
}
|
||||
|
||||
return nil
|
||||
},
|
||||
},
|
||||
}
|
||||
for _, tt := range tests {
|
||||
t.Run(tt.name, func(t *testing.T) {
|
||||
got := state.KeyOfStrings(tt.k...).Bytes()
|
||||
if err := tt.want(got); err != nil {
|
||||
t.Errorf("Bytes() = %v, err = %v", got, err)
|
||||
}
|
||||
})
|
||||
}
|
||||
}
|
23
modules/state/options.go
Normal file
23
modules/state/options.go
Normal file
|
@ -0,0 +1,23 @@
|
|||
package state
|
||||
|
||||
import (
|
||||
"time"
|
||||
|
||||
"code.icb4dc0.de/buildr/buildr/modules/state/ent"
|
||||
)
|
||||
|
||||
type EntryOptionFunc func(e *ent.KVEntryCreate)
|
||||
|
||||
func (f EntryOptionFunc) applyToEntry(e *ent.KVEntryCreate) {
|
||||
f(e)
|
||||
}
|
||||
|
||||
func WithTTL(t time.Time) EntryOption {
|
||||
return EntryOptionFunc(func(e *ent.KVEntryCreate) {
|
||||
if t.Location() != time.UTC {
|
||||
t = t.UTC()
|
||||
}
|
||||
|
||||
e.SetTTL(t)
|
||||
})
|
||||
}
|
46
modules/state/schema/kventry.go
Normal file
46
modules/state/schema/kventry.go
Normal file
|
@ -0,0 +1,46 @@
|
|||
package schema
|
||||
|
||||
import (
|
||||
"time"
|
||||
|
||||
"entgo.io/ent"
|
||||
"entgo.io/ent/schema/field"
|
||||
"entgo.io/ent/schema/index"
|
||||
"github.com/google/uuid"
|
||||
)
|
||||
|
||||
// KVEntry holds the schema definition for the KVEntry entity.
|
||||
type KVEntry struct {
|
||||
ent.Schema
|
||||
}
|
||||
|
||||
func (KVEntry) Indexes() []ent.Index {
|
||||
return []ent.Index{
|
||||
index.Fields("key").
|
||||
Unique(),
|
||||
}
|
||||
}
|
||||
|
||||
// Fields of the KVEntry.
|
||||
func (KVEntry) Fields() []ent.Field {
|
||||
return []ent.Field{
|
||||
field.UUID("id", uuid.UUID{}).
|
||||
Default(uuid.New),
|
||||
field.Bytes("key").
|
||||
Unique().
|
||||
Immutable().
|
||||
MaxLen(32).
|
||||
NotEmpty(),
|
||||
field.Time("modified_at").
|
||||
Default(time.Now),
|
||||
field.Time("ttl").
|
||||
Optional().
|
||||
Nillable(),
|
||||
field.Bytes("state"),
|
||||
}
|
||||
}
|
||||
|
||||
// Edges of the KVEntry.
|
||||
func (KVEntry) Edges() []ent.Edge {
|
||||
return nil
|
||||
}
|
104
modules/state/store.go
Normal file
104
modules/state/store.go
Normal file
|
@ -0,0 +1,104 @@
|
|||
package state
|
||||
|
||||
import (
|
||||
"context"
|
||||
"errors"
|
||||
"fmt"
|
||||
"time"
|
||||
|
||||
"code.icb4dc0.de/buildr/buildr/modules/state/ent"
|
||||
"code.icb4dc0.de/buildr/buildr/modules/state/ent/kventry"
|
||||
|
||||
"entgo.io/ent/dialect"
|
||||
|
||||
_ "modernc.org/sqlite"
|
||||
)
|
||||
|
||||
var ErrEmptyKey = errors.New("key may not be empty")
|
||||
|
||||
func NewEntStore(ctx context.Context, stateFilePath string) (*EntStore, error) {
|
||||
client, err := ent.Open(dialect.SQLite, fmt.Sprintf("file:%s?_fk=1&_pragma=foreign_keys(1)", stateFilePath))
|
||||
if err != nil {
|
||||
return nil, fmt.Errorf("failed to open SQLite database: %w at %s", err, stateFilePath)
|
||||
}
|
||||
|
||||
if err := client.Schema.Create(ctx); err != nil {
|
||||
return nil, fmt.Errorf("failed to create schema: %w at %s", err, stateFilePath)
|
||||
}
|
||||
|
||||
_, err = client.KVEntry.
|
||||
Delete().
|
||||
Where(kventry.TTLGT(time.Now().UTC())).
|
||||
Exec(ctx)
|
||||
|
||||
if err != nil {
|
||||
return nil, fmt.Errorf("failed to delete expired entries: %w at %s", err, stateFilePath)
|
||||
}
|
||||
|
||||
return &EntStore{
|
||||
client: client,
|
||||
}, nil
|
||||
}
|
||||
|
||||
var _ Store = (*EntStore)(nil)
|
||||
|
||||
type EntStore struct {
|
||||
client *ent.Client
|
||||
}
|
||||
|
||||
func (s *EntStore) Get(ctx context.Context, key Key) (state []byte, meta Metadata, err error) {
|
||||
keyHash := key.Bytes()
|
||||
if len(keyHash) == 0 {
|
||||
return nil, Metadata{}, ErrEmptyKey
|
||||
}
|
||||
kvEntry, err := s.client.KVEntry.
|
||||
Query().
|
||||
Where(kventry.Key(keyHash)).
|
||||
Only(ctx)
|
||||
if err != nil {
|
||||
if ent.IsNotFound(err) {
|
||||
return nil, Metadata{}, nil
|
||||
}
|
||||
|
||||
return nil, Metadata{}, err
|
||||
}
|
||||
|
||||
if kvEntry.TTL != nil && kvEntry.TTL.After(time.Now().UTC()) {
|
||||
return nil, Metadata{}, nil
|
||||
}
|
||||
|
||||
return kvEntry.State, metadataForEntry(*kvEntry), nil
|
||||
}
|
||||
|
||||
func (s *EntStore) Set(ctx context.Context, key Key, state []byte, opts ...EntryOption) error {
|
||||
keyHash := key.Bytes()
|
||||
if len(keyHash) == 0 {
|
||||
return ErrEmptyKey
|
||||
}
|
||||
|
||||
create := s.client.KVEntry.
|
||||
Create()
|
||||
|
||||
for i := range opts {
|
||||
opts[i].applyToEntry(create)
|
||||
}
|
||||
|
||||
return create.
|
||||
SetKey(keyHash).
|
||||
SetState(state).
|
||||
OnConflict().
|
||||
UpdateModifiedAt().
|
||||
UpdateState().
|
||||
Exec(ctx)
|
||||
}
|
||||
|
||||
func (s *EntStore) Close() error {
|
||||
return s.client.Close()
|
||||
}
|
||||
|
||||
func metadataForEntry(entry ent.KVEntry) Metadata {
|
||||
return Metadata{
|
||||
ModifiedAt: entry.ModifiedAt,
|
||||
TTL: entry.TTL,
|
||||
}
|
||||
}
|
125
modules/state/store_test.go
Normal file
125
modules/state/store_test.go
Normal file
|
@ -0,0 +1,125 @@
|
|||
package state_test
|
||||
|
||||
import (
|
||||
"context"
|
||||
"path/filepath"
|
||||
"testing"
|
||||
|
||||
"code.icb4dc0.de/buildr/buildr/modules"
|
||||
"code.icb4dc0.de/buildr/buildr/modules/state"
|
||||
)
|
||||
|
||||
func TestEntStore_Set(t *testing.T) {
|
||||
type args struct {
|
||||
key state.StringsKey
|
||||
state []byte
|
||||
}
|
||||
tests := []struct {
|
||||
name string
|
||||
args args
|
||||
wantErr bool
|
||||
}{
|
||||
{
|
||||
name: "Set empty JSON state",
|
||||
args: args{
|
||||
key: state.KeyOfStrings(
|
||||
"script_state",
|
||||
modules.CategoryTask,
|
||||
"go_test",
|
||||
),
|
||||
state: []byte(`{}`),
|
||||
},
|
||||
wantErr: false,
|
||||
},
|
||||
{
|
||||
name: "Set JSON state",
|
||||
args: args{
|
||||
key: state.KeyOfStrings(
|
||||
"tool_state",
|
||||
modules.CategoryTool,
|
||||
"mockery",
|
||||
),
|
||||
state: []byte(`{"InstalledVersion": "1.0.0"}`),
|
||||
},
|
||||
wantErr: false,
|
||||
},
|
||||
}
|
||||
for _, tt := range tests {
|
||||
t.Run(tt.name, func(t *testing.T) {
|
||||
ctx, cancel := context.WithCancel(context.Background())
|
||||
t.Cleanup(cancel)
|
||||
|
||||
store, err := state.NewEntStore(ctx, filepath.Join(t.TempDir(), "store.sqlite"))
|
||||
if err != nil {
|
||||
t.Errorf("NewEntStore() error = %v", err)
|
||||
return
|
||||
}
|
||||
|
||||
if err := store.Set(ctx, tt.args.key, tt.args.state); (err != nil) != tt.wantErr {
|
||||
t.Errorf("Set() error = %v, wantErr %v", err, tt.wantErr)
|
||||
}
|
||||
})
|
||||
}
|
||||
}
|
||||
|
||||
func TestEntStore_Get_NonExisting(t *testing.T) {
|
||||
t.Parallel()
|
||||
ctx, cancel := context.WithCancel(context.Background())
|
||||
t.Cleanup(cancel)
|
||||
|
||||
store, err := state.NewEntStore(ctx, filepath.Join(t.TempDir(), "store.sqlite"))
|
||||
if err != nil {
|
||||
t.Errorf("NewEntStore() error = %v", err)
|
||||
return
|
||||
}
|
||||
key := state.KeyOfStrings(
|
||||
"script_state",
|
||||
modules.CategoryTask,
|
||||
"go_test",
|
||||
)
|
||||
|
||||
gotState, _, err := store.Get(ctx, key)
|
||||
if err != nil {
|
||||
t.Errorf("Get() error = %v", err)
|
||||
return
|
||||
}
|
||||
|
||||
if len(gotState) != 0 {
|
||||
t.Errorf("Expected empty state, got %v", gotState)
|
||||
}
|
||||
}
|
||||
|
||||
func TestEntStore_SetGet(t *testing.T) {
|
||||
t.Parallel()
|
||||
ctx, cancel := context.WithCancel(context.Background())
|
||||
t.Cleanup(cancel)
|
||||
|
||||
store, err := state.NewEntStore(ctx, filepath.Join(t.TempDir(), "store.sqlite"))
|
||||
if err != nil {
|
||||
t.Errorf("NewEntStore() error = %v", err)
|
||||
return
|
||||
}
|
||||
key := state.KeyOfStrings(
|
||||
"script_state",
|
||||
modules.CategoryTask,
|
||||
"go_test",
|
||||
)
|
||||
|
||||
if err := store.Set(ctx, key, []byte(`{}`)); err != nil {
|
||||
t.Errorf("Set() error = %v", err)
|
||||
return
|
||||
}
|
||||
|
||||
gotState, gotMeta, err := store.Get(ctx, key)
|
||||
if err != nil {
|
||||
t.Errorf("Get() error = %v", err)
|
||||
return
|
||||
}
|
||||
if string(gotState) != `{}` {
|
||||
t.Errorf("Get() gotState = %s", gotState)
|
||||
}
|
||||
|
||||
if gotMeta.ModifiedAt.IsZero() {
|
||||
t.Errorf("Get() gotMeta.ModifiedAt = %v", gotMeta.ModifiedAt)
|
||||
}
|
||||
}
|
|
@ -1,12 +1,18 @@
|
|||
package golang
|
||||
|
||||
import (
|
||||
"context"
|
||||
"fmt"
|
||||
"net/url"
|
||||
"os"
|
||||
"os/exec"
|
||||
"path"
|
||||
"path/filepath"
|
||||
"regexp"
|
||||
"strings"
|
||||
"time"
|
||||
|
||||
"code.icb4dc0.de/buildr/buildr/internal/errs"
|
||||
|
||||
"golang.org/x/exp/slog"
|
||||
|
||||
|
@ -68,7 +74,9 @@ func (g GoTool) Category() modules.Category {
|
|||
func (g GoTool) Execute(ctx modules.ExecutionContext) (err error) {
|
||||
var (
|
||||
binName = g.BinaryName()
|
||||
stateKey = fmt.Sprintf("%s.state", binName)
|
||||
logger = ctx.Logger().With(slog.String("tool_name", binName))
|
||||
stateEncoder = modules.NewJSONStateEncoder[GoToolState](ctx)
|
||||
state = g.state()
|
||||
existingToolPath string
|
||||
)
|
||||
|
@ -81,6 +89,25 @@ func (g GoTool) Execute(ctx modules.ExecutionContext) (err error) {
|
|||
)
|
||||
return nil
|
||||
}
|
||||
stateCtx, cancel := context.WithTimeout(ctx, 100*time.Millisecond)
|
||||
currentState, _, _, err := stateEncoder.Get(stateCtx, stateKey)
|
||||
cancel()
|
||||
|
||||
if err != nil {
|
||||
logger.Warn("Failed to get state", errs.Attr(err))
|
||||
}
|
||||
|
||||
desiredState := GoToolState{
|
||||
InstalledVersion: g.Version,
|
||||
BuildArgs: g.BuildArgs,
|
||||
Env: g.Env,
|
||||
}
|
||||
|
||||
existingToolPath = filepath.Join(ctx.BinariesDir(), binName)
|
||||
if _, err := os.Stat(existingToolPath); err == nil && desiredState.Equals(currentState) {
|
||||
logger.Info("Tool is already installed according to state", slog.String("tool_path", existingToolPath))
|
||||
return nil
|
||||
}
|
||||
|
||||
logger.Debug("Installing Go tool")
|
||||
|
||||
|
@ -111,7 +138,13 @@ func (g GoTool) Execute(ctx modules.ExecutionContext) (err error) {
|
|||
|
||||
cmd.AddEnv(g.Env)
|
||||
|
||||
return cmd.Run()
|
||||
if err = cmd.Run(); err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
stateCtx, cancel = context.WithTimeout(ctx, 100*time.Millisecond)
|
||||
defer cancel()
|
||||
return stateEncoder.Set(stateCtx, stateKey, desiredState)
|
||||
}
|
||||
|
||||
func (g GoTool) version() string {
|
||||
|
|
31
modules/tool/golang/go_tool_state.go
Normal file
31
modules/tool/golang/go_tool_state.go
Normal file
|
@ -0,0 +1,31 @@
|
|||
package golang
|
||||
|
||||
import (
|
||||
"golang.org/x/exp/maps"
|
||||
"golang.org/x/exp/slices"
|
||||
)
|
||||
|
||||
type GoToolState struct {
|
||||
InstalledVersion string `json:"installed_version"`
|
||||
BuildArgs []string `json:"build_args,omitempty"`
|
||||
Env map[string]string `json:"env,omitempty"`
|
||||
}
|
||||
|
||||
func (s GoToolState) Equals(other GoToolState) bool {
|
||||
if s.InstalledVersion != other.InstalledVersion {
|
||||
return false
|
||||
}
|
||||
|
||||
if len(s.BuildArgs) != len(other.BuildArgs) {
|
||||
return false
|
||||
}
|
||||
|
||||
slices.Sort(s.BuildArgs)
|
||||
slices.Sort(other.BuildArgs)
|
||||
|
||||
if !slices.Equal(s.BuildArgs, other.BuildArgs) {
|
||||
return false
|
||||
}
|
||||
|
||||
return maps.Equal(s.Env, other.Env)
|
||||
}
|
Loading…
Reference in a new issue