From dbb18412f05989de8dfc130f8d18b2dc6d781fb1 Mon Sep 17 00:00:00 2001 From: Aman Sharma Date: Wed, 15 Dec 2021 15:33:04 +0100 Subject: [PATCH] fix(ImportCleaner): resolve imports of parent types (#4353) --- .../spoon/reflect/visitor/ImportCleaner.java | 33 +- .../reflect/visitor/ImportCleanerTest.java | 12 + .../importCleaner/DoNotRemoveSubType.java | 9 + .../TypeImportButUseSubType.java | 316 ++++++++++++++++++ 4 files changed, 366 insertions(+), 4 deletions(-) create mode 100644 src/test/resources/importCleaner/DoNotRemoveSubType.java create mode 100644 src/test/resources/importCleaner/TypeImportButUseSubType.java diff --git a/src/main/java/spoon/reflect/visitor/ImportCleaner.java b/src/main/java/spoon/reflect/visitor/ImportCleaner.java index ddc87de1beb..b1ec753af18 100644 --- a/src/main/java/spoon/reflect/visitor/ImportCleaner.java +++ b/src/main/java/spoon/reflect/visitor/ImportCleaner.java @@ -37,6 +37,7 @@ import spoon.support.Experimental; import spoon.support.util.ModelList; import spoon.support.visitor.ClassTypingContext; +import spoon.support.visitor.equals.EqualsVisitor; import java.util.ArrayList; import java.util.Collection; @@ -176,8 +177,7 @@ void addImport(CtReference ref) { // we would like to add an import, but we don't know to where return; } - if (ref instanceof CtFieldReference - && !isReferenceToFieldPresentInImports((CtFieldReference) ref)) { + if (ref instanceof CtFieldReference && !isReferencePresentInImports(ref)) { return; } CtTypeReference topLevelTypeRef = typeRef.getTopLevelType(); @@ -185,6 +185,15 @@ void addImport(CtReference ref) { //it is reference to a type of this CompilationUnit. Do not add it return; } + if (ref instanceof CtTypeReference + && !isReferencePresentInImports(topLevelTypeRef) + && topLevelTypeRef != ref + && !isReferencePresentInImports(ref)) { + // check if a top level type has been imported + // if it has been, we don't need to add a separate import for its subtype + // last condition ensures that if only the subtype has been imported, we do not remove it + return; + } CtPackageReference packageRef = topLevelTypeRef.getPackage(); if (packageRef == null) { return; @@ -210,10 +219,26 @@ void addImport(CtReference ref) { } } - private boolean isReferenceToFieldPresentInImports(CtFieldReference ref) { + private boolean isReferencePresentInImports(CtReference ref) { return compilationUnit.getImports() .stream() - .anyMatch(ctImport -> ctImport.getReference() != null && ctImport.getReference().equals(ref)); + .anyMatch(ctImport -> ctImport.getReference() != null + && isEqualAfterSkippingRole(ctImport.getReference(), ref, CtRole.TYPE_ARGUMENT)); + } + + /** + * Checks if element and other are equal if comparison for `role` value is skipped + */ + private boolean isEqualAfterSkippingRole(CtElement element, CtElement other, CtRole role) { + EqualsVisitor equalsVisitor = new EqualsVisitor(); + boolean isEqual = equalsVisitor.checkEquals(element, other); + if (isEqual) { + return true; + } + if (role == equalsVisitor.getNotEqualRole()) { + return true; + } + return false; } void onCompilationUnitProcessed(CtCompilationUnit compilationUnit) { diff --git a/src/test/java/spoon/reflect/visitor/ImportCleanerTest.java b/src/test/java/spoon/reflect/visitor/ImportCleanerTest.java index 3ccd5acd4f7..e993f272b71 100644 --- a/src/test/java/spoon/reflect/visitor/ImportCleanerTest.java +++ b/src/test/java/spoon/reflect/visitor/ImportCleanerTest.java @@ -15,6 +15,18 @@ public class ImportCleanerTest { + @Test + void testDoesNotRemoveImportOfSubType() { + // contract: The import cleaner should not remove import of subtype if it is used by its simply qualified name + testImportCleanerDoesNotAlterImports("src/test/resources/importCleaner/DoNotRemoveSubType.java", "importCleaner.DoNotRemoveSubType"); + } + + @Test + void testDoesNotImportTypeWhoseParentTypeIsAlreadyImported() { + // contract: The import cleaner should not import the subtype if its parent has already been imported + testImportCleanerDoesNotAlterImports("src/test/resources/importCleaner/TypeImportButUseSubType.java", "importCleaner.TypeImportButUseSubType"); + } + @Test void testDoesNotRemoveImportForStaticFieldOfStaticClass() { // contract: The import cleaner should not remove import of the static field diff --git a/src/test/resources/importCleaner/DoNotRemoveSubType.java b/src/test/resources/importCleaner/DoNotRemoveSubType.java new file mode 100644 index 00000000000..69d234e9b38 --- /dev/null +++ b/src/test/resources/importCleaner/DoNotRemoveSubType.java @@ -0,0 +1,9 @@ +package importCleaner; + +import java.util.Map.Entry; + +public class DoNotRemoveSubType { + public void hey() { + Entry f = null; + } +} diff --git a/src/test/resources/importCleaner/TypeImportButUseSubType.java b/src/test/resources/importCleaner/TypeImportButUseSubType.java new file mode 100644 index 00000000000..25fb99c096a --- /dev/null +++ b/src/test/resources/importCleaner/TypeImportButUseSubType.java @@ -0,0 +1,316 @@ +package importCleaner; + +import com.basho.riak.client.api.commands.buckets.StoreBucketProperties; +import com.basho.riak.client.api.commands.kv.StoreValue; +import com.basho.riak.client.api.commands.kv.UpdateValue; +import com.basho.riak.client.core.RiakFuture; +import com.basho.riak.client.core.query.RiakObject; +import com.basho.riak.client.core.query.indexes.LongIntIndex; +import com.basho.riak.client.core.util.BinaryValue; +import site.ycsb.*; + +import java.io.IOException; +import java.io.InputStream; +import java.util.*; +import java.util.concurrent.TimeUnit; +import java.util.concurrent.TimeoutException; + +import com.basho.riak.client.api.RiakClient; +import com.basho.riak.client.api.cap.Quorum; +import com.basho.riak.client.api.commands.indexes.IntIndexQuery; +import com.basho.riak.client.api.commands.kv.DeleteValue; +import com.basho.riak.client.api.commands.kv.FetchValue; +import com.basho.riak.client.core.RiakCluster; +import com.basho.riak.client.core.RiakNode; +import com.basho.riak.client.core.query.Location; +import com.basho.riak.client.core.query.Namespace; + +import static site.ycsb.db.riak.RiakUtils.createResultHashMap; +import static site.ycsb.db.riak.RiakUtils.getKeyAsLong; +import static site.ycsb.db.riak.RiakUtils.serializeTable; + +public class TypeImportButUseSubType { + private static final String HOST_PROPERTY = "riak.hosts"; + private static final String PORT_PROPERTY = "riak.port"; + private static final String BUCKET_TYPE_PROPERTY = "riak.bucket_type"; + private static final String R_VALUE_PROPERTY = "riak.r_val"; + private static final String W_VALUE_PROPERTY = "riak.w_val"; + private static final String READ_RETRY_COUNT_PROPERTY = "riak.read_retry_count"; + private static final String WAIT_TIME_BEFORE_RETRY_PROPERTY = "riak.wait_time_before_retry"; + private static final String TRANSACTION_TIME_LIMIT_PROPERTY = "riak.transaction_time_limit"; + private static final String STRONG_CONSISTENCY_PROPERTY = "riak.strong_consistency"; + private static final String STRONG_CONSISTENT_SCANS_BUCKET_TYPE_PROPERTY = "riak.strong_consistent_scans_bucket_type"; + private static final String DEBUG_PROPERTY = "riak.debug"; + + private static final int TIME_OUT = 1; + + private String[] hosts; + private int port; + private String bucketType; + private String bucketType2i; + private Quorum rvalue; + private Quorum wvalue; + private int readRetryCount; + private int waitTimeBeforeRetry; + private int transactionTimeLimit; + private boolean strongConsistency; + private String strongConsistentScansBucketType; + private boolean performStrongConsistentScans; + private boolean debug; + + private RiakClient riakClient; + private RiakCluster riakCluster; + + private void loadDefaultProperties() { + InputStream propFile = RiakKVClient.class.getClassLoader().getResourceAsStream("riak.properties"); + Properties propsPF = new Properties(System.getProperties()); + + try { + propsPF.load(propFile); + } catch (IOException e) { + e.printStackTrace(); + } + + hosts = propsPF.getProperty(HOST_PROPERTY).split(","); + port = Integer.parseInt(propsPF.getProperty(PORT_PROPERTY)); + bucketType = propsPF.getProperty(BUCKET_TYPE_PROPERTY); + rvalue = new Quorum(Integer.parseInt(propsPF.getProperty(R_VALUE_PROPERTY))); + wvalue = new Quorum(Integer.parseInt(propsPF.getProperty(W_VALUE_PROPERTY))); + readRetryCount = Integer.parseInt(propsPF.getProperty(READ_RETRY_COUNT_PROPERTY)); + waitTimeBeforeRetry = Integer.parseInt(propsPF.getProperty(WAIT_TIME_BEFORE_RETRY_PROPERTY)); + transactionTimeLimit = Integer.parseInt(propsPF.getProperty(TRANSACTION_TIME_LIMIT_PROPERTY)); + strongConsistency = Boolean.parseBoolean(propsPF.getProperty(STRONG_CONSISTENCY_PROPERTY)); + strongConsistentScansBucketType = propsPF.getProperty(STRONG_CONSISTENT_SCANS_BUCKET_TYPE_PROPERTY); + debug = Boolean.parseBoolean(propsPF.getProperty(DEBUG_PROPERTY)); + } + + private void loadProperties() { + // First, load the default properties... + loadDefaultProperties(); + + // ...then, check for some props set at command line! + Properties props = getProperties(); + + String portString = props.getProperty(PORT_PROPERTY); + if (portString != null) { + port = Integer.parseInt(portString); + } + + String hostsString = props.getProperty(HOST_PROPERTY); + if (hostsString != null) { + hosts = hostsString.split(","); + } + + String bucketTypeString = props.getProperty(BUCKET_TYPE_PROPERTY); + if (bucketTypeString != null) { + bucketType = bucketTypeString; + } + + String rValueString = props.getProperty(R_VALUE_PROPERTY); + if (rValueString != null) { + rvalue = new Quorum(Integer.parseInt(rValueString)); + } + + String wValueString = props.getProperty(W_VALUE_PROPERTY); + if (wValueString != null) { + wvalue = new Quorum(Integer.parseInt(wValueString)); + } + + String readRetryCountString = props.getProperty(READ_RETRY_COUNT_PROPERTY); + if (readRetryCountString != null) { + readRetryCount = Integer.parseInt(readRetryCountString); + } + + String waitTimeBeforeRetryString = props.getProperty(WAIT_TIME_BEFORE_RETRY_PROPERTY); + if (waitTimeBeforeRetryString != null) { + waitTimeBeforeRetry = Integer.parseInt(waitTimeBeforeRetryString); + } + + String transactionTimeLimitString = props.getProperty(TRANSACTION_TIME_LIMIT_PROPERTY); + if (transactionTimeLimitString != null) { + transactionTimeLimit = Integer.parseInt(transactionTimeLimitString); + } + + String strongConsistencyString = props.getProperty(STRONG_CONSISTENCY_PROPERTY); + if (strongConsistencyString != null) { + strongConsistency = Boolean.parseBoolean(strongConsistencyString); + } + + String strongConsistentScansBucketTypeString = props.getProperty(STRONG_CONSISTENT_SCANS_BUCKET_TYPE_PROPERTY); + if (strongConsistentScansBucketTypeString != null) { + strongConsistentScansBucketType = strongConsistentScansBucketTypeString; + } + + String debugString = props.getProperty(DEBUG_PROPERTY); + if (debugString != null) { + debug = Boolean.parseBoolean(debugString); + } + } + + public void init() throws Exception { + loadProperties(); + + RiakNode.Builder builder = new RiakNode.Builder().withRemotePort(port); + List nodes = RiakNode.Builder.buildNodes(builder, Arrays.asList(hosts)); + riakCluster = new RiakCluster.Builder(nodes).build(); + + try { + riakCluster.start(); + riakClient = new RiakClient(riakCluster); + } catch (Exception e) { + System.err.println("Unable to properly start up the cluster. Reason: " + e.toString()); + throw new Exception(e); + } + + // If strong consistency is in use, we need to change the bucket-type where the 2i indexes will be stored. + if (strongConsistency && !strongConsistentScansBucketType.isEmpty()) { + // The 2i indexes have to be stored in the appositely created strongConsistentScansBucketType: this however has + // to be done only if the user actually created it! So, if the latter doesn't exist, then the scan transactions + // will not be performed at all. + bucketType2i = strongConsistentScansBucketType; + performStrongConsistentScans = true; + } else { + // If instead eventual consistency is in use, then the 2i indexes have to be stored in the bucket-type + // indicated with the bucketType variable. + bucketType2i = bucketType; + performStrongConsistentScans = false; + } + + if (debug) { + System.err.println("DEBUG ENABLED. Configuration parameters:"); + System.err.println("-----------------------------------------"); + System.err.println("Hosts: " + Arrays.toString(hosts)); + System.err.println("Port: " + port); + System.err.println("Bucket Type: " + bucketType); + System.err.println("R Val: " + rvalue.toString()); + System.err.println("W Val: " + wvalue.toString()); + System.err.println("Read Retry Count: " + readRetryCount); + System.err.println("Wait Time Before Retry: " + waitTimeBeforeRetry + " ms"); + System.err.println("Transaction Time Limit: " + transactionTimeLimit + " s"); + System.err.println("Consistency model: " + (strongConsistency ? "Strong" : "Eventual")); + + if (strongConsistency) { + System.err.println("Strong Consistent Scan Transactions " + (performStrongConsistentScans ? "" : "NOT ") + + "allowed."); + } + } + } + + @Override + public int read(String table, String key, Set fields, Map result) { + Location location = new Location(new Namespace(bucketType, table), key); + FetchValue fv = new FetchValue.Builder(location).withOption(FetchValue.Option.R, rvalue).build(); + FetchValue.Response response; + + try { + response = fetch(fv); + + if (response.isNotFound()) { + if (debug) { + System.err.println("Unable to read key " + key + ". Reason: NOT FOUND"); + } + + return 2; + } + } catch (TimeoutException e) { + if (debug) { + System.err.println("Unable to read key " + key + ". Reason: TIME OUT"); + } + + return TIME_OUT; + } catch (Exception e) { + if (debug) { + System.err.println("Unable to read key " + key + ". Reason: " + e.toString()); + } + + return 3; + } + + // Create the result HashMap. + HashMap partialResult = new HashMap<>(); + createResultHashMap(fields, response, partialResult); + result.putAll(partialResult); + return 1; + } + + @Override + public int scan(String table, String startkey, int recordcount, Set fields, + Vector> result) { + if (strongConsistency && !performStrongConsistentScans) { + return 403; + } + + // The strong consistent bucket-type is not capable of storing 2i indexes. So, we need to read them from the fake + // one (which we use only to store indexes). This is why, when using such a consistency model, the bucketType2i + // variable is set to FAKE_BUCKET_TYPE. + IntIndexQuery iiq = new IntIndexQuery + .Builder(new Namespace(bucketType2i, table), "key", getKeyAsLong(startkey), Long.MAX_VALUE) + .withMaxResults(recordcount) + .withPaginationSort(true) + .build(); + + Location location; + RiakFuture future = riakClient.executeAsync(iiq); + + try { + IntIndexQuery.Response response = future.get(transactionTimeLimit, TimeUnit.SECONDS); + List entries = response.getEntries(); + + // If no entries were retrieved, then something bad happened... + if (entries.size() == 0) { + if (debug) { + System.err.println("Unable to scan any record starting from key " + startkey + ", aborting transaction. " + + "Reason: NOT FOUND"); + } + + return 404; + } + + for (IntIndexQuery.Response.Entry entry : entries) { + // If strong consistency is in use, then the actual location of the object we want to read is obtained by + // fetching the key from the one retrieved with the 2i indexes search operation. + if (strongConsistency) { + location = new Location(new Namespace(bucketType, table), entry.getRiakObjectLocation().getKeyAsString()); + } else { + location = entry.getRiakObjectLocation(); + } + + FetchValue fv = new FetchValue.Builder(location) + .withOption(FetchValue.Option.R, rvalue) + .build(); + + FetchValue.Response keyResponse = fetch(fv); + + if (keyResponse.isNotFound()) { + if (debug) { + System.err.println("Unable to scan all requested records starting from key " + startkey + ", aborting " + + "transaction. Reason: NOT FOUND"); + } + + return 404; + } + + // Create the partial result to add to the result vector. + HashMap partialResult = new HashMap<>(); + createResultHashMap(fields, keyResponse, partialResult); + result.add(partialResult); + } + } catch (TimeoutException e) { + if (debug) { + System.err.println("Unable to scan all requested records starting from key " + startkey + ", aborting " + + "transaction. Reason: TIME OUT"); + } + + return TIME_OUT; + } catch (Exception e) { + if (debug) { + System.err.println("Unable to scan all records starting from key " + startkey + ", aborting transaction. " + + "Reason: " + e.toString()); + } + + return 2; + } + + return 1; + } +}