Compare commits
	
		
			13 Commits
		
	
	
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|  | 68959a741a | ||
|  | c327898faa | ||
|  | 6768e02d91 | ||
|  | 656e7b2fc1 | ||
|  | 9624b2b9ad | ||
|  | 48d1cf9ff8 | ||
|  | 52d4a68d86 | ||
|  | e08d238243 | ||
|  | 92e846002e | ||
|  | 708a07fec6 | ||
|  | db43a4e741 | ||
|  | 7a37ec05c4 | ||
|  | 0868adad0a | 
							
								
								
									
										2
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							| @@ -15,3 +15,5 @@ | |||||||
| # Misc | # Misc | ||||||
| _gopath | _gopath | ||||||
| _gocache | _gocache | ||||||
|  |  | ||||||
|  | dist/ | ||||||
|   | |||||||
							
								
								
									
										37
									
								
								.goreleaser.yaml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										37
									
								
								.goreleaser.yaml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,37 @@ | |||||||
|  | # This is an example .goreleaser.yml file with some sensible defaults. | ||||||
|  | # Make sure to check the documentation at https://goreleaser.com | ||||||
|  | before: | ||||||
|  |   hooks: | ||||||
|  |     # You may remove this if you don't use go modules. | ||||||
|  |     - go mod tidy | ||||||
|  |     # you may remove this if you don't need go generate | ||||||
|  |     - go generate ./... | ||||||
|  | builds: | ||||||
|  |   - env: | ||||||
|  |       - CGO_ENABLED=0 | ||||||
|  |     goos: | ||||||
|  |       - linux | ||||||
|  |       - darwin | ||||||
|  |     goarch: | ||||||
|  |       - arm64 | ||||||
|  |       - amd64 | ||||||
|  |       - 386 | ||||||
|  |     main: ./cmd/gomodproxy/main.go | ||||||
|  | archives: | ||||||
|  |   - replacements: | ||||||
|  |       darwin: Darwin | ||||||
|  |       linux: Linux | ||||||
|  |       windows: Windows | ||||||
|  |       386: i386 | ||||||
|  |       amd64: x86_64 | ||||||
|  |       arm64: Arm64 | ||||||
|  | checksum: | ||||||
|  |   name_template: 'checksums.txt' | ||||||
|  | snapshot: | ||||||
|  |   name_template: "{{ incpatch .Version }}-next" | ||||||
|  | changelog: | ||||||
|  |   sort: asc | ||||||
|  |   filters: | ||||||
|  |     exclude: | ||||||
|  |       - '^docs:' | ||||||
|  |       - '^test:' | ||||||
							
								
								
									
										11
									
								
								Dockerfile
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										11
									
								
								Dockerfile
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,11 @@ | |||||||
|  | FROM golang:1.16.7 as build | ||||||
|  | ENV HOME /opt/app | ||||||
|  | COPY . $HOME | ||||||
|  | WORKDIR $HOME | ||||||
|  | RUN ls $HOME | ||||||
|  | RUN go build cmd/gomodproxy/main.go && \ | ||||||
|  |     go clean | ||||||
|  |  | ||||||
|  | FROM debian:buster | ||||||
|  | COPY --from=build /go/bin/ /go/bin/ | ||||||
|  | ENTRYPOINT ["/go/bin/gomodproxy"] | ||||||
| @@ -8,6 +8,10 @@ gomodproxy is a caching proxy for [Go modules]. | |||||||
|  |  | ||||||
| Go 1.11 has introduced optional proxy support via GOPROXY environment variable.  It is essential for use cases where you want to have better control over your dependencies and handle scenarios when GitHub is down or some open-source dependency has been removed. | Go 1.11 has introduced optional proxy support via GOPROXY environment variable.  It is essential for use cases where you want to have better control over your dependencies and handle scenarios when GitHub is down or some open-source dependency has been removed. | ||||||
|  |  | ||||||
|  | ## Releasing | ||||||
|  |  | ||||||
|  | See https://goreleaser.com/quick-start/ | ||||||
|  |  | ||||||
| ## Getting started | ## Getting started | ||||||
|  |  | ||||||
| gomodproxy requires Go 1.11 or newer. There are no plans to support `vgo` or Go 1.11 beta versions. | gomodproxy requires Go 1.11 or newer. There are no plans to support `vgo` or Go 1.11 beta versions. | ||||||
|   | |||||||
| @@ -16,7 +16,7 @@ import ( | |||||||
| 	"time" | 	"time" | ||||||
| 	"unicode" | 	"unicode" | ||||||
|  |  | ||||||
| 	"github.com/sixt/gomodproxy/pkg/api" | 	"github.com/bilus/gomodproxy/pkg/api" | ||||||
|  |  | ||||||
| 	"expvar" | 	"expvar" | ||||||
| 	_ "net/http/pprof" | 	_ "net/http/pprof" | ||||||
| @@ -137,7 +137,8 @@ func main() { | |||||||
| 		if len(kv) != 2 { | 		if len(kv) != 2 { | ||||||
| 			log.Fatal("bad git path:", path) | 			log.Fatal("bad git path:", path) | ||||||
| 		} | 		} | ||||||
| 		options = append(options, api.Git(kv[0], kv[1])) | 		password := os.Getenv("SSH_PASSPHRASE") | ||||||
|  | 		options = append(options, api.GitWithEphemeralTags(kv[0], kv[1], password)) | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	for _, path := range vcsPaths { | 	for _, path := range vcsPaths { | ||||||
|   | |||||||
							
								
								
									
										16
									
								
								go.mod
									
									
									
									
									
								
							
							
						
						
									
										16
									
								
								go.mod
									
									
									
									
									
								
							| @@ -1,13 +1,5 @@ | |||||||
| module github.com/sixt/gomodproxy | module github.com/bilus/gomodproxy | ||||||
|  |  | ||||||
| require ( | go 1.16 | ||||||
| 	github.com/emirpasic/gods v1.12.0 // indirect |  | ||||||
| 	github.com/pkg/errors v0.8.1 // indirect | require gopkg.in/src-d/go-git.v4 v4.13.1 | ||||||
| 	github.com/stretchr/testify v1.3.0 // indirect |  | ||||||
| 	golang.org/x/crypto v0.0.0-20190103213133-ff983b9c42bc // indirect |  | ||||||
| 	golang.org/x/net v0.0.0-20190110200230-915654e7eabc // indirect |  | ||||||
| 	golang.org/x/sys v0.0.0-20190109145017-48ac38b7c8cb // indirect |  | ||||||
| 	gopkg.in/src-d/go-billy.v4 v4.3.0 // indirect |  | ||||||
| 	gopkg.in/src-d/go-git-fixtures.v3 v3.3.0 // indirect |  | ||||||
| 	gopkg.in/src-d/go-git.v4 v4.8.1 |  | ||||||
| ) |  | ||||||
|   | |||||||
							
								
								
									
										72
									
								
								go.sum
									
									
									
									
									
								
							
							
						
						
									
										72
									
								
								go.sum
									
									
									
									
									
								
							| @@ -2,33 +2,33 @@ github.com/alcortesm/tgz v0.0.0-20161220082320-9c5fe88206d7 h1:uSoVVbwJiQipAclBb | |||||||
| github.com/alcortesm/tgz v0.0.0-20161220082320-9c5fe88206d7/go.mod h1:6zEj6s6u/ghQa61ZWa/C2Aw3RkjiTBOix7dkqa1VLIs= | github.com/alcortesm/tgz v0.0.0-20161220082320-9c5fe88206d7/go.mod h1:6zEj6s6u/ghQa61ZWa/C2Aw3RkjiTBOix7dkqa1VLIs= | ||||||
| github.com/anmitsu/go-shlex v0.0.0-20161002113705-648efa622239 h1:kFOfPq6dUM1hTo4JG6LR5AXSUEsOjtdm0kw0FtQtMJA= | github.com/anmitsu/go-shlex v0.0.0-20161002113705-648efa622239 h1:kFOfPq6dUM1hTo4JG6LR5AXSUEsOjtdm0kw0FtQtMJA= | ||||||
| github.com/anmitsu/go-shlex v0.0.0-20161002113705-648efa622239/go.mod h1:2FmKhYUyUczH0OGQWaF5ceTx0UBShxjsH6f8oGKYe2c= | github.com/anmitsu/go-shlex v0.0.0-20161002113705-648efa622239/go.mod h1:2FmKhYUyUczH0OGQWaF5ceTx0UBShxjsH6f8oGKYe2c= | ||||||
|  | github.com/armon/go-socks5 v0.0.0-20160902184237-e75332964ef5 h1:0CwZNZbxp69SHPdPJAN/hZIm0C4OItdklCFmMRWYpio= | ||||||
|  | github.com/armon/go-socks5 v0.0.0-20160902184237-e75332964ef5/go.mod h1:wHh0iHkYZB8zMSxRWpUBQtwG5a7fFgvEO+odwuTv2gs= | ||||||
|  | github.com/creack/pty v1.1.7/go.mod h1:lj5s0c3V2DBrqTV7llrYr5NG6My20zk30Fl46Y7DoTY= | ||||||
| github.com/davecgh/go-spew v1.1.0/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38= | github.com/davecgh/go-spew v1.1.0/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38= | ||||||
| github.com/davecgh/go-spew v1.1.1 h1:vj9j/u1bqnvCEfJOwUhtlOARqs3+rkHYY13jYWTU97c= | github.com/davecgh/go-spew v1.1.1 h1:vj9j/u1bqnvCEfJOwUhtlOARqs3+rkHYY13jYWTU97c= | ||||||
| github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38= | github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38= | ||||||
| github.com/emirpasic/gods v1.9.0/go.mod h1:YfzfFFoVP/catgzJb4IKIqXjX78Ha8FMSDh3ymbK86o= |  | ||||||
| github.com/emirpasic/gods v1.12.0 h1:QAUIPSaCu4G+POclxeqb3F+WPpdKqFGlw36+yOzGlrg= | github.com/emirpasic/gods v1.12.0 h1:QAUIPSaCu4G+POclxeqb3F+WPpdKqFGlw36+yOzGlrg= | ||||||
| github.com/emirpasic/gods v1.12.0/go.mod h1:YfzfFFoVP/catgzJb4IKIqXjX78Ha8FMSDh3ymbK86o= | github.com/emirpasic/gods v1.12.0/go.mod h1:YfzfFFoVP/catgzJb4IKIqXjX78Ha8FMSDh3ymbK86o= | ||||||
| github.com/flynn/go-shlex v0.0.0-20150515145356-3f9db97f8568 h1:BHsljHzVlRcyQhjrss6TZTdY2VfCqZPbv5k3iBFa2ZQ= |  | ||||||
| github.com/flynn/go-shlex v0.0.0-20150515145356-3f9db97f8568/go.mod h1:xEzjJPgXI435gkrCt3MPfRiAkVrwSbHsst4LCFVfpJc= | github.com/flynn/go-shlex v0.0.0-20150515145356-3f9db97f8568/go.mod h1:xEzjJPgXI435gkrCt3MPfRiAkVrwSbHsst4LCFVfpJc= | ||||||
| github.com/gliderlabs/ssh v0.1.1 h1:j3L6gSLQalDETeEg/Jg0mGY0/y/N6zI2xX1978P0Uqw= | github.com/gliderlabs/ssh v0.2.2 h1:6zsha5zo/TWhRhwqCD3+EarCAgZ2yN28ipRnGPnwkI0= | ||||||
| github.com/gliderlabs/ssh v0.1.1/go.mod h1:U7qILu1NlMHj9FlMhZLlkCdDnU1DBEAqr0aevW3Awn0= | github.com/gliderlabs/ssh v0.2.2/go.mod h1:U7qILu1NlMHj9FlMhZLlkCdDnU1DBEAqr0aevW3Awn0= | ||||||
| github.com/google/go-cmp v0.2.0 h1:+dTQ8DZQJz0Mb/HjFlkptS1FeQ4cWSnN941F8aEG4SQ= | github.com/google/go-cmp v0.3.0 h1:crn/baboCvb5fXaQ0IJ1SGTsTVrWpDsCWC8EGETZijY= | ||||||
| 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/jbenet/go-context v0.0.0-20150711004518-d14ea06fba99 h1:BQSFePA1RWJOlocH6Fxy8MmwDt+yVQYULKfN0RoTN8A= | github.com/jbenet/go-context v0.0.0-20150711004518-d14ea06fba99 h1:BQSFePA1RWJOlocH6Fxy8MmwDt+yVQYULKfN0RoTN8A= | ||||||
| github.com/jbenet/go-context v0.0.0-20150711004518-d14ea06fba99/go.mod h1:1lJo3i6rXxKeerYnT8Nvf0QmHCRC1n8sfWVwXF2Frvo= | github.com/jbenet/go-context v0.0.0-20150711004518-d14ea06fba99/go.mod h1:1lJo3i6rXxKeerYnT8Nvf0QmHCRC1n8sfWVwXF2Frvo= | ||||||
| github.com/jessevdk/go-flags v1.4.0/go.mod h1:4FA24M0QyGHXBuZZK/XkWh8h0e1EYbRYJSGM75WSRxI= | github.com/jessevdk/go-flags v1.4.0/go.mod h1:4FA24M0QyGHXBuZZK/XkWh8h0e1EYbRYJSGM75WSRxI= | ||||||
| github.com/kevinburke/ssh_config v0.0.0-20180830205328-81db2a75821e h1:RgQk53JHp/Cjunrr1WlsXSZpqXn+uREuHvUVcK82CV8= | github.com/kevinburke/ssh_config v0.0.0-20190725054713-01f96b0aa0cd h1:Coekwdh0v2wtGp9Gmz1Ze3eVRAWJMLokvN3QjdzCHLY= | ||||||
| github.com/kevinburke/ssh_config v0.0.0-20180830205328-81db2a75821e/go.mod h1:CT57kijsi8u/K/BOFA39wgDQJ9CxiF4nAY/ojJ6r6mM= | github.com/kevinburke/ssh_config v0.0.0-20190725054713-01f96b0aa0cd/go.mod h1:CT57kijsi8u/K/BOFA39wgDQJ9CxiF4nAY/ojJ6r6mM= | ||||||
| github.com/kr/pretty v0.1.0 h1:L/CwN0zerZDmRFUapSPitk6f+Q3+0za1rQkzVuMiMFI= | github.com/kr/pretty v0.1.0 h1:L/CwN0zerZDmRFUapSPitk6f+Q3+0za1rQkzVuMiMFI= | ||||||
| github.com/kr/pretty v0.1.0/go.mod h1:dAy3ld7l9f0ibDNOQOHHMYYIIbhfbHSm3C4ZsoJORNo= | github.com/kr/pretty v0.1.0/go.mod h1:dAy3ld7l9f0ibDNOQOHHMYYIIbhfbHSm3C4ZsoJORNo= | ||||||
| github.com/kr/pty v1.1.1/go.mod h1:pFQYn66WHrOpPYNljwOMqo10TkYh1fy3cYio2l3bCsQ= | github.com/kr/pty v1.1.1/go.mod h1:pFQYn66WHrOpPYNljwOMqo10TkYh1fy3cYio2l3bCsQ= | ||||||
|  | github.com/kr/pty v1.1.8/go.mod h1:O1sed60cT9XZ5uDucP5qwvh+TE3NnUj51EiZO/lmSfw= | ||||||
| github.com/kr/text v0.1.0 h1:45sCR5RtlFHMR4UwH9sdQ5TC8v0qDQCHnXt+kaKSTVE= | github.com/kr/text v0.1.0 h1:45sCR5RtlFHMR4UwH9sdQ5TC8v0qDQCHnXt+kaKSTVE= | ||||||
| github.com/kr/text v0.1.0/go.mod h1:4Jbv+DJW3UT/LiOwJeYQe1efqtUx/iVham/4vfdArNI= | github.com/kr/text v0.1.0/go.mod h1:4Jbv+DJW3UT/LiOwJeYQe1efqtUx/iVham/4vfdArNI= | ||||||
| github.com/mitchellh/go-homedir v1.0.0 h1:vKb8ShqSby24Yrqr/yDYkuFz8d0WUjys40rvnGC8aR0= | github.com/mitchellh/go-homedir v1.1.0 h1:lukF9ziXFxDFPkA1vsr5zpc1XuPDn/wFntq5mG+4E0Y= | ||||||
| github.com/mitchellh/go-homedir v1.0.0/go.mod h1:SfyaCUpYCn1Vlf4IUYiD9fPX4A5wJrkLzIz1N1q0pr0= | github.com/mitchellh/go-homedir v1.1.0/go.mod h1:SfyaCUpYCn1Vlf4IUYiD9fPX4A5wJrkLzIz1N1q0pr0= | ||||||
| github.com/pelletier/go-buffruneio v0.2.0 h1:U4t4R6YkofJ5xHm3dJzuRpPZ0mr5MMCoAWooScCR7aA= |  | ||||||
| github.com/pelletier/go-buffruneio v0.2.0/go.mod h1:JkE26KsDizTr40EUHkXVtNPvgGtbSNq5BcowyYOWdKo= | github.com/pelletier/go-buffruneio v0.2.0/go.mod h1:JkE26KsDizTr40EUHkXVtNPvgGtbSNq5BcowyYOWdKo= | ||||||
| github.com/pkg/errors v0.8.0/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0= |  | ||||||
| github.com/pkg/errors v0.8.1 h1:iURUrRGxPUNPdy5/HRSm+Yj6okJ6UtLINN0Q9M4+h3I= | github.com/pkg/errors v0.8.1 h1:iURUrRGxPUNPdy5/HRSm+Yj6okJ6UtLINN0Q9M4+h3I= | ||||||
| github.com/pkg/errors v0.8.1/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0= | github.com/pkg/errors v0.8.1/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0= | ||||||
| github.com/pmezard/go-difflib v1.0.0 h1:4DBwDE0NGyQoBHbLQYPwSUPoCMWR5BEzIk/f1lZbAQM= | github.com/pmezard/go-difflib v1.0.0 h1:4DBwDE0NGyQoBHbLQYPwSUPoCMWR5BEzIk/f1lZbAQM= | ||||||
| @@ -38,31 +38,37 @@ github.com/sergi/go-diff v1.0.0/go.mod h1:0CfEIISq7TuYL3j771MWULgwwjU+GofnZX9QAm | |||||||
| github.com/src-d/gcfg v1.4.0 h1:xXbNR5AlLSA315x2UO+fTSSAXCDf+Ar38/6oyGbDKQ4= | github.com/src-d/gcfg v1.4.0 h1:xXbNR5AlLSA315x2UO+fTSSAXCDf+Ar38/6oyGbDKQ4= | ||||||
| github.com/src-d/gcfg v1.4.0/go.mod h1:p/UMsR43ujA89BJY9duynAwIpvqEujIH/jFlfL7jWoI= | github.com/src-d/gcfg v1.4.0/go.mod h1:p/UMsR43ujA89BJY9duynAwIpvqEujIH/jFlfL7jWoI= | ||||||
| github.com/stretchr/objx v0.1.0/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME= | github.com/stretchr/objx v0.1.0/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME= | ||||||
| github.com/stretchr/testify v1.2.2/go.mod h1:a8OnRcib4nhh0OaRAV+Yts87kKdq0PP7pXfy6kDkUVs= | github.com/stretchr/objx v0.2.0/go.mod h1:qt09Ya8vawLte6SNmTgCsAVtYtaKzEcn8ATUoHMkEqE= | ||||||
| github.com/stretchr/testify v1.3.0 h1:TivCn/peBQ7UY8ooIcPgZFpTNSz0Q2U6UrFlUfqbe0Q= | github.com/stretchr/testify v1.3.0 h1:TivCn/peBQ7UY8ooIcPgZFpTNSz0Q2U6UrFlUfqbe0Q= | ||||||
| github.com/stretchr/testify v1.3.0/go.mod h1:M5WIy9Dh21IEIfnGCwXGc5bZfKNJtfHm1UVUgZn+9EI= | github.com/stretchr/testify v1.3.0/go.mod h1:M5WIy9Dh21IEIfnGCwXGc5bZfKNJtfHm1UVUgZn+9EI= | ||||||
| github.com/xanzy/ssh-agent v0.2.0 h1:Adglfbi5p9Z0BmK2oKU9nTG+zKfniSfnaMYB+ULd+Ro= | github.com/xanzy/ssh-agent v0.2.1 h1:TCbipTQL2JiiCprBWx9frJ2eJlCYT00NmctrHxVAr70= | ||||||
| github.com/xanzy/ssh-agent v0.2.0/go.mod h1:0NyE30eGUDliuLEHJgYte/zncp2zdTStcOnWhgSqHD8= | github.com/xanzy/ssh-agent v0.2.1/go.mod h1:mLlQY/MoOhWBj+gOGMQkOeiEvkx+8pJSI+0Bx9h2kr4= | ||||||
| golang.org/x/crypto v0.0.0-20180904163835-0709b304e793/go.mod h1:6SG95UA2DQfeDnfUPMdvaQW0Q7yPrPDi9nlGo2tz2b4= | golang.org/x/crypto v0.0.0-20190219172222-a4c6cb3142f2/go.mod h1:6SG95UA2DQfeDnfUPMdvaQW0Q7yPrPDi9nlGo2tz2b4= | ||||||
| golang.org/x/crypto v0.0.0-20190103213133-ff983b9c42bc h1:F5tKCVGp+MUAHhKp5MZtGqAlGX3+oCsiL1Q629FL90M= | golang.org/x/crypto v0.0.0-20190308221718-c2843e01d9a2/go.mod h1:djNgcEr1/C05ACkg1iLfiJU5Ep61QUkGW8qpdssI0+w= | ||||||
| golang.org/x/crypto v0.0.0-20190103213133-ff983b9c42bc/go.mod h1:6SG95UA2DQfeDnfUPMdvaQW0Q7yPrPDi9nlGo2tz2b4= | golang.org/x/crypto v0.0.0-20190701094942-4def268fd1a4 h1:HuIa8hRrWRSrqYzx1qI49NNxhdi2PrY7gxVSq1JjLDc= | ||||||
| golang.org/x/net v0.0.0-20180906233101-161cd47e91fd/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= | golang.org/x/crypto v0.0.0-20190701094942-4def268fd1a4/go.mod h1:yigFU9vqHzYiE8UmvKecakEJjdnWj3jj499lnFckfCI= | ||||||
| golang.org/x/net v0.0.0-20190110200230-915654e7eabc h1:Yx9JGxI1SBhVLFjpAkWMaO1TF+xyqtHLjZpvQboJGiM= | golang.org/x/net v0.0.0-20190404232315-eb5bcb51f2a3/go.mod h1:t9HGtf8HONx5eT2rtn7q6eTqICYqUVnKs3thJo3Qplg= | ||||||
| golang.org/x/net v0.0.0-20190110200230-915654e7eabc/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= | golang.org/x/net v0.0.0-20190620200207-3b0461eec859/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s= | ||||||
| golang.org/x/sys v0.0.0-20180903190138-2b024373dcd9/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= | golang.org/x/net v0.0.0-20190724013045-ca1201d0de80 h1:Ao/3l156eZf2AW5wK8a7/smtodRU+gha3+BeqJ69lRk= | ||||||
| golang.org/x/sys v0.0.0-20190109145017-48ac38b7c8cb h1:1w588/yEchbPNpa9sEvOcMZYbWHedwJjg4VOAdDHWHk= | golang.org/x/net v0.0.0-20190724013045-ca1201d0de80/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s= | ||||||
| golang.org/x/sys v0.0.0-20190109145017-48ac38b7c8cb/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= | golang.org/x/sync v0.0.0-20190423024810-112230192c58/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= | ||||||
| golang.org/x/text v0.3.0 h1:g61tztE5qeGQ89tm6NTjjM9VPIm088od1l6aSorWRWg= | golang.org/x/sys v0.0.0-20190215142949-d0b11bdaac8a/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= | ||||||
|  | golang.org/x/sys v0.0.0-20190221075227-b4e8571b14e0/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= | ||||||
|  | golang.org/x/sys v0.0.0-20190412213103-97732733099d/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||||
|  | golang.org/x/sys v0.0.0-20190726091711-fc99dfbffb4e h1:D5TXcfTk7xF7hvieo4QErS3qqCB4teTffacDWr7CI+0= | ||||||
|  | golang.org/x/sys v0.0.0-20190726091711-fc99dfbffb4e/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||||
| golang.org/x/text v0.3.0/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.2 h1:tW2bmiBqwgJj/UpqtC8EpXEZVYOwU0yG4iWbprSVAcs= | ||||||
|  | golang.org/x/text v0.3.2/go.mod h1:bEr9sfX3Q8Zfm5fL9x+3itogRgK3+ptLWKqgva+5dAk= | ||||||
|  | golang.org/x/tools v0.0.0-20180917221912-90fa682c2a6e/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ= | ||||||
|  | golang.org/x/tools v0.0.0-20190729092621-ff9f1409240a/go.mod h1:jcCCGcm9btYwXyDqrUWc6MKQKKGJCWEQ3AfLSRIbEuI= | ||||||
| gopkg.in/check.v1 v1.0.0-20180628173108-788fd7840127 h1:qIbj1fsPNlZgppZ+VLlY7N33q108Sa+fhmuc+sWQYwY= | gopkg.in/check.v1 v1.0.0-20180628173108-788fd7840127 h1:qIbj1fsPNlZgppZ+VLlY7N33q108Sa+fhmuc+sWQYwY= | ||||||
| gopkg.in/check.v1 v1.0.0-20180628173108-788fd7840127/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= | gopkg.in/check.v1 v1.0.0-20180628173108-788fd7840127/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= | ||||||
| gopkg.in/src-d/go-billy.v4 v4.2.1/go.mod h1:tm33zBoOwxjYHZIE+OV8bxTWFMJLrconzFMd38aARFk= | gopkg.in/src-d/go-billy.v4 v4.3.2 h1:0SQA1pRztfTFx2miS8sA97XvooFeNOmvUenF4o0EcVg= | ||||||
| gopkg.in/src-d/go-billy.v4 v4.3.0 h1:KtlZ4c1OWbIs4jCv5ZXrTqG8EQocr0g/d4DjNg70aek= | gopkg.in/src-d/go-billy.v4 v4.3.2/go.mod h1:nDjArDMp+XMs1aFAESLRjfGSgfvoYN0hDfzEk0GjC98= | ||||||
| gopkg.in/src-d/go-billy.v4 v4.3.0/go.mod h1:tm33zBoOwxjYHZIE+OV8bxTWFMJLrconzFMd38aARFk= | gopkg.in/src-d/go-git-fixtures.v3 v3.5.0 h1:ivZFOIltbce2Mo8IjzUHAFoq/IylO9WHhNOAJK+LsJg= | ||||||
| gopkg.in/src-d/go-git-fixtures.v3 v3.1.1/go.mod h1:dLBcvytrw/TYZsNTWCnkNF2DSIlzWYqTe3rJR56Ac7g= | gopkg.in/src-d/go-git-fixtures.v3 v3.5.0/go.mod h1:dLBcvytrw/TYZsNTWCnkNF2DSIlzWYqTe3rJR56Ac7g= | ||||||
| gopkg.in/src-d/go-git-fixtures.v3 v3.3.0 h1:AxUOwLW3at53ysFqs0Lg+H+8KSQXl7AEHBvWj8wEsT8= | gopkg.in/src-d/go-git.v4 v4.13.1 h1:SRtFyV8Kxc0UP7aCHcijOMQGPxHSmMOPrzulQWolkYE= | ||||||
| gopkg.in/src-d/go-git-fixtures.v3 v3.3.0/go.mod h1:dLBcvytrw/TYZsNTWCnkNF2DSIlzWYqTe3rJR56Ac7g= | gopkg.in/src-d/go-git.v4 v4.13.1/go.mod h1:nx5NYcxdKxq5fpltdHnPa2Exj4Sx0EclMWZQbYDu2z8= | ||||||
| gopkg.in/src-d/go-git.v4 v4.8.1 h1:aAyBmkdE1QUUEHcP4YFCGKmsMQRAuRmUcPEQR7lOAa0= |  | ||||||
| gopkg.in/src-d/go-git.v4 v4.8.1/go.mod h1:Vtut8izDyrM8BUVQnzJ+YvmNcem2J89EmfZYCkLokZk= |  | ||||||
| gopkg.in/warnings.v0 v0.1.2 h1:wFXVbFY8DY5/xOe1ECiWdKCzZlxgshcYVNkBHstARME= | gopkg.in/warnings.v0 v0.1.2 h1:wFXVbFY8DY5/xOe1ECiWdKCzZlxgshcYVNkBHstARME= | ||||||
| gopkg.in/warnings.v0 v0.1.2/go.mod h1:jksf8JmL6Qr/oQM2OXTHunEvvTAsrWBLb6OOjuVWRNI= | gopkg.in/warnings.v0 v0.1.2/go.mod h1:jksf8JmL6Qr/oQM2OXTHunEvvTAsrWBLb6OOjuVWRNI= | ||||||
|   | |||||||
| @@ -15,8 +15,8 @@ import ( | |||||||
| 	"time" | 	"time" | ||||||
| 	"unicode" | 	"unicode" | ||||||
|  |  | ||||||
| 	"github.com/sixt/gomodproxy/pkg/store" | 	"github.com/bilus/gomodproxy/pkg/store" | ||||||
| 	"github.com/sixt/gomodproxy/pkg/vcs" | 	"github.com/bilus/gomodproxy/pkg/vcs" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| type logger = func(v ...interface{}) | type logger = func(v ...interface{}) | ||||||
| @@ -27,6 +27,8 @@ type api struct { | |||||||
| 	vcsPaths []vcsPath | 	vcsPaths []vcsPath | ||||||
| 	stores   []store.Store | 	stores   []store.Store | ||||||
| 	semc     chan struct{} | 	semc     chan struct{} | ||||||
|  |  | ||||||
|  | 	ephemeralTagStorage *vcs.EphemeralTagStorage | ||||||
| } | } | ||||||
|  |  | ||||||
| type vcsPath struct { | type vcsPath struct { | ||||||
| @@ -42,6 +44,7 @@ var ( | |||||||
| 	apiInfo = regexp.MustCompile(`^/(?P<module>.*)/@v/(?P<version>.*).info$`) | 	apiInfo = regexp.MustCompile(`^/(?P<module>.*)/@v/(?P<version>.*).info$`) | ||||||
| 	apiMod  = regexp.MustCompile(`^/(?P<module>.*)/@v/(?P<version>.*).mod$`) | 	apiMod  = regexp.MustCompile(`^/(?P<module>.*)/@v/(?P<version>.*).mod$`) | ||||||
| 	apiZip  = regexp.MustCompile(`^/(?P<module>.*)/@v/(?P<version>.*).zip$`) | 	apiZip  = regexp.MustCompile(`^/(?P<module>.*)/@v/(?P<version>.*).zip$`) | ||||||
|  | 	apiTag  = regexp.MustCompile(`^/tags/(?P<module>.*)/@v/(?P<version>.*)$`) | ||||||
| ) | ) | ||||||
|  |  | ||||||
| var ( | var ( | ||||||
| @@ -71,9 +74,9 @@ func GitDir(dir string) Option { return func(api *api) { api.gitdir = dir } } | |||||||
| // Git configures API to use a specific git client when trying to download a | // Git configures API to use a specific git client when trying to download a | ||||||
| // repository with the given prefix. Auth string can be a path to the SSK key, | // repository with the given prefix. Auth string can be a path to the SSK key, | ||||||
| // or a colon-separated username:password string. | // or a colon-separated username:password string. | ||||||
| func Git(prefix string, auth string) Option { | func Git(prefix, key, password string) Option { | ||||||
| 	a := vcs.Key(auth) | 	a := vcs.Key(key, password) | ||||||
| 	if creds := strings.SplitN(auth, ":", 2); len(creds) == 2 { | 	if creds := strings.SplitN(key, ":", 2); len(creds) == 2 { | ||||||
| 		a = vcs.Password(creds[0], creds[1]) | 		a = vcs.Password(creds[0], creds[1]) | ||||||
| 	} | 	} | ||||||
| 	return func(api *api) { | 	return func(api *api) { | ||||||
| @@ -86,6 +89,29 @@ func Git(prefix string, auth string) Option { | |||||||
| 	} | 	} | ||||||
| } | } | ||||||
|  |  | ||||||
|  | // GitWithEphemeralTags configures API to use a specific git client when trying | ||||||
|  | // to download a repository with the given prefix. Auth string can be a path to | ||||||
|  | // the SSK key, or a colon-separated username:password string. | ||||||
|  | func GitWithEphemeralTags(prefix, key, password string) Option { | ||||||
|  | 	// TODO(bilus): Ugly but we don't want to mess with the : encoding so | ||||||
|  | 	// we'll work around the issue of having to pass a passphrase | ||||||
|  | 	// to decrypt a key. | ||||||
|  | 	storage := vcs.NewEphemeralTagStorage() | ||||||
|  | 	a := vcs.Key(key, password) | ||||||
|  | 	if creds := strings.SplitN(key, ":", 2); len(creds) == 2 { | ||||||
|  | 		a = vcs.Password(creds[0], creds[1]) | ||||||
|  | 	} | ||||||
|  | 	return func(api *api) { | ||||||
|  | 		api.ephemeralTagStorage = storage | ||||||
|  | 		api.vcsPaths = append(api.vcsPaths, vcsPath{ | ||||||
|  | 			prefix: prefix, | ||||||
|  | 			vcs: func(module string) vcs.VCS { | ||||||
|  | 				return vcs.NewGitWithEphemeralTags(api.log, api.gitdir, module, a, storage) | ||||||
|  | 			}, | ||||||
|  | 		}) | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
| func CustomVCS(prefix string, cmd string) Option { | func CustomVCS(prefix string, cmd string) Option { | ||||||
| 	return func(api *api) { | 	return func(api *api) { | ||||||
| 		api.vcsPaths = append(api.vcsPaths, vcsPath{ | 		api.vcsPaths = append(api.vcsPaths, vcsPath{ | ||||||
| @@ -151,6 +177,7 @@ func (api *api) ServeHTTP(w http.ResponseWriter, r *http.Request) { | |||||||
| 		{"info", apiInfo, api.info}, | 		{"info", apiInfo, api.info}, | ||||||
| 		{"api", apiMod, api.mod}, | 		{"api", apiMod, api.mod}, | ||||||
| 		{"zip", apiZip, api.zip}, | 		{"zip", apiZip, api.zip}, | ||||||
|  | 		{"tag", apiTag, api.tag}, | ||||||
| 	} { | 	} { | ||||||
| 		if m := route.regexp.FindStringSubmatch(r.URL.Path); m != nil { | 		if m := route.regexp.FindStringSubmatch(r.URL.Path); m != nil { | ||||||
| 			module, version := m[1], "" | 			module, version := m[1], "" | ||||||
| @@ -158,6 +185,10 @@ func (api *api) ServeHTTP(w http.ResponseWriter, r *http.Request) { | |||||||
| 				version = m[2] | 				version = m[2] | ||||||
| 			} | 			} | ||||||
| 			module = decodeBangs(module) | 			module = decodeBangs(module) | ||||||
|  | 			if r.Method == http.MethodDelete && version != "" { | ||||||
|  | 				api.delete(w, r, module, version) | ||||||
|  | 				return | ||||||
|  | 			} | ||||||
| 			httpRequests.Add(route.id, 1) | 			httpRequests.Add(route.id, 1) | ||||||
| 			defer func() { | 			defer func() { | ||||||
| 				v := &expvar.Float{} | 				v := &expvar.Float{} | ||||||
| @@ -179,7 +210,7 @@ func (api *api) vcs(ctx context.Context, module string) vcs.VCS { | |||||||
| 			return path.vcs(module) | 			return path.vcs(module) | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
| 	return vcs.NewGit(api.log, api.gitdir, module, vcs.NoAuth()) | 	return vcs.NewGoMod(api.log, module) | ||||||
| } | } | ||||||
|  |  | ||||||
| func (api *api) module(ctx context.Context, module string, version vcs.Version) ([]byte, time.Time, error) { | func (api *api) module(ctx context.Context, module string, version vcs.Version) ([]byte, time.Time, error) { | ||||||
| @@ -232,7 +263,7 @@ func (api *api) list(w http.ResponseWriter, r *http.Request, module, version str | |||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		api.log("api.list", "module", module, "error", err) | 		api.log("api.list", "module", module, "error", err) | ||||||
| 		httpErrors.Add(module, 1) | 		httpErrors.Add(module, 1) | ||||||
| 		http.Error(w, err.Error(), http.StatusInternalServerError) | 		http.Error(w, err.Error(), http.StatusNotFound) | ||||||
| 		return | 		return | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| @@ -248,7 +279,7 @@ func (api *api) info(w http.ResponseWriter, r *http.Request, module, version str | |||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		api.log("api.info", "module", module, "version", version, "error", err) | 		api.log("api.info", "module", module, "version", version, "error", err) | ||||||
| 		httpErrors.Add(module, 1) | 		httpErrors.Add(module, 1) | ||||||
| 		http.Error(w, err.Error(), http.StatusInternalServerError) | 		http.Error(w, err.Error(), http.StatusNotFound) | ||||||
| 		return | 		return | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| @@ -283,8 +314,49 @@ func (api *api) zip(w http.ResponseWriter, r *http.Request, module, version stri | |||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		api.log("api.zip", "module", module, "version", version, "error", err) | 		api.log("api.zip", "module", module, "version", version, "error", err) | ||||||
| 		httpErrors.Add(module, 1) | 		httpErrors.Add(module, 1) | ||||||
| 		http.Error(w, err.Error(), http.StatusInternalServerError) | 		http.Error(w, err.Error(), http.StatusNotFound) | ||||||
| 		return | 		return | ||||||
| 	} | 	} | ||||||
| 	io.Copy(w, bytes.NewReader(b)) | 	io.Copy(w, bytes.NewReader(b)) | ||||||
| } | } | ||||||
|  |  | ||||||
|  | func (api *api) delete(w http.ResponseWriter, r *http.Request, module, version string) { | ||||||
|  | 	for _, store := range api.stores { | ||||||
|  | 		if err := store.Del(r.Context(), module, vcs.Version(version)); err != nil { | ||||||
|  | 			http.Error(w, err.Error(), http.StatusNotFound) | ||||||
|  | 			return | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (api *api) tag(w http.ResponseWriter, r *http.Request, module, version string) { | ||||||
|  | 	api.log("api.tag", "module", module, "version", version) | ||||||
|  |  | ||||||
|  | 	taggable, ok := api.vcs(r.Context(), module).(vcs.Taggable) | ||||||
|  | 	if !ok { | ||||||
|  | 		err := fmt.Errorf("repository for module %v is not taggable", module) | ||||||
|  | 		api.log("api.tag", "module", module, "version", version, "error", err) | ||||||
|  | 		httpErrors.Add(module, 1) | ||||||
|  | 		http.Error(w, err.Error(), http.StatusNotFound) | ||||||
|  | 		return | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	defer r.Body.Close() | ||||||
|  |  | ||||||
|  | 	req := struct { | ||||||
|  | 		Short string `json:"short"` | ||||||
|  | 	}{} | ||||||
|  | 	err := json.NewDecoder(r.Body).Decode(&req) | ||||||
|  | 	if err != nil { | ||||||
|  | 		api.log("api.tag", "module", module, "version", version, "error", err) | ||||||
|  | 		http.Error(w, err.Error(), http.StatusBadRequest) | ||||||
|  | 		return | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	err = taggable.Tag(r.Context(), vcs.Version(version), req.Short) | ||||||
|  | 	if err != nil { | ||||||
|  | 		api.log("api.tag", "module", module, "version", version, "error", err) | ||||||
|  | 		http.Error(w, err.Error(), http.StatusBadRequest) | ||||||
|  | 		return | ||||||
|  | 	} | ||||||
|  | } | ||||||
|   | |||||||
| @@ -6,7 +6,7 @@ import ( | |||||||
| 	"os" | 	"os" | ||||||
| 	"path/filepath" | 	"path/filepath" | ||||||
|  |  | ||||||
| 	"github.com/sixt/gomodproxy/pkg/vcs" | 	"github.com/bilus/gomodproxy/pkg/vcs" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| type disk string | type disk string | ||||||
| @@ -47,4 +47,15 @@ func (d disk) Get(ctx context.Context, module string, version vcs.Version) (Snap | |||||||
| 	return s, err | 	return s, err | ||||||
| } | } | ||||||
|  |  | ||||||
|  | func (d disk) Del(ctx context.Context, module string, version vcs.Version) error { | ||||||
|  | 	dir := string(d) | ||||||
|  | 	s := Snapshot{Module: module, Version: version} | ||||||
|  | 	err := os.Remove(filepath.Join(dir, s.Key()+".time")) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  | 	err = os.Remove(filepath.Join(dir, s.Key()+".zip")) | ||||||
|  | 	return err | ||||||
|  | } | ||||||
|  |  | ||||||
| func (d disk) Close() error { return nil } | func (d disk) Close() error { return nil } | ||||||
|   | |||||||
| @@ -5,7 +5,7 @@ import ( | |||||||
| 	"errors" | 	"errors" | ||||||
| 	"sync" | 	"sync" | ||||||
|  |  | ||||||
| 	"github.com/sixt/gomodproxy/pkg/vcs" | 	"github.com/bilus/gomodproxy/pkg/vcs" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| type memory struct { | type memory struct { | ||||||
| @@ -45,17 +45,41 @@ func (m *memory) Put(ctx context.Context, snapshot Snapshot) error { | |||||||
| func (m *memory) Get(ctx context.Context, module string, version vcs.Version) (Snapshot, error) { | func (m *memory) Get(ctx context.Context, module string, version vcs.Version) (Snapshot, error) { | ||||||
| 	m.Lock() | 	m.Lock() | ||||||
| 	defer m.Unlock() | 	defer m.Unlock() | ||||||
| 	return m.lookup(module, version) | 	item, err := m.lookup(module, version) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return Snapshot{}, err | ||||||
|  | 	} | ||||||
|  | 	return item.Snapshot, nil | ||||||
| } | } | ||||||
|  |  | ||||||
| func (m *memory) lookup(module string, version vcs.Version) (Snapshot, error) { | func (m *memory) Del(ctx context.Context, module string, version vcs.Version) error { | ||||||
|  | 	m.Lock() | ||||||
|  | 	defer m.Unlock() | ||||||
|  | 	item, err := m.lookup(module, version) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  | 	if item.prev == nil { | ||||||
|  | 		m.head = item.next | ||||||
|  | 	} else { | ||||||
|  | 		item.prev.next = item.next | ||||||
|  | 	} | ||||||
|  | 	if item.next == nil { | ||||||
|  | 		m.tail = item.prev | ||||||
|  | 	} else { | ||||||
|  | 		item.next.prev = item.prev | ||||||
|  | 	} | ||||||
|  | 	return nil | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (m *memory) lookup(module string, version vcs.Version) (*lruItem, error) { | ||||||
| 	for item := m.head; item != nil; item = item.next { | 	for item := m.head; item != nil; item = item.next { | ||||||
| 		if item.Module == module && item.Version == version { | 		if item.Module == module && item.Version == version { | ||||||
| 			m.update(item) | 			m.update(item) | ||||||
| 			return item.Snapshot, nil | 			return item, nil | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
| 	return Snapshot{}, errors.New("not found") | 	return nil, errors.New("not found") | ||||||
| } | } | ||||||
|  |  | ||||||
| func (m *memory) insert(item *lruItem) { | func (m *memory) insert(item *lruItem) { | ||||||
|   | |||||||
| @@ -4,7 +4,7 @@ import ( | |||||||
| 	"context" | 	"context" | ||||||
| 	"time" | 	"time" | ||||||
|  |  | ||||||
| 	"github.com/sixt/gomodproxy/pkg/vcs" | 	"github.com/bilus/gomodproxy/pkg/vcs" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| type logger = func(...interface{}) | type logger = func(...interface{}) | ||||||
| @@ -14,6 +14,7 @@ type logger = func(...interface{}) | |||||||
| type Store interface { | type Store interface { | ||||||
| 	Put(ctx context.Context, snapshot Snapshot) error | 	Put(ctx context.Context, snapshot Snapshot) error | ||||||
| 	Get(ctx context.Context, module string, version vcs.Version) (Snapshot, error) | 	Get(ctx context.Context, module string, version vcs.Version) (Snapshot, error) | ||||||
|  | 	Del(ctx context.Context, module string, version vcs.Version) error | ||||||
| 	Close() error | 	Close() error | ||||||
| } | } | ||||||
|  |  | ||||||
|   | |||||||
| @@ -34,6 +34,8 @@ type gitVCS struct { | |||||||
| 	auth   Auth | 	auth   Auth | ||||||
| } | } | ||||||
|  |  | ||||||
|  | var ErrNoMatchingVersion = errors.New("no matching versions") | ||||||
|  |  | ||||||
| // NewGit return a go-git VCS client implementation that provides information | // NewGit return a go-git VCS client implementation that provides information | ||||||
| // about the specific module using the pgiven authentication mechanism. | // about the specific module using the pgiven authentication mechanism. | ||||||
| func NewGit(l logger, dir string, module string, auth Auth) VCS { | func NewGit(l logger, dir string, module string, auth Auth) VCS { | ||||||
| @@ -61,7 +63,6 @@ func (g *gitVCS) List(ctx context.Context) ([]Version, error) { | |||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	list := []Version{} | 	list := []Version{} | ||||||
| 	masterHash := "" | 	masterHash := "" | ||||||
| 	tagPrefix := "" | 	tagPrefix := "" | ||||||
| @@ -81,18 +82,55 @@ func (g *gitVCS) List(ctx context.Context) ([]Version, error) { | |||||||
| 		if masterHash == "" { | 		if masterHash == "" { | ||||||
| 			return nil, errors.New("no tags and no master branch found") | 			return nil, errors.New("no tags and no master branch found") | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		short := masterHash[:12] | 		short := masterHash[:12] | ||||||
| 		t, err := g.Timestamp(ctx, Version("v0.0.0-20060102150405-"+short)) | 		version, err := g.versionFromHash(ctx, short) | ||||||
| 		if err != nil { | 		if err != nil { | ||||||
| 			return nil, err | 			return nil, err | ||||||
| 		} | 		} | ||||||
| 		list = []Version{Version(fmt.Sprintf("v0.0.0-%s-%s", t.Format("20060102150405"), short))} |  | ||||||
|  | 		masterCommit, err := g.commit(ctx, version) | ||||||
|  | 		if err != nil { | ||||||
|  | 			return nil, err | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		tree, err := masterCommit.Tree() | ||||||
|  | 		if err != nil { | ||||||
|  | 			return nil, err | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		// No tags while it's a module. | ||||||
|  | 		if g.isModule(tree) { | ||||||
|  | 			return nil, ErrNoMatchingVersion | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		list = []Version{version} | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	g.log("gitVCS.List", "module", g.module, "list", list) | 	g.log("gitVCS.List", "module", g.module, "list", list) | ||||||
| 	return list, nil | 	return list, nil | ||||||
| } | } | ||||||
|  |  | ||||||
|  | func (g *gitVCS) versionFromHash(ctx context.Context, hash string) (Version, error) { | ||||||
|  | 	t, err := g.Timestamp(ctx, Version("v0.0.0-20060102150405-"+hash)) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return Version(""), err | ||||||
|  | 	} | ||||||
|  | 	v := Version(fmt.Sprintf("v0.0.0-%s-%s", t.Format("20060102150405"), hash)) | ||||||
|  | 	return v, nil | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (g *gitVCS) isModule(tree *object.Tree) bool { | ||||||
|  | 	mod := "go.mod" | ||||||
|  | 	for path := g.prefix; path != "."; path = filepath.Dir(path) { | ||||||
|  | 		_, err := tree.FindEntry(filepath.Join(path, mod)) | ||||||
|  | 		if err == nil { | ||||||
|  | 			return true | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 	return false | ||||||
|  | } | ||||||
|  |  | ||||||
| func (g *gitVCS) Timestamp(ctx context.Context, version Version) (time.Time, error) { | func (g *gitVCS) Timestamp(ctx context.Context, version Version) (time.Time, error) { | ||||||
| 	g.log("gitVCS.Timestamp", "module", g.module, "version", version) | 	g.log("gitVCS.Timestamp", "module", g.module, "version", version) | ||||||
| 	ci, err := g.commit(ctx, version) | 	ci, err := g.commit(ctx, version) | ||||||
| @@ -116,6 +154,11 @@ func isVendoredPackage(name string) bool { | |||||||
| } | } | ||||||
|  |  | ||||||
| func (g *gitVCS) Zip(ctx context.Context, version Version) (io.ReadCloser, error) { | func (g *gitVCS) Zip(ctx context.Context, version Version) (io.ReadCloser, error) { | ||||||
|  | 	dirName := g.module + "@" + string(version) | ||||||
|  | 	return g.zipAs(ctx, version, dirName) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (g *gitVCS) zipAs(ctx context.Context, version Version, dirName string) (io.ReadCloser, error) { | ||||||
| 	g.log("gitVCS.Zip", "module", g.module, "version", version) | 	g.log("gitVCS.Zip", "module", g.module, "version", version) | ||||||
| 	ci, err := g.commit(ctx, version) | 	ci, err := g.commit(ctx, version) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| @@ -176,7 +219,7 @@ func (g *gitVCS) Zip(ctx context.Context, version Version) (io.ReadCloser, error | |||||||
| 		} else { | 		} else { | ||||||
| 			continue | 			continue | ||||||
| 		} | 		} | ||||||
| 		w, err := zw.Create(filepath.Join(g.module+"@"+string(version), name)) | 		w, err := zw.Create(filepath.Join(dirName, name)) | ||||||
| 		if err != nil { | 		if err != nil { | ||||||
| 			return nil, err | 			return nil, err | ||||||
| 		} | 		} | ||||||
| @@ -235,10 +278,15 @@ func (g *gitVCS) commit(ctx context.Context, version Version) (*object.Commit, e | |||||||
| 	err = repo.FetchContext(ctx, &git.FetchOptions{ | 	err = repo.FetchContext(ctx, &git.FetchOptions{ | ||||||
| 		RemoteName: remoteName, | 		RemoteName: remoteName, | ||||||
| 		Auth:       auth, | 		Auth:       auth, | ||||||
|  | 		Tags:       git.AllTags, | ||||||
| 	}) | 	}) | ||||||
| 	if err != nil && err != git.NoErrAlreadyUpToDate { | 	if err != nil && err != git.NoErrAlreadyUpToDate { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
|  | 	tagPrefix := "" | ||||||
|  | 	if g.prefix != "" { | ||||||
|  | 		tagPrefix = g.prefix + "/" | ||||||
|  | 	} | ||||||
|  |  | ||||||
| 	version = Version(strings.TrimSuffix(string(version), "+incompatible")) | 	version = Version(strings.TrimSuffix(string(version), "+incompatible")) | ||||||
| 	hash := version.Hash() | 	hash := version.Hash() | ||||||
| @@ -248,7 +296,7 @@ func (g *gitVCS) commit(ctx context.Context, version Version) (*object.Commit, e | |||||||
| 			return nil, err | 			return nil, err | ||||||
| 		} | 		} | ||||||
| 		tags.ForEach(func(t *plumbing.Reference) error { | 		tags.ForEach(func(t *plumbing.Reference) error { | ||||||
| 			if t.Name().String() == "refs/tags/"+string(version) { | 			if t.Name().String() == path.Join("refs/tags", tagPrefix, string(version)) { | ||||||
| 				hash = t.Hash().String() | 				hash = t.Hash().String() | ||||||
| 				annotated, err := repo.TagObject(t.Hash()) | 				annotated, err := repo.TagObject(t.Hash()) | ||||||
| 				if err == nil { | 				if err == nil { | ||||||
| @@ -276,7 +324,7 @@ func (g *gitVCS) commit(ctx context.Context, version Version) (*object.Commit, e | |||||||
|  |  | ||||||
| func (g *gitVCS) authMethod() (transport.AuthMethod, error) { | func (g *gitVCS) authMethod() (transport.AuthMethod, error) { | ||||||
| 	if g.auth.Key != "" { | 	if g.auth.Key != "" { | ||||||
| 		return ssh.NewPublicKeysFromFile("git", g.auth.Key, "") | 		return ssh.NewPublicKeysFromFile("git", g.auth.Key, g.auth.Password) | ||||||
| 	} else if g.auth.Username != "" { | 	} else if g.auth.Username != "" { | ||||||
| 		return &http.BasicAuth{Username: g.auth.Username, Password: g.auth.Password}, nil | 		return &http.BasicAuth{Username: g.auth.Username, Password: g.auth.Password}, nil | ||||||
| 	} | 	} | ||||||
|   | |||||||
							
								
								
									
										87
									
								
								pkg/vcs/gomod.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										87
									
								
								pkg/vcs/gomod.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,87 @@ | |||||||
|  | package vcs | ||||||
|  |  | ||||||
|  | import ( | ||||||
|  | 	"bytes" | ||||||
|  | 	"context" | ||||||
|  | 	"encoding/json" | ||||||
|  | 	"io" | ||||||
|  | 	"io/ioutil" | ||||||
|  | 	"os" | ||||||
|  | 	"os/exec" | ||||||
|  | 	"path/filepath" | ||||||
|  | 	"strconv" | ||||||
|  | 	"strings" | ||||||
|  | 	"time" | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | type goVCS struct { | ||||||
|  | 	dir    string | ||||||
|  | 	log    logger | ||||||
|  | 	module string | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func NewGoMod(l logger, module string) VCS { | ||||||
|  | 	return &goVCS{log: l, module: module, dir: "/tmp/_go"} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (g *goVCS) List(ctx context.Context) ([]Version, error) { | ||||||
|  | 	if err := g.download(ctx, "latest"); err != nil { | ||||||
|  | 		return nil, err | ||||||
|  | 	} | ||||||
|  | 	b, err := g.file("list") | ||||||
|  | 	if err != nil { | ||||||
|  | 		return nil, err | ||||||
|  | 	} | ||||||
|  | 	versions := []Version{} | ||||||
|  | 	for _, line := range strings.Split(string(b), "\n") { | ||||||
|  | 		versions = append(versions, Version(line)) | ||||||
|  | 	} | ||||||
|  | 	return versions, nil | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (g *goVCS) Timestamp(ctx context.Context, version Version) (time.Time, error) { | ||||||
|  | 	if err := g.download(ctx, version.String()); err != nil { | ||||||
|  | 		return time.Time{}, err | ||||||
|  | 	} | ||||||
|  | 	b, err := g.file(version.String() + ".info") | ||||||
|  | 	if err != nil { | ||||||
|  | 		return time.Time{}, err | ||||||
|  | 	} | ||||||
|  | 	info := struct { | ||||||
|  | 		Version string | ||||||
|  | 		Time    time.Time | ||||||
|  | 	}{} | ||||||
|  | 	if json.Unmarshal(b, &info) == nil { | ||||||
|  | 		return info.Time, nil | ||||||
|  | 	} | ||||||
|  | 	if t, err := time.Parse(time.RFC3339, string(b)); err == nil { | ||||||
|  | 		return t, nil | ||||||
|  | 	} | ||||||
|  | 	if sec, err := strconv.ParseInt(string(b), 10, 64); err == nil { | ||||||
|  | 		return time.Unix(sec, 0), nil | ||||||
|  | 	} | ||||||
|  | 	return time.Time{}, nil | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (g *goVCS) Zip(ctx context.Context, version Version) (io.ReadCloser, error) { | ||||||
|  | 	if err := g.download(ctx, version.String()); err != nil { | ||||||
|  | 		return nil, err | ||||||
|  | 	} | ||||||
|  | 	b, err := g.file(version.String() + ".zip") | ||||||
|  | 	if err != nil { | ||||||
|  | 		return nil, err | ||||||
|  | 	} | ||||||
|  | 	return ioutil.NopCloser(bytes.NewReader(b)), nil | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (g *goVCS) download(ctx context.Context, version string) error { | ||||||
|  | 	cmd := exec.Command("go", "mod", "download", g.module+"@"+version) | ||||||
|  | 	cmd.Env = append(os.Environ(), "GOPATH="+g.dir) | ||||||
|  | 	cmd.Stderr = os.Stderr | ||||||
|  | 	return cmd.Run() | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (g *goVCS) file(name string) ([]byte, error) { | ||||||
|  | 	path := filepath.Join(g.dir, "pkg", "mod", "cache", "download", encodeBangs(g.module), "@v", name) | ||||||
|  | 	return ioutil.ReadFile(path) | ||||||
|  | } | ||||||
							
								
								
									
										142
									
								
								pkg/vcs/tags.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										142
									
								
								pkg/vcs/tags.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,142 @@ | |||||||
|  | package vcs | ||||||
|  |  | ||||||
|  | import ( | ||||||
|  | 	"context" | ||||||
|  | 	"fmt" | ||||||
|  | 	"io" | ||||||
|  | 	"time" | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | type ephemeralTag struct { | ||||||
|  | 	semVer Version | ||||||
|  | 	short  string | ||||||
|  | } | ||||||
|  |  | ||||||
|  | type EphemeralTagStorage struct { | ||||||
|  | 	tagsByModule map[moduleName][]ephemeralTag | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func NewEphemeralTagStorage() *EphemeralTagStorage { | ||||||
|  | 	return &EphemeralTagStorage{ | ||||||
|  | 		tagsByModule: make(map[moduleName][]ephemeralTag), | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (s *EphemeralTagStorage) Tag(module string, semVer Version, short string) error { | ||||||
|  | 	tags := s.tagsByModule[module] | ||||||
|  | 	tmp := tags[:0] | ||||||
|  | 	for _, t := range tags { | ||||||
|  | 		if t.semVer != semVer { | ||||||
|  | 			tmp = append(tmp, t) | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 	s.tagsByModule[module] = append(tmp, ephemeralTag{semVer, short}) | ||||||
|  | 	return nil | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (s *EphemeralTagStorage) tags(module string) []ephemeralTag { | ||||||
|  | 	return s.tagsByModule[module] | ||||||
|  | } | ||||||
|  |  | ||||||
|  | type moduleName = string | ||||||
|  |  | ||||||
|  | type taggableVCS struct { | ||||||
|  | 	wrapped *gitVCS | ||||||
|  | 	module  string | ||||||
|  | 	storage *EphemeralTagStorage | ||||||
|  | } | ||||||
|  |  | ||||||
|  | type Taggable interface { | ||||||
|  | 	Tag(ctx context.Context, semVer Version, short string) error | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // NewGitWithEphemeralTags return a go-git VCS client implementation that | ||||||
|  | // provides information about the specific module using the given | ||||||
|  | // authentication mechanism while adding support to ephemeral tags. | ||||||
|  | func NewGitWithEphemeralTags(l logger, dir string, module string, auth Auth, storage *EphemeralTagStorage) VCS { | ||||||
|  | 	git := NewGit(l, dir, module, auth).(*gitVCS) | ||||||
|  | 	return &taggableVCS{ | ||||||
|  | 		wrapped: git, | ||||||
|  | 		module:  module, | ||||||
|  | 		storage: storage, | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (v *taggableVCS) safeList(ctx context.Context) ([]Version, error) { | ||||||
|  | 	remoteVersions, err := v.wrapped.List(ctx) | ||||||
|  | 	if err != nil { | ||||||
|  | 		// Ignore this error, we can still count on ephemeral tags. | ||||||
|  | 		if err != ErrNoMatchingVersion { | ||||||
|  | 			return nil, err | ||||||
|  | 		} | ||||||
|  | 		v.wrapped.log("No remote version tags yet:", err) | ||||||
|  | 	} | ||||||
|  | 	return remoteVersions, nil | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (v *taggableVCS) Tag(ctx context.Context, semVer Version, short string) error { | ||||||
|  | 	remoteVersions, err := v.safeList(ctx) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  | 	if versionExists(remoteVersions, semVer) { | ||||||
|  | 		return fmt.Errorf("remote version %s already exists for module %s", semVer, v.module) | ||||||
|  | 	} | ||||||
|  | 	return v.storage.Tag(v.module, semVer, short) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (v *taggableVCS) List(ctx context.Context) ([]Version, error) { | ||||||
|  | 	remoteVersions, err := v.safeList(ctx) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return nil, err | ||||||
|  | 	} | ||||||
|  | 	tags := v.storage.tags(v.module) | ||||||
|  | 	// Remote versions win. | ||||||
|  | 	return appendEphemeralVersion(remoteVersions, tags...), nil | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func appendEphemeralVersion(versions []Version, tags ...ephemeralTag) []Version { | ||||||
|  | 	ephemeral := make([]Version, 0) | ||||||
|  | 	for _, tag := range tags { | ||||||
|  | 		if !versionExists(versions, tag.semVer) { | ||||||
|  | 			ephemeral = append(ephemeral, tag.semVer) | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 	return append(versions, ephemeral...) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func versionExists(versions []Version, v Version) bool { | ||||||
|  | 	for _, v2 := range versions { | ||||||
|  | 		if v == v2 { | ||||||
|  | 			return true | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 	return false | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (v *taggableVCS) Timestamp(ctx context.Context, version Version) (time.Time, error) { | ||||||
|  | 	version2, err := v.resolveVersion(ctx, version) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return time.Time{}, err | ||||||
|  | 	} | ||||||
|  | 	return v.wrapped.Timestamp(ctx, version2) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (v *taggableVCS) Zip(ctx context.Context, version Version) (io.ReadCloser, error) { | ||||||
|  | 	version2, err := v.resolveVersion(ctx, version) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return nil, err | ||||||
|  | 	} | ||||||
|  | 	// Zip must contain the ephemeral version. | ||||||
|  | 	dirName := v.module + "@" + string(version) | ||||||
|  | 	return v.wrapped.zipAs(ctx, version2, dirName) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (v *taggableVCS) resolveVersion(ctx context.Context, version Version) (Version, error) { | ||||||
|  | 	for _, tag := range v.storage.tags(v.module) { | ||||||
|  | 		if tag.semVer == version { | ||||||
|  | 			return v.wrapped.versionFromHash(ctx, tag.short) | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 	return version, nil | ||||||
|  | } | ||||||
| @@ -61,4 +61,4 @@ func NoAuth() Auth { return Auth{} } | |||||||
| func Password(username, password string) Auth { return Auth{Username: username, Password: password} } | func Password(username, password string) Auth { return Auth{Username: username, Password: password} } | ||||||
|  |  | ||||||
| // Key returns an Auth implementation that uses key file authentication mechanism. | // Key returns an Auth implementation that uses key file authentication mechanism. | ||||||
| func Key(key string) Auth { return Auth{Key: key} } | func Key(key, password string) Auth { return Auth{Key: key, Password: password} } | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user