23
23
import java .io .InputStream ;
24
24
import java .io .OutputStream ;
25
25
import java .nio .file .Files ;
26
+ import java .nio .file .Path ;
27
+ import java .nio .file .attribute .BasicFileAttributeView ;
28
+ import java .util .HashMap ;
29
+ import java .util .Map ;
26
30
import java .util .zip .GZIPOutputStream ;
27
31
28
32
import org .apache .commons .compress .archivers .tar .TarArchiveEntry ;
39
43
import org .codehaus .plexus .archiver .util .Streams ;
40
44
import org .codehaus .plexus .components .io .attributes .PlexusIoResourceAttributes ;
41
45
import org .codehaus .plexus .components .io .functions .SymlinkDestinationSupplier ;
46
+ import org .codehaus .plexus .components .io .resources .PlexusIoFileResource ;
42
47
import org .codehaus .plexus .components .io .resources .PlexusIoResource ;
43
48
import org .codehaus .plexus .util .IOUtil ;
44
49
import org .codehaus .plexus .util .StringUtils ;
@@ -65,6 +70,8 @@ public class TarArchiver extends AbstractArchiver {
65
70
66
71
private TarArchiveOutputStream tOut ;
67
72
73
+ private final Map <Object , String > seenFiles = new HashMap <>(10 );
74
+
68
75
/**
69
76
* Set how to handle long files, those with a path>100 chars.
70
77
* Optional, default=warn.
@@ -177,7 +184,8 @@ protected void tarFile(ArchiveEntry entry, TarArchiveOutputStream tOut, String v
177
184
return ;
178
185
}
179
186
180
- if (entry .getResource ().isDirectory () && !vPath .endsWith ("/" )) {
187
+ final PlexusIoResource ioResource = entry .getResource ();
188
+ if (ioResource .isDirectory () && !vPath .endsWith ("/" )) {
181
189
vPath += "/" ;
182
190
}
183
191
@@ -194,7 +202,7 @@ protected void tarFile(ArchiveEntry entry, TarArchiveOutputStream tOut, String v
194
202
InputStream fIn = null ;
195
203
196
204
try {
197
- TarArchiveEntry te ;
205
+ TarArchiveEntry te = null ;
198
206
if (!longFileMode .isGnuMode ()
199
207
&& pathLength >= org .apache .commons .compress .archivers .tar .TarConstants .NAMELEN ) {
200
208
int maxPosixPathLen = org .apache .commons .compress .archivers .tar .TarConstants .NAMELEN
@@ -233,18 +241,43 @@ protected void tarFile(ArchiveEntry entry, TarArchiveOutputStream tOut, String v
233
241
}
234
242
}
235
243
244
+ boolean doCopy = true ;
236
245
if (entry .getType () == ArchiveEntry .SYMLINK ) {
237
- final SymlinkDestinationSupplier plexusIoSymlinkResource =
238
- (SymlinkDestinationSupplier ) entry .getResource ();
246
+ final SymlinkDestinationSupplier plexusIoSymlinkResource = (SymlinkDestinationSupplier ) ioResource ;
239
247
240
248
te = new TarArchiveEntry (vPath , TarArchiveEntry .LF_SYMLINK );
241
249
te .setLinkName (plexusIoSymlinkResource .getSymlinkDestination ());
242
- } else {
250
+ doCopy = false ;
251
+ } else if (options .getPreserveHardLinks ()
252
+ && ioResource .isFile ()
253
+ && ioResource instanceof PlexusIoFileResource ) {
254
+ final PlexusIoFileResource fileResource = (PlexusIoFileResource ) ioResource ;
255
+ final Path file = fileResource .getFile ().toPath ();
256
+ if (Files .exists (file )) {
257
+ final BasicFileAttributeView fileAttributeView =
258
+ Files .getFileAttributeView (file , BasicFileAttributeView .class );
259
+ if (fileAttributeView != null ) {
260
+ final Object fileKey =
261
+ fileAttributeView .readAttributes ().fileKey ();
262
+ if (fileKey != null ) {
263
+ final String seenFile = this .seenFiles .get (fileKey );
264
+ if (seenFile != null ) {
265
+ te = new TarArchiveEntry (vPath , TarArchiveEntry .LF_LINK );
266
+ te .setLinkName (seenFile );
267
+ doCopy = false ;
268
+ } else {
269
+ this .seenFiles .put (fileKey , vPath );
270
+ }
271
+ }
272
+ }
273
+ }
274
+ }
275
+ if (te == null ) {
243
276
te = new TarArchiveEntry (vPath );
244
277
}
245
278
246
279
if (getLastModifiedTime () == null ) {
247
- long teLastModified = entry . getResource () .getLastModified ();
280
+ long teLastModified = ioResource .getLastModified ();
248
281
te .setModTime (
249
282
teLastModified == PlexusIoResource .UNKNOWN_MODIFICATION_DATE
250
283
? System .currentTimeMillis ()
@@ -253,11 +286,11 @@ protected void tarFile(ArchiveEntry entry, TarArchiveOutputStream tOut, String v
253
286
te .setModTime (getLastModifiedTime ().toMillis ());
254
287
}
255
288
256
- if (entry . getType () == ArchiveEntry . SYMLINK ) {
289
+ if (! doCopy ) {
257
290
te .setSize (0 );
258
291
259
- } else if (!entry . getResource () .isDirectory ()) {
260
- final long size = entry . getResource () .getSize ();
292
+ } else if (!ioResource .isDirectory ()) {
293
+ final long size = ioResource .getSize ();
261
294
te .setSize (size == PlexusIoResource .UNKNOWN_RESOURCE_SIZE ? 0 : size );
262
295
}
263
296
te .setMode (entry .getMode ());
@@ -289,7 +322,7 @@ protected void tarFile(ArchiveEntry entry, TarArchiveOutputStream tOut, String v
289
322
tOut .putArchiveEntry (te );
290
323
291
324
try {
292
- if (entry . getResource (). isFile () && !( entry . getType () == ArchiveEntry . SYMLINK ) ) {
325
+ if (ioResource . isFile () && doCopy ) {
293
326
fIn = entry .getInputStream ();
294
327
295
328
Streams .copyFullyDontCloseOutput (fIn , tOut , "xAR" );
@@ -320,6 +353,8 @@ public class TarOptions {
320
353
321
354
private boolean preserveLeadingSlashes = false ;
322
355
356
+ private boolean preserveHardLinks = true ;
357
+
323
358
/**
324
359
* The username for the tar entry
325
360
* This is not the same as the UID.
@@ -405,6 +440,14 @@ public boolean getPreserveLeadingSlashes() {
405
440
public void setPreserveLeadingSlashes (boolean preserveLeadingSlashes ) {
406
441
this .preserveLeadingSlashes = preserveLeadingSlashes ;
407
442
}
443
+
444
+ public boolean getPreserveHardLinks () {
445
+ return preserveHardLinks ;
446
+ }
447
+
448
+ public void setPreserveHardLinks (boolean preserveHardLinks ) {
449
+ this .preserveHardLinks = preserveHardLinks ;
450
+ }
408
451
}
409
452
410
453
/**
0 commit comments