@@ -183,6 +183,7 @@ func TestIntegration(t *testing.T) {
183
183
testSBOMScan ,
184
184
testSBOMScanSingleRef ,
185
185
testMultipleCacheExports ,
186
+ testMountStubsTimestamp ,
186
187
)
187
188
}
188
189
@@ -7902,6 +7903,75 @@ func testMultipleCacheExports(t *testing.T, sb integration.Sandbox) {
7902
7903
ensureFileContents (t , filepath .Join (destDir , "unique" ), string (uniqueFile ))
7903
7904
}
7904
7905
7906
+ // https://github.com/moby/buildkit/issues/3148
7907
+ func testMountStubsTimestamp (t * testing.T , sb integration.Sandbox ) {
7908
+ c , err := New (sb .Context (), sb .Address ())
7909
+ require .NoError (t , err )
7910
+ defer c .Close ()
7911
+
7912
+ const sourceDateEpoch = int64 (1234567890 ) // Fri Feb 13 11:31:30 PM UTC 2009
7913
+ st := llb .Image ("busybox:latest" ).Run (
7914
+ llb .Args ([]string {"/bin/touch" , fmt .Sprintf ("--date=@%d" , sourceDateEpoch ),
7915
+ "/bin" ,
7916
+ "/etc" ,
7917
+ "/var" ,
7918
+ "/var/foo" ,
7919
+ "/tmp" ,
7920
+ "/tmp/foo2" ,
7921
+ "/tmp/foo2/bar" ,
7922
+ }),
7923
+ llb .AddMount ("/var/foo" , llb .Scratch (), llb .Tmpfs ()),
7924
+ llb .AddMount ("/tmp/foo2/bar" , llb .Scratch (), llb .Tmpfs ()),
7925
+ )
7926
+ def , err := st .Marshal (sb .Context ())
7927
+ require .NoError (t , err )
7928
+
7929
+ tmpDir := t .TempDir ()
7930
+ tarFile := filepath .Join (tmpDir , "out.tar" )
7931
+ tarFileW , err := os .Create (tarFile )
7932
+ require .NoError (t , err )
7933
+ defer tarFileW .Close ()
7934
+
7935
+ _ , err = c .Solve (sb .Context (), def , SolveOpt {
7936
+ Exports : []ExportEntry {
7937
+ {
7938
+ Type : ExporterTar ,
7939
+ Output : fixedWriteCloser (tarFileW ),
7940
+ },
7941
+ },
7942
+ }, nil )
7943
+ require .NoError (t , err )
7944
+ tarFileW .Close ()
7945
+
7946
+ tarFileR , err := os .Open (tarFile )
7947
+ require .NoError (t , err )
7948
+ defer tarFileR .Close ()
7949
+ tarR := tar .NewReader (tarFileR )
7950
+ touched := map [string ]* tar.Header {
7951
+ "bin/" : nil , // Regular dir
7952
+ "etc/" : nil , // Parent of file mounts (etc/{resolv.conf, hosts})
7953
+ "var/" : nil , // Parent of dir mount (var/foo/)
7954
+ "tmp/" : nil , // Grandparent of dir mount (tmp/foo2/bar/)
7955
+ // No support for reproducing the timestamps of mount point directories such as var/foo/ and tmp/foo2/bar/,
7956
+ // because the touched timestamp value is lost when the mount is unmounted.
7957
+ }
7958
+ for {
7959
+ hd , err := tarR .Next ()
7960
+ if errors .Is (err , io .EOF ) {
7961
+ break
7962
+ }
7963
+ require .NoError (t , err )
7964
+ if x , ok := touched [hd .Name ]; ok && x == nil {
7965
+ touched [hd .Name ] = hd
7966
+ }
7967
+ }
7968
+ for name , hd := range touched {
7969
+ t .Logf ("Verifying %q (%+v)" , name , hd )
7970
+ require .NotNil (t , hd , name )
7971
+ require .Equal (t , sourceDateEpoch , hd .ModTime .Unix (), name )
7972
+ }
7973
+ }
7974
+
7905
7975
func ensureFile (t * testing.T , path string ) {
7906
7976
st , err := os .Stat (path )
7907
7977
require .NoError (t , err , "expected file at %s" , path )
0 commit comments