From 82e581196697bf0158f9ba34a926aca951ff73d8 Mon Sep 17 00:00:00 2001 From: schnee Date: Sat, 7 Dec 2024 17:47:52 +0800 Subject: [PATCH] feat: devsandbox add diffs api --- cnb-builder-shim/Makefile | 1 + cnb-builder-shim/go.mod | 10 +- cnb-builder-shim/go.sum | 32 +--- .../internal/devsandbox/config/loader.go | 5 +- .../internal/devsandbox/config/types.go | 2 +- .../devsandbox/filediffer/filediffer.go | 159 ++++++++++++++++ .../filediffer/filediffer_suite_test.go | 13 ++ .../devsandbox/filediffer/filediffer_test.go | 169 ++++++++++++++++++ .../internal/devsandbox/filediffer/types.go | 41 +++++ .../internal/devsandbox/webserver/server.go | 73 +++++++- 10 files changed, 457 insertions(+), 48 deletions(-) create mode 100644 cnb-builder-shim/internal/devsandbox/filediffer/filediffer.go create mode 100644 cnb-builder-shim/internal/devsandbox/filediffer/filediffer_suite_test.go create mode 100644 cnb-builder-shim/internal/devsandbox/filediffer/filediffer_test.go create mode 100644 cnb-builder-shim/internal/devsandbox/filediffer/types.go diff --git a/cnb-builder-shim/Makefile b/cnb-builder-shim/Makefile index 27b1c83243..dfe370b62a 100644 --- a/cnb-builder-shim/Makefile +++ b/cnb-builder-shim/Makefile @@ -45,6 +45,7 @@ dev-launcher: heroku-builder-bionic: ## Build builder image which is based on `cloudnative-buildpacks/builders/heroku-builder` docker buildx bake heroku-builder-bionic -f docker-build/heroku-builder/docker-bake.hcl + .PHONY: heroku-builder-jammy heroku-builder-jammy: ## Build builder image which is based on `cloudnative-buildpacks/builders/heroku-builder` **experimental** IMAGE_NAME=mirrors.tencent.com/bkpaas/bk-builder-heroku-jammy docker buildx bake heroku-builder-jammy -f docker-build/heroku-builder/docker-bake.hcl diff --git a/cnb-builder-shim/go.mod b/cnb-builder-shim/go.mod index 27b58b6013..88080c220a 100644 --- a/cnb-builder-shim/go.mod +++ b/cnb-builder-shim/go.mod @@ -6,6 +6,8 @@ require ( github.com/BurntSushi/toml v1.3.2 github.com/buildpacks/lifecycle v0.18.3 github.com/caarlos0/env/v10 v10.0.0 + github.com/docker/docker v27.3.1+incompatible + github.com/gin-contrib/cors v1.7.2 github.com/gin-gonic/gin v1.9.1 github.com/go-logr/logr v1.4.2 github.com/google/go-containerregistry v0.16.1 @@ -16,25 +18,24 @@ require ( github.com/pelletier/go-toml/v2 v2.2.1 github.com/pkg/errors v0.9.1 github.com/shabbywu/logfmtr v0.2.3 + github.com/spf13/cobra v1.8.1 github.com/spf13/pflag v1.0.5 gopkg.in/yaml.v3 v3.0.1 ) require ( + github.com/AdaLogics/go-fuzz-headers v0.0.0-20240806141605-e8a1dd7889d6 // indirect github.com/andybalholm/brotli v1.0.5 // indirect github.com/bytedance/sonic v1.11.6 // indirect github.com/bytedance/sonic/loader v0.1.1 // indirect - github.com/chenzhuoyu/base64x v0.0.0-20221115062448-fe3a3abad311 // indirect github.com/cloudwego/base64x v0.1.4 // indirect github.com/cloudwego/iasm v0.2.0 // indirect github.com/containerd/stargz-snapshotter/estargz v0.14.3 // indirect github.com/docker/cli v24.0.6+incompatible // indirect github.com/docker/distribution v2.8.3+incompatible // indirect - github.com/docker/docker v27.3.1+incompatible // indirect github.com/docker/docker-credential-helpers v0.8.0 // indirect github.com/dsnet/compress v0.0.2-0.20210315054119-f66993602bf5 // indirect github.com/gabriel-vasile/mimetype v1.4.3 // indirect - github.com/gin-contrib/cors v1.7.2 // indirect github.com/gin-contrib/sse v0.1.0 // indirect github.com/go-playground/locales v0.14.1 // indirect github.com/go-playground/universal-translator v0.18.1 // indirect @@ -63,8 +64,6 @@ require ( github.com/pierrec/lz4/v4 v4.1.17 // indirect github.com/rogpeppe/go-internal v1.10.0 // indirect github.com/sirupsen/logrus v1.9.3 // indirect - github.com/spf13/cobra v1.8.1 // indirect - github.com/stretchr/testify v1.9.0 // indirect github.com/twitchyliquid64/golang-asm v0.15.1 // indirect github.com/ugorji/go/codec v1.2.12 // indirect github.com/ulikunitz/xz v0.5.11 // indirect @@ -74,7 +73,6 @@ require ( go.opentelemetry.io/otel/trace v1.31.0 // indirect golang.org/x/arch v0.7.0 // indirect golang.org/x/crypto v0.28.0 // indirect - golang.org/x/exp v0.0.0-20241009180824-f66d83c29e7c // indirect golang.org/x/net v0.30.0 // indirect golang.org/x/sync v0.8.0 // indirect golang.org/x/sys v0.26.0 // indirect diff --git a/cnb-builder-shim/go.sum b/cnb-builder-shim/go.sum index 52b2a5098d..ac9383cf09 100644 --- a/cnb-builder-shim/go.sum +++ b/cnb-builder-shim/go.sum @@ -1,3 +1,5 @@ +github.com/AdaLogics/go-fuzz-headers v0.0.0-20240806141605-e8a1dd7889d6 h1:He8afgbRMd7mFxO99hRNu+6tazq8nFF9lIwo9JFroBk= +github.com/AdaLogics/go-fuzz-headers v0.0.0-20240806141605-e8a1dd7889d6/go.mod h1:8o94RPi1/7XTJvwPpRSzSUedZrtlirdB3r9Z20bi2f8= github.com/Azure/go-ansiterm v0.0.0-20230124172434-306776ec8161 h1:L/gRVlceqvL25UVaW/CKtUDjefjrs0SPonmDGUVOYP0= github.com/Azure/go-ansiterm v0.0.0-20230124172434-306776ec8161/go.mod h1:xomTg63KZ2rFqZQzSB4Vz2SUXa1BpHTVz9L5PTmPC4E= github.com/BurntSushi/toml v1.3.2 h1:o7IhLm0Msx3BaB+n3Ag7L8EVlByGnpq14C4YWiu/gL8= @@ -11,18 +13,12 @@ github.com/apex/log v1.9.0 h1:FHtw/xuaM8AgmvDDTI9fiwoAL25Sq2cxojnZICUU8l0= github.com/apex/log v1.9.0/go.mod h1:m82fZlWIuiWzWP04XCTXmnX0xRkYYbCdYn8jbJeLBEA= github.com/buildpacks/lifecycle v0.18.3 h1:vNrQmzOpgdvyu+hYeQJesdpAGcyyYFR+R4zHWQhBOVo= github.com/buildpacks/lifecycle v0.18.3/go.mod h1:lLVVv+RTs2NA5T1Dh/cuvjW4LcUhYsBuY8+mxO3V3HI= -github.com/bytedance/sonic v1.5.0/go.mod h1:ED5hyg4y6t3/9Ku1R6dU/4KyJ48DZ4jPhfY1O2AihPM= -github.com/bytedance/sonic v1.9.1 h1:6iJ6NqdoxCDr6mbY8h18oSO+cShGSMRGCEo7F2h0x8s= -github.com/bytedance/sonic v1.9.1/go.mod h1:i736AoUSYt75HyZLoJW9ERYxcy6eaN6h4BZXU064P/U= github.com/bytedance/sonic v1.11.6 h1:oUp34TzMlL+OY1OUWxHqsdkgC/Zfc85zGqw9siXjrc0= github.com/bytedance/sonic v1.11.6/go.mod h1:LysEHSvpvDySVdC2f87zGWf6CIKJcAvqab1ZaiQtds4= github.com/bytedance/sonic/loader v0.1.1 h1:c+e5Pt1k/cy5wMveRDyk2X4B9hF4g7an8N3zCYjJFNM= github.com/bytedance/sonic/loader v0.1.1/go.mod h1:ncP89zfokxS5LZrJxl5z0UJcsk4M4yY2JpfqGeCtNLU= github.com/caarlos0/env/v10 v10.0.0 h1:yIHUBZGsyqCnpTkbjk8asUlx6RFhhEs+h7TOBdgdzXA= github.com/caarlos0/env/v10 v10.0.0/go.mod h1:ZfulV76NvVPw3tm591U4SwL3Xx9ldzBP9aGxzeN7G18= -github.com/chenzhuoyu/base64x v0.0.0-20211019084208-fb5309c8db06/go.mod h1:DH46F32mSOjUmXrMHnKwZdA8wcEefY7UVqBKYGjpdQY= -github.com/chenzhuoyu/base64x v0.0.0-20221115062448-fe3a3abad311 h1:qSGYFH7+jGhDF8vLC+iwCD4WpbV1EBDSzWkJODFLams= -github.com/chenzhuoyu/base64x v0.0.0-20221115062448-fe3a3abad311/go.mod h1:b583jCggY9gE99b6G5LEC39OIiVsWj+R97kbl5odCEk= 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= @@ -56,8 +52,6 @@ github.com/dsnet/compress v0.0.2-0.20210315054119-f66993602bf5/go.mod h1:qssHWj6 github.com/dsnet/golib v0.0.0-20171103203638-1ea166775780/go.mod h1:Lj+Z9rebOhdfkVLjJ8T6VcRQv3SXugXy999NBtR9aFY= github.com/felixge/httpsnoop v1.0.4 h1:NFTV2Zj1bL4mc9sqWACXbQFVBBg2W3GPvqp8/ESS2Wg= github.com/felixge/httpsnoop v1.0.4/go.mod h1:m8KPJKqk1gH5J9DgRY2ASl2lWCfGKXixSwevea8zH2U= -github.com/gabriel-vasile/mimetype v1.4.2 h1:w5qFW6JKBz9Y393Y4q372O9A7cUSequkh1Q7OhCmWKU= -github.com/gabriel-vasile/mimetype v1.4.2/go.mod h1:zApsH/mKG4w07erKIaJPFiX0Tsq9BFQgN3qGY5GnNgA= github.com/gabriel-vasile/mimetype v1.4.3 h1:in2uUcidCuFcDKtdcBxlR0rJ1+fsokWf+uqxgUFjbI0= github.com/gabriel-vasile/mimetype v1.4.3/go.mod h1:d8uq/6HKRL6CGdk+aubisF/M5GcPfT7nKyLpA0lbSSk= github.com/gin-contrib/cors v1.7.2 h1:oLDHxdg8W/XDoN/8zamqk/Drgt4oVZDvaV0YmvVICQw= @@ -78,8 +72,6 @@ github.com/go-playground/locales v0.14.1 h1:EWaQ/wswjilfKLTECiXz7Rh+3BjFhfDFKv/o github.com/go-playground/locales v0.14.1/go.mod h1:hxrqLVvrK65+Rwrd5Fc6F2O76J/NuW9t0sjnWqG1slY= github.com/go-playground/universal-translator v0.18.1 h1:Bcnm0ZwsGyWbCzImXv+pAJnYK9S473LQFuzCbDbfSFY= github.com/go-playground/universal-translator v0.18.1/go.mod h1:xekY+UJKNuX9WP91TpwSH2VMlDf28Uj24BCp08ZFTUY= -github.com/go-playground/validator/v10 v10.14.0 h1:vgvQWe3XCz3gIeFDm/HnTIbj6UGmg/+t63MyGU2n5js= -github.com/go-playground/validator/v10 v10.14.0/go.mod h1:9iXMNT7sEkjXb0I+enO7QXmzG6QCsPWY4zveKFVRSyU= github.com/go-playground/validator/v10 v10.20.0 h1:K9ISHbSaI0lyB2eWMPJo+kOS/FBExVwjEviJTixqxL8= github.com/go-playground/validator/v10 v10.20.0/go.mod h1:dbuPbCMFw/DrkbEynArYaCwl3amGuJotoKCe95atGMM= github.com/go-task/slim-sprig v0.0.0-20230315185526-52ccab3ef572 h1:tfuBGBXKqDEevZMzYi5KSi8KkcZtzBcTgAUUtapy0OI= @@ -118,8 +110,6 @@ github.com/klauspost/compress v1.17.1 h1:NE3C767s2ak2bweCZo3+rdP4U/HoyVXLv/X9f2g github.com/klauspost/compress v1.17.1/go.mod h1:ntbaceVETuRiXiv4DpjP66DpAtAGkEQskQzEyD//IeE= github.com/klauspost/cpuid v1.2.0/go.mod h1:Pj4uuM528wm8OyEC2QMXAi2YiTZ96dNQPGgoMS4s3ek= github.com/klauspost/cpuid/v2 v2.0.9/go.mod h1:FInQzS24/EEf25PyTYn52gqo7WaD8xa0213Md/qVLRg= -github.com/klauspost/cpuid/v2 v2.2.4 h1:acbojRNwl3o09bUq+yDCtZFc1aiwaAAxtcn8YkZXnvk= -github.com/klauspost/cpuid/v2 v2.2.4/go.mod h1:RVVoqg1df56z8g3pUjL/3lE5UfnlrJX8tyFgg4nqhuY= github.com/klauspost/cpuid/v2 v2.2.7 h1:ZWSB3igEs+d0qvnxR/ZBzXVmxkgt8DdzP6m9pfuVLDM= github.com/klauspost/cpuid/v2 v2.2.7/go.mod h1:Lcz8mBdAVJIBVzewtcLocK12l3Y+JytZYpaMropDUws= github.com/klauspost/pgzip v1.2.5/go.mod h1:Ch1tH69qFZu15pkjo5kYi6mth2Zzwzt50oCQKQE9RUs= @@ -130,8 +120,6 @@ github.com/kr/pretty v0.3.1 h1:flRD4NNwYAUpkphVc1HcthR4KEIFJ65n8Mw5qdRn3LE= github.com/kr/pretty v0.3.1/go.mod h1:hoEshYVHaxMs3cyo3Yncou5ZscifuDolrwPKZanG3xk= github.com/kr/text v0.2.0 h1:5Nx0Ya0ZqY2ygV366QzturHI13Jq95ApcVaJBhpS+AY= github.com/kr/text v0.2.0/go.mod h1:eLer722TekiGuMkidMxC/pM04lWEeraHUUmBw8l2grE= -github.com/leodido/go-urn v1.2.4 h1:XlAE/cm/ms7TE/VMVoduSpNBoyc2dOxHs5MZSwAN63Q= -github.com/leodido/go-urn v1.2.4/go.mod h1:7ZrI8mTSeBSHl/UaRyKQW1qZeMgak41ANeCNaVckg+4= github.com/leodido/go-urn v1.4.0 h1:WT9HwE9SGECu3lg4d/dIA+jxlljEa1/ffXKmRjqdmIQ= github.com/leodido/go-urn v1.4.0/go.mod h1:bvxc+MVxLKB4z00jd1z+Dvzr47oO32F/QSNjSBOlFxI= github.com/mattn/go-isatty v0.0.20 h1:xfD0iDuEKnDkl03q4limB+vH+GxLEtL/jb4xVJSWWEY= @@ -162,8 +150,6 @@ github.com/opencontainers/go-digest v1.0.0 h1:apOUWs51W5PlhuyGyz9FCeeBIOUDA/6nW8 github.com/opencontainers/go-digest v1.0.0/go.mod h1:0JzlMkj0TRzQZfJkVvzbP0HBR3IKzErnv2BNG4W4MAM= github.com/opencontainers/image-spec v1.1.0-rc5 h1:Ygwkfw9bpDvs+c9E34SdgGOj41dX/cbdlwvlWt0pnFI= github.com/opencontainers/image-spec v1.1.0-rc5/go.mod h1:X4pATf0uXsnn3g5aiGIsVnJBR4mxhKzfwmvK/B2NTm8= -github.com/pelletier/go-toml/v2 v2.0.8 h1:0ctb6s9mE31h0/lhu+J6OPmVeDxJn+kYnJc2jZR9tGQ= -github.com/pelletier/go-toml/v2 v2.0.8/go.mod h1:vuYfssBdrU2XDZ9bYydBu6t+6a6PYNcZljzZR9VXg+4= github.com/pelletier/go-toml/v2 v2.2.1 h1:9TA9+T8+8CUCO2+WYnDLCgrYi9+omqKXyjDtosvtEhg= github.com/pelletier/go-toml/v2 v2.2.1/go.mod h1:1t835xjRzz80PqgE6HHgN2JOsmgYu/h4qDAS4n929Rs= github.com/pierrec/lz4/v4 v4.1.2/go.mod h1:gZWDp/Ze/IJXGXf23ltt2EXimqmTUXEy0GFuRQyBid4= @@ -198,15 +184,11 @@ github.com/stretchr/testify v1.7.0/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/ 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/go.mod h1:w2LPCIKwWwSfY2zedu0+kehJoqGctiVI29o6fzry7u4= -github.com/stretchr/testify v1.8.2/go.mod h1:w2LPCIKwWwSfY2zedu0+kehJoqGctiVI29o6fzry7u4= -github.com/stretchr/testify v1.8.3/go.mod h1:sz/lmYIOXD/1dqDmKjjqLyZ2RngseejIcXlSw2iwfAo= github.com/stretchr/testify v1.8.4/go.mod h1:sz/lmYIOXD/1dqDmKjjqLyZ2RngseejIcXlSw2iwfAo= github.com/stretchr/testify v1.9.0 h1:HtqpIVDClZ4nwg75+f6Lvsy/wHu+3BoSGCbBAcpTsTg= github.com/stretchr/testify v1.9.0/go.mod h1:r2ic/lqez/lEtzL7wO/rwa5dbSLXVDPFyf8C91i36aY= github.com/twitchyliquid64/golang-asm v0.15.1 h1:SU5vSMR7hnwNxj24w34ZyCi/FmDZTkS4MhqMhdFk5YI= github.com/twitchyliquid64/golang-asm v0.15.1/go.mod h1:a1lVb/DtPvCB8fslRZhAngC2+aY1QWCk3Cedj/Gdt08= -github.com/ugorji/go/codec v1.2.11 h1:BMaWp1Bb6fHwEtbplGBGJ498wD+LKlNSl25MjdZY4dU= -github.com/ugorji/go/codec v1.2.11/go.mod h1:UNopzCgEMSXjBc6AOMqYvWC1ktqTAfzJZUZgYf6w6lg= github.com/ugorji/go/codec v1.2.12 h1:9LC83zGrHhuUA9l16C9AHXAqEV/2wBQ4nkvumAE65EE= github.com/ugorji/go/codec v1.2.12/go.mod h1:UNopzCgEMSXjBc6AOMqYvWC1ktqTAfzJZUZgYf6w6lg= github.com/ulikunitz/xz v0.5.8/go.mod h1:nbz6k7qbPmH4IRqmfOplQw/tblSgqTqBwxkY0oWt/14= @@ -226,23 +208,17 @@ go.opentelemetry.io/otel/metric v1.31.0/go.mod h1:C3dEloVbLuYoX41KpmAhOqNriGbA+q go.opentelemetry.io/otel/trace v1.31.0 h1:ffjsj1aRouKewfr85U2aGagJ46+MvodynlQ1HYdmJys= go.opentelemetry.io/otel/trace v1.31.0/go.mod h1:TXZkRk7SM2ZQLtR6eoAWQFIHPvzQ06FJAsO1tJg480A= golang.org/x/arch v0.0.0-20210923205945-b76863e36670/go.mod h1:5om86z9Hs0C8fWVUuoMHwpExlXzs5Tkyp9hOrfG7pp8= -golang.org/x/arch v0.3.0 h1:02VY4/ZcO/gBOH6PUaoiptASxtXU10jazRCP865E97k= -golang.org/x/arch v0.3.0/go.mod h1:5om86z9Hs0C8fWVUuoMHwpExlXzs5Tkyp9hOrfG7pp8= golang.org/x/arch v0.7.0 h1:pskyeJh/3AmoQ8CPE95vxHLqp1G1GfGNXTmcl9NEKTc= golang.org/x/arch v0.7.0/go.mod h1:FEVrYAQjsQXMVJ1nsMoVVXPZg6p2JE2mx8psSWTDQys= golang.org/x/crypto v0.28.0 h1:GBDwsMXVQi34v5CCYUm2jkJvu4cbtru2U4TN2PSyQnw= golang.org/x/crypto v0.28.0/go.mod h1:rmgy+3RHxRZMyY0jjAJShp2zgEdOqj2AO7U0pYmeQ7U= -golang.org/x/exp v0.0.0-20241009180824-f66d83c29e7c h1:7dEasQXItcW1xKJ2+gg5VOiBnqWrJc+rq0DPKyvvdbY= -golang.org/x/exp v0.0.0-20241009180824-f66d83c29e7c/go.mod h1:NQtJDoLvd6faHhE7m4T/1IY708gDefGGjR/iUW8yQQ8= -golang.org/x/mod v0.17.0 h1:zY54UmvipHiNd+pm+m0x9KhZ9hl1/7QNMyxXbc6ICqA= -golang.org/x/mod v0.17.0/go.mod h1:hTbmBsO62+eylJbnUtE2MGJUyE7QWk4xUqPFrRgJ+7c= golang.org/x/mod v0.21.0 h1:vvrHzRwRfVKSiLrG+d4FMl/Qi4ukBCE6kZlTUkDYRT0= +golang.org/x/mod v0.21.0/go.mod h1:6SkKJ3Xj0I0BrPOZoBy3bdMptDDU9oJrpohJ3eWZ1fY= golang.org/x/net v0.30.0 h1:AcW1SDZMkb8IpzCdQUaIq2sP4sZ4zw+55h6ynffypl4= golang.org/x/net v0.30.0/go.mod h1:2wGyMJ5iFasEhkwi13ChkO/t1ECNC4X4eBKkVFyYFlU= golang.org/x/sync v0.8.0 h1:3NFvSEYkUoMifnESzZl15y791HH1qU2xm6eCJU5ZPXQ= golang.org/x/sync v0.8.0/go.mod h1:Czt+wKu1gCyEFDUtn0jG5QVvpJ6rzVqr5aXyt9drQfk= golang.org/x/sys v0.0.0-20191204072324-ce4227a45e2e/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= -golang.org/x/sys v0.0.0-20220704084225-05e143d24a9e/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/sys v0.0.0-20220715151400-c0bba94af5f8/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/sys v0.5.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/sys v0.6.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= @@ -250,8 +226,6 @@ golang.org/x/sys v0.26.0 h1:KHjCJyddX0LoSTb3J+vWpupP9p0oznkqVk/IfjymZbo= golang.org/x/sys v0.26.0/go.mod h1:/VUhepiaJMQUp4+oa/7Zr1D23ma6VTLIYjOOTFZPUcA= golang.org/x/text v0.19.0 h1:kTxAhCbGbxhK0IwgSKiMO5awPoDQ0RpfiVYBfK860YM= golang.org/x/text v0.19.0/go.mod h1:BuEKDfySbSR4drPmRPG/7iBdf8hvFMuRexcpahXilzY= -golang.org/x/tools v0.21.1-0.20240508182429-e35e4ccd0d2d h1:vU5i/LfpvrRCpgM/VPfJLg5KjxD3E+hfT1SH+d9zLwg= -golang.org/x/tools v0.21.1-0.20240508182429-e35e4ccd0d2d/go.mod h1:aiJjzUbINMkxbQROHiO6hDPo2LHcIPhhQsa9DLh0yGk= golang.org/x/tools v0.26.0 h1:v/60pFQmzmT9ExmjDv2gGIfi3OqfKoEP6I5+umXlbnQ= golang.org/x/tools v0.26.0/go.mod h1:TPVVj70c7JJ3WCazhD8OdXcZg/og+b9+tH/KxylGwH0= golang.org/x/xerrors v0.0.0-20191204190536-9bdfabe68543/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= diff --git a/cnb-builder-shim/internal/devsandbox/config/loader.go b/cnb-builder-shim/internal/devsandbox/config/loader.go index ce4318209e..6d344040e4 100644 --- a/cnb-builder-shim/internal/devsandbox/config/loader.go +++ b/cnb-builder-shim/internal/devsandbox/config/loader.go @@ -48,14 +48,13 @@ func loadCORSConfigFromEnv() (CORSConfig, error) { AllowCredentials: allowCredentials, }, nil } + func loadServiceConfigFromEnv() (ServiceConfig, error) { corsConfig, err := loadCORSConfigFromEnv() if err != nil { return ServiceConfig{}, err } - return ServiceConfig{ - CORS: corsConfig, - }, nil + return ServiceConfig{CORS: corsConfig}, nil } func loadConfigFromEnv() (*Config, error) { diff --git a/cnb-builder-shim/internal/devsandbox/config/types.go b/cnb-builder-shim/internal/devsandbox/config/types.go index 7f4077fcd6..6d35dcc319 100644 --- a/cnb-builder-shim/internal/devsandbox/config/types.go +++ b/cnb-builder-shim/internal/devsandbox/config/types.go @@ -25,7 +25,7 @@ type SourceCodeConfig struct { type CORSConfig struct { // 允许的来源 AllowOrigins []string - // 允许的HTTP方法 + // 允许的 HTTP 方法 AllowMethods []string // 允许的请求头 AllowHeaders []string diff --git a/cnb-builder-shim/internal/devsandbox/filediffer/filediffer.go b/cnb-builder-shim/internal/devsandbox/filediffer/filediffer.go new file mode 100644 index 0000000000..f4ef0ef500 --- /dev/null +++ b/cnb-builder-shim/internal/devsandbox/filediffer/filediffer.go @@ -0,0 +1,159 @@ +/* + * TencentBlueKing is pleased to support the open source community by making + * 蓝鲸智云 - PaaS 平台 (BlueKing - PaaS System) available. + * Copyright (C) 2017 THL A29 Limited, a Tencent company. All rights reserved. + * Licensed under the MIT License (the "License"); you may not use this file except + * in compliance with the License. You may obtain a copy of the License at + * + * http://opensource.org/licenses/MIT + * + * Unless required by applicable law or agreed to in writing, software distributed under + * the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, + * either express or implied. See the License for the specific language governing permissions and + * limitations under the License. + * + * We undertake not to change the open source license (MIT license) applicable + * to the current version of the project delivered to anyone in the future. + */ +package filediffer + +import ( + "bytes" + "io" + "os" + "os/exec" + "path" + "strings" + + "github.com/pkg/errors" +) + +// FileDiffer 文件变更对比器 +type FileDiffer struct { + srcPath string +} + +// New ... +func New() *FileDiffer { + return &FileDiffer{} +} + +// Prepare 准备步骤 +func (d *FileDiffer) Prepare(srcPath string) error { + d.srcPath = srcPath + + _, err := os.Stat(path.Join(d.srcPath, ".git")) + // 如果对应目录下存在 .git 目录,跳过 + if err == nil { + return nil + } + + // 没有 .git 目录(使用子目录部署的情况),执行以下命令以初始化 + // 1. git init + // 2. git add . + // 3. git commit --author="bkpaas " -m "init" + if os.IsNotExist(err) { + commands := [][]string{ + {"init"}, + {"add", "."}, + {"commit", "--author=bkpaas ", "-m", "init"}, + } + for _, cmd := range commands { + if _, err = d.runGitCommand(cmd...); err != nil { + return err + } + } + return nil + } + + // 其他错误,直接返回 + return err +} + +// Diff 对比输出文件变更信息 +func (d *FileDiffer) Diff() ([]File, error) { + if _, err := d.runGitCommand("add", "."); err != nil { + return nil, err + } + output, err := d.runGitCommand("diff", "--cached", "--name-status") + if err != nil { + return nil, err + } + + lines := strings.Split(output, "\n") + files := []File{} + for _, line := range lines { + if line == "" { + continue + } + fields := strings.Fields(line) + if len(fields) != 2 { + return nil, errors.Errorf("invalid line: %s", line) + } + var action FileAction + switch fields[0] { + case "A": + action = FileActionAdded + case "M": + action = FileActionModified + case "D": + action = FileActionDeleted + default: + return nil, errors.Errorf("unknown action: %s", fields[0]) + } + // 强制忽略部分变更文件 + if d.mustIgnoreFile(fields[1]) { + continue + } + + var content string + // 如果是删除操作,不加载文件 + if action != FileActionDeleted { + if content, err = d.loadFileContent(fields[1]); err != nil { + return nil, err + } + } + files = append(files, File{Action: action, Path: fields[1], Content: content}) + } + return files, nil +} + +// 执行 Git 命令 +func (d *FileDiffer) runGitCommand(args ...string) (string, error) { + cmd := exec.Command("git", args...) + cmd.Dir = d.srcPath + + var out bytes.Buffer + cmd.Stdout = &out + cmd.Stderr = &out + if err := cmd.Run(); err != nil { + return "", errors.Errorf("git command failed: %s, output: %s", err, out.String()) + } + + return out.String(), nil +} + +// 加载指定文件内容 +func (d *FileDiffer) loadFileContent(filepath string) (string, error) { + file, err := os.Open(path.Join(d.srcPath, filepath)) + if err != nil { + return "", err + } + defer file.Close() + + content, err := io.ReadAll(file) + if err != nil { + return "", err + } + return string(content), nil +} + +// 判断变更的文件是否需要被忽略 +func (d *FileDiffer) mustIgnoreFile(filepath string) bool { + for _, prefix := range forceIgnoreFilePathPrefixes { + if strings.HasPrefix(filepath, prefix) { + return true + } + } + return false +} diff --git a/cnb-builder-shim/internal/devsandbox/filediffer/filediffer_suite_test.go b/cnb-builder-shim/internal/devsandbox/filediffer/filediffer_suite_test.go new file mode 100644 index 0000000000..206e8ca5ca --- /dev/null +++ b/cnb-builder-shim/internal/devsandbox/filediffer/filediffer_suite_test.go @@ -0,0 +1,13 @@ +package filediffer_test + +import ( + "testing" + + . "github.com/onsi/ginkgo/v2" + . "github.com/onsi/gomega" +) + +func TestFileDiffer(t *testing.T) { + RegisterFailHandler(Fail) + RunSpecs(t, "File Differ Suite") +} diff --git a/cnb-builder-shim/internal/devsandbox/filediffer/filediffer_test.go b/cnb-builder-shim/internal/devsandbox/filediffer/filediffer_test.go new file mode 100644 index 0000000000..3dfcd2de06 --- /dev/null +++ b/cnb-builder-shim/internal/devsandbox/filediffer/filediffer_test.go @@ -0,0 +1,169 @@ +/* + * TencentBlueKing is pleased to support the open source community by making + * 蓝鲸智云 - PaaS 平台 (BlueKing - PaaS System) available. + * Copyright (C) 2017 THL A29 Limited, a Tencent company. All rights reserved. + * Licensed under the MIT License (the "License"); you may not use this file except + * in compliance with the License. You may obtain a copy of the License at + * + * http://opensource.org/licenses/MIT + * + * Unless required by applicable law or agreed to in writing, software distributed under + * the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, + * either express or implied. See the License for the specific language governing permissions and + * limitations under the License. + * + * We undertake not to change the open source license (MIT license) applicable + * to the current version of the project delivered to anyone in the future. + */ + +package filediffer + +import ( + "os" + "os/exec" + "path" + + . "github.com/onsi/ginkgo/v2" + . "github.com/onsi/gomega" +) + +var _ = Describe("Test Differ", func() { + var tmpDir string + + var initFile = func(dir, filename, content string) error { + if err := os.MkdirAll(dir, os.ModePerm); err != nil { + return err + } + + file, err := os.Create(path.Join(dir, filename)) + if err != nil { + return err + } + defer file.Close() + + // 把文件名写入文件作为内容 + _, err = file.WriteString(content) + return err + } + + var editFile = func(dir, filename, content string) error { + file, err := os.OpenFile(path.Join(dir, filename), os.O_WRONLY|os.O_TRUNC, 0644) + if err != nil { + return err + } + defer file.Close() + + // 追加内容到文件 + _, err = file.WriteString(content) + return err + } + + var runGitCommand = func(dir string, args ...string) error { + cmd := exec.Command("git", args...) + cmd.Dir = dir + return cmd.Run() + } + + // 初始化临时目录 & 文件 + BeforeEach(func() { + tmpDir, _ = os.MkdirTemp("", "filediffer") + for _, filename := range []string{"example.txt", "example.py", "example.go"} { + Expect(initFile(tmpDir, filename, path.Ext(filename))).To(BeNil()) + } + }) + // 清理临时目录 + AfterEach(func() { + Expect(os.RemoveAll(tmpDir)).To(BeNil()) + }) + + Context("Test FileDiffer", func() { + It("with .git", func() { + Expect(runGitCommand(tmpDir, "init")).To(BeNil()) + Expect(runGitCommand(tmpDir, "add", ".")).To(BeNil()) + Expect(runGitCommand(tmpDir, "commit", "--author=bkpaas ", "-m", "init")).To(BeNil()) + + differ := New() + Expect(differ.Prepare(tmpDir)).To(BeNil()) + + files, err := differ.Diff() + Expect(err).To(BeNil()) + Expect(files).To(HaveLen(0)) + + _ = initFile(tmpDir, "example.html", ".html") + _ = initFile(tmpDir, "example.js", ".js") + + files, err = differ.Diff() + Expect(err).To(BeNil()) + Expect(files).To(Equal([]File{ + {Action: FileActionAdded, Path: "example.html", Content: ".html"}, + {Action: FileActionAdded, Path: "example.js", Content: ".js"}, + })) + + _ = os.Remove(path.Join(tmpDir, "example.html")) + _ = os.Remove(path.Join(tmpDir, "example.py")) + files, err = differ.Diff() + Expect(err).To(BeNil()) + Expect(files).To(Equal([]File{ + {Action: FileActionAdded, Path: "example.js", Content: ".js"}, + {Action: FileActionDeleted, Path: "example.py", Content: ""}, + })) + + _ = editFile(tmpDir, "example.go", "gogo") + files, err = differ.Diff() + Expect(err).To(BeNil()) + Expect(files).To(Equal([]File{ + {Action: FileActionModified, Path: "example.go", Content: "gogo"}, + {Action: FileActionAdded, Path: "example.js", Content: ".js"}, + {Action: FileActionDeleted, Path: "example.py", Content: ""}, + })) + }) + + It("without .git", func() { + differ := New() + Expect(differ.Prepare(tmpDir)).To(BeNil()) + + files, err := differ.Diff() + Expect(err).To(BeNil()) + Expect(files).To(HaveLen(0)) + + _ = initFile(path.Join(tmpDir, "webfe/templates"), "example.html", ".html") + _ = initFile(path.Join(tmpDir, "webfe/static"), "example.js", ".js") + _ = initFile(path.Join(tmpDir, "webfe/static"), "example.css", ".css") + _ = os.Remove(path.Join(tmpDir, "webfe/templates/example.html")) + _ = os.Remove(path.Join(tmpDir, "example.py")) + _ = editFile(tmpDir, "example.go", "gogo") + _ = editFile(path.Join(tmpDir, "webfe/static"), "example.js", "js-js") + _ = editFile(tmpDir, "example.txt", "txt no.1") + + files, err = differ.Diff() + Expect(err).To(BeNil()) + Expect(files).To(HaveLen(5)) + Expect(files).To(Equal([]File{ + {Action: FileActionModified, Path: "example.go", Content: "gogo"}, + {Action: FileActionDeleted, Path: "example.py", Content: ""}, + {Action: FileActionModified, Path: "example.txt", Content: "txt no.1"}, + {Action: FileActionAdded, Path: "webfe/static/example.css", Content: ".css"}, + {Action: FileActionAdded, Path: "webfe/static/example.js", Content: "js-js"}, + })) + }) + + It("with ignore", func() { + differ := New() + Expect(differ.Prepare(tmpDir)).To(BeNil()) + + _ = initFile(path.Join(tmpDir, "webfe/templates"), "example.html", ".html") + _ = initFile(path.Join(tmpDir, "v3logs"), "celery.log", "celery is running...") + _ = initFile(path.Join(tmpDir, "v3logs"), "gunicorn.log", "gunicorn is running...") + _ = editFile(tmpDir, "example.txt", "txt no.1") + _ = os.Remove(path.Join(tmpDir, "example.go")) + + files, err := differ.Diff() + Expect(err).To(BeNil()) + Expect(files).To(Equal([]File{ + {Action: FileActionDeleted, Path: "example.go", Content: ""}, + {Action: FileActionModified, Path: "example.txt", Content: "txt no.1"}, + {Action: FileActionAdded, Path: "webfe/templates/example.html", Content: ".html"}, + })) + }) + }) +}) diff --git a/cnb-builder-shim/internal/devsandbox/filediffer/types.go b/cnb-builder-shim/internal/devsandbox/filediffer/types.go new file mode 100644 index 0000000000..66090d5f25 --- /dev/null +++ b/cnb-builder-shim/internal/devsandbox/filediffer/types.go @@ -0,0 +1,41 @@ +/* + * TencentBlueKing is pleased to support the open source community by making + * 蓝鲸智云 - PaaS 平台 (BlueKing - PaaS System) available. + * Copyright (C) 2017 THL A29 Limited, a Tencent company. All rights reserved. + * Licensed under the MIT License (the "License"); you may not use this file except + * in compliance with the License. You may obtain a copy of the License at + * + * http://opensource.org/licenses/MIT + * + * Unless required by applicable law or agreed to in writing, software distributed under + * the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, + * either express or implied. See the License for the specific language governing permissions and + * limitations under the License. + * + * We undertake not to change the open source license (MIT license) applicable + * to the current version of the project delivered to anyone in the future. + */ + +package filediffer + +// 强制忽略的文件路径前缀 +var forceIgnoreFilePathPrefixes = []string{"v3logs"} + +// FileAction 文件操作类型 +type FileAction string + +const ( + // FileActionAdded 文件增加 + FileActionAdded FileAction = "added" + // FileActionModified 文件变更 + FileActionModified FileAction = "modified" + // FileActionDeleted 文件删除 + FileActionDeleted FileAction = "deleted" +) + +// File 文件详情 +type File struct { + Action FileAction `json:"action"` + Path string `json:"path"` + Content string `json:"content"` +} diff --git a/cnb-builder-shim/internal/devsandbox/webserver/server.go b/cnb-builder-shim/internal/devsandbox/webserver/server.go index cce8589e9f..eed9d260e8 100644 --- a/cnb-builder-shim/internal/devsandbox/webserver/server.go +++ b/cnb-builder-shim/internal/devsandbox/webserver/server.go @@ -35,6 +35,7 @@ import ( "github.com/TencentBlueking/bkpaas/cnb-builder-shim/internal/devsandbox" "github.com/TencentBlueking/bkpaas/cnb-builder-shim/internal/devsandbox/config" + "github.com/TencentBlueking/bkpaas/cnb-builder-shim/internal/devsandbox/filediffer" "github.com/TencentBlueking/bkpaas/cnb-builder-shim/internal/devsandbox/webserver/service" "github.com/TencentBlueking/bkpaas/cnb-builder-shim/pkg/appdesc" "github.com/TencentBlueking/bkpaas/cnb-builder-shim/pkg/utils" @@ -93,6 +94,7 @@ func New(lg *logr.Logger) (*WebServer, error) { r.GET("/app_logs", AppLogHandler()) r.GET("/processes/status", ProcessStatusHandler()) r.GET("/processes/list", ProcessListHandler()) + r.GET("/diffs", DiffsHandler()) return s, nil } @@ -139,41 +141,56 @@ func DeployHandler(s *WebServer, svc service.DeployServiceHandler) gin.HandlerFu // 创建临时文件夹 tmpDir, err := os.MkdirTemp("", "source-*") if err != nil { - c.JSON(http.StatusInternalServerError, gin.H{"message": fmt.Sprintf("create tmp dir err: %s", err.Error())}) + c.JSON( + http.StatusInternalServerError, + gin.H{"message": fmt.Sprintf("create tmp dir err: %s", err.Error())}, + ) return } defer os.RemoveAll(tmpDir) file, err := c.FormFile("file") if err != nil { - c.JSON(http.StatusInternalServerError, gin.H{"message": fmt.Sprintf("get form err: %s", err.Error())}) + c.JSON( + http.StatusInternalServerError, + gin.H{"message": fmt.Sprintf("get form err: %s", err.Error())}, + ) return } fileName := filepath.Base(file.Filename) dst := path.Join(s.env.UploadDir, fileName) if len(dst) > 0 && dst[len(dst)-1] == '.' { - c.JSON(http.StatusBadRequest, gin.H{"message": fmt.Sprintf("invalid file name: %s", file.Filename)}) + c.JSON( + http.StatusBadRequest, + gin.H{"message": fmt.Sprintf("invalid file name: %s", file.Filename)}, + ) return } if err = c.SaveUploadedFile(file, dst); err != nil { - c.JSON(http.StatusInternalServerError, gin.H{"message": fmt.Sprintf("upload file err: %s", err.Error())}) + c.JSON( + http.StatusInternalServerError, + gin.H{"message": fmt.Sprintf("upload file err: %s", err.Error())}, + ) return } // 解压文件到临时目录 if err = utils.Unzip(dst, tmpDir); err != nil { - c.JSON(http.StatusInternalServerError, gin.H{"message": fmt.Sprintf("unzip file err: %s", err.Error())}) + c.JSON( + http.StatusInternalServerError, + gin.H{"message": fmt.Sprintf("unzip file err: %s", err.Error())}, + ) return } srcFilePath = path.Join(tmpDir, strings.TrimSuffix(fileName, filepath.Ext(fileName))) case config.BK_REPO: srcFilePath = config.G.SourceCode.Workspace case config.GIT: - c.JSON(http.StatusBadRequest, gin.H{"message": fmt.Sprintf("unsupported source fetch method: %s", config.G.SourceCode.FetchMethod)}) - return + fallthrough default: - c.JSON(http.StatusBadRequest, gin.H{"message": fmt.Sprintf("unsupported source fetch method: %s", config.G.SourceCode.FetchMethod)}) + errMsg := fmt.Sprintf("unsupported source fetch method: %s", config.G.SourceCode.FetchMethod) + c.JSON(http.StatusBadRequest, gin.H{"message": errMsg}) return } @@ -184,7 +201,11 @@ func DeployHandler(s *WebServer, svc service.DeployServiceHandler) gin.HandlerFu } select { - case s.ch <- devsandbox.AppReloadEvent{ID: status.DeployID, Rebuild: status.StepOpts.Rebuild, Relaunch: status.StepOpts.Relaunch}: + case s.ch <- devsandbox.AppReloadEvent{ + ID: status.DeployID, + Rebuild: status.StepOpts.Rebuild, + Relaunch: status.StepOpts.Relaunch, + }: c.JSON(http.StatusOK, gin.H{"deployID": status.DeployID}) default: c.JSON( @@ -262,6 +283,40 @@ func ProcessListHandler() gin.HandlerFunc { } } +// DiffsHandler 提供文件变更信息 +func DiffsHandler() gin.HandlerFunc { + return func(c *gin.Context) { + // 由于目前 HTTP 附带文件的源码初始化逻辑不同,暂时不支持 TODO 后续重构时需要统一 + if config.G.SourceCode.FetchMethod != config.BK_REPO { + c.JSON( + http.StatusBadRequest, + gin.H{"message": fmt.Sprintf("unsupported fetch method: %s", config.G.SourceCode.FetchMethod)}, + ) + return + } + + differ := filediffer.New() + // 初始化 + if err := differ.Prepare(config.G.SourceCode.Workspace); err != nil { + c.JSON( + http.StatusInternalServerError, + gin.H{"message": fmt.Sprintf("file differ prepare failed: %s", err)}, + ) + return + } + // 获取文件变更信息 + files, err := differ.Diff() + if err != nil { + c.JSON( + http.StatusInternalServerError, + gin.H{"message": fmt.Sprintf("failed to diff files: %s", err)}, + ) + return + } + c.JSON(http.StatusOK, gin.H{"data": files}) + } +} + // HealthzHandler ... func HealthzHandler() gin.HandlerFunc { return func(c *gin.Context) {