mirror of https://github.com/apache/nifi.git
NIFI-4715: Update currentKeys after listing loop
ListS3 used to update currentKeys within listing loop, that causes duplicates. Because S3 returns object list in lexicographic order, if we clear currentKeys during the loop, we cannot tell if the object has been listed or not, in a case where newer object has a lexicographically former name. Signed-off-by: James Wing <jvwing@gmail.com> This closes #3116, closes #2361.
This commit is contained in:
parent
0a014dcdb1
commit
37a0e1b304
|
@ -230,7 +230,7 @@ public class ListS3 extends AbstractS3Processor {
|
||||||
final AmazonS3 client = getClient();
|
final AmazonS3 client = getClient();
|
||||||
int listCount = 0;
|
int listCount = 0;
|
||||||
int totalListCount = 0;
|
int totalListCount = 0;
|
||||||
long maxTimestamp = currentTimestamp;
|
long latestListedTimestampInThisCycle = currentTimestamp;
|
||||||
String delimiter = context.getProperty(DELIMITER).getValue();
|
String delimiter = context.getProperty(DELIMITER).getValue();
|
||||||
String prefix = context.getProperty(PREFIX).evaluateAttributeExpressions().getValue();
|
String prefix = context.getProperty(PREFIX).evaluateAttributeExpressions().getValue();
|
||||||
|
|
||||||
|
@ -252,6 +252,9 @@ public class ListS3 extends AbstractS3Processor {
|
||||||
}
|
}
|
||||||
|
|
||||||
VersionListing versionListing;
|
VersionListing versionListing;
|
||||||
|
final Set<String> listedKeys = new HashSet<>();
|
||||||
|
getLogger().trace("Start listing, listingTimestamp={}, currentTimestamp={}, currentKeys={}", new Object[]{listingTimestamp, currentTimestamp, currentKeys});
|
||||||
|
|
||||||
do {
|
do {
|
||||||
versionListing = bucketLister.listVersions();
|
versionListing = bucketLister.listVersions();
|
||||||
for (S3VersionSummary versionSummary : versionListing.getVersionSummaries()) {
|
for (S3VersionSummary versionSummary : versionListing.getVersionSummaries()) {
|
||||||
|
@ -262,6 +265,8 @@ public class ListS3 extends AbstractS3Processor {
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
getLogger().trace("Listed key={}, lastModified={}, currentKeys={}", new Object[]{versionSummary.getKey(), lastModified, currentKeys});
|
||||||
|
|
||||||
// Create the attributes
|
// Create the attributes
|
||||||
final Map<String, String> attributes = new HashMap<>();
|
final Map<String, String> attributes = new HashMap<>();
|
||||||
attributes.put(CoreAttributes.FILENAME.key(), versionSummary.getKey());
|
attributes.put(CoreAttributes.FILENAME.key(), versionSummary.getKey());
|
||||||
|
@ -287,14 +292,17 @@ public class ListS3 extends AbstractS3Processor {
|
||||||
flowFile = session.putAllAttributes(flowFile, attributes);
|
flowFile = session.putAllAttributes(flowFile, attributes);
|
||||||
session.transfer(flowFile, REL_SUCCESS);
|
session.transfer(flowFile, REL_SUCCESS);
|
||||||
|
|
||||||
// Update state
|
// Track the latest lastModified timestamp and keys having that timestamp.
|
||||||
if (lastModified > maxTimestamp) {
|
// NOTE: Amazon S3 lists objects in UTF-8 character encoding in lexicographical order. Not ordered by timestamps.
|
||||||
maxTimestamp = lastModified;
|
if (lastModified > latestListedTimestampInThisCycle) {
|
||||||
currentKeys.clear();
|
latestListedTimestampInThisCycle = lastModified;
|
||||||
}
|
listedKeys.clear();
|
||||||
if (lastModified == maxTimestamp) {
|
listedKeys.add(versionSummary.getKey());
|
||||||
currentKeys.add(versionSummary.getKey());
|
|
||||||
|
} else if (lastModified == latestListedTimestampInThisCycle) {
|
||||||
|
listedKeys.add(versionSummary.getKey());
|
||||||
}
|
}
|
||||||
|
|
||||||
listCount++;
|
listCount++;
|
||||||
}
|
}
|
||||||
bucketLister.setNextMarker();
|
bucketLister.setNextMarker();
|
||||||
|
@ -304,8 +312,14 @@ public class ListS3 extends AbstractS3Processor {
|
||||||
listCount = 0;
|
listCount = 0;
|
||||||
} while (bucketLister.isTruncated());
|
} while (bucketLister.isTruncated());
|
||||||
|
|
||||||
|
// Update currentKeys.
|
||||||
|
if (latestListedTimestampInThisCycle > currentTimestamp) {
|
||||||
|
currentKeys.clear();
|
||||||
|
}
|
||||||
|
currentKeys.addAll(listedKeys);
|
||||||
|
|
||||||
// Update stateManger with the most recent timestamp
|
// Update stateManger with the most recent timestamp
|
||||||
currentTimestamp = maxTimestamp;
|
currentTimestamp = latestListedTimestampInThisCycle;
|
||||||
persistState(context);
|
persistState(context);
|
||||||
|
|
||||||
final long listMillis = TimeUnit.NANOSECONDS.toMillis(System.nanoTime() - startNanos);
|
final long listMillis = TimeUnit.NANOSECONDS.toMillis(System.nanoTime() - startNanos);
|
||||||
|
|
Loading…
Reference in New Issue