image lookup: do not match *any* tags
For reasons buried in the history of Podman, looking up an untagged image would match any tag of matching image. For instance, looking up centos would match a local image centos:foobar. Change that behavior to only match the latest tag. Fix: #11964 Signed-off-by: Valentin Rothberg <rothberg@redhat.com>
This commit is contained in:
		
							parent
							
								
									a55473bea8
								
							
						
					
					
						commit
						0d1aaf080e
					
				
							
								
								
									
										4
									
								
								go.mod
								
								
								
								
							
							
						
						
									
										4
									
								
								go.mod
								
								
								
								
							|  | @ -12,12 +12,12 @@ require ( | |||
| 	github.com/containernetworking/cni v1.0.1 | ||||
| 	github.com/containernetworking/plugins v1.0.1 | ||||
| 	github.com/containers/buildah v1.23.1 | ||||
| 	github.com/containers/common v0.46.1-0.20211115170340-7ae7bd1c3f8e | ||||
| 	github.com/containers/common v0.46.1-0.20211122213330-d4e7724a0c58 | ||||
| 	github.com/containers/conmon v2.0.20+incompatible | ||||
| 	github.com/containers/image/v5 v5.17.0 | ||||
| 	github.com/containers/ocicrypt v1.1.2 | ||||
| 	github.com/containers/psgo v1.7.1 | ||||
| 	github.com/containers/storage v1.37.1-0.20211014130921-5c5bf639ed01 | ||||
| 	github.com/containers/storage v1.37.1-0.20211122214631-59ba58582415 | ||||
| 	github.com/coreos/go-systemd/v22 v22.3.2 | ||||
| 	github.com/coreos/stream-metadata-go v0.0.0-20210225230131-70edb9eb47b3 | ||||
| 	github.com/cyphar/filepath-securejoin v0.2.3 | ||||
|  |  | |||
							
								
								
									
										22
									
								
								go.sum
								
								
								
								
							
							
						
						
									
										22
									
								
								go.sum
								
								
								
								
							|  | @ -66,8 +66,9 @@ github.com/Microsoft/go-winio v0.4.16/go.mod h1:XB6nPKklQyQ7GC9LdcBEcBl8PF76WugX | |||
| github.com/Microsoft/go-winio v0.4.17-0.20210211115548-6eac466e5fa3/go.mod h1:JPGBdM1cNvN/6ISo+n8V5iA4v8pBzdOpzfwIujj1a84= | ||||
| github.com/Microsoft/go-winio v0.4.17-0.20210324224401-5516f17a5958/go.mod h1:JPGBdM1cNvN/6ISo+n8V5iA4v8pBzdOpzfwIujj1a84= | ||||
| github.com/Microsoft/go-winio v0.4.17/go.mod h1:JPGBdM1cNvN/6ISo+n8V5iA4v8pBzdOpzfwIujj1a84= | ||||
| github.com/Microsoft/go-winio v0.5.0 h1:Elr9Wn+sGKPlkaBvwu4mTrxtmOp3F3yV9qhaHbXGjwU= | ||||
| github.com/Microsoft/go-winio v0.5.0/go.mod h1:JPGBdM1cNvN/6ISo+n8V5iA4v8pBzdOpzfwIujj1a84= | ||||
| github.com/Microsoft/go-winio v0.5.1 h1:aPJp2QD7OOrhO5tQXqQoGSJc+DjDtWTGLOmNyAm6FgY= | ||||
| github.com/Microsoft/go-winio v0.5.1/go.mod h1:JPGBdM1cNvN/6ISo+n8V5iA4v8pBzdOpzfwIujj1a84= | ||||
| github.com/Microsoft/hcsshim v0.8.6/go.mod h1:Op3hHsoHPAvb6lceZHDtd9OkTew38wNoXnJs8iY7rUg= | ||||
| github.com/Microsoft/hcsshim v0.8.7-0.20190325164909-8abdbb8205e4/go.mod h1:Op3hHsoHPAvb6lceZHDtd9OkTew38wNoXnJs8iY7rUg= | ||||
| github.com/Microsoft/hcsshim v0.8.7/go.mod h1:OHd7sQqRFrYd3RmSgbgji+ctCwkbq2wbEYNSzOYtcBQ= | ||||
|  | @ -79,8 +80,8 @@ github.com/Microsoft/hcsshim v0.8.18/go.mod h1:+w2gRZ5ReXQhFOrvSQeNfhrYB/dg3oDwT | |||
| github.com/Microsoft/hcsshim v0.8.20/go.mod h1:+w2gRZ5ReXQhFOrvSQeNfhrYB/dg3oDwTOcER2fw4I4= | ||||
| github.com/Microsoft/hcsshim v0.8.21/go.mod h1:+w2gRZ5ReXQhFOrvSQeNfhrYB/dg3oDwTOcER2fw4I4= | ||||
| github.com/Microsoft/hcsshim v0.8.22/go.mod h1:91uVCVzvX2QD16sMCenoxxXo6L1wJnLMX2PSufFMtF0= | ||||
| github.com/Microsoft/hcsshim v0.9.0 h1:BBgYMxl5YZDZVIijz02AlDINpYZOzQqRNCl9CZM13vk= | ||||
| github.com/Microsoft/hcsshim v0.9.0/go.mod h1:VBJWdC71NSWPlEo7lwde1aL21748J8B6Sdgno7NqEGE= | ||||
| github.com/Microsoft/hcsshim v0.9.1 h1:VfDCj+QnY19ktX5TsH22JHcjaZ05RWQiwDbOyEg5ziM= | ||||
| github.com/Microsoft/hcsshim v0.9.1/go.mod h1:Y/0uV2jUab5kBI7SQgl62at0AVX7uaruzADAVmxm3eM= | ||||
| github.com/Microsoft/hcsshim/test v0.0.0-20201218223536-d3e5debf77da/go.mod h1:5hlzMzRKMLyo42nCZ9oml8AdTlq/0cvIaBv6tK1RehU= | ||||
| github.com/Microsoft/hcsshim/test v0.0.0-20210227013316-43a75bb4edd3/go.mod h1:mw7qgWloBUl75W/gVH3cQszUg1+gUITj7D6NY7ywVnY= | ||||
| github.com/NYTimes/gziphandler v0.0.0-20170623195520-56545f4a5d46/go.mod h1:3wb06e3pkSAbeQ52E9H9iFoQsEEwGN64994WTCIhntQ= | ||||
|  | @ -129,6 +130,7 @@ github.com/buger/jsonparser v1.1.1/go.mod h1:6RYKKt7H4d4+iWqouImQ9R2FZql3VbhNgx2 | |||
| github.com/bugsnag/bugsnag-go v0.0.0-20141110184014-b1d153021fcd/go.mod h1:2oa8nejYd4cQ/b0hMIopN0lCRxU0bueqREvZLWFrtK8= | ||||
| github.com/bugsnag/osext v0.0.0-20130617224835-0dd3f918b21b/go.mod h1:obH5gd0BsqsP2LwDJ9aOkm/6J86V6lyAXCoQWGw3K50= | ||||
| github.com/bugsnag/panicwrap v0.0.0-20151223152923-e2c28503fcd0/go.mod h1:D/8v3kj0zr8ZAKg1AQ6crr+5VwKN5eIywRkfhyM/+dE= | ||||
| github.com/cenkalti/backoff/v4 v4.1.1/go.mod h1:scbssz8iZGpm3xbr14ovlUdkxfGXNInqkPWOWmG2CLw= | ||||
| github.com/census-instrumentation/opencensus-proto v0.2.1/go.mod h1:f6KPmirojxKA12rnyqOA5BBL4O983OfeGPqjHWSTneU= | ||||
| github.com/cespare/xxhash v1.1.0 h1:a6HrQnmkObjyL+Gs60czilIUGqrzKutQD6XZog3p+ko= | ||||
| github.com/cespare/xxhash v1.1.0/go.mod h1:XrSqR1VqqWfGrhpAt58auRo0WTKS1nRRg3ghfAqPWnc= | ||||
|  | @ -231,13 +233,15 @@ github.com/containerd/nri v0.0.0-20210316161719-dbaa18c31c14/go.mod h1:lmxnXF6oM | |||
| github.com/containerd/nri v0.1.0/go.mod h1:lmxnXF6oMkbqs39FiCt1s0R2HSMhcLel9vNL3m4AaeY= | ||||
| github.com/containerd/stargz-snapshotter/estargz v0.4.1/go.mod h1:x7Q9dg9QYb4+ELgxmo4gBUeJB0tl5dqH1Sdz0nJU1QM= | ||||
| github.com/containerd/stargz-snapshotter/estargz v0.8.0/go.mod h1:mwIwuwb+D8FX2t45Trwi0hmWmZm5VW7zPP/rekwhWQU= | ||||
| github.com/containerd/stargz-snapshotter/estargz v0.9.0 h1:PkB6BSTfOKX23erT2GkoUKkJEcXfNcyKskIViK770v8= | ||||
| github.com/containerd/stargz-snapshotter/estargz v0.9.0/go.mod h1:aE5PCyhFMwR8sbrErO5eM2GcvkyXTTJremG883D4qF0= | ||||
| github.com/containerd/stargz-snapshotter/estargz v0.10.1 h1:hd1EoVjI2Ax8Cr64tdYqnJ4i4pZU49FkEf5kU8KxQng= | ||||
| github.com/containerd/stargz-snapshotter/estargz v0.10.1/go.mod h1:aE5PCyhFMwR8sbrErO5eM2GcvkyXTTJremG883D4qF0= | ||||
| github.com/containerd/ttrpc v0.0.0-20190828154514-0e0f228740de/go.mod h1:PvCDdDGpgqzQIzDW1TphrGLssLDZp2GuS+X5DkEJB8o= | ||||
| github.com/containerd/ttrpc v0.0.0-20190828172938-92c8520ef9f8/go.mod h1:PvCDdDGpgqzQIzDW1TphrGLssLDZp2GuS+X5DkEJB8o= | ||||
| github.com/containerd/ttrpc v0.0.0-20191028202541-4f1b8fe65a5c/go.mod h1:LPm1u0xBw8r8NOKoOdNMeVHSawSsltak+Ihv+etqsE8= | ||||
| github.com/containerd/ttrpc v1.0.1/go.mod h1:UAxOpgT9ziI0gJrmKvgcZivgxOp8iFPSk8httJEt98Y= | ||||
| github.com/containerd/ttrpc v1.0.2/go.mod h1:UAxOpgT9ziI0gJrmKvgcZivgxOp8iFPSk8httJEt98Y= | ||||
| github.com/containerd/ttrpc v1.1.0/go.mod h1:XX4ZTnoOId4HklF4edwc4DcqskFZuvXB1Evzy5KFQpQ= | ||||
| github.com/containerd/typeurl v0.0.0-20180627222232-a93fcdb778cd/go.mod h1:Cm3kwCdlkCfMSHURc+r6fwoGH6/F1hH3S4sg0rLFWPc= | ||||
| github.com/containerd/typeurl v0.0.0-20190911142611-5eb25027c9fd/go.mod h1:GeKYzf2pQcqv7tJ0AoCuuhtnqhva5LNU3U+OyKxxJpk= | ||||
| github.com/containerd/typeurl v1.0.1/go.mod h1:TB1hUtrpaiO88KEK56ijojHS1+NeF0izUACaJW2mdXg= | ||||
|  | @ -259,8 +263,8 @@ github.com/containernetworking/plugins v1.0.1/go.mod h1:QHCfGpaTwYTbbH+nZXKVTxNB | |||
| github.com/containers/buildah v1.23.1 h1:Tpc9DsRuU+0Oofewpxb6OJVNQjCu7yloN/obUqzfDTY= | ||||
| github.com/containers/buildah v1.23.1/go.mod h1:4WnrN0yrA7ab0ppgunixu2WM1rlD2rG8QLJAKbEkZlQ= | ||||
| github.com/containers/common v0.44.2/go.mod h1:7sdP4vmI5Bm6FPFxb3lvAh1Iktb6tiO1MzjUzhxdoGo= | ||||
| github.com/containers/common v0.46.1-0.20211115170340-7ae7bd1c3f8e h1:YSuo3zGivcgQhRV1TOJ6zW3VjyjoU7BJMRyh71v/Zdc= | ||||
| github.com/containers/common v0.46.1-0.20211115170340-7ae7bd1c3f8e/go.mod h1:bu8gizEkgAz6gXHvUw2cMtI5ErxB+fn/hv49RWk5N1A= | ||||
| github.com/containers/common v0.46.1-0.20211122213330-d4e7724a0c58 h1:d99ZfYePYt1gU5dPvtIdnORNtv/7mkAZUHhCJzR5D5k= | ||||
| github.com/containers/common v0.46.1-0.20211122213330-d4e7724a0c58/go.mod h1:GrXYaGvQtdKA+fCQLudCTOSGRwZ06MVmRnC7KlI+syY= | ||||
| github.com/containers/conmon v2.0.20+incompatible h1:YbCVSFSCqFjjVwHTPINGdMX1F6JXHGTUje2ZYobNrkg= | ||||
| github.com/containers/conmon v2.0.20+incompatible/go.mod h1:hgwZ2mtuDrppv78a/cOBNiCm6O0UMWGx1mu7P00nu5I= | ||||
| github.com/containers/image/v5 v5.16.0/go.mod h1:XgTpfAPLRGOd1XYyCU5cISFr777bLmOerCSpt/v7+Q4= | ||||
|  | @ -280,8 +284,9 @@ github.com/containers/storage v1.23.5/go.mod h1:ha26Q6ngehFNhf3AWoXldvAvwI4jFe3E | |||
| github.com/containers/storage v1.35.0/go.mod h1:qzYhasQP2/V9D9XdO+vRwkHBhsBO0oznMLzzRDQ8s20= | ||||
| github.com/containers/storage v1.36.0/go.mod h1:vbd3SKVQNHdmU5qQI6hTEcKPxnZkGqydG4f6uwrI5a8= | ||||
| github.com/containers/storage v1.37.0/go.mod h1:kqeJeS0b7DO2ZT1nVWs0XufrmPFbgV3c+Q/45RlH6r4= | ||||
| github.com/containers/storage v1.37.1-0.20211014130921-5c5bf639ed01 h1:bCJAZyAq5rLWwlDAsRNuk7hVqnAugvTJrYqZWjy9POk= | ||||
| github.com/containers/storage v1.37.1-0.20211014130921-5c5bf639ed01/go.mod h1:4QYCZMzSvM3XZtyrdMqC3jW/u5/MXAo8hjwyzug9noc= | ||||
| github.com/containers/storage v1.37.1-0.20211119174841-bf170b3ddac0/go.mod h1:XjCNlt5JUUmRuTJXhFxHb9hHGPho7DNg3o4N/14prdQ= | ||||
| github.com/containers/storage v1.37.1-0.20211122214631-59ba58582415 h1:iqTDpYOxZibrkC7Mo7gCXuJDIT7wyIU432RTmRhlZqY= | ||||
| github.com/containers/storage v1.37.1-0.20211122214631-59ba58582415/go.mod h1:hvKpaiPRALDI7oz4Jx+AEch8iS/viRnc22HPilQROWU= | ||||
| github.com/coreos/bbolt v1.3.2/go.mod h1:iRUV2dpdMOn7Bo10OQBFzIJO9kkE559Wcmn+qkEiiKk= | ||||
| github.com/coreos/etcd v3.3.10+incompatible/go.mod h1:uF7uidLiAD3TWHmW31ZFd/JWoc32PjwdhPthX9715RE= | ||||
| github.com/coreos/etcd v3.3.13+incompatible/go.mod h1:uF7uidLiAD3TWHmW31ZFd/JWoc32PjwdhPthX9715RE= | ||||
|  | @ -338,7 +343,6 @@ github.com/docker/distribution v2.7.1+incompatible h1:a5mlkVzth6W5A4fOsS3D2EO5BU | |||
| github.com/docker/distribution v2.7.1+incompatible/go.mod h1:J2gT2udsDAN96Uj4KfcMRqY0/ypR+oyYUYmja8H+y+w= | ||||
| github.com/docker/docker v1.4.2-0.20190924003213-a8608b5b67c7/go.mod h1:eEKB0N0r5NX/I1kEveEz05bcu8tLC/8azJZsviup8Sk= | ||||
| github.com/docker/docker v20.10.8+incompatible/go.mod h1:eEKB0N0r5NX/I1kEveEz05bcu8tLC/8azJZsviup8Sk= | ||||
| github.com/docker/docker v20.10.10+incompatible/go.mod h1:eEKB0N0r5NX/I1kEveEz05bcu8tLC/8azJZsviup8Sk= | ||||
| github.com/docker/docker v20.10.11+incompatible h1:OqzI/g/W54LczvhnccGqniFoQghHx3pklbLuhfXpqGo= | ||||
| github.com/docker/docker v20.10.11+incompatible/go.mod h1:eEKB0N0r5NX/I1kEveEz05bcu8tLC/8azJZsviup8Sk= | ||||
| github.com/docker/docker-credential-helpers v0.6.3/go.mod h1:WRaJzqw3CTB9bk10avuGsjVBZsD05qeibJ1/TYlvc0Y= | ||||
|  |  | |||
|  | @ -48,7 +48,7 @@ var _ = Describe("Podman pull", func() { | |||
| 		found, _ := session.ErrorGrepString(expectedError) | ||||
| 		Expect(found).To(Equal(true)) | ||||
| 
 | ||||
| 		session = podmanTest.Podman([]string{"rmi", "busybox", "alpine", "testdigest_v2s2", "quay.io/libpod/cirros"}) | ||||
| 		session = podmanTest.Podman([]string{"rmi", "busybox:musl", "alpine", "quay.io/libpod/cirros", "testdigest_v2s2@sha256:755f4d90b3716e2bf57060d249e2cd61c9ac089b1233465c5c2cb2d7ee550fdb"}) | ||||
| 		session.WaitWithDefaultTimeout() | ||||
| 		Expect(session).Should(Exit(0)) | ||||
| 	}) | ||||
|  | @ -104,8 +104,13 @@ var _ = Describe("Podman pull", func() { | |||
| 		session.WaitWithDefaultTimeout() | ||||
| 		Expect(session).Should(Exit(0)) | ||||
| 
 | ||||
| 		// Without a tag/digest the input is normalized with the "latest" tag, see #11964
 | ||||
| 		session = podmanTest.Podman([]string{"rmi", "testdigest_v2s2"}) | ||||
| 		session.WaitWithDefaultTimeout() | ||||
| 		Expect(session).Should(Exit(1)) | ||||
| 
 | ||||
| 		session = podmanTest.Podman([]string{"rmi", "testdigest_v2s2@sha256:755f4d90b3716e2bf57060d249e2cd61c9ac089b1233465c5c2cb2d7ee550fdb"}) | ||||
| 		session.WaitWithDefaultTimeout() | ||||
| 		Expect(session).Should(Exit(0)) | ||||
| 	}) | ||||
| 
 | ||||
|  |  | |||
|  | @ -11,12 +11,27 @@ package. | |||
| 
 | ||||
| Please see the LICENSE file for licensing information. | ||||
| 
 | ||||
| This project has adopted the [Microsoft Open Source Code of | ||||
| Conduct](https://opensource.microsoft.com/codeofconduct/). For more information | ||||
| see the [Code of Conduct | ||||
| FAQ](https://opensource.microsoft.com/codeofconduct/faq/) or contact | ||||
| [opencode@microsoft.com](mailto:opencode@microsoft.com) with any additional | ||||
| questions or comments. | ||||
| ## Contributing | ||||
| 
 | ||||
| This project welcomes contributions and suggestions. Most contributions require you to agree to a Contributor License Agreement (CLA) | ||||
| declaring that you have the right to, and actually do, grant us the rights to use your contribution. For details, visit https://cla.microsoft.com. | ||||
| 
 | ||||
| When you submit a pull request, a CLA-bot will automatically determine whether you need to provide a CLA and decorate the PR | ||||
| appropriately (e.g., label, comment). Simply follow the instructions provided by the bot. You will only need to do this once across all repos using our CLA. | ||||
| 
 | ||||
| We also require that contributors sign their commits using git commit -s or git commit --signoff to certify they either authored the work themselves | ||||
| or otherwise have permission to use it in this project. Please see https://developercertificate.org/ for more info, as well as to make sure that you can | ||||
| attest to the rules listed. Our CI uses the DCO Github app to ensure that all commits in a given PR are signed-off. | ||||
| 
 | ||||
| 
 | ||||
| ## Code of Conduct | ||||
| 
 | ||||
| This project has adopted the [Microsoft Open Source Code of Conduct](https://opensource.microsoft.com/codeofconduct/). | ||||
| For more information see the [Code of Conduct FAQ](https://opensource.microsoft.com/codeofconduct/faq/) or | ||||
| contact [opencode@microsoft.com](mailto:opencode@microsoft.com) with any additional questions or comments. | ||||
| 
 | ||||
| 
 | ||||
| 
 | ||||
| ## Special Thanks | ||||
| Thanks to natefinch for the inspiration for this library. See https://github.com/natefinch/npipe | ||||
| for another named pipe implementation. | ||||
|  |  | |||
|  | @ -5,7 +5,6 @@ package backuptar | |||
| import ( | ||||
| 	"archive/tar" | ||||
| 	"encoding/base64" | ||||
| 	"errors" | ||||
| 	"fmt" | ||||
| 	"io" | ||||
| 	"io/ioutil" | ||||
|  | @ -42,19 +41,14 @@ const ( | |||
| 	hdrCreationTime = "LIBARCHIVE.creationtime" | ||||
| ) | ||||
| 
 | ||||
| func writeZeroes(w io.Writer, count int64) error { | ||||
| 	buf := make([]byte, 8192) | ||||
| 	c := len(buf) | ||||
| 	for i := int64(0); i < count; i += int64(c) { | ||||
| 		if int64(c) > count-i { | ||||
| 			c = int(count - i) | ||||
| 		} | ||||
| 		_, err := w.Write(buf[:c]) | ||||
| 		if err != nil { | ||||
| 			return err | ||||
| 		} | ||||
| // zeroReader is an io.Reader that always returns 0s.
 | ||||
| type zeroReader struct{} | ||||
| 
 | ||||
| func (zr zeroReader) Read(b []byte) (int, error) { | ||||
| 	for i := range b { | ||||
| 		b[i] = 0 | ||||
| 	} | ||||
| 	return nil | ||||
| 	return len(b), nil | ||||
| } | ||||
| 
 | ||||
| func copySparse(t *tar.Writer, br *winio.BackupStreamReader) error { | ||||
|  | @ -71,16 +65,26 @@ func copySparse(t *tar.Writer, br *winio.BackupStreamReader) error { | |||
| 			return fmt.Errorf("unexpected stream %d", bhdr.Id) | ||||
| 		} | ||||
| 
 | ||||
| 		// We can't seek backwards, since we have already written that data to the tar.Writer.
 | ||||
| 		if bhdr.Offset < curOffset { | ||||
| 			return fmt.Errorf("cannot seek back from %d to %d", curOffset, bhdr.Offset) | ||||
| 		} | ||||
| 		// archive/tar does not support writing sparse files
 | ||||
| 		// so just write zeroes to catch up to the current offset.
 | ||||
| 		err = writeZeroes(t, bhdr.Offset-curOffset) | ||||
| 		if _, err := io.CopyN(t, zeroReader{}, bhdr.Offset-curOffset); err != nil { | ||||
| 			return fmt.Errorf("seek to offset %d: %s", bhdr.Offset, err) | ||||
| 		} | ||||
| 		if bhdr.Size == 0 { | ||||
| 			// A sparse block with size = 0 is used to mark the end of the sparse blocks.
 | ||||
| 			break | ||||
| 		} | ||||
| 		n, err := io.Copy(t, br) | ||||
| 		if err != nil { | ||||
| 			return err | ||||
| 		} | ||||
| 		if n != bhdr.Size { | ||||
| 			return fmt.Errorf("copied %d bytes instead of %d at offset %d", n, bhdr.Size, bhdr.Offset) | ||||
| 		} | ||||
| 		curOffset = bhdr.Offset + n | ||||
| 	} | ||||
| 	return nil | ||||
|  | @ -221,20 +225,44 @@ func WriteTarFileFromBackupStream(t *tar.Writer, r io.Reader, name string, size | |||
| 		} | ||||
| 	} | ||||
| 
 | ||||
| 	// The logic for copying file contents is fairly complicated due to the need for handling sparse files,
 | ||||
| 	// and the weird ways they are represented by BackupRead. A normal file will always either have a data stream
 | ||||
| 	// with size and content, or no data stream at all (if empty). However, for a sparse file, the content can also
 | ||||
| 	// be represented using a series of sparse block streams following the data stream. Additionally, the way sparse
 | ||||
| 	// files are handled by BackupRead has changed in the OS recently. The specifics of the representation are described
 | ||||
| 	// in the list at the bottom of this block comment.
 | ||||
| 	//
 | ||||
| 	// Sparse files can be represented in four different ways, based on the specifics of the file.
 | ||||
| 	// - Size = 0:
 | ||||
| 	//     Previously: BackupRead yields no data stream and no sparse block streams.
 | ||||
| 	//     Recently: BackupRead yields a data stream with size = 0. There are no following sparse block streams.
 | ||||
| 	// - Size > 0, no allocated ranges:
 | ||||
| 	//     BackupRead yields a data stream with size = 0. Following is a single sparse block stream with
 | ||||
| 	//     size = 0 and offset = <file size>.
 | ||||
| 	// - Size > 0, one allocated range:
 | ||||
| 	//     BackupRead yields a data stream with size = <file size> containing the file contents. There are no
 | ||||
| 	//     sparse block streams. This is the case if you take a normal file with contents and simply set the
 | ||||
| 	//     sparse flag on it.
 | ||||
| 	// - Size > 0, multiple allocated ranges:
 | ||||
| 	//     BackupRead yields a data stream with size = 0. Following are sparse block streams for each allocated
 | ||||
| 	//     range of the file containing the range contents. Finally there is a sparse block stream with
 | ||||
| 	//     size = 0 and offset = <file size>.
 | ||||
| 
 | ||||
| 	if dataHdr != nil { | ||||
| 		// A data stream was found. Copy the data.
 | ||||
| 		if (dataHdr.Attributes & winio.StreamSparseAttributes) == 0 { | ||||
| 		// We assume that we will either have a data stream size > 0 XOR have sparse block streams.
 | ||||
| 		if dataHdr.Size > 0 || (dataHdr.Attributes&winio.StreamSparseAttributes) == 0 { | ||||
| 			if size != dataHdr.Size { | ||||
| 				return fmt.Errorf("%s: mismatch between file size %d and header size %d", name, size, dataHdr.Size) | ||||
| 			} | ||||
| 			_, err = io.Copy(t, br) | ||||
| 			if err != nil { | ||||
| 				return err | ||||
| 			if _, err = io.Copy(t, br); err != nil { | ||||
| 				return fmt.Errorf("%s: copying contents from data stream: %s", name, err) | ||||
| 			} | ||||
| 		} else { | ||||
| 			err = copySparse(t, br) | ||||
| 			if err != nil { | ||||
| 				return err | ||||
| 		} else if size > 0 { | ||||
| 			// As of a recent OS change, BackupRead now returns a data stream for empty sparse files.
 | ||||
| 			// These files have no sparse block streams, so skip the copySparse call if file size = 0.
 | ||||
| 			if err = copySparse(t, br); err != nil { | ||||
| 				return fmt.Errorf("%s: copying contents from sparse block stream: %s", name, err) | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
|  | @ -279,7 +307,7 @@ func WriteTarFileFromBackupStream(t *tar.Writer, r io.Reader, name string, size | |||
| 			} else { | ||||
| 				// Unsupported for now, since the size of the alternate stream is not present
 | ||||
| 				// in the backup stream until after the data has been read.
 | ||||
| 				return errors.New("tar of sparse alternate data streams is unsupported") | ||||
| 				return fmt.Errorf("%s: tar of sparse alternate data streams is unsupported", name) | ||||
| 			} | ||||
| 		case winio.BackupEaData, winio.BackupLink, winio.BackupPropertyData, winio.BackupObjectId, winio.BackupTxfsData: | ||||
| 			// ignore these streams
 | ||||
|  |  | |||
|  | @ -2,6 +2,6 @@ package security | |||
| 
 | ||||
| //go:generate go run mksyscall_windows.go -output zsyscall_windows.go syscall_windows.go
 | ||||
| 
 | ||||
| //sys getSecurityInfo(handle syscall.Handle, objectType uint32, si uint32, ppsidOwner **uintptr, ppsidGroup **uintptr, ppDacl *uintptr, ppSacl *uintptr, ppSecurityDescriptor *uintptr) (err error) [failretval!=0] = advapi32.GetSecurityInfo
 | ||||
| //sys setSecurityInfo(handle syscall.Handle, objectType uint32, si uint32, psidOwner uintptr, psidGroup uintptr, pDacl uintptr, pSacl uintptr) (err error) [failretval!=0] = advapi32.SetSecurityInfo
 | ||||
| //sys setEntriesInAcl(count uintptr, pListOfEEs uintptr, oldAcl uintptr, newAcl *uintptr) (err error) [failretval!=0] = advapi32.SetEntriesInAclW
 | ||||
| //sys getSecurityInfo(handle syscall.Handle, objectType uint32, si uint32, ppsidOwner **uintptr, ppsidGroup **uintptr, ppDacl *uintptr, ppSacl *uintptr, ppSecurityDescriptor *uintptr) (win32err error) = advapi32.GetSecurityInfo
 | ||||
| //sys setSecurityInfo(handle syscall.Handle, objectType uint32, si uint32, psidOwner uintptr, psidGroup uintptr, pDacl uintptr, pSacl uintptr) (win32err error) = advapi32.SetSecurityInfo
 | ||||
| //sys setEntriesInAcl(count uintptr, pListOfEEs uintptr, oldAcl uintptr, newAcl *uintptr) (win32err error) = advapi32.SetEntriesInAclW
 | ||||
|  |  | |||
|  | @ -45,26 +45,26 @@ var ( | |||
| 	procSetSecurityInfo  = modadvapi32.NewProc("SetSecurityInfo") | ||||
| ) | ||||
| 
 | ||||
| func getSecurityInfo(handle syscall.Handle, objectType uint32, si uint32, ppsidOwner **uintptr, ppsidGroup **uintptr, ppDacl *uintptr, ppSacl *uintptr, ppSecurityDescriptor *uintptr) (err error) { | ||||
| 	r1, _, e1 := syscall.Syscall9(procGetSecurityInfo.Addr(), 8, uintptr(handle), uintptr(objectType), uintptr(si), uintptr(unsafe.Pointer(ppsidOwner)), uintptr(unsafe.Pointer(ppsidGroup)), uintptr(unsafe.Pointer(ppDacl)), uintptr(unsafe.Pointer(ppSacl)), uintptr(unsafe.Pointer(ppSecurityDescriptor)), 0) | ||||
| 	if r1 != 0 { | ||||
| 		err = errnoErr(e1) | ||||
| func getSecurityInfo(handle syscall.Handle, objectType uint32, si uint32, ppsidOwner **uintptr, ppsidGroup **uintptr, ppDacl *uintptr, ppSacl *uintptr, ppSecurityDescriptor *uintptr) (win32err error) { | ||||
| 	r0, _, _ := syscall.Syscall9(procGetSecurityInfo.Addr(), 8, uintptr(handle), uintptr(objectType), uintptr(si), uintptr(unsafe.Pointer(ppsidOwner)), uintptr(unsafe.Pointer(ppsidGroup)), uintptr(unsafe.Pointer(ppDacl)), uintptr(unsafe.Pointer(ppSacl)), uintptr(unsafe.Pointer(ppSecurityDescriptor)), 0) | ||||
| 	if r0 != 0 { | ||||
| 		win32err = syscall.Errno(r0) | ||||
| 	} | ||||
| 	return | ||||
| } | ||||
| 
 | ||||
| func setEntriesInAcl(count uintptr, pListOfEEs uintptr, oldAcl uintptr, newAcl *uintptr) (err error) { | ||||
| 	r1, _, e1 := syscall.Syscall6(procSetEntriesInAclW.Addr(), 4, uintptr(count), uintptr(pListOfEEs), uintptr(oldAcl), uintptr(unsafe.Pointer(newAcl)), 0, 0) | ||||
| 	if r1 != 0 { | ||||
| 		err = errnoErr(e1) | ||||
| func setEntriesInAcl(count uintptr, pListOfEEs uintptr, oldAcl uintptr, newAcl *uintptr) (win32err error) { | ||||
| 	r0, _, _ := syscall.Syscall6(procSetEntriesInAclW.Addr(), 4, uintptr(count), uintptr(pListOfEEs), uintptr(oldAcl), uintptr(unsafe.Pointer(newAcl)), 0, 0) | ||||
| 	if r0 != 0 { | ||||
| 		win32err = syscall.Errno(r0) | ||||
| 	} | ||||
| 	return | ||||
| } | ||||
| 
 | ||||
| func setSecurityInfo(handle syscall.Handle, objectType uint32, si uint32, psidOwner uintptr, psidGroup uintptr, pDacl uintptr, pSacl uintptr) (err error) { | ||||
| 	r1, _, e1 := syscall.Syscall9(procSetSecurityInfo.Addr(), 7, uintptr(handle), uintptr(objectType), uintptr(si), uintptr(psidOwner), uintptr(psidGroup), uintptr(pDacl), uintptr(pSacl), 0, 0) | ||||
| 	if r1 != 0 { | ||||
| 		err = errnoErr(e1) | ||||
| func setSecurityInfo(handle syscall.Handle, objectType uint32, si uint32, psidOwner uintptr, psidGroup uintptr, pDacl uintptr, pSacl uintptr) (win32err error) { | ||||
| 	r0, _, _ := syscall.Syscall9(procSetSecurityInfo.Addr(), 7, uintptr(handle), uintptr(objectType), uintptr(si), uintptr(psidOwner), uintptr(psidGroup), uintptr(pDacl), uintptr(pSacl), 0, 0) | ||||
| 	if r0 != 0 { | ||||
| 		win32err = syscall.Errno(r0) | ||||
| 	} | ||||
| 	return | ||||
| } | ||||
|  |  | |||
|  | @ -13,11 +13,11 @@ import ( | |||
| 
 | ||||
| //go:generate go run mksyscall_windows.go -output zvhd_windows.go vhd.go
 | ||||
| 
 | ||||
| //sys createVirtualDisk(virtualStorageType *VirtualStorageType, path string, virtualDiskAccessMask uint32, securityDescriptor *uintptr, createVirtualDiskFlags uint32, providerSpecificFlags uint32, parameters *CreateVirtualDiskParameters, overlapped *syscall.Overlapped, handle *syscall.Handle) (err error) [failretval != 0] = virtdisk.CreateVirtualDisk
 | ||||
| //sys openVirtualDisk(virtualStorageType *VirtualStorageType, path string, virtualDiskAccessMask uint32, openVirtualDiskFlags uint32, parameters *OpenVirtualDiskParameters, handle *syscall.Handle) (err error) [failretval != 0] = virtdisk.OpenVirtualDisk
 | ||||
| //sys attachVirtualDisk(handle syscall.Handle, securityDescriptor *uintptr, attachVirtualDiskFlag uint32, providerSpecificFlags uint32, parameters *AttachVirtualDiskParameters, overlapped *syscall.Overlapped) (err error) [failretval != 0] = virtdisk.AttachVirtualDisk
 | ||||
| //sys detachVirtualDisk(handle syscall.Handle, detachVirtualDiskFlags uint32, providerSpecificFlags uint32) (err error) [failretval != 0] = virtdisk.DetachVirtualDisk
 | ||||
| //sys getVirtualDiskPhysicalPath(handle syscall.Handle, diskPathSizeInBytes *uint32, buffer *uint16) (err error) [failretval != 0] = virtdisk.GetVirtualDiskPhysicalPath
 | ||||
| //sys createVirtualDisk(virtualStorageType *VirtualStorageType, path string, virtualDiskAccessMask uint32, securityDescriptor *uintptr, createVirtualDiskFlags uint32, providerSpecificFlags uint32, parameters *CreateVirtualDiskParameters, overlapped *syscall.Overlapped, handle *syscall.Handle) (win32err error) = virtdisk.CreateVirtualDisk
 | ||||
| //sys openVirtualDisk(virtualStorageType *VirtualStorageType, path string, virtualDiskAccessMask uint32, openVirtualDiskFlags uint32, parameters *OpenVirtualDiskParameters, handle *syscall.Handle) (win32err error) = virtdisk.OpenVirtualDisk
 | ||||
| //sys attachVirtualDisk(handle syscall.Handle, securityDescriptor *uintptr, attachVirtualDiskFlag uint32, providerSpecificFlags uint32, parameters *AttachVirtualDiskParameters, overlapped *syscall.Overlapped) (win32err error) = virtdisk.AttachVirtualDisk
 | ||||
| //sys detachVirtualDisk(handle syscall.Handle, detachVirtualDiskFlags uint32, providerSpecificFlags uint32) (win32err error) = virtdisk.DetachVirtualDisk
 | ||||
| //sys getVirtualDiskPhysicalPath(handle syscall.Handle, diskPathSizeInBytes *uint32, buffer *uint16) (win32err error) = virtdisk.GetVirtualDiskPhysicalPath
 | ||||
| 
 | ||||
| type ( | ||||
| 	CreateVirtualDiskFlag uint32 | ||||
|  |  | |||
|  | @ -47,60 +47,60 @@ var ( | |||
| 	procOpenVirtualDisk            = modvirtdisk.NewProc("OpenVirtualDisk") | ||||
| ) | ||||
| 
 | ||||
| func attachVirtualDisk(handle syscall.Handle, securityDescriptor *uintptr, attachVirtualDiskFlag uint32, providerSpecificFlags uint32, parameters *AttachVirtualDiskParameters, overlapped *syscall.Overlapped) (err error) { | ||||
| 	r1, _, e1 := syscall.Syscall6(procAttachVirtualDisk.Addr(), 6, uintptr(handle), uintptr(unsafe.Pointer(securityDescriptor)), uintptr(attachVirtualDiskFlag), uintptr(providerSpecificFlags), uintptr(unsafe.Pointer(parameters)), uintptr(unsafe.Pointer(overlapped))) | ||||
| 	if r1 != 0 { | ||||
| 		err = errnoErr(e1) | ||||
| func attachVirtualDisk(handle syscall.Handle, securityDescriptor *uintptr, attachVirtualDiskFlag uint32, providerSpecificFlags uint32, parameters *AttachVirtualDiskParameters, overlapped *syscall.Overlapped) (win32err error) { | ||||
| 	r0, _, _ := syscall.Syscall6(procAttachVirtualDisk.Addr(), 6, uintptr(handle), uintptr(unsafe.Pointer(securityDescriptor)), uintptr(attachVirtualDiskFlag), uintptr(providerSpecificFlags), uintptr(unsafe.Pointer(parameters)), uintptr(unsafe.Pointer(overlapped))) | ||||
| 	if r0 != 0 { | ||||
| 		win32err = syscall.Errno(r0) | ||||
| 	} | ||||
| 	return | ||||
| } | ||||
| 
 | ||||
| func createVirtualDisk(virtualStorageType *VirtualStorageType, path string, virtualDiskAccessMask uint32, securityDescriptor *uintptr, createVirtualDiskFlags uint32, providerSpecificFlags uint32, parameters *CreateVirtualDiskParameters, overlapped *syscall.Overlapped, handle *syscall.Handle) (err error) { | ||||
| func createVirtualDisk(virtualStorageType *VirtualStorageType, path string, virtualDiskAccessMask uint32, securityDescriptor *uintptr, createVirtualDiskFlags uint32, providerSpecificFlags uint32, parameters *CreateVirtualDiskParameters, overlapped *syscall.Overlapped, handle *syscall.Handle) (win32err error) { | ||||
| 	var _p0 *uint16 | ||||
| 	_p0, err = syscall.UTF16PtrFromString(path) | ||||
| 	if err != nil { | ||||
| 	_p0, win32err = syscall.UTF16PtrFromString(path) | ||||
| 	if win32err != nil { | ||||
| 		return | ||||
| 	} | ||||
| 	return _createVirtualDisk(virtualStorageType, _p0, virtualDiskAccessMask, securityDescriptor, createVirtualDiskFlags, providerSpecificFlags, parameters, overlapped, handle) | ||||
| } | ||||
| 
 | ||||
| func _createVirtualDisk(virtualStorageType *VirtualStorageType, path *uint16, virtualDiskAccessMask uint32, securityDescriptor *uintptr, createVirtualDiskFlags uint32, providerSpecificFlags uint32, parameters *CreateVirtualDiskParameters, overlapped *syscall.Overlapped, handle *syscall.Handle) (err error) { | ||||
| 	r1, _, e1 := syscall.Syscall9(procCreateVirtualDisk.Addr(), 9, uintptr(unsafe.Pointer(virtualStorageType)), uintptr(unsafe.Pointer(path)), uintptr(virtualDiskAccessMask), uintptr(unsafe.Pointer(securityDescriptor)), uintptr(createVirtualDiskFlags), uintptr(providerSpecificFlags), uintptr(unsafe.Pointer(parameters)), uintptr(unsafe.Pointer(overlapped)), uintptr(unsafe.Pointer(handle))) | ||||
| 	if r1 != 0 { | ||||
| 		err = errnoErr(e1) | ||||
| func _createVirtualDisk(virtualStorageType *VirtualStorageType, path *uint16, virtualDiskAccessMask uint32, securityDescriptor *uintptr, createVirtualDiskFlags uint32, providerSpecificFlags uint32, parameters *CreateVirtualDiskParameters, overlapped *syscall.Overlapped, handle *syscall.Handle) (win32err error) { | ||||
| 	r0, _, _ := syscall.Syscall9(procCreateVirtualDisk.Addr(), 9, uintptr(unsafe.Pointer(virtualStorageType)), uintptr(unsafe.Pointer(path)), uintptr(virtualDiskAccessMask), uintptr(unsafe.Pointer(securityDescriptor)), uintptr(createVirtualDiskFlags), uintptr(providerSpecificFlags), uintptr(unsafe.Pointer(parameters)), uintptr(unsafe.Pointer(overlapped)), uintptr(unsafe.Pointer(handle))) | ||||
| 	if r0 != 0 { | ||||
| 		win32err = syscall.Errno(r0) | ||||
| 	} | ||||
| 	return | ||||
| } | ||||
| 
 | ||||
| func detachVirtualDisk(handle syscall.Handle, detachVirtualDiskFlags uint32, providerSpecificFlags uint32) (err error) { | ||||
| 	r1, _, e1 := syscall.Syscall(procDetachVirtualDisk.Addr(), 3, uintptr(handle), uintptr(detachVirtualDiskFlags), uintptr(providerSpecificFlags)) | ||||
| 	if r1 != 0 { | ||||
| 		err = errnoErr(e1) | ||||
| func detachVirtualDisk(handle syscall.Handle, detachVirtualDiskFlags uint32, providerSpecificFlags uint32) (win32err error) { | ||||
| 	r0, _, _ := syscall.Syscall(procDetachVirtualDisk.Addr(), 3, uintptr(handle), uintptr(detachVirtualDiskFlags), uintptr(providerSpecificFlags)) | ||||
| 	if r0 != 0 { | ||||
| 		win32err = syscall.Errno(r0) | ||||
| 	} | ||||
| 	return | ||||
| } | ||||
| 
 | ||||
| func getVirtualDiskPhysicalPath(handle syscall.Handle, diskPathSizeInBytes *uint32, buffer *uint16) (err error) { | ||||
| 	r1, _, e1 := syscall.Syscall(procGetVirtualDiskPhysicalPath.Addr(), 3, uintptr(handle), uintptr(unsafe.Pointer(diskPathSizeInBytes)), uintptr(unsafe.Pointer(buffer))) | ||||
| 	if r1 != 0 { | ||||
| 		err = errnoErr(e1) | ||||
| func getVirtualDiskPhysicalPath(handle syscall.Handle, diskPathSizeInBytes *uint32, buffer *uint16) (win32err error) { | ||||
| 	r0, _, _ := syscall.Syscall(procGetVirtualDiskPhysicalPath.Addr(), 3, uintptr(handle), uintptr(unsafe.Pointer(diskPathSizeInBytes)), uintptr(unsafe.Pointer(buffer))) | ||||
| 	if r0 != 0 { | ||||
| 		win32err = syscall.Errno(r0) | ||||
| 	} | ||||
| 	return | ||||
| } | ||||
| 
 | ||||
| func openVirtualDisk(virtualStorageType *VirtualStorageType, path string, virtualDiskAccessMask uint32, openVirtualDiskFlags uint32, parameters *OpenVirtualDiskParameters, handle *syscall.Handle) (err error) { | ||||
| func openVirtualDisk(virtualStorageType *VirtualStorageType, path string, virtualDiskAccessMask uint32, openVirtualDiskFlags uint32, parameters *OpenVirtualDiskParameters, handle *syscall.Handle) (win32err error) { | ||||
| 	var _p0 *uint16 | ||||
| 	_p0, err = syscall.UTF16PtrFromString(path) | ||||
| 	if err != nil { | ||||
| 	_p0, win32err = syscall.UTF16PtrFromString(path) | ||||
| 	if win32err != nil { | ||||
| 		return | ||||
| 	} | ||||
| 	return _openVirtualDisk(virtualStorageType, _p0, virtualDiskAccessMask, openVirtualDiskFlags, parameters, handle) | ||||
| } | ||||
| 
 | ||||
| func _openVirtualDisk(virtualStorageType *VirtualStorageType, path *uint16, virtualDiskAccessMask uint32, openVirtualDiskFlags uint32, parameters *OpenVirtualDiskParameters, handle *syscall.Handle) (err error) { | ||||
| 	r1, _, e1 := syscall.Syscall6(procOpenVirtualDisk.Addr(), 6, uintptr(unsafe.Pointer(virtualStorageType)), uintptr(unsafe.Pointer(path)), uintptr(virtualDiskAccessMask), uintptr(openVirtualDiskFlags), uintptr(unsafe.Pointer(parameters)), uintptr(unsafe.Pointer(handle))) | ||||
| 	if r1 != 0 { | ||||
| 		err = errnoErr(e1) | ||||
| func _openVirtualDisk(virtualStorageType *VirtualStorageType, path *uint16, virtualDiskAccessMask uint32, openVirtualDiskFlags uint32, parameters *OpenVirtualDiskParameters, handle *syscall.Handle) (win32err error) { | ||||
| 	r0, _, _ := syscall.Syscall6(procOpenVirtualDisk.Addr(), 6, uintptr(unsafe.Pointer(virtualStorageType)), uintptr(unsafe.Pointer(path)), uintptr(virtualDiskAccessMask), uintptr(openVirtualDiskFlags), uintptr(unsafe.Pointer(parameters)), uintptr(unsafe.Pointer(handle))) | ||||
| 	if r0 != 0 { | ||||
| 		win32err = syscall.Errno(r0) | ||||
| 	} | ||||
| 	return | ||||
| } | ||||
|  |  | |||
|  | @ -5,11 +5,12 @@ go 1.13 | |||
| require ( | ||||
| 	github.com/BurntSushi/toml v0.3.1 | ||||
| 	github.com/Microsoft/go-winio v0.4.17 | ||||
| 	github.com/cenkalti/backoff/v4 v4.1.1 | ||||
| 	github.com/containerd/cgroups v1.0.1 | ||||
| 	github.com/containerd/console v1.0.2 | ||||
| 	github.com/containerd/containerd v1.5.7 | ||||
| 	github.com/containerd/go-runc v1.0.0 | ||||
| 	github.com/containerd/ttrpc v1.0.2 | ||||
| 	github.com/containerd/ttrpc v1.1.0 | ||||
| 	github.com/containerd/typeurl v1.0.2 | ||||
| 	github.com/gogo/protobuf v1.3.2 | ||||
| 	github.com/golang/mock v1.6.0 | ||||
|  |  | |||
|  | @ -86,6 +86,8 @@ github.com/buger/jsonparser v0.0.0-20180808090653-f4dd9f5a6b44/go.mod h1:bbYlZJ7 | |||
| github.com/bugsnag/bugsnag-go v0.0.0-20141110184014-b1d153021fcd/go.mod h1:2oa8nejYd4cQ/b0hMIopN0lCRxU0bueqREvZLWFrtK8= | ||||
| github.com/bugsnag/osext v0.0.0-20130617224835-0dd3f918b21b/go.mod h1:obH5gd0BsqsP2LwDJ9aOkm/6J86V6lyAXCoQWGw3K50= | ||||
| github.com/bugsnag/panicwrap v0.0.0-20151223152923-e2c28503fcd0/go.mod h1:D/8v3kj0zr8ZAKg1AQ6crr+5VwKN5eIywRkfhyM/+dE= | ||||
| github.com/cenkalti/backoff/v4 v4.1.1 h1:G2HAfAmvm/GcKan2oOQpBXOd2tT2G57ZnZGWa1PxPBQ= | ||||
| github.com/cenkalti/backoff/v4 v4.1.1/go.mod h1:scbssz8iZGpm3xbr14ovlUdkxfGXNInqkPWOWmG2CLw= | ||||
| github.com/census-instrumentation/opencensus-proto v0.2.1/go.mod h1:f6KPmirojxKA12rnyqOA5BBL4O983OfeGPqjHWSTneU= | ||||
| github.com/cespare/xxhash v1.1.0/go.mod h1:XrSqR1VqqWfGrhpAt58auRo0WTKS1nRRg3ghfAqPWnc= | ||||
| github.com/cespare/xxhash/v2 v2.1.1/go.mod h1:VGX0DQ3Q6kWi7AoAeZDth3/j3BFtOZR5XLFGgcrjCOs= | ||||
|  | @ -172,8 +174,9 @@ github.com/containerd/ttrpc v0.0.0-20190828154514-0e0f228740de/go.mod h1:PvCDdDG | |||
| github.com/containerd/ttrpc v0.0.0-20190828172938-92c8520ef9f8/go.mod h1:PvCDdDGpgqzQIzDW1TphrGLssLDZp2GuS+X5DkEJB8o= | ||||
| github.com/containerd/ttrpc v0.0.0-20191028202541-4f1b8fe65a5c/go.mod h1:LPm1u0xBw8r8NOKoOdNMeVHSawSsltak+Ihv+etqsE8= | ||||
| github.com/containerd/ttrpc v1.0.1/go.mod h1:UAxOpgT9ziI0gJrmKvgcZivgxOp8iFPSk8httJEt98Y= | ||||
| github.com/containerd/ttrpc v1.0.2 h1:2/O3oTZN36q2xRolk0a2WWGgh7/Vf/liElg5hFYLX9U= | ||||
| github.com/containerd/ttrpc v1.0.2/go.mod h1:UAxOpgT9ziI0gJrmKvgcZivgxOp8iFPSk8httJEt98Y= | ||||
| github.com/containerd/ttrpc v1.1.0 h1:GbtyLRxb0gOLR0TYQWt3O6B0NvT8tMdorEHqIQo/lWI= | ||||
| github.com/containerd/ttrpc v1.1.0/go.mod h1:XX4ZTnoOId4HklF4edwc4DcqskFZuvXB1Evzy5KFQpQ= | ||||
| github.com/containerd/typeurl v0.0.0-20180627222232-a93fcdb778cd/go.mod h1:Cm3kwCdlkCfMSHURc+r6fwoGH6/F1hH3S4sg0rLFWPc= | ||||
| github.com/containerd/typeurl v0.0.0-20190911142611-5eb25027c9fd/go.mod h1:GeKYzf2pQcqv7tJ0AoCuuhtnqhva5LNU3U+OyKxxJpk= | ||||
| github.com/containerd/typeurl v1.0.1/go.mod h1:TB1hUtrpaiO88KEK56ijojHS1+NeF0izUACaJW2mdXg= | ||||
|  | @ -904,8 +907,9 @@ google.golang.org/protobuf v1.22.0/go.mod h1:EGpADcykh3NcUnDUJcl1+ZksZNG86OlYog2 | |||
| google.golang.org/protobuf v1.23.0/go.mod h1:EGpADcykh3NcUnDUJcl1+ZksZNG86OlYog2l/sGQquU= | ||||
| 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 h1:bxAC2xTBsZGibn2RTntX0oH50xLsqy1OxA9tTL3p/lk= | ||||
| google.golang.org/protobuf v1.26.0/go.mod h1:9q0QmTI4eRPtz6boOQmLYwt+qCgq0jsYwAQnmE0givc= | ||||
| google.golang.org/protobuf v1.27.1 h1:SnqbnDw1V7RiZcXPx5MEeqPv2s79L9i7BJUlG/+RurQ= | ||||
| google.golang.org/protobuf v1.27.1/go.mod h1:9q0QmTI4eRPtz6boOQmLYwt+qCgq0jsYwAQnmE0givc= | ||||
| gopkg.in/airbrake/gobrake.v2 v2.0.9/go.mod h1:/h5ZAUhDkGaJfjzjKLSjv6zCL6O0LLBxU4K+aSYdM/U= | ||||
| gopkg.in/alecthomas/kingpin.v2 v2.2.6/go.mod h1:FMv+mEhP44yOT+4EoQTLFTRgOQ1FBLkstjWtayDeSgw= | ||||
| gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= | ||||
|  |  | |||
|  | @ -118,7 +118,7 @@ func Open(sr *io.SectionReader, opt ...OpenOption) (*Reader, error) { | |||
| 		} | ||||
| 	} | ||||
| 
 | ||||
| 	gzipCompressors := []Decompressor{new(GzipDecompressor), new(legacyGzipDecompressor)} | ||||
| 	gzipCompressors := []Decompressor{new(GzipDecompressor), new(LegacyGzipDecompressor)} | ||||
| 	decompressors := append(gzipCompressors, opts.decompressors...) | ||||
| 
 | ||||
| 	// Determine the size to fetch. Try to fetch as many bytes as possible.
 | ||||
|  | @ -184,7 +184,7 @@ func OpenFooter(sr *io.SectionReader) (tocOffset int64, footerSize int64, rErr e | |||
| 		return 0, 0, fmt.Errorf("error reading footer: %v", err) | ||||
| 	} | ||||
| 	var allErr []error | ||||
| 	for _, d := range []Decompressor{new(GzipDecompressor), new(legacyGzipDecompressor)} { | ||||
| 	for _, d := range []Decompressor{new(GzipDecompressor), new(LegacyGzipDecompressor)} { | ||||
| 		fSize := d.FooterSize() | ||||
| 		fOffset := positive(int64(len(footer)) - fSize) | ||||
| 		_, tocOffset, _, err := d.ParseFooter(footer[fOffset:]) | ||||
|  | @ -279,12 +279,12 @@ func (r *Reader) initFields() error { | |||
| 		pdir := r.getOrCreateDir(pdirName) | ||||
| 		ent.NumLink++ // at least one name(ent.Name) references this entry.
 | ||||
| 		if ent.Type == "hardlink" { | ||||
| 			if org, ok := r.m[cleanEntryName(ent.LinkName)]; ok { | ||||
| 				org.NumLink++ // original entry is referenced by this ent.Name.
 | ||||
| 				ent = org | ||||
| 			} else { | ||||
| 				return fmt.Errorf("%q is a hardlink but the linkname %q isn't found", ent.Name, ent.LinkName) | ||||
| 			org, err := r.getSource(ent) | ||||
| 			if err != nil { | ||||
| 				return err | ||||
| 			} | ||||
| 			org.NumLink++ // original entry is referenced by this ent.Name.
 | ||||
| 			ent = org | ||||
| 		} | ||||
| 		pdir.addChild(path.Base(name), ent) | ||||
| 	} | ||||
|  | @ -303,6 +303,20 @@ func (r *Reader) initFields() error { | |||
| 	return nil | ||||
| } | ||||
| 
 | ||||
| func (r *Reader) getSource(ent *TOCEntry) (_ *TOCEntry, err error) { | ||||
| 	if ent.Type == "hardlink" { | ||||
| 		org, ok := r.m[cleanEntryName(ent.LinkName)] | ||||
| 		if !ok { | ||||
| 			return nil, fmt.Errorf("%q is a hardlink but the linkname %q isn't found", ent.Name, ent.LinkName) | ||||
| 		} | ||||
| 		ent, err = r.getSource(org) | ||||
| 		if err != nil { | ||||
| 			return nil, err | ||||
| 		} | ||||
| 	} | ||||
| 	return ent, nil | ||||
| } | ||||
| 
 | ||||
| func parentDir(p string) string { | ||||
| 	dir, _ := path.Split(p) | ||||
| 	return strings.TrimSuffix(dir, "/") | ||||
|  | @ -464,7 +478,11 @@ func (r *Reader) Lookup(path string) (e *TOCEntry, ok bool) { | |||
| 	} | ||||
| 	e, ok = r.m[path] | ||||
| 	if ok && e.Type == "hardlink" { | ||||
| 		e, ok = r.m[e.LinkName] | ||||
| 		var err error | ||||
| 		e, err = r.getSource(e) | ||||
| 		if err != nil { | ||||
| 			return nil, false | ||||
| 		} | ||||
| 	} | ||||
| 	return | ||||
| } | ||||
|  |  | |||
|  | @ -38,30 +38,34 @@ import ( | |||
| ) | ||||
| 
 | ||||
| type gzipCompression struct { | ||||
| 	*gzipCompressor | ||||
| 	*GzipCompressor | ||||
| 	*GzipDecompressor | ||||
| } | ||||
| 
 | ||||
| func newGzipCompressionWithLevel(level int) Compression { | ||||
| 	return &gzipCompression{ | ||||
| 		&gzipCompressor{level}, | ||||
| 		&GzipCompressor{level}, | ||||
| 		&GzipDecompressor{}, | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| func NewGzipCompressorWithLevel(level int) Compressor { | ||||
| 	return &gzipCompressor{level} | ||||
| func NewGzipCompressor() *GzipCompressor { | ||||
| 	return &GzipCompressor{gzip.BestCompression} | ||||
| } | ||||
| 
 | ||||
| type gzipCompressor struct { | ||||
| func NewGzipCompressorWithLevel(level int) *GzipCompressor { | ||||
| 	return &GzipCompressor{level} | ||||
| } | ||||
| 
 | ||||
| type GzipCompressor struct { | ||||
| 	compressionLevel int | ||||
| } | ||||
| 
 | ||||
| func (gc *gzipCompressor) Writer(w io.Writer) (io.WriteCloser, error) { | ||||
| func (gc *GzipCompressor) Writer(w io.Writer) (io.WriteCloser, error) { | ||||
| 	return gzip.NewWriterLevel(w, gc.compressionLevel) | ||||
| } | ||||
| 
 | ||||
| func (gc *gzipCompressor) WriteTOCAndFooter(w io.Writer, off int64, toc *JTOC, diffHash hash.Hash) (digest.Digest, error) { | ||||
| func (gc *GzipCompressor) WriteTOCAndFooter(w io.Writer, off int64, toc *JTOC, diffHash hash.Hash) (digest.Digest, error) { | ||||
| 	tocJSON, err := json.MarshalIndent(toc, "", "\t") | ||||
| 	if err != nil { | ||||
| 		return "", err | ||||
|  | @ -155,17 +159,21 @@ func (gz *GzipDecompressor) FooterSize() int64 { | |||
| 	return FooterSize | ||||
| } | ||||
| 
 | ||||
| type legacyGzipDecompressor struct{} | ||||
| func (gz *GzipDecompressor) DecompressTOC(r io.Reader) (tocJSON io.ReadCloser, err error) { | ||||
| 	return decompressTOCEStargz(r) | ||||
| } | ||||
| 
 | ||||
| func (gz *legacyGzipDecompressor) Reader(r io.Reader) (io.ReadCloser, error) { | ||||
| type LegacyGzipDecompressor struct{} | ||||
| 
 | ||||
| func (gz *LegacyGzipDecompressor) Reader(r io.Reader) (io.ReadCloser, error) { | ||||
| 	return gzip.NewReader(r) | ||||
| } | ||||
| 
 | ||||
| func (gz *legacyGzipDecompressor) ParseTOC(r io.Reader) (toc *JTOC, tocDgst digest.Digest, err error) { | ||||
| func (gz *LegacyGzipDecompressor) ParseTOC(r io.Reader) (toc *JTOC, tocDgst digest.Digest, err error) { | ||||
| 	return parseTOCEStargz(r) | ||||
| } | ||||
| 
 | ||||
| func (gz *legacyGzipDecompressor) ParseFooter(p []byte) (blobPayloadSize, tocOffset, tocSize int64, err error) { | ||||
| func (gz *LegacyGzipDecompressor) ParseFooter(p []byte) (blobPayloadSize, tocOffset, tocSize int64, err error) { | ||||
| 	if len(p) != legacyFooterSize { | ||||
| 		return 0, 0, 0, fmt.Errorf("legacy: invalid length %d cannot be parsed", len(p)) | ||||
| 	} | ||||
|  | @ -188,29 +196,43 @@ func (gz *legacyGzipDecompressor) ParseFooter(p []byte) (blobPayloadSize, tocOff | |||
| 	return tocOffset, tocOffset, 0, nil | ||||
| } | ||||
| 
 | ||||
| func (gz *legacyGzipDecompressor) FooterSize() int64 { | ||||
| func (gz *LegacyGzipDecompressor) FooterSize() int64 { | ||||
| 	return legacyFooterSize | ||||
| } | ||||
| 
 | ||||
| func (gz *LegacyGzipDecompressor) DecompressTOC(r io.Reader) (tocJSON io.ReadCloser, err error) { | ||||
| 	return decompressTOCEStargz(r) | ||||
| } | ||||
| 
 | ||||
| func parseTOCEStargz(r io.Reader) (toc *JTOC, tocDgst digest.Digest, err error) { | ||||
| 	zr, err := gzip.NewReader(r) | ||||
| 	tr, err := decompressTOCEStargz(r) | ||||
| 	if err != nil { | ||||
| 		return nil, "", fmt.Errorf("malformed TOC gzip header: %v", err) | ||||
| 	} | ||||
| 	defer zr.Close() | ||||
| 	zr.Multistream(false) | ||||
| 	tr := tar.NewReader(zr) | ||||
| 	h, err := tr.Next() | ||||
| 	if err != nil { | ||||
| 		return nil, "", fmt.Errorf("failed to find tar header in TOC gzip stream: %v", err) | ||||
| 	} | ||||
| 	if h.Name != TOCTarName { | ||||
| 		return nil, "", fmt.Errorf("TOC tar entry had name %q; expected %q", h.Name, TOCTarName) | ||||
| 		return nil, "", err | ||||
| 	} | ||||
| 	dgstr := digest.Canonical.Digester() | ||||
| 	toc = new(JTOC) | ||||
| 	if err := json.NewDecoder(io.TeeReader(tr, dgstr.Hash())).Decode(&toc); err != nil { | ||||
| 		return nil, "", fmt.Errorf("error decoding TOC JSON: %v", err) | ||||
| 	} | ||||
| 	if err := tr.Close(); err != nil { | ||||
| 		return nil, "", err | ||||
| 	} | ||||
| 	return toc, dgstr.Digest(), nil | ||||
| } | ||||
| 
 | ||||
| func decompressTOCEStargz(r io.Reader) (tocJSON io.ReadCloser, err error) { | ||||
| 	zr, err := gzip.NewReader(r) | ||||
| 	if err != nil { | ||||
| 		return nil, fmt.Errorf("malformed TOC gzip header: %v", err) | ||||
| 	} | ||||
| 	zr.Multistream(false) | ||||
| 	tr := tar.NewReader(zr) | ||||
| 	h, err := tr.Next() | ||||
| 	if err != nil { | ||||
| 		return nil, fmt.Errorf("failed to find tar header in TOC gzip stream: %v", err) | ||||
| 	} | ||||
| 	if h.Name != TOCTarName { | ||||
| 		return nil, fmt.Errorf("TOC tar entry had name %q; expected %q", h.Name, TOCTarName) | ||||
| 	} | ||||
| 	return readCloser{tr, zr.Close}, nil | ||||
| } | ||||
|  |  | |||
|  | @ -624,8 +624,7 @@ func (i *Image) NamedRepoTags() ([]reference.Named, error) { | |||
| } | ||||
| 
 | ||||
| // inRepoTags looks for the specified name/tag pair in the image's repo tags.
 | ||||
| // Note that tag may be empty.
 | ||||
| func (i *Image) inRepoTags(name, tag string) (reference.Named, error) { | ||||
| func (i *Image) inRepoTags(namedTagged reference.NamedTagged) (reference.Named, error) { | ||||
| 	repoTags, err := i.NamedRepoTags() | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
|  | @ -636,8 +635,10 @@ func (i *Image) inRepoTags(name, tag string) (reference.Named, error) { | |||
| 		return nil, err | ||||
| 	} | ||||
| 
 | ||||
| 	name := namedTagged.Name() | ||||
| 	tag := namedTagged.Tag() | ||||
| 	for _, pair := range pairs { | ||||
| 		if tag != "" && tag != pair.Tag { | ||||
| 		if tag != pair.Tag { | ||||
| 			continue | ||||
| 		} | ||||
| 		if !strings.HasSuffix(pair.Name, name) { | ||||
|  |  | |||
|  | @ -389,16 +389,17 @@ func (r *Runtime) lookupImageInDigestsAndRepoTags(name string, options *LookupIm | |||
| 		return nil, "", err | ||||
| 	} | ||||
| 
 | ||||
| 	if !shortnames.IsShortName(name) { | ||||
| 		named, err := reference.ParseNormalizedNamed(name) | ||||
| 		if err != nil { | ||||
| 			return nil, "", err | ||||
| 		} | ||||
| 		digested, hasDigest := named.(reference.Digested) | ||||
| 		if !hasDigest { | ||||
| 			return nil, "", errors.Wrap(storage.ErrImageUnknown, name) | ||||
| 		} | ||||
| 	ref, err := reference.Parse(name) // Warning! This is not ParseNormalizedNamed
 | ||||
| 	if err != nil { | ||||
| 		return nil, "", err | ||||
| 	} | ||||
| 	named, isNamed := ref.(reference.Named) | ||||
| 	if !isNamed { | ||||
| 		return nil, "", errors.Wrap(storage.ErrImageUnknown, name) | ||||
| 	} | ||||
| 
 | ||||
| 	digested, isDigested := named.(reference.Digested) | ||||
| 	if isDigested { | ||||
| 		logrus.Debug("Looking for image with matching recorded digests") | ||||
| 		digest := digested.Digest() | ||||
| 		for _, image := range allImages { | ||||
|  | @ -408,22 +409,23 @@ func (r *Runtime) lookupImageInDigestsAndRepoTags(name string, options *LookupIm | |||
| 				} | ||||
| 			} | ||||
| 		} | ||||
| 
 | ||||
| 		return nil, "", errors.Wrap(storage.ErrImageUnknown, name) | ||||
| 	} | ||||
| 
 | ||||
| 	// Podman compat: if we're looking for a short name but couldn't
 | ||||
| 	// resolve it via the registries.conf dance, we need to look at *all*
 | ||||
| 	// images and check if the name we're looking for matches a repo tag.
 | ||||
| 	// Split the name into a repo/tag pair
 | ||||
| 	split := strings.SplitN(name, ":", 2) | ||||
| 	repo := split[0] | ||||
| 	tag := "" | ||||
| 	if len(split) == 2 { | ||||
| 		tag = split[1] | ||||
| 	if !shortnames.IsShortName(name) { | ||||
| 		return nil, "", errors.Wrap(storage.ErrImageUnknown, name) | ||||
| 	} | ||||
| 
 | ||||
| 	named = reference.TagNameOnly(named) // Make sure to add ":latest" if needed
 | ||||
| 	namedTagged, isNammedTagged := named.(reference.NamedTagged) | ||||
| 	if !isNammedTagged { | ||||
| 		// NOTE: this should never happen since we already know it's
 | ||||
| 		// not a digested reference.
 | ||||
| 		return nil, "", fmt.Errorf("%s: %w (could not cast to tagged)", name, storage.ErrImageUnknown) | ||||
| 	} | ||||
| 
 | ||||
| 	for _, image := range allImages { | ||||
| 		named, err := image.inRepoTags(repo, tag) | ||||
| 		named, err := image.inRepoTags(namedTagged) | ||||
| 		if err != nil { | ||||
| 			return nil, "", err | ||||
| 		} | ||||
|  |  | |||
|  | @ -215,6 +215,12 @@ type EngineConfig struct { | |||
| 	// The first path pointing to a valid file will be used.
 | ||||
| 	ConmonPath []string `toml:"conmon_path,omitempty"` | ||||
| 
 | ||||
| 	// CompatAPIEnforceDockerHub enforces using docker.io for completing
 | ||||
| 	// short names in Podman's compatibility REST API.  Note that this will
 | ||||
| 	// ignore unqualified-search-registries and short-name aliases defined
 | ||||
| 	// in containers-registries.conf(5).
 | ||||
| 	CompatAPIEnforceDockerHub bool `toml:"compat_api_enforce_docker_hub,omitempty"` | ||||
| 
 | ||||
| 	// DetachKeys is the sequence of keys used to detach a container.
 | ||||
| 	DetachKeys string `toml:"detach_keys,omitempty"` | ||||
| 
 | ||||
|  |  | |||
|  | @ -317,6 +317,11 @@ default_sysctls = [ | |||
| #  "/usr/local/sbin/conmon" | ||||
| #] | ||||
| 
 | ||||
| # Enforces using docker.io for completing short names in Podman's compatibility | ||||
| # REST API. Note that this will ignore unqualified-search-registries and | ||||
| # short-name aliases defined in containers-registries.conf(5). | ||||
| #compat_api_enforce_docker_hub = true | ||||
| 
 | ||||
| # Specify the keys sequence used to detach a container. | ||||
| # Format is a single character [a-Z] or a comma separated sequence of | ||||
| # `ctrl-<value>`, where `<value>` is one of: | ||||
|  |  | |||
|  | @ -190,6 +190,7 @@ func DefaultConfig() (*Config, error) { | |||
| 			IPCNS:              "private", | ||||
| 			LogDriver:          defaultLogDriver(), | ||||
| 			LogSizeMax:         DefaultLogSizeMax, | ||||
| 			NetNS:              "private", | ||||
| 			NoHosts:            false, | ||||
| 			PidsLimit:          DefaultPidsLimit, | ||||
| 			PidNS:              "private", | ||||
|  | @ -225,7 +226,7 @@ func defaultSecretConfig() SecretConfig { | |||
| func defaultMachineConfig() MachineConfig { | ||||
| 	return MachineConfig{ | ||||
| 		CPUs:     1, | ||||
| 		DiskSize: 10, | ||||
| 		DiskSize: 100, | ||||
| 		Image:    "testing", | ||||
| 		Memory:   2048, | ||||
| 	} | ||||
|  | @ -243,6 +244,8 @@ func defaultConfigFromMemory() (*EngineConfig, error) { | |||
| 
 | ||||
| 	c.EventsLogFilePath = filepath.Join(c.TmpDir, "events", "events.log") | ||||
| 
 | ||||
| 	c.CompatAPIEnforceDockerHub = true | ||||
| 
 | ||||
| 	if path, ok := os.LookupEnv("CONTAINERS_STORAGE_CONF"); ok { | ||||
| 		types.SetDefaultConfigFilePath(path) | ||||
| 	} | ||||
|  |  | |||
|  | @ -3,34 +3,44 @@ Package report provides helper structs/methods/funcs for formatting output | |||
| 
 | ||||
| To format output for an array of structs: | ||||
| 
 | ||||
| 	w := report.NewWriterDefault(os.Stdout) | ||||
| 	defer w.Flush() | ||||
| 
 | ||||
| ExamplePodman: | ||||
| 	headers := report.Headers(struct { | ||||
| 		ID string | ||||
| 	}{}, nil) | ||||
| 	t, _ := report.NewTemplate("command name").Parse("{{range .}}{{.ID}}{{end}}") | ||||
| 	t.Execute(t, headers) | ||||
| 	t.Execute(t, map[string]string{ | ||||
| 
 | ||||
| 	f := report.New(os.Stdout, "Command Name") | ||||
| 	f, _ := f.Parse(report.OriginPodman, "{{range .}}{{.ID}}{{end}}") | ||||
| 	defer f.Flush() | ||||
| 
 | ||||
| 	if f.RenderHeaders { | ||||
| 		f.Execute(headers) | ||||
| 	} | ||||
| 	f.Execute( map[string]string{ | ||||
| 		"ID":"fa85da03b40141899f3af3de6d27852b", | ||||
| 	}) | ||||
| 	// t.IsTable() == false
 | ||||
| 
 | ||||
| or | ||||
| 
 | ||||
| 	w := report.NewWriterDefault(os.Stdout) | ||||
| 	defer w.Flush() | ||||
| 	// Output:
 | ||||
| 	// ID
 | ||||
| 	// fa85da03b40141899f3af3de6d27852b
 | ||||
| 
 | ||||
| ExampleUser: | ||||
| 	headers := report.Headers(struct { | ||||
| 		CID string | ||||
| 	}{}, map[string]string{ | ||||
| 		"CID":"ID"}) | ||||
| 	t, _ := report.NewTemplate("command name").Parse("table {{.CID}}") | ||||
| 	t.Execute(t, headers) | ||||
| 	}{}, map[string]string{"CID":"ID"}) | ||||
| 
 | ||||
| 	f, _ := report.New(os.Stdout, "Command Name").Parse(report.OriginUser, "table {{.CID}}") | ||||
| 	defer f.Flush() | ||||
| 
 | ||||
| 	if f.RenderHeaders { | ||||
| 		t.Execute(t, headers) | ||||
| 	} | ||||
| 	t.Execute(t,map[string]string{ | ||||
| 		"CID":"fa85da03b40141899f3af3de6d27852b", | ||||
| 	}) | ||||
| 	// t.IsTable() == true
 | ||||
| 
 | ||||
| 	// Output:
 | ||||
| 	// ID
 | ||||
| 	// fa85da03b40141899f3af3de6d27852b
 | ||||
| 
 | ||||
| Helpers: | ||||
| 
 | ||||
|  | @ -38,13 +48,20 @@ Helpers: | |||
| 		... process JSON and output | ||||
| 	} | ||||
| 
 | ||||
| 	if report.HasTable(cmd.Flag("format").Value.String()) { | ||||
| 		... "table" keyword prefix in format text | ||||
| 	} | ||||
| 
 | ||||
| Template Functions: | ||||
| 
 | ||||
| The following template functions are added to the template when parsed: | ||||
| 	- join  strings.Join, {{join .Field separator}} | ||||
| 	- json encode field as JSON {{ json .Field }} | ||||
| 	- lower strings.ToLower {{ .Field | lower }} | ||||
| 	- pad add spaces as prefix and suffix {{ pad . 2 2 }} | ||||
| 	- split strings.Split {{ .Field | split }} | ||||
| 	- title strings.Title {{ .Field | title }} | ||||
| 	- truncate limit field length {{ truncate . 10 }} | ||||
| 	- upper strings.ToUpper {{ .Field | upper }} | ||||
| 
 | ||||
| report.Funcs() may be used to add additional template functions. | ||||
|  |  | |||
|  | @ -0,0 +1,151 @@ | |||
| package report | ||||
| 
 | ||||
| import ( | ||||
| 	"io" | ||||
| 	"strings" | ||||
| 	"text/tabwriter" | ||||
| 	"text/template" | ||||
| ) | ||||
| 
 | ||||
| // Flusher is the interface that wraps the Flush method.
 | ||||
| type Flusher interface { | ||||
| 	Flush() error | ||||
| } | ||||
| 
 | ||||
| // NopFlusher represents a type which flush operation is nop.
 | ||||
| type NopFlusher struct{} | ||||
| 
 | ||||
| // Flush is a nop operation.
 | ||||
| func (f *NopFlusher) Flush() (err error) { return } | ||||
| 
 | ||||
| type Origin int | ||||
| 
 | ||||
| const ( | ||||
| 	OriginUnknown Origin = iota | ||||
| 	OriginPodman | ||||
| 	OriginUser | ||||
| ) | ||||
| 
 | ||||
| func (o Origin) String() string { | ||||
| 	switch o { | ||||
| 	case OriginPodman: | ||||
| 		return "OriginPodman" | ||||
| 	case OriginUser: | ||||
| 		return "OriginUser" | ||||
| 	default: | ||||
| 		return "OriginUnknown" | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| // Formatter holds the configured Writer and parsed Template, additional state fields are
 | ||||
| // maintained to assist in the podman command report writing.
 | ||||
| type Formatter struct { | ||||
| 	Origin        Origin             // Source of go template. OriginUser or OriginPodman
 | ||||
| 	RenderHeaders bool               // Hint, default behavior for given template is to include headers
 | ||||
| 	RenderTable   bool               // Does template have "table" keyword
 | ||||
| 	flusher       Flusher            // Flush any buffered formatted output
 | ||||
| 	template      *template.Template // Go text/template for formatting output
 | ||||
| 	text          string             // value of canonical template after processing
 | ||||
| 	writer        io.Writer          // Destination for formatted output
 | ||||
| } | ||||
| 
 | ||||
| // Parse parses golang template returning a formatter
 | ||||
| //
 | ||||
| // - OriginPodman implies text is a template from podman code. Output will
 | ||||
| //   be filtered through a tabwriter.
 | ||||
| //
 | ||||
| // - OriginUser implies text is a template from a user. If template includes
 | ||||
| //   keyword "table" output will be filtered through a tabwriter.
 | ||||
| func (f *Formatter) Parse(origin Origin, text string) (*Formatter, error) { | ||||
| 	f.Origin = origin | ||||
| 
 | ||||
| 	switch { | ||||
| 	case strings.HasPrefix(text, "table "): | ||||
| 		f.RenderTable = true | ||||
| 		text = "{{range .}}" + NormalizeFormat(text) + "{{end -}}" | ||||
| 	case OriginUser == origin: | ||||
| 		text = EnforceRange(NormalizeFormat(text)) | ||||
| 	default: | ||||
| 		text = NormalizeFormat(text) | ||||
| 	} | ||||
| 	f.text = text | ||||
| 
 | ||||
| 	if f.RenderTable || origin == OriginPodman { | ||||
| 		tw := tabwriter.NewWriter(f.writer, 12, 2, 2, ' ', tabwriter.StripEscape) | ||||
| 		f.writer = tw | ||||
| 		f.flusher = tw | ||||
| 		f.RenderHeaders = true | ||||
| 	} | ||||
| 
 | ||||
| 	tmpl, err := f.template.Funcs(template.FuncMap(DefaultFuncs)).Parse(text) | ||||
| 	if err != nil { | ||||
| 		return f, err | ||||
| 	} | ||||
| 	f.template = tmpl | ||||
| 	return f, nil | ||||
| } | ||||
| 
 | ||||
| // Funcs adds the elements of the argument map to the template's function map.
 | ||||
| // A default template function will be replaced if there is a key collision.
 | ||||
| func (f *Formatter) Funcs(funcMap template.FuncMap) *Formatter { | ||||
| 	m := make(template.FuncMap, len(DefaultFuncs)+len(funcMap)) | ||||
| 	for k, v := range DefaultFuncs { | ||||
| 		m[k] = v | ||||
| 	} | ||||
| 	for k, v := range funcMap { | ||||
| 		m[k] = v | ||||
| 	} | ||||
| 	f.template = f.template.Funcs(funcMap) | ||||
| 	return f | ||||
| } | ||||
| 
 | ||||
| // Init either resets the given tabwriter with new values or wraps w in tabwriter with given values
 | ||||
| func (f *Formatter) Init(w io.Writer, minwidth, tabwidth, padding int, padchar byte, flags uint) *Formatter { | ||||
| 	flags |= tabwriter.StripEscape | ||||
| 
 | ||||
| 	if tw, ok := f.writer.(*tabwriter.Writer); ok { | ||||
| 		tw = tw.Init(w, minwidth, tabwidth, padding, padchar, flags) | ||||
| 		f.writer = tw | ||||
| 		f.flusher = tw | ||||
| 	} else { | ||||
| 		tw = tabwriter.NewWriter(w, minwidth, tabwidth, padding, padchar, flags) | ||||
| 		f.writer = tw | ||||
| 		f.flusher = tw | ||||
| 	} | ||||
| 	return f | ||||
| } | ||||
| 
 | ||||
| // Execute applies a parsed template to the specified data object,
 | ||||
| // and writes the output to Formatter.Writer.
 | ||||
| func (f *Formatter) Execute(data interface{}) error { | ||||
| 	return f.template.Execute(f.writer, data) | ||||
| } | ||||
| 
 | ||||
| // Flush should be called after the last call to Write to ensure
 | ||||
| // that any data buffered in the Formatter is written to output. Any
 | ||||
| // incomplete escape sequence at the end is considered
 | ||||
| // complete for formatting purposes.
 | ||||
| func (f *Formatter) Flush() error { | ||||
| 	// Indirection is required here to prevent caller from having to know when
 | ||||
| 	// value of Flusher may be changed.
 | ||||
| 	return f.flusher.Flush() | ||||
| } | ||||
| 
 | ||||
| // Writer returns the embedded io.Writer from Formatter
 | ||||
| func (f *Formatter) Writer() io.Writer { | ||||
| 	return f.writer | ||||
| } | ||||
| 
 | ||||
| // New allocates a new, undefined Formatter with the given name and Writer
 | ||||
| func New(output io.Writer, name string) *Formatter { | ||||
| 	f := new(Formatter) | ||||
| 
 | ||||
| 	f.flusher = new(NopFlusher) | ||||
| 	if flusher, ok := output.(Flusher); ok { | ||||
| 		f.flusher = flusher | ||||
| 	} | ||||
| 
 | ||||
| 	f.template = template.New(name) | ||||
| 	f.writer = output | ||||
| 	return f | ||||
| } | ||||
|  | @ -25,17 +25,19 @@ var tableReplacer = strings.NewReplacer( | |||
| 	"table ", "", | ||||
| 	`\t`, "\t", | ||||
| 	" ", "\t", | ||||
| 	`\n`, "\n", | ||||
| ) | ||||
| 
 | ||||
| // escapedReplacer will clean up escaped characters from CLI
 | ||||
| var escapedReplacer = strings.NewReplacer( | ||||
| 	`\t`, "\t", | ||||
| 	`\n`, "\n", | ||||
| ) | ||||
| 
 | ||||
| var DefaultFuncs = FuncMap{ | ||||
| 	"join": strings.Join, | ||||
| 	"json": func(v interface{}) string { | ||||
| 		buf := &bytes.Buffer{} | ||||
| 		buf := new(bytes.Buffer) | ||||
| 		enc := json.NewEncoder(buf) | ||||
| 		enc.SetEscapeHTML(false) | ||||
| 		enc.Encode(v) | ||||
|  | @ -157,7 +159,7 @@ func (t *Template) IsTable() bool { | |||
| 	return t.isTable | ||||
| } | ||||
| 
 | ||||
| var rangeRegex = regexp.MustCompile(`{{\s*range\s*\.\s*}}.*{{\s*end\s*-?\s*}}`) | ||||
| var rangeRegex = regexp.MustCompile(`(?s){{\s*range\s*\.\s*}}.*{{\s*end\s*-?\s*}}`) | ||||
| 
 | ||||
| // EnforceRange ensures that the format string contains a range
 | ||||
| func EnforceRange(format string) string { | ||||
|  |  | |||
|  | @ -155,6 +155,15 @@ func hasMetacopyOption(opts []string) bool { | |||
| 	return false | ||||
| } | ||||
| 
 | ||||
| func stripOption(opts []string, option string) []string { | ||||
| 	for i, s := range opts { | ||||
| 		if s == option { | ||||
| 			return stripOption(append(opts[:i], opts[i+1:]...), option) | ||||
| 		} | ||||
| 	} | ||||
| 	return opts | ||||
| } | ||||
| 
 | ||||
| func hasVolatileOption(opts []string) bool { | ||||
| 	for _, s := range opts { | ||||
| 		if s == "volatile" { | ||||
|  | @ -881,11 +890,18 @@ func (d *Driver) create(id, parent string, opts *graphdriver.CreateOpts, disable | |||
| 	if err != nil { | ||||
| 		return err | ||||
| 	} | ||||
| 
 | ||||
| 	idPair := idtools.IDPair{ | ||||
| 		UID: rootUID, | ||||
| 		GID: rootGID, | ||||
| 	} | ||||
| 
 | ||||
| 	// Make the link directory if it does not exist
 | ||||
| 	if err := idtools.MkdirAllAs(path.Join(d.home, linkDir), 0700, rootUID, rootGID); err != nil { | ||||
| 	if err := idtools.MkdirAllAndChownNew(path.Join(d.home, linkDir), 0700, idPair); err != nil { | ||||
| 		return err | ||||
| 	} | ||||
| 	if err := idtools.MkdirAllAs(path.Dir(dir), 0700, rootUID, rootGID); err != nil { | ||||
| 
 | ||||
| 	if err := idtools.MkdirAllAndChownNew(path.Dir(dir), 0700, idPair); err != nil { | ||||
| 		return err | ||||
| 	} | ||||
| 	if parent != "" { | ||||
|  | @ -896,7 +912,7 @@ func (d *Driver) create(id, parent string, opts *graphdriver.CreateOpts, disable | |||
| 		rootUID = int(st.UID()) | ||||
| 		rootGID = int(st.GID()) | ||||
| 	} | ||||
| 	if err := idtools.MkdirAs(dir, 0700, rootUID, rootGID); err != nil { | ||||
| 	if err := idtools.MkdirAllAndChownNew(dir, 0700, idPair); err != nil { | ||||
| 		return err | ||||
| 	} | ||||
| 
 | ||||
|  | @ -1175,7 +1191,7 @@ func (d *Driver) recreateSymlinks() error { | |||
| 			// Read the "link" file under each layer to get the name of the symlink
 | ||||
| 			data, err := ioutil.ReadFile(path.Join(d.dir(dir.Name()), "link")) | ||||
| 			if err != nil { | ||||
| 				errs = multierror.Append(errs, errors.Wrapf(err, "reading name of symlink for %q", dir)) | ||||
| 				errs = multierror.Append(errs, errors.Wrapf(err, "reading name of symlink for %q", dir.Name())) | ||||
| 				continue | ||||
| 			} | ||||
| 			linkPath := path.Join(d.home, linkDir, strings.Trim(string(data), "\n")) | ||||
|  | @ -1254,6 +1270,10 @@ func (d *Driver) get(id string, disableShifting bool, options graphdriver.MountO | |||
| 		disableShifting = true | ||||
| 	} | ||||
| 
 | ||||
| 	logLevel := logrus.WarnLevel | ||||
| 	if unshare.IsRootless() { | ||||
| 		logLevel = logrus.DebugLevel | ||||
| 	} | ||||
| 	optsList := options.Options | ||||
| 	if len(optsList) == 0 { | ||||
| 		optsList = strings.Split(d.options.mountOptions, ",") | ||||
|  | @ -1262,16 +1282,18 @@ func (d *Driver) get(id string, disableShifting bool, options graphdriver.MountO | |||
| 		// options otherwise the kernel refuses to follow the metacopy xattr.
 | ||||
| 		if hasMetacopyOption(strings.Split(d.options.mountOptions, ",")) && !hasMetacopyOption(options.Options) { | ||||
| 			if d.usingMetacopy { | ||||
| 				logrus.StandardLogger().Logf(logrus.DebugLevel, "Adding metacopy option, configured globally") | ||||
| 				optsList = append(optsList, "metacopy=on") | ||||
| 			} else { | ||||
| 				logLevel := logrus.WarnLevel | ||||
| 				if unshare.IsRootless() { | ||||
| 					logLevel = logrus.DebugLevel | ||||
| 				} | ||||
| 				logrus.StandardLogger().Logf(logLevel, "Ignoring metacopy option from storage.conf, not supported with booted kernel") | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
| 	if !d.usingMetacopy { | ||||
| 		if hasMetacopyOption(optsList) { | ||||
| 			logrus.StandardLogger().Logf(logLevel, "Ignoring global metacopy option, not supported with booted kernel") | ||||
| 		} | ||||
| 		optsList = stripOption(optsList, "metacopy=on") | ||||
| 	} | ||||
| 
 | ||||
| 	for _, o := range optsList { | ||||
| 		if o == "ro" { | ||||
| 			readWrite = false | ||||
|  |  | |||
|  | @ -4,9 +4,9 @@ module github.com/containers/storage | |||
| 
 | ||||
| require ( | ||||
| 	github.com/BurntSushi/toml v0.4.1 | ||||
| 	github.com/Microsoft/go-winio v0.5.0 | ||||
| 	github.com/Microsoft/hcsshim v0.9.0 | ||||
| 	github.com/containerd/stargz-snapshotter/estargz v0.9.0 | ||||
| 	github.com/Microsoft/go-winio v0.5.1 | ||||
| 	github.com/Microsoft/hcsshim v0.9.1 | ||||
| 	github.com/containerd/stargz-snapshotter/estargz v0.10.1 | ||||
| 	github.com/docker/go-units v0.4.0 | ||||
| 	github.com/google/go-intervals v0.0.2 | ||||
| 	github.com/hashicorp/go-multierror v1.1.1 | ||||
|  | @ -15,11 +15,11 @@ require ( | |||
| 	github.com/klauspost/pgzip v1.2.5 | ||||
| 	github.com/mattn/go-shellwords v1.0.12 | ||||
| 	github.com/mistifyio/go-zfs v2.1.2-0.20190413222219-f784269be439+incompatible | ||||
| 	github.com/moby/sys/mountinfo v0.4.1 | ||||
| 	github.com/moby/sys/mountinfo v0.5.0 | ||||
| 	github.com/opencontainers/go-digest v1.0.0 | ||||
| 	github.com/opencontainers/runc v1.0.2 | ||||
| 	github.com/opencontainers/runtime-spec v1.0.3-0.20210326190908-1c3f411f0417 | ||||
| 	github.com/opencontainers/selinux v1.9.1 | ||||
| 	github.com/opencontainers/selinux v1.10.0 | ||||
| 	github.com/pkg/errors v0.9.1 | ||||
| 	github.com/sirupsen/logrus v1.8.1 | ||||
| 	github.com/stretchr/testify v1.7.0 | ||||
|  | @ -28,6 +28,6 @@ require ( | |||
| 	github.com/ulikunitz/xz v0.5.10 | ||||
| 	github.com/vbatts/tar-split v0.11.2 | ||||
| 	golang.org/x/net v0.0.0-20210825183410-e898025ed96a | ||||
| 	golang.org/x/sys v0.0.0-20210820121016-41cdb8703e55 | ||||
| 	golang.org/x/sys v0.0.0-20211025201205-69cdffdb9359 | ||||
| 	gotest.tools v2.2.0+incompatible | ||||
| ) | ||||
|  |  | |||
|  | @ -47,8 +47,8 @@ github.com/Microsoft/go-winio v0.4.16/go.mod h1:XB6nPKklQyQ7GC9LdcBEcBl8PF76WugX | |||
| github.com/Microsoft/go-winio v0.4.17-0.20210211115548-6eac466e5fa3/go.mod h1:JPGBdM1cNvN/6ISo+n8V5iA4v8pBzdOpzfwIujj1a84= | ||||
| github.com/Microsoft/go-winio v0.4.17-0.20210324224401-5516f17a5958/go.mod h1:JPGBdM1cNvN/6ISo+n8V5iA4v8pBzdOpzfwIujj1a84= | ||||
| github.com/Microsoft/go-winio v0.4.17/go.mod h1:JPGBdM1cNvN/6ISo+n8V5iA4v8pBzdOpzfwIujj1a84= | ||||
| github.com/Microsoft/go-winio v0.5.0 h1:Elr9Wn+sGKPlkaBvwu4mTrxtmOp3F3yV9qhaHbXGjwU= | ||||
| github.com/Microsoft/go-winio v0.5.0/go.mod h1:JPGBdM1cNvN/6ISo+n8V5iA4v8pBzdOpzfwIujj1a84= | ||||
| github.com/Microsoft/go-winio v0.5.1 h1:aPJp2QD7OOrhO5tQXqQoGSJc+DjDtWTGLOmNyAm6FgY= | ||||
| github.com/Microsoft/go-winio v0.5.1/go.mod h1:JPGBdM1cNvN/6ISo+n8V5iA4v8pBzdOpzfwIujj1a84= | ||||
| github.com/Microsoft/hcsshim v0.8.6/go.mod h1:Op3hHsoHPAvb6lceZHDtd9OkTew38wNoXnJs8iY7rUg= | ||||
| github.com/Microsoft/hcsshim v0.8.7-0.20190325164909-8abdbb8205e4/go.mod h1:Op3hHsoHPAvb6lceZHDtd9OkTew38wNoXnJs8iY7rUg= | ||||
| github.com/Microsoft/hcsshim v0.8.7/go.mod h1:OHd7sQqRFrYd3RmSgbgji+ctCwkbq2wbEYNSzOYtcBQ= | ||||
|  | @ -57,8 +57,8 @@ github.com/Microsoft/hcsshim v0.8.14/go.mod h1:NtVKoYxQuTLx6gEq0L96c9Ju4JbRJ4nY2 | |||
| github.com/Microsoft/hcsshim v0.8.15/go.mod h1:x38A4YbHbdxJtc0sF6oIz+RG0npwSCAvn69iY6URG00= | ||||
| github.com/Microsoft/hcsshim v0.8.16/go.mod h1:o5/SZqmR7x9JNKsW3pu+nqHm0MF8vbA+VxGOoXdC600= | ||||
| github.com/Microsoft/hcsshim v0.8.21/go.mod h1:+w2gRZ5ReXQhFOrvSQeNfhrYB/dg3oDwTOcER2fw4I4= | ||||
| github.com/Microsoft/hcsshim v0.9.0 h1:BBgYMxl5YZDZVIijz02AlDINpYZOzQqRNCl9CZM13vk= | ||||
| github.com/Microsoft/hcsshim v0.9.0/go.mod h1:VBJWdC71NSWPlEo7lwde1aL21748J8B6Sdgno7NqEGE= | ||||
| github.com/Microsoft/hcsshim v0.9.1 h1:VfDCj+QnY19ktX5TsH22JHcjaZ05RWQiwDbOyEg5ziM= | ||||
| github.com/Microsoft/hcsshim v0.9.1/go.mod h1:Y/0uV2jUab5kBI7SQgl62at0AVX7uaruzADAVmxm3eM= | ||||
| github.com/Microsoft/hcsshim/test v0.0.0-20201218223536-d3e5debf77da/go.mod h1:5hlzMzRKMLyo42nCZ9oml8AdTlq/0cvIaBv6tK1RehU= | ||||
| github.com/Microsoft/hcsshim/test v0.0.0-20210227013316-43a75bb4edd3/go.mod h1:mw7qgWloBUl75W/gVH3cQszUg1+gUITj7D6NY7ywVnY= | ||||
| github.com/NYTimes/gziphandler v0.0.0-20170623195520-56545f4a5d46/go.mod h1:3wb06e3pkSAbeQ52E9H9iFoQsEEwGN64994WTCIhntQ= | ||||
|  | @ -92,6 +92,7 @@ github.com/buger/jsonparser v0.0.0-20180808090653-f4dd9f5a6b44/go.mod h1:bbYlZJ7 | |||
| github.com/bugsnag/bugsnag-go v0.0.0-20141110184014-b1d153021fcd/go.mod h1:2oa8nejYd4cQ/b0hMIopN0lCRxU0bueqREvZLWFrtK8= | ||||
| github.com/bugsnag/osext v0.0.0-20130617224835-0dd3f918b21b/go.mod h1:obH5gd0BsqsP2LwDJ9aOkm/6J86V6lyAXCoQWGw3K50= | ||||
| github.com/bugsnag/panicwrap v0.0.0-20151223152923-e2c28503fcd0/go.mod h1:D/8v3kj0zr8ZAKg1AQ6crr+5VwKN5eIywRkfhyM/+dE= | ||||
| github.com/cenkalti/backoff/v4 v4.1.1/go.mod h1:scbssz8iZGpm3xbr14ovlUdkxfGXNInqkPWOWmG2CLw= | ||||
| github.com/census-instrumentation/opencensus-proto v0.2.1/go.mod h1:f6KPmirojxKA12rnyqOA5BBL4O983OfeGPqjHWSTneU= | ||||
| github.com/cespare/xxhash v1.1.0/go.mod h1:XrSqR1VqqWfGrhpAt58auRo0WTKS1nRRg3ghfAqPWnc= | ||||
| github.com/cespare/xxhash/v2 v2.1.1/go.mod h1:VGX0DQ3Q6kWi7AoAeZDth3/j3BFtOZR5XLFGgcrjCOs= | ||||
|  | @ -172,13 +173,14 @@ github.com/containerd/nri v0.0.0-20201007170849-eb1350a75164/go.mod h1:+2wGSDGFY | |||
| github.com/containerd/nri v0.0.0-20210316161719-dbaa18c31c14/go.mod h1:lmxnXF6oMkbqs39FiCt1s0R2HSMhcLel9vNL3m4AaeY= | ||||
| github.com/containerd/nri v0.1.0/go.mod h1:lmxnXF6oMkbqs39FiCt1s0R2HSMhcLel9vNL3m4AaeY= | ||||
| github.com/containerd/stargz-snapshotter/estargz v0.4.1/go.mod h1:x7Q9dg9QYb4+ELgxmo4gBUeJB0tl5dqH1Sdz0nJU1QM= | ||||
| github.com/containerd/stargz-snapshotter/estargz v0.9.0 h1:PkB6BSTfOKX23erT2GkoUKkJEcXfNcyKskIViK770v8= | ||||
| github.com/containerd/stargz-snapshotter/estargz v0.9.0/go.mod h1:aE5PCyhFMwR8sbrErO5eM2GcvkyXTTJremG883D4qF0= | ||||
| github.com/containerd/stargz-snapshotter/estargz v0.10.1 h1:hd1EoVjI2Ax8Cr64tdYqnJ4i4pZU49FkEf5kU8KxQng= | ||||
| github.com/containerd/stargz-snapshotter/estargz v0.10.1/go.mod h1:aE5PCyhFMwR8sbrErO5eM2GcvkyXTTJremG883D4qF0= | ||||
| github.com/containerd/ttrpc v0.0.0-20190828154514-0e0f228740de/go.mod h1:PvCDdDGpgqzQIzDW1TphrGLssLDZp2GuS+X5DkEJB8o= | ||||
| github.com/containerd/ttrpc v0.0.0-20190828172938-92c8520ef9f8/go.mod h1:PvCDdDGpgqzQIzDW1TphrGLssLDZp2GuS+X5DkEJB8o= | ||||
| github.com/containerd/ttrpc v0.0.0-20191028202541-4f1b8fe65a5c/go.mod h1:LPm1u0xBw8r8NOKoOdNMeVHSawSsltak+Ihv+etqsE8= | ||||
| github.com/containerd/ttrpc v1.0.1/go.mod h1:UAxOpgT9ziI0gJrmKvgcZivgxOp8iFPSk8httJEt98Y= | ||||
| github.com/containerd/ttrpc v1.0.2/go.mod h1:UAxOpgT9ziI0gJrmKvgcZivgxOp8iFPSk8httJEt98Y= | ||||
| github.com/containerd/ttrpc v1.1.0/go.mod h1:XX4ZTnoOId4HklF4edwc4DcqskFZuvXB1Evzy5KFQpQ= | ||||
| github.com/containerd/typeurl v0.0.0-20180627222232-a93fcdb778cd/go.mod h1:Cm3kwCdlkCfMSHURc+r6fwoGH6/F1hH3S4sg0rLFWPc= | ||||
| github.com/containerd/typeurl v0.0.0-20190911142611-5eb25027c9fd/go.mod h1:GeKYzf2pQcqv7tJ0AoCuuhtnqhva5LNU3U+OyKxxJpk= | ||||
| github.com/containerd/typeurl v1.0.1/go.mod h1:TB1hUtrpaiO88KEK56ijojHS1+NeF0izUACaJW2mdXg= | ||||
|  | @ -457,8 +459,9 @@ github.com/mitchellh/mapstructure v1.1.2/go.mod h1:FVVH3fgwuzCH5S8UJGiWEs2h04kUh | |||
| github.com/mitchellh/osext v0.0.0-20151018003038-5e2d6d41470f/go.mod h1:OkQIRizQZAeMln+1tSwduZz7+Af5oFlKirV/MSYes2A= | ||||
| github.com/moby/locker v1.0.1/go.mod h1:S7SDdo5zpBK84bzzVlKr2V0hz+7x9hWbYC/kq7oQppc= | ||||
| github.com/moby/sys/mountinfo v0.4.0/go.mod h1:rEr8tzG/lsIZHBtN/JjGG+LMYx9eXgW2JI+6q0qou+A= | ||||
| github.com/moby/sys/mountinfo v0.4.1 h1:1O+1cHA1aujwEwwVMa2Xm2l+gIpUHyd3+D+d7LZh1kM= | ||||
| github.com/moby/sys/mountinfo v0.4.1/go.mod h1:rEr8tzG/lsIZHBtN/JjGG+LMYx9eXgW2JI+6q0qou+A= | ||||
| github.com/moby/sys/mountinfo v0.5.0 h1:2Ks8/r6lopsxWi9m58nlwjaeSzUX9iiL1vj5qB/9ObI= | ||||
| github.com/moby/sys/mountinfo v0.5.0/go.mod h1:3bMD3Rg+zkqx8MRYPi7Pyb0Ie97QEBmdxbhnCLlSvSU= | ||||
| github.com/moby/sys/symlink v0.1.0/go.mod h1:GGDODQmbFOjFsXvfLVn3+ZRxkch54RkSiGqsZeMYowQ= | ||||
| github.com/moby/term v0.0.0-20200312100748-672ec06f55cd/go.mod h1:DdlQx2hp0Ss5/fLikoLlEeIYiATotOjgB//nb973jeo= | ||||
| github.com/modern-go/concurrent v0.0.0-20180228061459-e0a39a4cb421/go.mod h1:6dJC0mAP4ikYIbvyc7fijjWJddQyLn8Ig3JB5CqoB9Q= | ||||
|  | @ -522,8 +525,8 @@ github.com/opencontainers/runtime-tools v0.0.0-20181011054405-1d69bd0f9c39/go.mo | |||
| github.com/opencontainers/selinux v1.6.0/go.mod h1:VVGKuOLlE7v4PJyT6h7mNWvq1rzqiriPsEqVhc+svHE= | ||||
| github.com/opencontainers/selinux v1.8.0/go.mod h1:RScLhm78qiWa2gbVCcGkC7tCGdgk3ogry1nUQF8Evvo= | ||||
| github.com/opencontainers/selinux v1.8.2/go.mod h1:MUIHuUEvKB1wtJjQdOyYRgOnLD2xAPP8dBsCoU0KuF8= | ||||
| github.com/opencontainers/selinux v1.9.1 h1:b4VPEF3O5JLZgdTDBmGepaaIbAo0GqoF6EBRq5f/g3Y= | ||||
| github.com/opencontainers/selinux v1.9.1/go.mod h1:2i0OySw99QjzBBQByd1Gr9gSjvuho1lHsJxIJ3gGbJI= | ||||
| github.com/opencontainers/selinux v1.10.0 h1:rAiKF8hTcgLI3w0DHm6i0ylVVcOrlgR1kK99DRLDhyU= | ||||
| github.com/opencontainers/selinux v1.10.0/go.mod h1:2i0OySw99QjzBBQByd1Gr9gSjvuho1lHsJxIJ3gGbJI= | ||||
| github.com/pelletier/go-toml v1.2.0/go.mod h1:5z9KED0ma1S8pY6P1sdut58dfprrGBbd/94hg7ilaic= | ||||
| github.com/pelletier/go-toml v1.8.1/go.mod h1:T2/BmBdy8dvIRq1a/8aqjN41wvWlN4lrapLU/GW4pbc= | ||||
| github.com/peterbourgon/diskv v2.0.1+incompatible/go.mod h1:uqqh8zWWbv1HBMNONnaR/tNboyR3/BZd58JJSHlUSCU= | ||||
|  | @ -833,8 +836,8 @@ golang.org/x/sys v0.0.0-20210330210617-4fbd30eecc44/go.mod h1:h1NjWce9XRLGQEsW7w | |||
| golang.org/x/sys v0.0.0-20210423082822-04245dca01da/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20210426230700-d19ff857e887/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20210510120138-977fb7262007/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= | ||||
| golang.org/x/sys v0.0.0-20210820121016-41cdb8703e55 h1:rw6UNGRMfarCepjI8qOepea/SXwIBVfTKjztZ5gBbq4= | ||||
| golang.org/x/sys v0.0.0-20210820121016-41cdb8703e55/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= | ||||
| golang.org/x/sys v0.0.0-20211025201205-69cdffdb9359 h1:2B5p2L5IfGiD7+b9BOoRMC6DgObAVZV+Fsp050NqXik= | ||||
| golang.org/x/sys v0.0.0-20211025201205-69cdffdb9359/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= | ||||
| golang.org/x/term v0.0.0-20201126162022-7de9c90e9dd1/go.mod h1:bj7SfCRtBDWHUb9snDiAeCFNEtKQo2Wmx5Cou7ajbmo= | ||||
| 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= | ||||
|  | @ -972,6 +975,7 @@ google.golang.org/protobuf v1.24.0/go.mod h1:r/3tXBNzIEhYS9I1OUVjXDlt8tc493IdKGj | |||
| 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.27.1/go.mod h1:9q0QmTI4eRPtz6boOQmLYwt+qCgq0jsYwAQnmE0givc= | ||||
| gopkg.in/airbrake/gobrake.v2 v2.0.9/go.mod h1:/h5ZAUhDkGaJfjzjKLSjv6zCL6O0LLBxU4K+aSYdM/U= | ||||
| gopkg.in/alecthomas/kingpin.v2 v2.2.6/go.mod h1:FMv+mEhP44yOT+4EoQTLFTRgOQ1FBLkstjWtayDeSgw= | ||||
| gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= | ||||
|  |  | |||
|  | @ -67,6 +67,24 @@ func timeToTimespec(time time.Time) (ts unix.Timespec) { | |||
| 	return unix.NsecToTimespec(time.UnixNano()) | ||||
| } | ||||
| 
 | ||||
| func doHardLink(srcFd int, destDirFd int, destBase string) error { | ||||
| 	doLink := func() error { | ||||
| 		// Using unix.AT_EMPTY_PATH requires CAP_DAC_READ_SEARCH while this variant that uses
 | ||||
| 		// /proc/self/fd doesn't and can be used with rootless.
 | ||||
| 		srcPath := fmt.Sprintf("/proc/self/fd/%d", srcFd) | ||||
| 		return unix.Linkat(unix.AT_FDCWD, srcPath, destDirFd, destBase, unix.AT_SYMLINK_FOLLOW) | ||||
| 	} | ||||
| 
 | ||||
| 	err := doLink() | ||||
| 
 | ||||
| 	// if the destination exists, unlink it first and try again
 | ||||
| 	if err != nil && os.IsExist(err) { | ||||
| 		unix.Unlinkat(destDirFd, destBase, 0) | ||||
| 		return doLink() | ||||
| 	} | ||||
| 	return err | ||||
| } | ||||
| 
 | ||||
| func copyFileContent(srcFd int, destFile string, dirfd int, mode os.FileMode, useHardLinks bool) (*os.File, int64, error) { | ||||
| 	src := fmt.Sprintf("/proc/self/fd/%d", srcFd) | ||||
| 	st, err := os.Stat(src) | ||||
|  | @ -83,20 +101,7 @@ func copyFileContent(srcFd int, destFile string, dirfd int, mode os.FileMode, us | |||
| 		if err == nil { | ||||
| 			defer destDir.Close() | ||||
| 
 | ||||
| 			doLink := func() error { | ||||
| 				// Using unix.AT_EMPTY_PATH requires CAP_DAC_READ_SEARCH while this variant that uses
 | ||||
| 				// /proc/self/fd doesn't and can be used with rootless.
 | ||||
| 				srcPath := fmt.Sprintf("/proc/self/fd/%d", srcFd) | ||||
| 				return unix.Linkat(unix.AT_FDCWD, srcPath, int(destDir.Fd()), destBase, unix.AT_SYMLINK_FOLLOW) | ||||
| 			} | ||||
| 
 | ||||
| 			err := doLink() | ||||
| 
 | ||||
| 			// if the destination exists, unlink it first and try again
 | ||||
| 			if err != nil && os.IsExist(err) { | ||||
| 				unix.Unlinkat(int(destDir.Fd()), destBase, 0) | ||||
| 				err = doLink() | ||||
| 			} | ||||
| 			err := doHardLink(srcFd, int(destDir.Fd()), destBase) | ||||
| 			if err == nil { | ||||
| 				return nil, st.Size(), nil | ||||
| 			} | ||||
|  | @ -797,7 +802,7 @@ func safeLink(dirfd int, mode os.FileMode, metadata *internal.FileMetadata, opti | |||
| 		destDirFd = int(f.Fd()) | ||||
| 	} | ||||
| 
 | ||||
| 	err = unix.Linkat(int(sourceFile.Fd()), "", destDirFd, destBase, unix.AT_EMPTY_PATH) | ||||
| 	err = doHardLink(int(sourceFile.Fd()), destDirFd, destBase) | ||||
| 	if err != nil { | ||||
| 		return err | ||||
| 	} | ||||
|  | @ -861,7 +866,7 @@ func (d whiteoutHandler) Mknod(path string, mode uint32, dev int) error { | |||
| 
 | ||||
| func checkChownErr(err error, name string, uid, gid int) error { | ||||
| 	if errors.Is(err, syscall.EINVAL) { | ||||
| 		return errors.Wrapf(err, "potentially insufficient UIDs or GIDs available in user namespace (requested %d:%d for %s): Check /etc/subuid and /etc/subgid", uid, gid, name) | ||||
| 		return errors.Wrapf(err, "potentially insufficient UIDs or GIDs available in user namespace (requested %d:%d for %s): Check /etc/subuid and /etc/subgid if configured locally", uid, gid, name) | ||||
| 	} | ||||
| 	return err | ||||
| } | ||||
|  |  | |||
|  | @ -185,18 +185,14 @@ func (i *IDMappings) RootPair() IDPair { | |||
| // Remapping is only performed if the ids aren't already the remapped root ids
 | ||||
| func (i *IDMappings) ToHost(pair IDPair) (IDPair, error) { | ||||
| 	var err error | ||||
| 	target := i.RootPair() | ||||
| 	var target IDPair | ||||
| 
 | ||||
| 	if pair.UID != target.UID { | ||||
| 		target.UID, err = toHost(pair.UID, i.uids) | ||||
| 		if err != nil { | ||||
| 			return target, err | ||||
| 		} | ||||
| 	target.UID, err = toHost(pair.UID, i.uids) | ||||
| 	if err != nil { | ||||
| 		return target, err | ||||
| 	} | ||||
| 
 | ||||
| 	if pair.GID != target.GID { | ||||
| 		target.GID, err = toHost(pair.GID, i.gids) | ||||
| 	} | ||||
| 	target.GID, err = toHost(pair.GID, i.gids) | ||||
| 	return target, err | ||||
| } | ||||
| 
 | ||||
|  | @ -293,7 +289,7 @@ func parseSubidFile(path, username string) (ranges, error) { | |||
| 
 | ||||
| func checkChownErr(err error, name string, uid, gid int) error { | ||||
| 	if e, ok := err.(*os.PathError); ok && e.Err == syscall.EINVAL { | ||||
| 		return errors.Wrapf(err, "potentially insufficient UIDs or GIDs available in user namespace (requested %d:%d for %s): Check /etc/subuid and /etc/subgid", uid, gid, name) | ||||
| 		return errors.Wrapf(err, "potentially insufficient UIDs or GIDs available in user namespace (requested %d:%d for %s): Check /etc/subuid and /etc/subgid if configured locally", uid, gid, name) | ||||
| 	} | ||||
| 	return err | ||||
| } | ||||
|  |  | |||
|  | @ -46,6 +46,9 @@ func mkdirAs(path string, mode os.FileMode, ownerUID, ownerGID int, mkAll, chown | |||
| 		// walk back to "/" looking for directories which do not exist
 | ||||
| 		// and add them to the paths array for chown after creation
 | ||||
| 		dirPath := path | ||||
| 		if !filepath.IsAbs(dirPath) { | ||||
| 			return fmt.Errorf("path: %s should be absolute", dirPath) | ||||
| 		} | ||||
| 		for { | ||||
| 			dirPath = filepath.Dir(dirPath) | ||||
| 			if dirPath == "/" { | ||||
|  |  | |||
|  | @ -17,6 +17,7 @@ func Self() string { | |||
| // This will use the in-memory version (/proc/self/exe) of the current binary,
 | ||||
| // it is thus safe to delete or replace the on-disk binary (os.Args[0]).
 | ||||
| func Command(args ...string) *exec.Cmd { | ||||
| 	panicIfNotInitialized() | ||||
| 	cmd := exec.Command(Self()) | ||||
| 	cmd.Args = args | ||||
| 	return cmd | ||||
|  | @ -26,6 +27,7 @@ func Command(args ...string) *exec.Cmd { | |||
| // This will use the in-memory version (/proc/self/exe) of the current binary,
 | ||||
| // it is thus safe to delete or replace the on-disk binary (os.Args[0]).
 | ||||
| func CommandContext(ctx context.Context, args ...string) *exec.Cmd { | ||||
| 	panicIfNotInitialized() | ||||
| 	cmd := exec.CommandContext(ctx, Self()) | ||||
| 	cmd.Args = args | ||||
| 	return cmd | ||||
|  |  | |||
|  | @ -17,6 +17,7 @@ func Self() string { | |||
| // For example if current binary is "docker" at "/usr/bin/", then cmd.Path will
 | ||||
| // be set to "/usr/bin/docker".
 | ||||
| func Command(args ...string) *exec.Cmd { | ||||
| 	panicIfNotInitialized() | ||||
| 	cmd := exec.Command(Self()) | ||||
| 	cmd.Args = args | ||||
| 	return cmd | ||||
|  | @ -24,6 +25,7 @@ func Command(args ...string) *exec.Cmd { | |||
| 
 | ||||
| // CommandContext returns *exec.Cmd which has Path as current binary.
 | ||||
| func CommandContext(ctx context.Context, args ...string) *exec.Cmd { | ||||
| 	panicIfNotInitialized() | ||||
| 	cmd := exec.CommandContext(ctx, Self()) | ||||
| 	cmd.Args = args | ||||
| 	return cmd | ||||
|  |  | |||
|  | @ -9,10 +9,12 @@ import ( | |||
| 
 | ||||
| // Command is unsupported on operating systems apart from Linux, Windows, Solaris and Darwin.
 | ||||
| func Command(args ...string) *exec.Cmd { | ||||
| 	panicIfNotInitialized() | ||||
| 	return nil | ||||
| } | ||||
| 
 | ||||
| // CommandContext is unsupported on operating systems apart from Linux, Windows, Solaris and Darwin.
 | ||||
| func CommandContext(ctx context.Context, args ...string) *exec.Cmd { | ||||
| 	panicIfNotInitialized() | ||||
| 	return nil | ||||
| } | ||||
|  |  | |||
|  | @ -17,6 +17,7 @@ func Self() string { | |||
| // For example if current binary is "docker.exe" at "C:\", then cmd.Path will
 | ||||
| // be set to "C:\docker.exe".
 | ||||
| func Command(args ...string) *exec.Cmd { | ||||
| 	panicIfNotInitialized() | ||||
| 	cmd := exec.Command(Self()) | ||||
| 	cmd.Args = args | ||||
| 	return cmd | ||||
|  | @ -26,6 +27,7 @@ func Command(args ...string) *exec.Cmd { | |||
| // For example if current binary is "docker.exe" at "C:\", then cmd.Path will
 | ||||
| // be set to "C:\docker.exe".
 | ||||
| func CommandContext(ctx context.Context, args ...string) *exec.Cmd { | ||||
| 	panicIfNotInitialized() | ||||
| 	cmd := exec.CommandContext(ctx, Self()) | ||||
| 	cmd.Args = args | ||||
| 	return cmd | ||||
|  |  | |||
|  | @ -7,7 +7,10 @@ import ( | |||
| 	"path/filepath" | ||||
| ) | ||||
| 
 | ||||
| var registeredInitializers = make(map[string]func()) | ||||
| var ( | ||||
| 	registeredInitializers = make(map[string]func()) | ||||
| 	initWasCalled          = false | ||||
| ) | ||||
| 
 | ||||
| // Register adds an initialization func under the specified name
 | ||||
| func Register(name string, initializer func()) { | ||||
|  | @ -22,6 +25,7 @@ func Register(name string, initializer func()) { | |||
| // initialization function was called.
 | ||||
| func Init() bool { | ||||
| 	initializer, exists := registeredInitializers[os.Args[0]] | ||||
| 	initWasCalled = true | ||||
| 	if exists { | ||||
| 		initializer() | ||||
| 
 | ||||
|  | @ -30,6 +34,21 @@ func Init() bool { | |||
| 	return false | ||||
| } | ||||
| 
 | ||||
| func panicIfNotInitialized() { | ||||
| 	if !initWasCalled { | ||||
| 		// The reexec package is used to run subroutines in
 | ||||
| 		// subprocesses which would otherwise have unacceptable side
 | ||||
| 		// effects on the main thread.  If you found this error, then
 | ||||
| 		// your program uses a package which needs to do this.  In
 | ||||
| 		// order for that to work, main() should start with this
 | ||||
| 		// boilerplate, or an equivalent:
 | ||||
| 		//     if reexec.Init() {
 | ||||
| 		//         return
 | ||||
| 		//     }
 | ||||
| 		panic("a library subroutine needed to run a subprocess, but reexec.Init() was not called in main()") | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| func naiveSelf() string { | ||||
| 	name := os.Args[0] | ||||
| 	if filepath.Base(name) == name { | ||||
|  |  | |||
|  | @ -1,5 +1,14 @@ | |||
| # This file is is the configuration file for all tools | ||||
| # that use the containers/storage library. | ||||
| # that use the containers/storage library. The storage.conf file | ||||
| # overrides all other storage.conf files. Container engines using the | ||||
| # container/storage library do not inherit fields from other storage.conf | ||||
| # files. | ||||
| # | ||||
| #  Note: The storage.conf file overrides other storage.conf files based on this precedence: | ||||
| #      /usr/containers/storage.conf | ||||
| #      /etc/containers/storage.conf | ||||
| #      $HOME/.config/containers/storage.conf | ||||
| #      $XDG_CONFIG_HOME/containers/storage.conf (If XDG_CONFIG_HOME is set) | ||||
| # See man 5 containers-storage.conf for more information | ||||
| # The "container storage" table contains all of the server options. | ||||
| [storage] | ||||
|  |  | |||
|  | @ -2371,22 +2371,16 @@ func (s *store) DeleteImage(id string, commit bool) (layers []string, err error) | |||
| 		if err != nil { | ||||
| 			return nil, err | ||||
| 		} | ||||
| 		childrenByParent := make(map[string]*[]string) | ||||
| 		childrenByParent := make(map[string][]string) | ||||
| 		for _, layer := range layers { | ||||
| 			parent := layer.Parent | ||||
| 			if list, ok := childrenByParent[parent]; ok { | ||||
| 				newList := append(*list, layer.ID) | ||||
| 				childrenByParent[parent] = &newList | ||||
| 			} else { | ||||
| 				childrenByParent[parent] = &([]string{layer.ID}) | ||||
| 			} | ||||
| 			childrenByParent[layer.Parent] = append(childrenByParent[layer.Parent], layer.ID) | ||||
| 		} | ||||
| 		otherImagesByTopLayer := make(map[string]string) | ||||
| 		otherImagesTopLayers := make(map[string]struct{}) | ||||
| 		for _, img := range images { | ||||
| 			if img.ID != id { | ||||
| 				otherImagesByTopLayer[img.TopLayer] = img.ID | ||||
| 				otherImagesTopLayers[img.TopLayer] = struct{}{} | ||||
| 				for _, layerID := range img.MappedTopLayers { | ||||
| 					otherImagesByTopLayer[layerID] = img.ID | ||||
| 					otherImagesTopLayers[layerID] = struct{}{} | ||||
| 				} | ||||
| 			} | ||||
| 		} | ||||
|  | @ -2396,43 +2390,46 @@ func (s *store) DeleteImage(id string, commit bool) (layers []string, err error) | |||
| 			} | ||||
| 		} | ||||
| 		layer := image.TopLayer | ||||
| 		lastRemoved := "" | ||||
| 		layersToRemoveMap := make(map[string]struct{}) | ||||
| 		for layer != "" { | ||||
| 			if rcstore.Exists(layer) { | ||||
| 				break | ||||
| 			} | ||||
| 			if _, ok := otherImagesByTopLayer[layer]; ok { | ||||
| 			if _, used := otherImagesTopLayers[layer]; used { | ||||
| 				break | ||||
| 			} | ||||
| 			parent := "" | ||||
| 			if l, err := rlstore.Get(layer); err == nil { | ||||
| 				parent = l.Parent | ||||
| 			} | ||||
| 			hasOtherRefs := func() bool { | ||||
| 			hasChildrenNotBeingRemoved := func() bool { | ||||
| 				layersToCheck := []string{layer} | ||||
| 				if layer == image.TopLayer { | ||||
| 					layersToCheck = append(layersToCheck, image.MappedTopLayers...) | ||||
| 				} | ||||
| 				for _, layer := range layersToCheck { | ||||
| 					if childList, ok := childrenByParent[layer]; ok && childList != nil { | ||||
| 						children := *childList | ||||
| 						for _, child := range children { | ||||
| 							if child != lastRemoved { | ||||
| 								return true | ||||
| 					if childList := childrenByParent[layer]; len(childList) > 0 { | ||||
| 						for _, child := range childList { | ||||
| 							if _, childIsSlatedForRemoval := layersToRemoveMap[child]; childIsSlatedForRemoval { | ||||
| 								continue | ||||
| 							} | ||||
| 							return true | ||||
| 						} | ||||
| 					} | ||||
| 				} | ||||
| 				return false | ||||
| 			} | ||||
| 			if hasOtherRefs() { | ||||
| 			if hasChildrenNotBeingRemoved() { | ||||
| 				break | ||||
| 			} | ||||
| 			lastRemoved = layer | ||||
| 			if layer == image.TopLayer { | ||||
| 				layersToRemove = append(layersToRemove, image.MappedTopLayers...) | ||||
| 				for _, mappedTopLayer := range image.MappedTopLayers { | ||||
| 					layersToRemoveMap[mappedTopLayer] = struct{}{} | ||||
| 				} | ||||
| 			} | ||||
| 			layersToRemove = append(layersToRemove, lastRemoved) | ||||
| 			layersToRemove = append(layersToRemove, layer) | ||||
| 			layersToRemoveMap[layer] = struct{}{} | ||||
| 			layer = parent | ||||
| 		} | ||||
| 	} else { | ||||
|  |  | |||
|  | @ -5,13 +5,13 @@ github.com/Azure/go-ansiterm/winterm | |||
| ## explicit | ||||
| github.com/BurntSushi/toml | ||||
| github.com/BurntSushi/toml/internal | ||||
| # github.com/Microsoft/go-winio v0.5.0 | ||||
| # github.com/Microsoft/go-winio v0.5.1 | ||||
| github.com/Microsoft/go-winio | ||||
| github.com/Microsoft/go-winio/backuptar | ||||
| github.com/Microsoft/go-winio/pkg/guid | ||||
| github.com/Microsoft/go-winio/pkg/security | ||||
| github.com/Microsoft/go-winio/vhd | ||||
| # github.com/Microsoft/hcsshim v0.9.0 | ||||
| # github.com/Microsoft/hcsshim v0.9.1 | ||||
| github.com/Microsoft/hcsshim | ||||
| github.com/Microsoft/hcsshim/computestorage | ||||
| github.com/Microsoft/hcsshim/internal/cow | ||||
|  | @ -69,7 +69,7 @@ github.com/containerd/containerd/log | |||
| github.com/containerd/containerd/pkg/userns | ||||
| github.com/containerd/containerd/platforms | ||||
| github.com/containerd/containerd/sys | ||||
| # github.com/containerd/stargz-snapshotter/estargz v0.9.0 | ||||
| # github.com/containerd/stargz-snapshotter/estargz v0.10.1 | ||||
| github.com/containerd/stargz-snapshotter/estargz | ||||
| github.com/containerd/stargz-snapshotter/estargz/errorutil | ||||
| # github.com/containernetworking/cni v1.0.1 | ||||
|  | @ -106,7 +106,7 @@ github.com/containers/buildah/pkg/rusage | |||
| github.com/containers/buildah/pkg/sshagent | ||||
| github.com/containers/buildah/pkg/util | ||||
| github.com/containers/buildah/util | ||||
| # github.com/containers/common v0.46.1-0.20211115170340-7ae7bd1c3f8e | ||||
| # github.com/containers/common v0.46.1-0.20211122213330-d4e7724a0c58 | ||||
| ## explicit | ||||
| github.com/containers/common/libimage | ||||
| github.com/containers/common/libimage/manifests | ||||
|  | @ -219,7 +219,7 @@ github.com/containers/psgo/internal/dev | |||
| github.com/containers/psgo/internal/host | ||||
| github.com/containers/psgo/internal/proc | ||||
| github.com/containers/psgo/internal/process | ||||
| # github.com/containers/storage v1.37.1-0.20211014130921-5c5bf639ed01 | ||||
| # github.com/containers/storage v1.37.1-0.20211122214631-59ba58582415 | ||||
| ## explicit | ||||
| github.com/containers/storage | ||||
| github.com/containers/storage/drivers | ||||
|  |  | |||
		Loading…
	
		Reference in New Issue