@@ -343,22 +343,26 @@ func Stat(root string, directory string, options StatOptions, globs []string) ([
343
343
344
344
// GetOptions controls parts of Get()'s behavior.
345
345
type GetOptions struct {
346
- UIDMap , GIDMap []idtools.IDMap // map from hostIDs to containerIDs in the output archive
347
- Excludes []string // contents to pretend don't exist, using the OS-specific path separator
348
- ExpandArchives bool // extract the contents of named items that are archives
349
- ChownDirs * idtools.IDPair // set ownership on directories. no effect on archives being extracted
350
- ChmodDirs * os.FileMode // set permissions on directories. no effect on archives being extracted
351
- ChownFiles * idtools.IDPair // set ownership of files. no effect on archives being extracted
352
- ChmodFiles * os.FileMode // set permissions on files. no effect on archives being extracted
353
- StripSetuidBit bool // strip the setuid bit off of items being copied. no effect on archives being extracted
354
- StripSetgidBit bool // strip the setgid bit off of items being copied. no effect on archives being extracted
355
- StripStickyBit bool // strip the sticky bit off of items being copied. no effect on archives being extracted
356
- StripXattrs bool // don't record extended attributes of items being copied. no effect on archives being extracted
357
- KeepDirectoryNames bool // don't strip the top directory's basename from the paths of items in subdirectories
358
- Rename map [string ]string // rename items with the specified names, or under the specified names
359
- NoDerefSymlinks bool // don't follow symlinks when globs match them
360
- IgnoreUnreadable bool // ignore errors reading items, instead of returning an error
361
- NoCrossDevice bool // if a subdirectory is a mountpoint with a different device number, include it but skip its contents
346
+ UIDMap , GIDMap []idtools.IDMap // map from hostIDs to containerIDs in the output archive
347
+ Excludes []string // contents to pretend don't exist, using the OS-specific path separator
348
+ ExpandArchives bool // extract the contents of named items that are archives
349
+ ChownDirs * idtools.IDPair // set ownership on directories. no effect on archives being extracted
350
+ ChmodDirs * os.FileMode // set permissions on directories. no effect on archives being extracted
351
+ ChownFiles * idtools.IDPair // set ownership of files. no effect on archives being extracted
352
+ ChmodFiles * os.FileMode // set permissions on files. no effect on archives being extracted
353
+ // ParentsPrefixToRemove is removed from source path and what is left is used as name to maintain
354
+ // the sources parent directory in the destination if is empty is used only file name as name in tar.
355
+ // Note: Parent directories are copied to the tarstream
356
+ ParentsPrefixToRemove string
357
+ StripSetuidBit bool // strip the setuid bit off of items being copied. no effect on archives being extracted
358
+ StripSetgidBit bool // strip the setgid bit off of items being copied. no effect on archives being extracted
359
+ StripStickyBit bool // strip the sticky bit off of items being copied. no effect on archives being extracted
360
+ StripXattrs bool // don't record extended attributes of items being copied. no effect on archives being extracted
361
+ KeepDirectoryNames bool // don't strip the top directory's basename from the paths of items in subdirectories
362
+ Rename map [string ]string // rename items with the specified names, or under the specified names
363
+ NoDerefSymlinks bool // don't follow symlinks when globs match them
364
+ IgnoreUnreadable bool // ignore errors reading items, instead of returning an error
365
+ NoCrossDevice bool // if a subdirectory is a mountpoint with a different device number, include it but skip its contents
362
366
}
363
367
364
368
// Get produces an archive containing items that match the specified glob
@@ -1215,6 +1219,7 @@ func copierHandlerGet(bulkWriter io.Writer, req request, pm *fileutils.PatternMa
1215
1219
return errorResponse ("copier: get: error reading info about directory %q: %v" , req .Directory , err )
1216
1220
}
1217
1221
cb := func () error {
1222
+ alreadyCopied := map [string ]struct {}{}
1218
1223
tw := tar .NewWriter (bulkWriter )
1219
1224
defer tw .Close ()
1220
1225
hardlinkChecker := new (hardlinkChecker )
@@ -1354,6 +1359,9 @@ func copierHandlerGet(bulkWriter io.Writer, req request, pm *fileutils.PatternMa
1354
1359
ok = filepath .SkipDir
1355
1360
}
1356
1361
}
1362
+ if len (req .GetOptions .ParentsPrefixToRemove ) > 0 {
1363
+ rel = filepath .Clean (strings .TrimPrefix (path , filepath .Join (req .Directory , req .GetOptions .ParentsPrefixToRemove )))
1364
+ }
1357
1365
// add the item to the outgoing tar stream
1358
1366
if err := copierHandlerGetOne (info , symlinkTarget , rel , path , options , tw , hardlinkChecker , idMappings ); err != nil {
1359
1367
if req .GetOptions .IgnoreUnreadable && errorIsPermission (err ) {
@@ -1379,17 +1387,37 @@ func copierHandlerGet(bulkWriter io.Writer, req request, pm *fileutils.PatternMa
1379
1387
if skip {
1380
1388
continue
1381
1389
}
1382
- // add the item to the outgoing tar stream. in
1383
- // cases where this was a symlink that we
1384
- // dereferenced, be sure to use the name of the
1385
- // link.
1386
- if err := copierHandlerGetOne (info , "" , filepath .Base (queue [i ]), item , req .GetOptions , tw , hardlinkChecker , idMappings ); err != nil {
1390
+
1391
+ copyFunc := func (name string , path string , fileInfo os.FileInfo ) error {
1392
+ // add the item to the outgoing tar stream. in
1393
+ // cases where this was a symlink that we
1394
+ // dereferenced, be sure to use the name of the
1395
+ // link.
1396
+ if err := copierHandlerGetOne (fileInfo , "" , name , path , req .GetOptions , tw , hardlinkChecker , idMappings ); err != nil {
1397
+ return err
1398
+ }
1399
+ itemsCopied ++
1400
+ return nil
1401
+ }
1402
+
1403
+ name := filepath .Base (queue [i ])
1404
+ if len (req .GetOptions .ParentsPrefixToRemove ) > 0 {
1405
+ name = filepath .Clean (strings .TrimPrefix (item , filepath .Join (req .Directory , req .GetOptions .ParentsPrefixToRemove )))
1406
+ alreadyCopied , err = copyParentsDirs (name , item , alreadyCopied , copyFunc )
1407
+ if err != nil {
1408
+ if req .GetOptions .IgnoreUnreadable && errorIsPermission (err ) {
1409
+ continue
1410
+ }
1411
+ return fmt .Errorf ("copier: get: %q: %w" , queue [i ], err )
1412
+ }
1413
+ }
1414
+
1415
+ if err := copyFunc (name , item , info ); err != nil {
1387
1416
if req .GetOptions .IgnoreUnreadable && errorIsPermission (err ) {
1388
1417
continue
1389
1418
}
1390
1419
return fmt .Errorf ("copier: get: %q: %w" , queue [i ], err )
1391
1420
}
1392
- itemsCopied ++
1393
1421
}
1394
1422
}
1395
1423
if itemsCopied == 0 {
@@ -1400,6 +1428,31 @@ func copierHandlerGet(bulkWriter io.Writer, req request, pm *fileutils.PatternMa
1400
1428
return & response {Stat : statResponse .Stat , Get : getResponse {}}, cb , nil
1401
1429
}
1402
1430
1431
+ func copyParentsDirs (name string , path string , alreadyCopied map [string ]struct {}, copy func (string , string , os.FileInfo ) error ) (map [string ]struct {}, error ) {
1432
+ parentName := filepath .Dir (name )
1433
+ parentPath := filepath .Dir (path )
1434
+ for parentName != "/" && parentName != "." {
1435
+ if _ , ok := alreadyCopied [parentPath ]; ok {
1436
+ parentName = filepath .Dir (parentName )
1437
+ parentPath = filepath .Dir (parentPath )
1438
+ continue
1439
+ }
1440
+
1441
+ parentInfo , err := os .Lstat (parentPath )
1442
+ if err != nil {
1443
+ return alreadyCopied , fmt .Errorf ("copier: get: lstat %q: %w" , parentPath , err )
1444
+ }
1445
+ if err := copy (strings .TrimPrefix (parentName , "/" ), parentPath , parentInfo ); err != nil {
1446
+ return alreadyCopied , err
1447
+ }
1448
+
1449
+ alreadyCopied [parentPath ] = struct {}{}
1450
+ parentName = filepath .Dir (parentName )
1451
+ parentPath = filepath .Dir (parentPath )
1452
+ }
1453
+ return alreadyCopied , nil
1454
+ }
1455
+
1403
1456
func handleRename (rename map [string ]string , name string ) string {
1404
1457
if rename == nil {
1405
1458
return name
0 commit comments